18
18
19
19
import static com .google .errorprone .BugPattern .SeverityLevel .ERROR ;
20
20
import static com .google .errorprone .matchers .Description .NO_MATCH ;
21
+ import static java .util .Arrays .stream ;
21
22
22
23
import com .google .common .collect .ImmutableMap ;
23
24
import com .google .common .collect .Iterables ;
28
29
import com .google .errorprone .matchers .Matcher ;
29
30
import com .google .errorprone .matchers .method .MethodMatchers ;
30
31
import com .google .errorprone .util .ASTHelpers ;
32
+ import com .google .protobuf .ByteString ;
31
33
import com .sun .source .tree .ExpressionTree ;
32
34
import com .sun .source .tree .MethodInvocationTree ;
33
35
import com .sun .tools .javac .code .Symbol .MethodSymbol ;
36
+ import java .lang .reflect .InvocationTargetException ;
34
37
import java .util .function .Consumer ;
35
38
36
39
/** A {@link BugChecker}; see the associated {@link BugPattern} annotation for details. */
40
43
severity = ERROR )
41
44
public class AlwaysThrows extends BugChecker implements MethodInvocationTreeMatcher {
42
45
43
- // TODO(cushon): generalize assumptions here (e.g. about 'parse' and 'CharSequence')
44
46
@ SuppressWarnings ("UnnecessarilyFullyQualified" )
45
47
private static final ImmutableMap <String , Consumer <CharSequence >> VALIDATORS =
46
48
ImmutableMap .<String , Consumer <CharSequence >>builder ()
@@ -58,25 +60,62 @@ public class AlwaysThrows extends BugChecker implements MethodInvocationTreeMatc
58
60
.put ("java.time.ZonedDateTime" , java .time .ZonedDateTime ::parse )
59
61
.build ();
60
62
61
- private static final Matcher <ExpressionTree > MATCHER =
62
- MethodMatchers .staticMethod ()
63
- .onClassAny (VALIDATORS .keySet ())
64
- .named ("parse" )
65
- .withParameters ("java.lang.CharSequence" );
63
+ enum Apis {
64
+ PARSE_TIME (
65
+ MethodMatchers .staticMethod ()
66
+ .onClassAny (VALIDATORS .keySet ())
67
+ .named ("parse" )
68
+ .withParameters ("java.lang.CharSequence" )) {
69
+ @ Override
70
+ void validate (MethodInvocationTree tree , String argument , VisitorState state )
71
+ throws Throwable {
72
+ MethodSymbol sym = ASTHelpers .getSymbol (tree );
73
+ VALIDATORS .get (sym .owner .getQualifiedName ().toString ()).accept (argument );
74
+ }
75
+ },
76
+ BYTE_STRING (
77
+ MethodMatchers .staticMethod ()
78
+ .onClass ("com.google.protobuf.ByteString" )
79
+ .named ("fromHex" )
80
+ .withParameters ("java.lang.String" )) {
81
+ @ Override
82
+ void validate (MethodInvocationTree tree , String argument , VisitorState state )
83
+ throws Throwable {
84
+ try {
85
+ ByteString .class .getMethod ("fromHex" , String .class ).invoke (null , argument );
86
+ } catch (NoSuchMethodException | IllegalAccessException e ) {
87
+ return ;
88
+ } catch (InvocationTargetException e ) {
89
+ throw e .getCause ();
90
+ }
91
+ }
92
+ };
93
+
94
+ Apis (Matcher <ExpressionTree > matcher ) {
95
+ this .matcher = matcher ;
96
+ }
97
+
98
+ @ SuppressWarnings ("ImmutableEnumChecker" ) // is immutable
99
+ private final Matcher <ExpressionTree > matcher ;
100
+
101
+ abstract void validate (MethodInvocationTree tree , String argument , VisitorState state )
102
+ throws Throwable ;
103
+ }
66
104
67
105
@ Override
68
106
public Description matchMethodInvocation (MethodInvocationTree tree , VisitorState state ) {
69
- if (!MATCHER .matches (tree , state )) {
107
+ Apis api =
108
+ stream (Apis .values ()).filter (m -> m .matcher .matches (tree , state )).findAny ().orElse (null );
109
+ if (api == null ) {
70
110
return NO_MATCH ;
71
111
}
72
112
String argument =
73
113
ASTHelpers .constValue (Iterables .getOnlyElement (tree .getArguments ()), String .class );
74
114
if (argument == null ) {
75
115
return NO_MATCH ;
76
116
}
77
- MethodSymbol sym = ASTHelpers .getSymbol (tree );
78
117
try {
79
- VALIDATORS . get ( sym . owner . getQualifiedName (). toString ()). accept ( argument );
118
+ api . validate ( tree , argument , state );
80
119
} catch (Throwable t ) {
81
120
return buildDescription (tree )
82
121
.setMessage (
0 commit comments