src.main.lombok.ast.syntaxChecks.KeywordChecks Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of lombok-ast Show documentation
Show all versions of lombok-ast Show documentation
This is a very small fork of lombok.ast as some Android tools needed a few modifications. The normal repository for lombok.ast is here https://github.com/rzwitserloot/lombok.ast and our changes for 0.2.3 are in this pull request: https://github.com/rzwitserloot/lombok.ast/pull/8
The newest version!
/*
* Copyright (C) 2010 The Project Lombok Authors.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package lombok.ast.syntaxChecks;
import static lombok.ast.syntaxChecks.MessageKey.*;
import static lombok.ast.Message.*;
import java.util.List;
import java.util.Map;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Lists;
import lombok.ast.AnnotationDeclaration;
import lombok.ast.AnnotationMethodDeclaration;
import lombok.ast.ClassDeclaration;
import lombok.ast.CompilationUnit;
import lombok.ast.EmptyDeclaration;
import lombok.ast.EnumDeclaration;
import lombok.ast.InterfaceDeclaration;
import lombok.ast.KeywordModifier;
import lombok.ast.MethodDeclaration;
import lombok.ast.Modifiers;
import lombok.ast.Node;
import lombok.ast.StaticInitializer;
import lombok.ast.TypeDeclaration;
import lombok.ast.VariableDeclaration;
import lombok.ast.VariableDefinition;
import lombok.ast.VariableDefinitionEntry;
import lombok.ast.template.SyntaxCheck;
@SyntaxCheck
public class KeywordChecks {
private static final int K_PUBLIC = 0x0001;
private static final int K_PRIVATE = 0x0002;
private static final int K_PROTECTED = 0x0004;
private static final int K_STATIC = 0x0008;
private static final int K_FINAL = 0x0010;
private static final int K_SYNCHRONIZED = 0x0020;
private static final int K_VOLATILE = 0x0040;
private static final int K_TRANSIENT = 0x0080;
private static final int K_NATIVE = 0x0100;
private static final int K_ABSTRACT = 0x0400;
private static final int K_STRICTFP = 0x0800;
private static final Map TO_FLAG_MAP = ImmutableMap.builder()
.put("private", K_PRIVATE)
.put("protected", K_PROTECTED)
.put("public", K_PUBLIC)
.put("final", K_FINAL)
.put("native", K_NATIVE)
.put("strictfp", K_STRICTFP)
.put("synchronized", K_SYNCHRONIZED)
.put("abstract", K_ABSTRACT)
.put("static", K_STATIC)
.put("transient", K_TRANSIENT)
.put("volatile", K_VOLATILE)
.build();
private static final int[] METHOD_MODIFIERS_EXCLUSIVITY = {
K_PRIVATE | K_PROTECTED,
K_PRIVATE | K_PUBLIC,
K_PROTECTED | K_PUBLIC,
K_NATIVE | K_STRICTFP,
K_ABSTRACT | K_FINAL,
K_ABSTRACT | K_NATIVE,
K_ABSTRACT | K_STATIC,
K_ABSTRACT | K_STRICTFP,
K_ABSTRACT | K_SYNCHRONIZED,
};
private static final int METHOD_MODIFIERS_LEGAL =
K_PRIVATE | K_PROTECTED | K_PUBLIC | K_ABSTRACT | K_FINAL | K_NATIVE |
K_STATIC | K_STRICTFP | K_SYNCHRONIZED
;
private static final int[] FIELD_MODIFIERS_EXCLUSIVITY = {
K_PRIVATE | K_PROTECTED,
K_PRIVATE | K_PUBLIC,
K_PROTECTED | K_PUBLIC,
K_VOLATILE | K_FINAL,
};
private static final int FIELD_MODIFIERS_LEGAL =
K_PRIVATE | K_PROTECTED | K_PUBLIC | K_STATIC | K_FINAL | K_TRANSIENT | K_VOLATILE;
private static final int[] TYPE_MODIFIERS_EXCLUSIVITY = {
K_PRIVATE | K_PROTECTED,
K_PRIVATE | K_PUBLIC,
K_PROTECTED | K_PUBLIC,
K_FINAL | K_ABSTRACT,
};
private static final int TYPE_MODIFIERS_LEGAL =
K_PRIVATE | K_PROTECTED | K_PUBLIC | K_STATIC | K_FINAL | K_ABSTRACT | K_STRICTFP;
public void duplicateKeywordModifierCheck(Modifiers modifiers) {
List keywords = Lists.newArrayList();
for (KeywordModifier n : modifiers.astKeywords()) {
String k = n.astName();
if (k != null && !k.isEmpty() ) {
if (keywords.contains(k)) {
n.addMessage(error(MODIFIERS_DUPLICATE_KEYWORD, "Duplicate modifier: " + k));
} else {
keywords.add(k);
}
}
}
}
public void methodModifiersCheck(MethodDeclaration md) {
modifiersCheck(md.astModifiers(),
METHOD_MODIFIERS_EXCLUSIVITY, METHOD_MODIFIERS_LEGAL, "method declarations");
checkStaticChain(md.astModifiers());
}
public void annotationMethodModifiersCheck(AnnotationMethodDeclaration md) {
modifiersCheck(md.astModifiers(),
METHOD_MODIFIERS_EXCLUSIVITY, METHOD_MODIFIERS_LEGAL & ~K_STRICTFP,
"annotation method declarations");
checkStaticChain(md.astModifiers());
}
public void fieldModifiersCheck(VariableDeclaration vd) {
TypeDeclaration td = vd.upUpToTypeDeclaration();
if (td == null) return; //not a field.
VariableDefinition def = vd.astDefinition();
if (def != null) {
Modifiers m = def.astModifiers();
modifiersCheck(m, FIELD_MODIFIERS_EXCLUSIVITY, FIELD_MODIFIERS_LEGAL, "field declarations");
boolean allFieldsHaveInitializers = true;
for (VariableDefinitionEntry entry : def.astVariables()) {
if (entry.rawInitializer() == null) {
allFieldsHaveInitializers = false;
break;
}
}
if (m.isStatic() && !m.isFinal() && !allFieldsHaveInitializers) {
// Initialized static final fields, assuming the initializer expression is a compile time constant, are 'special' and
// do not need to adhere to the static chain rule. However, we can't determine CTC nature without resolution.
checkStaticChain(m);
}
}
}
public void localVariableModifiersCheck(VariableDefinition vd) {
boolean applies = vd.upUpIfLocalVariableToBlock() != null;
if (!applies) applies = vd.upToForEach() != null;
if (!applies) applies = vd.upToFor() != null;
if (!applies) return;
modifiersCheck(vd.astModifiers(), new int[0], K_FINAL, "local variable declarations");
}
public void classDeclarationModifiersCheck(ClassDeclaration cd) {
typeModifiersCheck(cd, "class declarations");
}
public void interfaceDeclarationModifiersCheck(InterfaceDeclaration id) {
int flags = typeModifiersCheck(id, "interface declarations");
generateNotAllowedKeywordError(id.astModifiers(), flags, K_FINAL, "final", "Interfaces cannot be final");
}
public void annotationDeclarationModifiersCheck(AnnotationDeclaration ad) {
int flags = typeModifiersCheck(ad, "annotation declarations");
generateNotAllowedKeywordError(ad.astModifiers(), flags, K_FINAL, "final", "Annotations cannot be final");
}
public void enumDeclarationModifiersCheck(EnumDeclaration ad) {
int flags = typeModifiersCheck(ad, "enum declarations");
generateNotAllowedKeywordError(ad.astModifiers(), flags, K_FINAL, "final", "Enums cannot be marked final");
}
private int typeModifiersCheck(TypeDeclaration td, String desc) {
int flags = modifiersCheck(td.astModifiers(),
TYPE_MODIFIERS_EXCLUSIVITY, TYPE_MODIFIERS_LEGAL, desc);
boolean staticWarningEmitted = false;
if (td.upIfTopLevelToCompilationUnit() != null) {
generateNotAllowedKeywordError(td.astModifiers(), flags, K_PRIVATE, "private", "Top-level types cannot be private.");
staticWarningEmitted |= generateNotAllowedKeywordError(td.astModifiers(), flags, K_STATIC, "static", "Top-level types cannot be static.");
} else if (td.upToBlock() != null) {
generateNotAllowedKeywordError(td.astModifiers(), flags, K_PRIVATE, "private", "Method-local types cannot be private.");
generateNotAllowedKeywordError(td.astModifiers(), flags, K_PROTECTED, "protected", "Method-local types cannot be protected.");
generateNotAllowedKeywordError(td.astModifiers(), flags, K_PUBLIC, "public", "Method-local types cannot be public.");
staticWarningEmitted |= generateNotAllowedKeywordError(td.astModifiers(), flags, K_STATIC, "static", "Method-local types cannot be static.");
}
if (!staticWarningEmitted) checkStaticChain(td.astModifiers());
return flags;
}
public void checkStaticInitializerInNonStaticType(StaticInitializer node) {
TypeDeclaration parent = node.upUpToTypeDeclaration();
if (parent != null) {
if (!parent.astModifiers().isStatic()) {
node.addMessage(error(INITIALIZER_STATIC_IN_NON_STATIC_TYPE,
"static initializers are only allowed in top-level or static types declarations."));
}
}
}
private void checkStaticChain(Modifiers modifiers) {
if (!modifiers.isStatic()) return;
Node p = modifiers.getParent();
while (p != null) {
if (p instanceof CompilationUnit) return;
if (p instanceof TypeDeclaration) {
Modifiers pMods = ((TypeDeclaration)p).astModifiers();
if (!pMods.isStatic()) {
modifiers.getParent().addMessage(error(MODIFIERS_STATIC_CHAIN,
"This declaration is (effectively) static; static declarations or only legal in top-level and static declarations."));
}
}
p = p.getParent();
}
}
private int modifiersCheck(Modifiers modifiers, int[] exclusivity, int legality, String desc) {
int flags = modifiers.getEffectiveModifierFlags();
int implicits = flags & ~modifiers.getExplicitModifierFlags();
for (Node n : modifiers.rawKeywords()) {
if (n instanceof KeywordModifier) {
String k = ((KeywordModifier)n).astName();
if (k == null || k.isEmpty()) {
n.addMessage(error(MODIFIERS_EMPTY_MODIFIER, "Empty/null modifier."));
}
if (!TO_FLAG_MAP.containsKey(k)) {
n.addMessage(error(MODIFIERS_UNKNOWN_MODIFIER, "Unknown modifier: " + k));
continue;
}
int flag = TO_FLAG_MAP.get(k);
if ((legality & flag) == 0) {
n.addMessage(error(MODIFIERS_MODIFIER_NOT_ALLOWED, "Modifier not allowed on " + desc + ": " + k));
continue;
}
flags |= flag;
}
}
for (int exclusive : exclusivity) {
if ((flags & exclusive) == exclusive) {
generateExclusivityError(implicits, exclusive, modifiers);
}
}
return flags;
}
private boolean generateNotAllowedKeywordError(Modifiers modifiers, int flags, int flag, String keyword, String error) {
if ((flags & flag) == 0) return false;
for (KeywordModifier n : modifiers.astKeywords()) {
if (keyword.equals(n.astName())) {
n.addMessage(error(MODIFIERS_MODIFIER_NOT_ALLOWED, error));
return true;
}
}
return false;
}
public void emptyDeclarationMustHaveNoModifiers(EmptyDeclaration node) {
Modifiers modifiers = node.astModifiers();
if (!modifiers.astKeywords().isEmpty() || !modifiers.astAnnotations().isEmpty()) {
node.addMessage(error(MODIFIERS_MODIFIER_NOT_ALLOWED, "Empty Declarations cannot have modifiers."));
}
}
private void generateExclusivityError(int implicit, int exclusive, Modifiers modifiers) {
String hit = null;
int responsibleImplicit = implicit & exclusive;
if (responsibleImplicit != 0) {
String nameOfResponsibleImplicit = "(unknown)";
for (Map.Entry x : TO_FLAG_MAP.entrySet()) {
if (x.getValue() == responsibleImplicit) nameOfResponsibleImplicit = x.getKey();
}
for (Node n : modifiers.rawKeywords()) {
if (n instanceof KeywordModifier) {
String k = ((KeywordModifier)n).astName();
if (!TO_FLAG_MAP.containsKey(k)) continue;
int f = TO_FLAG_MAP.get(k);
if ((f & exclusive) == 0) continue;
modifiers.addMessage(error(MODIFIERS_MODIFIER_CONFLICT, String.format(
"Modifier %s cannot be used here; it is already implicitly %s.", k, nameOfResponsibleImplicit)));
}
}
} else {
for (Node n : modifiers.rawKeywords()) {
if (n instanceof KeywordModifier) {
String k = ((KeywordModifier)n).astName();
if (!TO_FLAG_MAP.containsKey(k)) continue;
int f = TO_FLAG_MAP.get(k);
if ((f & exclusive) == 0) continue;
if (hit == null) {
hit = k;
continue;
}
modifiers.addMessage(error(MODIFIERS_MODIFIER_CONFLICT, String.format(
"Modifier %s cannot be used together with %s here.", k, hit)));
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy