1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20 package com.puppycrawl.tools.checkstyle.checks.coding;
21
22 import java.util.ArrayDeque;
23 import java.util.Deque;
24 import java.util.regex.Pattern;
25
26 import com.puppycrawl.tools.checkstyle.FileStatefulCheck;
27 import com.puppycrawl.tools.checkstyle.api.AbstractCheck;
28 import com.puppycrawl.tools.checkstyle.api.DetailAST;
29 import com.puppycrawl.tools.checkstyle.api.TokenTypes;
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58 @FileStatefulCheck
59 public final class ReturnCountCheck extends AbstractCheck {
60
61
62
63
64
65 public static final String MSG_KEY = "return.count";
66
67
68
69
70 public static final String MSG_KEY_VOID = "return.countVoid";
71
72
73 private final Deque<Context> contextStack = new ArrayDeque<>();
74
75
76 private Pattern format = Pattern.compile("^equals$");
77
78
79 private int max = 2;
80
81 private int maxForVoid = 1;
82
83 private Context context;
84
85 @Override
86 public int[] getDefaultTokens() {
87 return new int[] {
88 TokenTypes.CTOR_DEF,
89 TokenTypes.METHOD_DEF,
90 TokenTypes.LAMBDA,
91 TokenTypes.LITERAL_RETURN,
92 };
93 }
94
95 @Override
96 public int[] getRequiredTokens() {
97 return new int[] {TokenTypes.LITERAL_RETURN};
98 }
99
100 @Override
101 public int[] getAcceptableTokens() {
102 return new int[] {
103 TokenTypes.CTOR_DEF,
104 TokenTypes.METHOD_DEF,
105 TokenTypes.LAMBDA,
106 TokenTypes.LITERAL_RETURN,
107 };
108 }
109
110
111
112
113
114
115
116 public void setFormat(Pattern pattern) {
117 format = pattern;
118 }
119
120
121
122
123
124
125
126
127 public void setMax(int max) {
128 this.max = max;
129 }
130
131
132
133
134
135
136
137
138 public void setMaxForVoid(int maxForVoid) {
139 this.maxForVoid = maxForVoid;
140 }
141
142 @Override
143 public void beginTree(DetailAST rootAST) {
144 context = new Context(false);
145 contextStack.clear();
146 }
147
148 @Override
149 public void visitToken(DetailAST ast) {
150 switch (ast.getType()) {
151 case TokenTypes.CTOR_DEF,
152 TokenTypes.METHOD_DEF -> visitMethodDef(ast);
153 case TokenTypes.LAMBDA -> visitLambda();
154 case TokenTypes.LITERAL_RETURN -> visitReturn(ast);
155 default -> throw new IllegalStateException(ast.toString());
156 }
157 }
158
159 @Override
160 public void leaveToken(DetailAST ast) {
161 switch (ast.getType()) {
162 case TokenTypes.CTOR_DEF:
163 case TokenTypes.METHOD_DEF:
164 case TokenTypes.LAMBDA:
165 leave(ast);
166 break;
167 case TokenTypes.LITERAL_RETURN:
168
169 break;
170 default:
171 throw new IllegalStateException(ast.toString());
172 }
173 }
174
175
176
177
178
179
180 private void visitMethodDef(DetailAST ast) {
181 contextStack.push(context);
182 final DetailAST methodNameAST = ast.findFirstToken(TokenTypes.IDENT);
183 final boolean check = !format.matcher(methodNameAST.getText()).find();
184 context = new Context(check);
185 }
186
187
188
189
190
191
192 private void leave(DetailAST ast) {
193 context.checkCount(ast);
194 context = contextStack.pop();
195 }
196
197
198
199
200 private void visitLambda() {
201 contextStack.push(context);
202 context = new Context(true);
203 }
204
205
206
207
208
209
210 private void visitReturn(DetailAST ast) {
211
212
213 if (ast.getFirstChild().getType() == TokenTypes.SEMI) {
214 context.visitLiteralReturn(maxForVoid, Boolean.TRUE);
215 }
216 else {
217 context.visitLiteralReturn(max, Boolean.FALSE);
218 }
219 }
220
221
222
223
224 private final class Context {
225
226
227 private final boolean checking;
228
229 private int count;
230
231 private Integer maxAllowed;
232
233 private boolean isVoidContext;
234
235
236
237
238
239
240 private Context(boolean checking) {
241 this.checking = checking;
242 }
243
244
245
246
247
248
249
250 public void visitLiteralReturn(int maxAssigned, Boolean voidReturn) {
251 isVoidContext = voidReturn;
252 maxAllowed = maxAssigned;
253
254 ++count;
255 }
256
257
258
259
260
261
262
263 public void checkCount(DetailAST ast) {
264 if (checking && maxAllowed != null && count > maxAllowed) {
265 if (isVoidContext) {
266 log(ast, MSG_KEY_VOID, count, maxAllowed);
267 }
268 else {
269 log(ast, MSG_KEY, count, maxAllowed);
270 }
271 }
272 }
273
274 }
275
276 }