14
14
15
15
import static java .util .Collections .emptyList ;
16
16
import static java .util .Collections .unmodifiableList ;
17
+ import static java .util .Collections .unmodifiableSet ;
17
18
import static java .util .Objects .requireNonNull ;
18
19
import static java .util .stream .Collectors .joining ;
19
- import static java .util .stream .Collectors .toList ;
20
20
import static org .assertj .core .util .Lists .list ;
21
+ import static org .assertj .core .util .Sets .newLinkedHashSet ;
21
22
22
23
import java .util .ArrayList ;
23
24
import java .util .List ;
24
25
import java .util .Objects ;
26
+ import java .util .Set ;
25
27
import java .util .regex .Pattern ;
26
28
27
29
/**
@@ -33,7 +35,7 @@ public final class FieldLocation implements Comparable<FieldLocation> {
33
35
34
36
private final String pathToUseInRules ;
35
37
private final List <String > decomposedPath ;
36
- private final List <String > pathsHierarchyToUseInRules ;
38
+ private final Set <String > pathsHierarchyToUseInRules ;
37
39
38
40
public FieldLocation (List <String > path ) {
39
41
decomposedPath = unmodifiableList (requireNonNull (path , "path cannot be null" ));
@@ -45,6 +47,46 @@ public FieldLocation(String s) {
45
47
this (list (s .split ("\\ ." )));
46
48
}
47
49
50
+ @ Override
51
+ public int compareTo (final FieldLocation other ) {
52
+ return pathToUseInRules .compareTo (other .pathToUseInRules );
53
+ }
54
+
55
+ @ Override
56
+ public boolean equals (Object obj ) {
57
+ if (this == obj ) return true ;
58
+ if (!(obj instanceof FieldLocation )) return false ;
59
+ FieldLocation that = (FieldLocation ) obj ;
60
+ return Objects .equals (pathToUseInRules , that .pathToUseInRules )
61
+ && Objects .equals (decomposedPath , that .decomposedPath )
62
+ && Objects .equals (pathsHierarchyToUseInRules , that .pathsHierarchyToUseInRules );
63
+ }
64
+
65
+ @ Override
66
+ public int hashCode () {
67
+ int result = Objects .hashCode (pathToUseInRules );
68
+ result = 31 * result + Objects .hashCode (decomposedPath );
69
+ result = 31 * result + Objects .hashCode (pathsHierarchyToUseInRules );
70
+ return result ;
71
+ }
72
+
73
+ @ Override
74
+ public String toString () {
75
+ return String .format ("<%s>" , pathToUseInRules );
76
+ }
77
+
78
+ public String shortDescription () {
79
+ return pathToUseInRules ;
80
+ }
81
+
82
+ private static String pathToUseInRules (List <String > path ) {
83
+ // remove the array sub-path, so person.children.[2].name -> person.children.name
84
+ // rules for ignoring fields don't apply at the element level (ex: children.[2]) but at the group level (ex: children).
85
+ return path .stream ()
86
+ .filter (subpath -> !subpath .startsWith ("[" ))
87
+ .collect (joining ("." ));
88
+ }
89
+
48
90
public boolean exactlyMatches (FieldLocation field ) {
49
91
return exactlyMatches (field .pathToUseInRules );
50
92
}
@@ -126,7 +168,7 @@ public boolean hierarchyMatches(String fieldPath) {
126
168
* </code></pre>
127
169
*
128
170
* @param regex the regex to test
129
- * @return true this fieldLocation or any of its parent matches the given regex., false otherwise.
171
+ * @return true, this fieldLocation or any of its parent matches the given regex., false otherwise.
130
172
*/
131
173
public boolean hierarchyMatchesRegex (Pattern regex ) {
132
174
return pathsHierarchyToUseInRules .stream ().anyMatch (path -> regex .matcher (path ).matches ());
@@ -146,43 +188,6 @@ public FieldLocation field(String field) {
146
188
return new FieldLocation (decomposedPathWithField );
147
189
}
148
190
149
- @ Override
150
- public int compareTo (final FieldLocation other ) {
151
- return pathToUseInRules .compareTo (other .pathToUseInRules );
152
- }
153
-
154
- @ Override
155
- public boolean equals (Object obj ) {
156
- if (this == obj ) return true ;
157
- if (!(obj instanceof FieldLocation )) return false ;
158
- FieldLocation that = (FieldLocation ) obj ;
159
- return Objects .equals (pathToUseInRules , that .pathToUseInRules )
160
- && Objects .equals (decomposedPath , that .decomposedPath )
161
- && Objects .equals (pathsHierarchyToUseInRules , that .pathsHierarchyToUseInRules );
162
- }
163
-
164
- @ Override
165
- public int hashCode () {
166
- return Objects .hash (pathToUseInRules , decomposedPath , pathsHierarchyToUseInRules );
167
- }
168
-
169
- @ Override
170
- public String toString () {
171
- return String .format ("<%s>" , pathToUseInRules );
172
- }
173
-
174
- public String shortDescription () {
175
- return pathToUseInRules ;
176
- }
177
-
178
- private static String pathToUseInRules (List <String > path ) {
179
- // remove the array sub-path, so person.children.[2].name -> person.children.name
180
- // rules for ignoring fields don't apply at the element level (ex: children.[2]) but at the group level (ex: children).
181
- return path .stream ()
182
- .filter (subpath -> !subpath .startsWith ("[" ))
183
- .collect (joining ("." ));
184
- }
185
-
186
191
public String getPathToUseInErrorReport () {
187
192
return String .join ("." , decomposedPath );
188
193
}
@@ -193,8 +198,13 @@ public String getFieldName() {
193
198
}
194
199
195
200
public boolean isRoot () {
196
- // root is the top level object compared or in case of the top level is a iterable/array the elements are considered as roots.
197
- // we don't do it for optional it has a 'value' field so for the moment
201
+ // Root is the top level object compared or in case of the top level is an iterable/array the elements are
202
+ // considered as roots.
203
+ // We don't do it for optional since it has a 'value' field (at least for now)
204
+ return isRootPath (pathToUseInRules );
205
+ }
206
+
207
+ private boolean isRootPath (String pathToUseInRules ) {
198
208
return pathToUseInRules .isEmpty ();
199
209
}
200
210
@@ -256,21 +266,23 @@ public boolean hasChild(FieldLocation child) {
256
266
return child .hasParent (this );
257
267
}
258
268
259
- private List <String > pathsHierarchyToUseInRules () {
260
- List <FieldLocation > fieldAndParentFields = list ();
261
- FieldLocation currentLocation = this ;
262
- while (!currentLocation .isRoot ()) {
263
- fieldAndParentFields .add (currentLocation );
264
- currentLocation = currentLocation .parent ();
269
+ private Set <String > pathsHierarchyToUseInRules () {
270
+ // using LinkedHashSet to maintain leaf to root iteration order
271
+ // so that hierarchyMatchesRegex can try matching from the longest to the shortest path
272
+ Set <String > fieldAndParentFields = newLinkedHashSet ();
273
+ String currentPath = this .pathToUseInRules ;
274
+ while (!isRootPath (currentPath )) {
275
+ fieldAndParentFields .add (currentPath );
276
+ currentPath = parent (currentPath );
265
277
}
266
- return fieldAndParentFields .stream ()
267
- .map (fieldLocation -> fieldLocation .pathToUseInRules )
268
- .collect (toList ());
278
+ return unmodifiableSet (fieldAndParentFields );
269
279
}
270
280
271
- private FieldLocation parent () {
272
- List <String > parentPath = new ArrayList <>(decomposedPath );
273
- parentPath .remove (decomposedPath .size () - 1 );
274
- return new FieldLocation (parentPath );
281
+ private String parent (String currentPath ) {
282
+ int lastDot = currentPath .lastIndexOf ('.' );
283
+ if (lastDot < 0 ) {
284
+ return "" ;
285
+ }
286
+ return currentPath .substring (0 , lastDot );
275
287
}
276
288
}
0 commit comments