001/////////////////////////////////////////////////////////////////////////////////////////////// 002// checkstyle: Checks Java source code and other text files for adherence to a set of rules. 003// Copyright (C) 2001-2026 the original author or authors. 004// 005// This library is free software; you can redistribute it and/or 006// modify it under the terms of the GNU Lesser General Public 007// License as published by the Free Software Foundation; either 008// version 2.1 of the License, or (at your option) any later version. 009// 010// This library is distributed in the hope that it will be useful, 011// but WITHOUT ANY WARRANTY; without even the implied warranty of 012// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 013// Lesser General Public License for more details. 014// 015// You should have received a copy of the GNU Lesser General Public 016// License along with this library; if not, write to the Free Software 017// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 018/////////////////////////////////////////////////////////////////////////////////////////////// 019 020package com.puppycrawl.tools.checkstyle.utils; 021 022import java.util.Optional; 023 024import com.puppycrawl.tools.checkstyle.api.DetailAST; 025import com.puppycrawl.tools.checkstyle.api.Scope; 026import com.puppycrawl.tools.checkstyle.api.TokenTypes; 027 028/** 029 * Contains utility methods for working on scope. 030 * 031 */ 032public final class ScopeUtil { 033 034 /** Prevent instantiation. */ 035 private ScopeUtil() { 036 } 037 038 /** 039 * Returns the {@code Scope} explicitly specified by the modifier set. 040 * Returns {@code null} if there are no modifiers. 041 * 042 * @param mods root node of a modifier set 043 * @return a {@code Scope} value or {@code null} 044 */ 045 public static Optional<Scope> getDeclaredScopeFromMods(DetailAST mods) { 046 Optional<Scope> result = Optional.empty(); 047 for (DetailAST token = mods.getFirstChild(); token != null; 048 token = token.getNextSibling()) { 049 result = switch (token.getType()) { 050 case TokenTypes.LITERAL_PUBLIC -> Optional.of(Scope.PUBLIC); 051 case TokenTypes.LITERAL_PROTECTED -> Optional.of(Scope.PROTECTED); 052 case TokenTypes.LITERAL_PRIVATE -> Optional.of(Scope.PRIVATE); 053 default -> result; 054 }; 055 } 056 return result; 057 } 058 059 /** 060 * Returns the {@code Scope} for a given {@code DetailAST}. 061 * 062 * @param ast the DetailAST to examine 063 * @return a {@code Scope} value 064 */ 065 public static Scope getScope(DetailAST ast) { 066 return Optional.ofNullable(ast.findFirstToken(TokenTypes.MODIFIERS)) 067 .flatMap(ScopeUtil::getDeclaredScopeFromMods) 068 .orElseGet(() -> getDefaultScope(ast)); 069 } 070 071 /** 072 * Returns the {@code Scope} specified by the modifier set. If no modifiers are present, 073 * the default scope is used. 074 * 075 * @param mods root node of a modifier set 076 * @return a {@code Scope} value 077 * @see #getDefaultScope(DetailAST) 078 */ 079 public static Scope getScopeFromMods(DetailAST mods) { 080 return getDeclaredScopeFromMods(mods) 081 .orElseGet(() -> getDefaultScope(mods)); 082 } 083 084 /** 085 * Returns the default {@code Scope} for a {@code DetailAST}. 086 * 087 * <p>The following rules are taken into account:</p> 088 * <ul> 089 * <li>enum constants are public</li> 090 * <li>enum constructors are private</li> 091 * <li>interface members are public</li> 092 * <li>everything else is package private</li> 093 * </ul> 094 * 095 * @param ast DetailAST to process 096 * @return a {@code Scope} value 097 */ 098 private static Scope getDefaultScope(DetailAST ast) { 099 final Scope result; 100 if (isInEnumBlock(ast)) { 101 if (ast.getType() == TokenTypes.ENUM_CONSTANT_DEF) { 102 result = Scope.PUBLIC; 103 } 104 else if (ast.getType() == TokenTypes.CTOR_DEF) { 105 result = Scope.PRIVATE; 106 } 107 else { 108 result = Scope.PACKAGE; 109 } 110 } 111 else if (isInInterfaceOrAnnotationBlock(ast)) { 112 result = Scope.PUBLIC; 113 } 114 else { 115 result = Scope.PACKAGE; 116 } 117 return result; 118 } 119 120 /** 121 * Returns the scope of the surrounding "block". 122 * 123 * @param node the node to return the scope for 124 * @return the Scope of the surrounding block 125 */ 126 public static Optional<Scope> getSurroundingScope(DetailAST node) { 127 Optional<Scope> returnValue = Optional.empty(); 128 DetailAST child = null; 129 for (DetailAST token = node; 130 token != null; 131 child = token, token = token.getParent()) { 132 final int type = token.getType(); 133 if (TokenUtil.isTypeDeclaration(type)) { 134 final Scope tokenScope = getScope(token); 135 if (returnValue.isEmpty() || returnValue.get().isIn(tokenScope)) { 136 returnValue = Optional.of(tokenScope); 137 } 138 } 139 else if (type == TokenTypes.LITERAL_NEW) { 140 returnValue = Optional.of(Scope.ANONINNER); 141 // because Scope.ANONINNER is not in any other Scope 142 break; 143 } 144 else if (type == TokenTypes.COMPACT_COMPILATION_UNIT 145 && !TokenUtil.isOfType(child, TokenTypes.IMPORT, 146 TokenTypes.STATIC_IMPORT, TokenTypes.MODULE_IMPORT) 147 && (returnValue.isEmpty() || returnValue.get().isIn(Scope.PACKAGE))) { 148 returnValue = Optional.of(Scope.PACKAGE); 149 } 150 } 151 152 return returnValue; 153 } 154 155 /** 156 * Returns whether a node is directly contained within a class block. 157 * 158 * @param node the node to check if directly contained within a class block. 159 * @return a {@code boolean} value 160 */ 161 public static boolean isInClassBlock(DetailAST node) { 162 return isInBlockOf(node, TokenTypes.CLASS_DEF); 163 } 164 165 /** 166 * Returns whether a node is directly contained within a record block. 167 * 168 * @param node the node to check if directly contained within a record block. 169 * @return a {@code boolean} value 170 */ 171 public static boolean isInRecordBlock(DetailAST node) { 172 return isInBlockOf(node, TokenTypes.RECORD_DEF); 173 } 174 175 /** 176 * Returns whether a node is directly contained within an interface block. 177 * 178 * @param node the node to check if directly contained within an interface block. 179 * @return a {@code boolean} value 180 */ 181 public static boolean isInInterfaceBlock(DetailAST node) { 182 return isInBlockOf(node, TokenTypes.INTERFACE_DEF); 183 } 184 185 /** 186 * Returns whether a node is directly contained within an annotation block. 187 * 188 * @param node the node to check if directly contained within an annotation block. 189 * @return a {@code boolean} value 190 */ 191 public static boolean isInAnnotationBlock(DetailAST node) { 192 return isInBlockOf(node, TokenTypes.ANNOTATION_DEF); 193 } 194 195 /** 196 * Returns whether a node is directly contained within a specified block. 197 * 198 * @param node the node to check if directly contained within a specified block. 199 * @param tokenType type of token. 200 * @return a {@code boolean} value 201 */ 202 public static boolean isInBlockOf(DetailAST node, int tokenType) { 203 boolean returnValue = false; 204 205 // Loop up looking for a containing interface block 206 for (DetailAST token = node.getParent(); 207 token != null; token = token.getParent()) { 208 if (TokenUtil.isOfType(token, TokenTypes.LITERAL_NEW, tokenType) 209 || TokenUtil.isTypeDeclaration(token.getType())) { 210 returnValue = token.getType() == tokenType; 211 break; 212 } 213 } 214 return returnValue; 215 } 216 217 /** 218 * Returns whether a node is directly contained within an interface or 219 * annotation block. 220 * 221 * @param node the node to check if directly contained within an interface 222 * or annotation block. 223 * @return a {@code boolean} value 224 */ 225 public static boolean isInInterfaceOrAnnotationBlock(DetailAST node) { 226 return isInInterfaceBlock(node) || isInAnnotationBlock(node); 227 } 228 229 /** 230 * Returns whether a node is directly contained within an enum block. 231 * 232 * @param node the node to check if directly contained within an enum block. 233 * @return a {@code boolean} value 234 */ 235 public static boolean isInEnumBlock(DetailAST node) { 236 boolean returnValue = false; 237 238 // Loop up looking for a containing interface block 239 for (DetailAST token = node.getParent(); 240 token != null; token = token.getParent()) { 241 if (TokenUtil.isOfType(token, TokenTypes.INTERFACE_DEF, 242 TokenTypes.ANNOTATION_DEF, TokenTypes.CLASS_DEF, 243 TokenTypes.LITERAL_NEW, TokenTypes.ENUM_DEF)) { 244 returnValue = token.getType() == TokenTypes.ENUM_DEF; 245 break; 246 } 247 } 248 249 return returnValue; 250 } 251 252 /** 253 * Returns whether the scope of a node is restricted to a code block. 254 * A code block is a method or constructor body, an initializer block, or lambda body. 255 * 256 * @param node the node to check 257 * @return a {@code boolean} value 258 */ 259 public static boolean isInCodeBlock(DetailAST node) { 260 boolean returnValue = false; 261 final int[] tokenTypes = { 262 TokenTypes.METHOD_DEF, 263 TokenTypes.CTOR_DEF, 264 TokenTypes.INSTANCE_INIT, 265 TokenTypes.STATIC_INIT, 266 TokenTypes.LAMBDA, 267 TokenTypes.COMPACT_CTOR_DEF, 268 }; 269 270 // Loop up looking for a containing code block 271 for (DetailAST token = node.getParent(); 272 token != null; 273 token = token.getParent()) { 274 if (TokenUtil.isOfType(token, tokenTypes)) { 275 returnValue = true; 276 break; 277 } 278 } 279 280 return returnValue; 281 } 282 283 /** 284 * Returns whether a node is contained in the outermost type block. 285 * 286 * @param node the node to check 287 * @return a {@code boolean} value 288 */ 289 public static boolean isOuterMostType(DetailAST node) { 290 boolean returnValue = true; 291 for (DetailAST parent = node.getParent(); 292 parent != null; 293 parent = parent.getParent()) { 294 if (TokenUtil.isTypeDeclaration(parent.getType()) 295 || parent.getType() == TokenTypes.COMPACT_COMPILATION_UNIT) { 296 returnValue = false; 297 break; 298 } 299 } 300 301 return returnValue; 302 } 303 304 /** 305 * Determines whether a node is a local variable definition. 306 * I.e. if it is declared in a code block, a for initializer, 307 * or a catch parameter. 308 * 309 * @param node the node to check. 310 * @return whether aAST is a local variable definition. 311 */ 312 public static boolean isLocalVariableDef(DetailAST node) { 313 final boolean localVariableDef; 314 // variable declaration? 315 if (node.getType() == TokenTypes.VARIABLE_DEF) { 316 final DetailAST parent = node.getParent(); 317 localVariableDef = TokenUtil.isOfType(parent, TokenTypes.SLIST, 318 TokenTypes.FOR_INIT, TokenTypes.FOR_EACH_CLAUSE); 319 } 320 321 else if (node.getType() == TokenTypes.RESOURCE) { 322 localVariableDef = node.getChildCount() > 1; 323 } 324 325 // catch parameter? 326 else if (node.getType() == TokenTypes.PARAMETER_DEF) { 327 final DetailAST parent = node.getParent(); 328 localVariableDef = parent.getType() == TokenTypes.LITERAL_CATCH; 329 } 330 331 else { 332 localVariableDef = false; 333 } 334 335 return localVariableDef; 336 } 337 338 /** 339 * Determines whether a node is a class field definition. 340 * I.e. if a variable is not declared in a code block, a for initializer, 341 * or a catch parameter. 342 * 343 * @param node the node to check. 344 * @return whether a node is a class field definition. 345 */ 346 public static boolean isClassFieldDef(DetailAST node) { 347 return node.getType() == TokenTypes.VARIABLE_DEF 348 && !isLocalVariableDef(node); 349 } 350 351 /** 352 * Checks whether ast node is in a specific scope. 353 * 354 * @param ast the node to check. 355 * @param scope a {@code Scope} value. 356 * @return true if the ast node is in the scope. 357 */ 358 public static boolean isInScope(DetailAST ast, Scope scope) { 359 return getSurroundingScope(ast) 360 .map(surroundingScope -> { 361 return surroundingScope == scope; 362 }) 363 .orElse(Boolean.FALSE); 364 } 365 366}