001///////////////////////////////////////////////////////////////////////////////////////////////
002// checkstyle: Checks Java source code and other text files for adherence to a set of rules.
003// Copyright (C) 2001-2025 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.checks.naming;
021
022import com.puppycrawl.tools.checkstyle.api.DetailAST;
023import com.puppycrawl.tools.checkstyle.api.TokenTypes;
024import com.puppycrawl.tools.checkstyle.utils.ScopeUtil;
025
026/**
027 * Abstract class for checking a class member (field/method)'s name conforms to
028 * a specified pattern.
029 *
030 * <p>
031 * This class extends {@link AbstractNameCheck} with support for access level
032 * restrictions. This allows the check to be configured to be applied to one of
033 * the four Java access levels: {@code public}, {@code protected},
034 * {@code "package"}, and {@code private}.
035 * </p>
036 *
037 * <p>Level is configured using the following properties:
038 * <ol>
039 * <li>applyToPublic, default true;</li>
040 * <li>applyToProtected, default true;</li>
041 * <li>applyToPackage, default true;</li>
042 * <li>applyToPrivate, default true;</li>
043 * </ol>
044 *
045 */
046public abstract class AbstractAccessControlNameCheck
047    extends AbstractNameCheck {
048
049    /** If true, applies the check be public members. */
050    private boolean applyToPublic = true;
051
052    /** If true, applies the check be protected members. */
053    private boolean applyToProtected = true;
054
055    /** If true, applies the check be "package" members. */
056    private boolean applyToPackage = true;
057
058    /** If true, applies the check be private members. */
059    private boolean applyToPrivate = true;
060
061    /**
062     * Creates a new {@code AbstractAccessControlNameCheck} instance.
063     *
064     * @param format
065     *                format to check with
066     */
067    protected AbstractAccessControlNameCheck(String format) {
068        super(format);
069    }
070
071    @Override
072    protected boolean mustCheckName(DetailAST ast) {
073        return shouldCheckInScope(ast.findFirstToken(TokenTypes.MODIFIERS));
074    }
075
076    /**
077     * Should we check member with given modifiers.
078     *
079     * @param modifiers
080     *                modifiers of member to check.
081     * @return true if we should check such member.
082     */
083    protected boolean shouldCheckInScope(DetailAST modifiers) {
084        final boolean isProtected = modifiers
085                .findFirstToken(TokenTypes.LITERAL_PROTECTED) != null;
086        final boolean isPrivate = modifiers
087                .findFirstToken(TokenTypes.LITERAL_PRIVATE) != null;
088        final boolean isPublic = isPublic(modifiers);
089
090        final boolean isPackage = !(isPublic || isProtected || isPrivate);
091
092        return applyToPublic && isPublic
093                || applyToProtected && isProtected
094                || applyToPackage && isPackage
095                || applyToPrivate && isPrivate;
096    }
097
098    /**
099     * Checks if given modifiers has public access.
100     * There are 2 cases - it is either has explicit modifier, or it is
101     * in annotation or interface.
102     *
103     * @param modifiers - modifiers to check
104     * @return true if public
105     */
106    private static boolean isPublic(DetailAST modifiers) {
107        return modifiers.findFirstToken(TokenTypes.LITERAL_PUBLIC) != null
108                || ScopeUtil.isInAnnotationBlock(modifiers)
109                || ScopeUtil.isInInterfaceBlock(modifiers)
110                    // interface methods can be private
111                    && modifiers.findFirstToken(TokenTypes.LITERAL_PRIVATE) == null;
112    }
113
114    /**
115     * Setter to control if check should apply to public members.
116     *
117     * @param applyTo new value of the property.
118     */
119    public void setApplyToPublic(boolean applyTo) {
120        applyToPublic = applyTo;
121    }
122
123    /**
124     * Setter to control if check should apply to protected members.
125     *
126     * @param applyTo new value of the property.
127     */
128    public void setApplyToProtected(boolean applyTo) {
129        applyToProtected = applyTo;
130    }
131
132    /**
133     * Setter to control if check should apply to package-private members.
134     *
135     * @param applyTo new value of the property.
136     */
137    public void setApplyToPackage(boolean applyTo) {
138        applyToPackage = applyTo;
139    }
140
141    /**
142     * Setter to control if check should apply to private members.
143     *
144     * @param applyTo new value of the property.
145     */
146    public void setApplyToPrivate(boolean applyTo) {
147        applyToPrivate = applyTo;
148    }
149
150}