1 ///////////////////////////////////////////////////////////////////////////////////////////////
2 // checkstyle: Checks Java source code and other text files for adherence to a set of rules.
3 // Copyright (C) 2001-2025 the original author or authors.
4 //
5 // This library is free software; you can redistribute it and/or
6 // modify it under the terms of the GNU Lesser General Public
7 // License as published by the Free Software Foundation; either
8 // version 2.1 of the License, or (at your option) any later version.
9 //
10 // This library is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 // Lesser General Public License for more details.
14 //
15 // You should have received a copy of the GNU Lesser General Public
16 // License along with this library; if not, write to the Free Software
17 // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 ///////////////////////////////////////////////////////////////////////////////////////////////
19
20 package com.puppycrawl.tools.checkstyle.checks.design;
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 * <div>
33 * Ensures that exception classes (classes with names conforming to some pattern
34 * and explicitly extending classes with names conforming to other
35 * pattern) are immutable, that is, that they have only final fields.
36 * </div>
37 *
38 * <p>
39 * The current algorithm is very simple: it checks that all members of exception are final.
40 * The user can still mutate an exception's instance (e.g. Throwable has a method called
41 * {@code setStackTrace} which changes the exception's stack trace). But, at least, all
42 * information provided by this exception type is unchangeable.
43 * </p>
44 *
45 * <p>
46 * Rationale: Exception instances should represent an error
47 * condition. Having non-final fields not only allows the state to be
48 * modified by accident and therefore mask the original condition but
49 * also allows developers to accidentally forget to set the initial state.
50 * In both cases, code catching the exception could draw incorrect
51 * conclusions based on the state.
52 * </p>
53 *
54 * @since 3.2
55 */
56 @FileStatefulCheck
57 public final class MutableExceptionCheck extends AbstractCheck {
58
59 /**
60 * A key is pointing to the warning message text in "messages.properties"
61 * file.
62 */
63 public static final String MSG_KEY = "mutable.exception";
64
65 /** Default value for format and extendedClassNameFormat properties. */
66 private static final String DEFAULT_FORMAT = "^.*Exception$|^.*Error$|^.*Throwable$";
67 /** Stack of checking information for classes. */
68 private final Deque<Boolean> checkingStack = new ArrayDeque<>();
69 /** Specify pattern for extended class names. */
70 private Pattern extendedClassNameFormat = Pattern.compile(DEFAULT_FORMAT);
71 /** Should we check current class or not. */
72 private boolean checking;
73 /** Specify pattern for exception class names. */
74 private Pattern format = extendedClassNameFormat;
75
76 /**
77 * Setter to specify pattern for extended class names.
78 *
79 * @param extendedClassNameFormat a {@code String} value
80 * @since 6.2
81 */
82 public void setExtendedClassNameFormat(Pattern extendedClassNameFormat) {
83 this.extendedClassNameFormat = extendedClassNameFormat;
84 }
85
86 /**
87 * Setter to specify pattern for exception class names.
88 *
89 * @param pattern the new pattern
90 * @since 3.2
91 */
92 public void setFormat(Pattern pattern) {
93 format = pattern;
94 }
95
96 @Override
97 public int[] getDefaultTokens() {
98 return getRequiredTokens();
99 }
100
101 @Override
102 public int[] getRequiredTokens() {
103 return new int[] {TokenTypes.CLASS_DEF, TokenTypes.VARIABLE_DEF};
104 }
105
106 @Override
107 public int[] getAcceptableTokens() {
108 return getRequiredTokens();
109 }
110
111 @Override
112 public void visitToken(DetailAST ast) {
113 switch (ast.getType()) {
114 case TokenTypes.CLASS_DEF -> visitClassDef(ast);
115 case TokenTypes.VARIABLE_DEF -> visitVariableDef(ast);
116 default -> throw new IllegalStateException(ast.toString());
117 }
118 }
119
120 @Override
121 public void leaveToken(DetailAST ast) {
122 if (ast.getType() == TokenTypes.CLASS_DEF) {
123 leaveClassDef();
124 }
125 }
126
127 /**
128 * Called when we start processing class definition.
129 *
130 * @param ast class definition node
131 */
132 private void visitClassDef(DetailAST ast) {
133 checkingStack.push(checking);
134 checking = isNamedAsException(ast) && isExtendedClassNamedAsException(ast);
135 }
136
137 /** Called when we leave class definition. */
138 private void leaveClassDef() {
139 checking = checkingStack.pop();
140 }
141
142 /**
143 * Checks variable definition.
144 *
145 * @param ast variable def node for check
146 */
147 private void visitVariableDef(DetailAST ast) {
148 if (checking && ast.getParent().getType() == TokenTypes.OBJBLOCK) {
149 final DetailAST modifiersAST =
150 ast.findFirstToken(TokenTypes.MODIFIERS);
151
152 if (modifiersAST.findFirstToken(TokenTypes.FINAL) == null) {
153 log(ast, MSG_KEY, ast.findFirstToken(TokenTypes.IDENT).getText());
154 }
155 }
156 }
157
158 /**
159 * Checks that a class name conforms to specified format.
160 *
161 * @param ast class definition node
162 * @return true if a class name conforms to specified format
163 */
164 private boolean isNamedAsException(DetailAST ast) {
165 final String className = ast.findFirstToken(TokenTypes.IDENT).getText();
166 return format.matcher(className).find();
167 }
168
169 /**
170 * Checks that if extended class name conforms to specified format.
171 *
172 * @param ast class definition node
173 * @return true if extended class name conforms to specified format
174 */
175 private boolean isExtendedClassNamedAsException(DetailAST ast) {
176 boolean result = false;
177 final DetailAST extendsClause = ast.findFirstToken(TokenTypes.EXTENDS_CLAUSE);
178 if (extendsClause != null) {
179 DetailAST currentNode = extendsClause;
180 while (currentNode.getLastChild() != null) {
181 currentNode = currentNode.getLastChild();
182 }
183 final String extendedClassName = currentNode.getText();
184 result = extendedClassNameFormat.matcher(extendedClassName).matches();
185 }
186 return result;
187 }
188
189 }