30
30
import java .util .List ;
31
31
import java .util .Map ;
32
32
import java .util .Map .Entry ;
33
+ import java .util .Queue ;
33
34
34
35
/**
35
36
* Thrown when a query failed on all the coordinators it was tried on. This exception may wrap
36
37
* multiple errors, that are available either as {@linkplain #getSuppressed() suppressed
37
38
* exceptions}, or via {@link #getAllErrors()} where they are grouped by node.
38
39
*/
39
40
public class AllNodesFailedException extends DriverException {
40
-
41
41
/** @deprecated Use {@link #fromErrors(List)} instead. */
42
42
@ NonNull
43
43
@ Deprecated
44
44
public static AllNodesFailedException fromErrors (@ Nullable Map <Node , Throwable > errors ) {
45
45
if (errors == null || errors .isEmpty ()) {
46
46
return new NoNodeAvailableException ();
47
47
} else {
48
- return new AllNodesFailedException (groupByNode (errors ));
48
+ return new AllNodesFailedException (groupByNode (errors ), null );
49
+ }
50
+ }
51
+
52
+ @ NonNull
53
+ public static AllNodesFailedException fromErrors (
54
+ @ Nullable List <Entry <Node , Throwable >> errors , Queue <Node > queryPlan ) {
55
+ if (errors == null || errors .isEmpty ()) {
56
+ return new NoNodeAvailableException (queryPlan );
57
+ } else {
58
+ return new AllNodesFailedException (groupByNode (errors ), queryPlan );
49
59
}
50
60
}
51
61
@@ -59,6 +69,7 @@ public static AllNodesFailedException fromErrors(@Nullable List<Entry<Node, Thro
59
69
}
60
70
61
71
private final Map <Node , List <Throwable >> errors ;
72
+ private final Queue <Node > queryPlan ;
62
73
63
74
/** @deprecated Use {@link #AllNodesFailedException(String, ExecutionInfo, Iterable)} instead. */
64
75
@ Deprecated
@@ -68,6 +79,7 @@ protected AllNodesFailedException(
68
79
@ NonNull Map <Node , Throwable > errors ) {
69
80
super (message , executionInfo , null , true );
70
81
this .errors = toDeepImmutableMap (groupByNode (errors ));
82
+ this .queryPlan = null ;
71
83
addSuppressedErrors ();
72
84
}
73
85
@@ -77,6 +89,18 @@ protected AllNodesFailedException(
77
89
@ NonNull Iterable <Entry <Node , List <Throwable >>> errors ) {
78
90
super (message , executionInfo , null , true );
79
91
this .errors = toDeepImmutableMap (errors );
92
+ this .queryPlan = null ;
93
+ addSuppressedErrors ();
94
+ }
95
+
96
+ protected AllNodesFailedException (
97
+ @ NonNull String message ,
98
+ @ Nullable ExecutionInfo executionInfo ,
99
+ @ NonNull Iterable <Entry <Node , List <Throwable >>> errors ,
100
+ @ Nullable Queue <Node > queryPlan ) {
101
+ super (message , executionInfo , null , true );
102
+ this .errors = toDeepImmutableMap (errors );
103
+ this .queryPlan = queryPlan ;
80
104
addSuppressedErrors ();
81
105
}
82
106
@@ -91,12 +115,26 @@ private void addSuppressedErrors() {
91
115
private AllNodesFailedException (Map <Node , List <Throwable >> errors ) {
92
116
this (
93
117
buildMessage (
94
- String .format ("All %d node(s) tried for the query failed" , errors .size ()), errors ),
118
+ String .format ("All %d node(s) tried for the query failed" , errors .size ()),
119
+ errors ,
120
+ null ),
95
121
null ,
96
122
errors .entrySet ());
97
123
}
98
124
99
- private static String buildMessage (String baseMessage , Map <Node , List <Throwable >> errors ) {
125
+ private AllNodesFailedException (Map <Node , List <Throwable >> errors , Queue <Node > queryPlan ) {
126
+ this (
127
+ buildMessage (
128
+ String .format ("All %d node(s) tried for the query failed" , errors .size ()),
129
+ errors ,
130
+ queryPlan ),
131
+ null ,
132
+ errors .entrySet (),
133
+ queryPlan );
134
+ }
135
+
136
+ private static String buildMessage (
137
+ String baseMessage , Map <Node , List <Throwable >> errors , Queue <Node > queryPlan ) {
100
138
int limit = Math .min (errors .size (), 3 );
101
139
Iterator <Entry <Node , List <Throwable >>> iterator =
102
140
Iterables .limit (errors .entrySet (), limit ).iterator ();
@@ -108,9 +146,14 @@ private static String buildMessage(String baseMessage, Map<Node, List<Throwable>
108
146
details .append (", " );
109
147
}
110
148
}
149
+ if (queryPlan == null ) {
150
+ return String .format (
151
+ "%s (showing first %d nodes, use getAllErrors() for more): %s" ,
152
+ baseMessage , limit , details );
153
+ }
111
154
return String .format (
112
- "%s (showing first %d nodes, use getAllErrors() for more): %s" ,
113
- baseMessage , limit , details );
155
+ "%s\n Query Plan: %s \n (showing first %d nodes, use getAllErrors() for more): %s" ,
156
+ baseMessage , queryPlan , limit , details );
114
157
}
115
158
116
159
/**
@@ -131,6 +174,10 @@ public Map<Node, Throwable> getErrors() {
131
174
return builder .build ();
132
175
}
133
176
177
+ protected Queue <Node > getQueryPlan () {
178
+ return this .queryPlan ;
179
+ }
180
+
134
181
/** An immutable map containing all errors on each tried node. */
135
182
@ NonNull
136
183
public Map <Node , List <Throwable >> getAllErrors () {
@@ -146,7 +193,7 @@ public DriverException copy() {
146
193
@ NonNull
147
194
public AllNodesFailedException reword (String newMessage ) {
148
195
return new AllNodesFailedException (
149
- buildMessage (newMessage , errors ), getExecutionInfo (), errors .entrySet ());
196
+ buildMessage (newMessage , errors , queryPlan ), getExecutionInfo (), errors .entrySet ());
150
197
}
151
198
152
199
private static Map <Node , List <Throwable >> groupByNode (Map <Node , Throwable > errors ) {
0 commit comments