All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.puppycrawl.tools.checkstyle.checks.design.FinalClassCheck Maven / Gradle / Ivy

////////////////////////////////////////////////////////////////////////////////
// checkstyle: Checks Java source code for adherence to a set of rules.
// Copyright (C) 2001-2015 the original author or authors.
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2.1 of the License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
////////////////////////////////////////////////////////////////////////////////

package com.puppycrawl.tools.checkstyle.checks.design;

import java.util.ArrayDeque;
import java.util.Deque;

import com.puppycrawl.tools.checkstyle.api.Check;
import com.puppycrawl.tools.checkstyle.api.DetailAST;
import com.puppycrawl.tools.checkstyle.api.TokenTypes;
import com.puppycrawl.tools.checkstyle.utils.ScopeUtils;

/**
 * 

* Checks that class which has only private ctors * is declared as final. Doesn't check for classes nested in interfaces * or annotations, as they are always final there. *

*

* An example of how to configure the check is: *

*
 * <module name="FinalClass"/>
 * 
* @author o_sukhodolsky */ public class FinalClassCheck extends Check { /** * A key is pointing to the warning message text in "messages.properties" * file. */ public static final String MSG_KEY = "final.class"; /** Keeps ClassDesc objects for stack of declared classes. */ private final Deque classes = new ArrayDeque<>(); @Override public int[] getDefaultTokens() { return getAcceptableTokens(); } @Override public int[] getAcceptableTokens() { return new int[] {TokenTypes.CLASS_DEF, TokenTypes.CTOR_DEF}; } @Override public int[] getRequiredTokens() { return getAcceptableTokens(); } @Override public void visitToken(DetailAST ast) { final DetailAST modifiers = ast.findFirstToken(TokenTypes.MODIFIERS); if (ast.getType() == TokenTypes.CLASS_DEF) { final boolean isFinal = modifiers.branchContains(TokenTypes.FINAL); final boolean isAbstract = modifiers.branchContains(TokenTypes.ABSTRACT); classes.push(new ClassDesc(isFinal, isAbstract)); } // ctors in enums don't matter else if (!ScopeUtils.isInEnumBlock(ast)) { final ClassDesc desc = classes.peek(); if (modifiers.branchContains(TokenTypes.LITERAL_PRIVATE)) { desc.reportPrivateCtor(); } else { desc.reportNonPrivateCtor(); } } } @Override public void leaveToken(DetailAST ast) { if (ast.getType() != TokenTypes.CLASS_DEF) { return; } final ClassDesc desc = classes.pop(); if (!desc.isDeclaredAsFinal() && !desc.isDeclaredAsAbstract() && desc.isWithPrivateCtor() && !desc.isWithNonPrivateCtor() && !ScopeUtils.isInInterfaceOrAnnotationBlock(ast)) { final String className = ast.findFirstToken(TokenTypes.IDENT).getText(); log(ast.getLineNo(), MSG_KEY, className); } } /** Maintains information about class' ctors. */ private static final class ClassDesc { /** Is class declared as final. */ private final boolean declaredAsFinal; /** Is class declared as abstract. */ private final boolean declaredAsAbstract; /** Does class have non-private ctors. */ private boolean withNonPrivateCtor; /** Does class have private ctors. */ private boolean withPrivateCtor; /** * Create a new ClassDesc instance. * @param declaredAsFinal indicates if the * class declared as final * @param declaredAsAbstract indicates if the * class declared as abstract */ ClassDesc(boolean declaredAsFinal, boolean declaredAsAbstract) { this.declaredAsFinal = declaredAsFinal; this.declaredAsAbstract = declaredAsAbstract; } /** Adds private ctor. */ private void reportPrivateCtor() { withPrivateCtor = true; } /** Adds non-private ctor. */ private void reportNonPrivateCtor() { withNonPrivateCtor = true; } /** * Does class have private ctors. * @return true if class has private ctors */ private boolean isWithPrivateCtor() { return withPrivateCtor; } /** * Does class have non-private ctors. * @return true if class has non-private ctors */ private boolean isWithNonPrivateCtor() { return withNonPrivateCtor; } /** * Is class declared as final. * @return true if class is declared as final */ private boolean isDeclaredAsFinal() { return declaredAsFinal; } /** * Is class declared as abstract. * @return true if class is declared as final */ private boolean isDeclaredAsAbstract() { return declaredAsAbstract; } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy