net.sourceforge.pmd.lang.java.symbols.JClassSymbol Maven / Gradle / Ivy
/*
* BSD-style license; for more info see http://pmd.sourceforge.net/license.html
*/
package net.sourceforge.pmd.lang.java.symbols;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.lang.reflect.Modifier;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import org.checkerframework.checker.nullness.qual.NonNull;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.pcollections.HashTreePSet;
import org.pcollections.PSet;
import net.sourceforge.pmd.lang.java.ast.ASTTypeDeclaration;
import net.sourceforge.pmd.lang.java.symbols.SymbolicValue.SymAnnot;
import net.sourceforge.pmd.lang.java.symbols.SymbolicValue.SymEnum;
import net.sourceforge.pmd.lang.java.types.JArrayType;
import net.sourceforge.pmd.lang.java.types.JClassType;
import net.sourceforge.pmd.lang.java.types.JPrimitiveType;
import net.sourceforge.pmd.lang.java.types.JTypeMirror;
import net.sourceforge.pmd.lang.java.types.Substitution;
/**
* Abstraction over a {@link Class} instance. This is not a type, it's
* the *declaration* of a type. For example, a class symbol representing
* a generic class can provide access to the formal type parameters, but
* the symbol does not represent a specific parametrization of a type.
*
* Class symbols represent the full range of types represented by {@link Class}:
* classes, interfaces, arrays, and primitives. This excludes type variables,
* intersection types, parameterized types, wildcard types, etc., which are only
* compile-time constructs.
*
*
Class symbols are used to back {@link JClassType}, {@link JArrayType},
* and {@link JPrimitiveType}. See {@link JTypeMirror#getSymbol()}.
*
* @since 7.0.0
*/
public interface JClassSymbol extends JTypeDeclSymbol,
JTypeParameterOwnerSymbol,
BoundToNode {
/**
* Returns the binary name of this type, as specified by the JLS:
* the JLS.
* For array types this returns the binary name of the component followed by "[]".
* This differs from {@link Class#getName()}, which for array types outputs an
* internal name.
*
* For example:
*
{@code
* int.class.getName() == "int"
* int[].class.getName() == "[I"
* String.class.getName() == "java.lang.String"
* String[].class.getName() == "[Ljava.lang.String;"
* }
* whereas
* {@code
* symbolOf(int.class).getBinaryName() == "int"
* symbolOf(int[].class).getBinaryName() == "int[]"
* symbolOf(String.class).getBinaryName() == "java.lang.String"
* symbolOf(String[].class).getBinaryName() == "java.lang.String[]"
* }
*/
@NonNull
String getBinaryName();
/**
* Returns the simple name of this class, as specified by
* {@link Class#getCanonicalName()}.
*/
@Nullable
String getCanonicalName();
/**
* Returns the method or constructor this symbol is declared in, if
* it represents a {@linkplain #isLocalClass() local class declaration}.
*
* Notice, that this returns null also if this class is local to
* a class or instance initializer.
*/
@Nullable JExecutableSymbol getEnclosingMethod();
@Override
default JTypeParameterOwnerSymbol getEnclosingTypeParameterOwner() {
JExecutableSymbol enclosingMethod = getEnclosingMethod();
return enclosingMethod != null ? enclosingMethod : getEnclosingClass();
}
/**
* Returns the member classes declared directly in this class.
*
* @see Class#getDeclaredClasses()
*/
List getDeclaredClasses();
/** Returns a class with the given name defined in this class. */
@Nullable
default JClassSymbol getDeclaredClass(String name) {
for (JClassSymbol klass : getDeclaredClasses()) {
if (klass.nameEquals(name)) {
return klass;
}
}
return null;
}
/**
* Returns the methods declared directly in this class.
* This excludes bridges and other synthetic methods.
*
* For an array type T[], to the difference of {@link Class},
* this method returns a one-element list with the {@link Object#clone()}
* method, as if declared like so: {@code public final T[] clone() {...}}.
*
* @see Class#getDeclaredMethods()
*/
List getDeclaredMethods();
/**
* Returns the constructors declared by this class.
* This excludes synthetic constructors.
*
* For an array type T[], and to the difference of {@link Class},
* this should return a one-element list with a constructor
* having the same modifiers as the array type, and a single
* {@code int} parameter.
*
* @see Class#getDeclaredConstructors()
*/
List getConstructors();
/**
* Returns the fields declared directly in this class.
* This excludes synthetic fields.
*
* For arrays, and to the difference of {@link Class},
* this should return a one-element list with the
* {@code public final int length} field.
*
* @see Class#getDeclaredFields()
*/
List getDeclaredFields();
/** Returns a field with the given name defined in this class. */
@Nullable
default JFieldSymbol getDeclaredField(String name) {
for (JFieldSymbol field : getDeclaredFields()) {
if (field.nameEquals(name)) {
return field;
}
}
return null;
}
/**
* Returns a list with all enum constants. If this symbol does
* not represent an enum, returns an empty list. The returned list
* is a subset of {@link #getDeclaredFields()}. The order of fields
* denotes the normal order of enum constants.
*/
default @NonNull List getEnumConstants() {
return Collections.emptyList();
}
/**
* Returns a list with all record components. If this symbol does
* not represent a record, returns an empty list. The order of values
* denotes the normal order of components.
*/
default @NonNull List getRecordComponents() {
return Collections.emptyList();
}
/** Returns the list of super interface types, under the given substitution. */
List getSuperInterfaceTypes(Substitution substitution);
/** Returns the superclass type, under the given substitution. */
@Nullable JClassType getSuperclassType(Substitution substitution);
/**
* Returns the superclass symbol if it exists. Returns null if this
* class represents the class {@link Object}, or a primitive type.
* If this symbol is an interface, returns the symbol for {@link Object}.
*/
@Nullable
JClassSymbol getSuperclass();
/** Returns the direct super-interfaces of this class or interface symbol. */
List getSuperInterfaces();
default boolean isAbstract() {
return Modifier.isAbstract(getModifiers());
}
/** Returns the component symbol, returns null if this is not an array. */
@Nullable
JTypeDeclSymbol getArrayComponent();
boolean isArray();
boolean isPrimitive();
boolean isEnum();
boolean isRecord();
boolean isAnnotation();
boolean isLocalClass();
boolean isAnonymousClass();
/**
* Return the list of permitted subclasses or subinterfaces, as defined in the
* {@code permits} clause of a sealed class or interface. If this class is sealed
* but has no permits clause, the permitted subtypes are inferred from the types
* in the compilation unit. If the class is not sealed, returns an empty list.
*
* Note that an enum class for which some constants declare a body is technically
* implicitly sealed, and implicitly permits only the anonymous classes for those enum
* constants. For consistency, this method will return only symbols that have a canonical
* name, and therefore always return an empty list for enums.
*
* @see #isSealed()
*/
default List getPermittedSubtypes() {
return Collections.emptyList();
}
/**
* Return true if this type is sealed. Then it has a non-empty list of permitted
* subclasses (or it is a compile-time error). Note that there is no trace of the
* non-sealed modifier in class files. A class must have the {@code non-sealed}
* modifier if it is not sealed, not final, and has a sealed supertype.
*
* Note that an enum class for which some constants declare a body is technically
* implicitly sealed, and implicitly permits only the anonymous classes for those enum
* constants. For consistency with {@link #getPermittedSubtypes()}, we treat such enums
* as not sealed.
*
* @see #getPermittedSubtypes()
*/
default boolean isSealed() {
return !getPermittedSubtypes().isEmpty();
}
/**
* Return true if this type is final, that is, does not admit subtypes. Note that
* array types have both modifiers final and abstract. Note also that enum classes
* may be non-final if they have constants that declare an anonymous body.
*/
default boolean isFinal() {
return Modifier.isFinal(getModifiers());
}
/**
* Return the simple names of all annotation attributes. If this
* is not an annotation type, return an empty set.
*/
default PSet getAnnotationAttributeNames() {
return HashTreePSet.empty();
}
/**
* Return the default value of the attribute if this is an annotation type
* with a default. Return null if this is not an annotation type, if there
* is no such attribute, or the attribute has no default value. If the name
* is in the {@linkplain #getAnnotationAttributeNames() attribute name set},
* then the null return value can only mean that the attribute exists but has
* no default value.
*
* @param attrName Attribute name
*/
default @Nullable SymbolicValue getDefaultAnnotationAttributeValue(String attrName) {
if (!isAnnotation()) {
return null;
}
for (JMethodSymbol m : getDeclaredMethods()) {
if (m.nameEquals(attrName) && m.isAnnotationAttribute()) {
return m.getDefaultAnnotationValue(); // nullable
}
}
return null;
}
/**
* Returns the retention policy of this annotation, if this is an
* annotation symbol. Otherwise returns null.
*/
default @Nullable RetentionPolicy getAnnotationRetention() {
if (!isAnnotation()) {
return null;
}
return Optional.ofNullable(getDeclaredAnnotation(Retention.class))
.map(annot -> annot.getAttribute("value"))
.filter(value -> value instanceof SymEnum)
.map(value -> ((SymEnum) value).toEnum(RetentionPolicy.class))
.orElse(RetentionPolicy.CLASS);
}
/**
* Return whether annotations of this annotation type apply to the
* given construct, as per the {@link Target} annotation. Return
* false if this is not an annotation.
*/
default boolean annotationAppliesTo(ElementType elementType) {
if (!isAnnotation()) {
return false;
}
SymAnnot target = getDeclaredAnnotation(Target.class);
if (target == null) {
// If an @Target meta-annotation is not present on an annotation type T,
// then an annotation of type T may be written as a modifier
// for any declaration except a type parameter declaration.
return elementType != ElementType.TYPE_PARAMETER;
}
return target.attributeContains("value", elementType).isTrue();
}
/**
* This returns true if this is not an interface, primitive or array.
* Note that this includes in particular records and enums.
*/
default boolean isClass() {
return !isInterface() && !isArray() && !isPrimitive();
}
/**
* Returns the toplevel class containing this class. If this is a
* toplevel class, returns this.
*/
default @NonNull JClassSymbol getNestRoot() {
JClassSymbol e = this;
while (e.getEnclosingClass() != null) {
e = e.getEnclosingClass();
}
return e;
}
@Override
default R acceptVisitor(SymbolVisitor visitor, P param) {
return visitor.visitClass(this, param);
}
}