org.jboss.jandex.ClassInfo Maven / Gradle / Ivy
/*
* JBoss, Home of Professional Open Source.
* Copyright 2013 Red Hat, Inc., and individual contributors
* as indicated by the @author tags.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jboss.jandex;
import java.util.AbstractList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
/**
* Represents a class entry in an index. A ClassInfo is only a partial view of a
* Java class, it is not intended as a complete replacement for Java reflection.
* Only the methods and fields which are references by an annotation are stored.
*
* Global information including the parent class, implemented methodParameters, and
* access flags are also provided since this information is often necessary.
*
*
Note that a parent class and interface may exist outside of the scope of the
* index (e.g. classes in a different jar) so the references are stored as names
* instead of direct references. It is expected that multiple indexes may need
* to be queried to assemble a full hierarchy in a complex multi-jar environment
* (e.g. an application server).
*
*
Thread-Safety
* This class is immutable and can be shared between threads without safe publication.
*
* @author Jason T. Greene
*
*/
public final class ClassInfo implements AnnotationTarget {
private final DotName name;
private final Map> annotations;
// Not final to allow lazy initialization, immutable once published
private short flags;
private Type[] interfaceTypes;
private Type superClassType;
private Type[] typeParameters;
private MethodInternal[] methods;
private FieldInternal[] fields;
private boolean hasNoArgsConstructor;
private NestingInfo nestingInfo;
/** Describes the form of nesting used by a class */
public enum NestingType {
/** A standard class declared within its own source unit */
TOP_LEVEL,
/** A named class enclosed by another class */
INNER,
/** A named class enclosed within a code block */
LOCAL,
/** An unnamed class enclosed within a code block */
ANONYMOUS}
private static final class NestingInfo {
private DotName enclosingClass;
private String simpleName;
private EnclosingMethodInfo enclosingMethod;
}
/**
* Provides information on the enclosing method or constructor for a local or anonymous class,
* if available.
*/
public static final class EnclosingMethodInfo {
private String name;
private Type returnType;
private Type[] parameters;
private DotName enclosingClass;
/**
* The name of the method or constructor
*
* @return the name of the method or constructor
*/
public String name() {
return name;
}
/**
* Returns the return type of the method.
*
* @return the return type
*/
public Type returnType() {
return returnType;
}
/**
* Returns the list of parameters declared by this method or constructor.
* This may be empty, but never null.
*
* @return the list of parameters.
*/
public List parameters() {
return Collections.unmodifiableList(Arrays.asList(parameters));
}
Type[] parametersArray() {
return parameters;
}
/**
* Returns the class name which declares this method or constructor.
*
* @return the name of the class which declared this method or constructor
*/
public DotName enclosingClass() {
return enclosingClass;
}
EnclosingMethodInfo(String name, Type returnType, Type[] parameters, DotName enclosingClass) {
this.name = name;
this.returnType = returnType;
this.parameters = parameters;
this.enclosingClass = enclosingClass;
}
public String toString() {
StringBuilder builder = new StringBuilder();
builder.append(returnType).append(' ').append(enclosingClass).append('.').append(name).append('(');
for (int i = 0; i < parameters.length; i++) {
builder.append(parameters[i]);
if (i + 1 < parameters.length)
builder.append(", ");
}
builder.append(')');
return builder.toString();
}
}
ClassInfo(DotName name, Type superClassType, short flags, Type[] interfaceTypes, Map> annotations) {
this(name, superClassType, flags, interfaceTypes, annotations, false);
}
ClassInfo(DotName name, Type superClassType, short flags, Type[] interfaceTypes, Map> annotations, boolean hasNoArgsConstructor) {
this.name = name;
this.superClassType = superClassType;
this.flags = flags;
this.interfaceTypes = interfaceTypes.length == 0 ? Type.EMPTY_ARRAY : interfaceTypes;
this.annotations = Collections.unmodifiableMap(annotations); // FIXME
this.hasNoArgsConstructor = hasNoArgsConstructor;
this.typeParameters = Type.EMPTY_ARRAY;
this.methods = MethodInternal.EMPTY_ARRAY;
this.fields = FieldInternal.EMPTY_ARRAY;
}
/**
* Constructs a "mock" ClassInfo using the passed values. All passed values MUST NOT BE MODIFIED AFTER THIS CALL.
* Otherwise the resulting object would not conform to the contract outlined above.
*
* @param name the name of this class
* @param superName the name of the parent class
* @param flags the class attributes
* @param interfaces the methodParameters this class implements
* @param annotations the annotations on this class
* @param hasNoArgsConstructor whether this class has a no arg constructor
* @return a new mock class representation
*/
@Deprecated
public static ClassInfo create(DotName name, DotName superName, short flags, DotName[] interfaces, Map> annotations, boolean hasNoArgsConstructor) {
Type[] interfaceTypes = new Type[interfaces.length];
for (int i = 0; i < interfaces.length; i++) {
interfaceTypes[i] = new ClassType(interfaces[i]);
}
ClassType superClassType = superName == null ? null : new ClassType(superName);
return new ClassInfo(name, superClassType, flags, interfaceTypes, annotations, hasNoArgsConstructor);
}
@Override
public final Kind kind() {
return Kind.CLASS;
}
public String toString() {
return name.toString();
}
/**
* Returns the name of the class
*
* @return the name of the class
*/
public final DotName name() {
return name;
}
/**
* Returns the access flags for this class. The standard {@link java.lang.reflect.Modifier}
* can be used to decode the value.
*
* @return the access flags
*/
public final short flags() {
return flags;
}
/**
* Returns the name of the super class declared by the extends clause of this class. This
* information is also available from the {@link #superClassType} method. For all classes,
* with the one exception of java.lang.Object
, which is the one class in the
* Java language without a super-type, this method will always return a non-null value.
*
* @return the name of the super class of this class, or null if this class is java.lang.Object
*/
public final DotName superName() {
return superClassType == null ? null : superClassType.name();
}
/**
* Returns an array of interface names implemented by this class. Every call to this method
* performs a defensive copy, so {@link #interfaceNames()} should be used instead.
*
* @return an array of interface names implemented by this class
*/
@Deprecated
public final DotName[] interfaces() {
DotName[] interfaces = new DotName[interfaceTypes.length];
for (int i = 0; i < interfaceTypes.length; i++) {
interfaces[i] = interfaceTypes[i].name();
}
return interfaces;
}
/**
* Returns a map indexed by annotation name, with a value list of annotation instances.
* The annotation instances in this map correspond to both annotations on the class,
* and every nested element of the class (fields, types, methods, etc).
*
* The target of the annotation instance can be used to determine the location of
* the annotation usage.
*
* @return the annotations specified on this class and its elements
*/
public final Map> annotations() {
return annotations;
}
/**
* Returns a list of all annotations directly declared on this class.
*
* @return the list of annotations declared on this class
*/
public final Collection classAnnotations() {
return new AnnotationTargetFilterCollection(annotations, ClassInfo.class);
}
/**
* Returns the annotation with the specified name directly declared on this class.
*
* @param name the annotation name to look for
* @return the declared annotation or null if not found
*/
public final AnnotationInstance classAnnotation(DotName name) {
List instances = annotations.get(name);
if (instances != null) {
for (AnnotationInstance instance : instances) {
if (instance.target() == this) {
return instance;
}
}
}
return null;
}
/**
* Returns a list of all methods declared in this class. This includes constructors
* and static initializer blocks which have the special JVM assigned names of "<init>"
* and "<clinit>", respectively. It does not, however, include inherited methods.
* These must be discovered by traversing the class hierarchy.
*
* This list may be empty, but never null.
*
* @return the list of methods declared in this class
*/
public final List methods() {
return new MethodInfoGenerator(this, methods);
}
final MethodInternal[] methodArray() {
return methods;
}
/**
* Retrieves a method based on its signature, which includes a method name and an argument list.
* The argument list is compared based on the underlying raw type of the type arguments. As an example,
* a generic type parameter "T" is equivalent to java.lang.Object
, since the raw form
* of a type parameter is its upper bound.
*
* Eligible methods include constructors and static initializer blocks which have the special JVM
* assigned names of "<init>" and "<clinit>", respectively. This does not, however, include
* inherited methods. These must be discovered by traversing the class hierarchy.
*
* @param name the name of the method to find
* @param parameters the type parameters of the method
* @return the located method or null if not found
*/
public final MethodInfo method(String name, Type... parameters) {
MethodInternal key = new MethodInternal(Utils.toUTF8(name), MethodInternal.EMPTY_PARAMETER_NAMES, parameters, null, (short) 0);
int i = Arrays.binarySearch(methods, key, MethodInternal.NAME_AND_PARAMETER_COMPONENT_COMPARATOR);
return i >= 0 ? new MethodInfo(this, methods[i]) : null;
}
/**
* Retrieves the "first" occurrence of a method by the given name. Note that the order of methods
* is not defined, and may change in the future. Therefore, this method should not be used when
* overloading is possible. It's merely intended to provide a handy shortcut for throw away or test
* code.
*
* @param name the name of the method
* @return the first discovered method matching this name, or null if no match is found
*/
public final MethodInfo firstMethod(String name) {
MethodInternal key = new MethodInternal(Utils.toUTF8(name), MethodInternal.EMPTY_PARAMETER_NAMES, Type.EMPTY_ARRAY, null, (short) 0);
int i = Arrays.binarySearch(methods, key, MethodInternal.NAME_AND_PARAMETER_COMPONENT_COMPARATOR);
if (i < -methods.length) {
return null;
}
MethodInfo method = new MethodInfo(this,i >= 0 ? methods[i] : methods[++i * -1]);
return method.name().equals(name) ? method : null;
}
/**
* Retrieves a field by the given name. Only fields declared in this class are available.
* Locating inherited fields requires traversing the class hierarchy.
*
* @param name the name of the field
* @return the field
*/
public final FieldInfo field(String name) {
FieldInternal key = new FieldInternal(Utils.toUTF8(name), VoidType.VOID, (short)0);
int i = Arrays.binarySearch(fields, key, FieldInternal.NAME_COMPARATOR);
if (i < 0) {
return null;
}
return new FieldInfo(this, fields[i]);
}
/**
* Returns a list of all available fields. Only fields declared in this class are available.
* Locating inherited fields requires traversing the class hierarchy. This list may be
* empty, but never null.
*
* @return a list of fields
*/
public final List fields() {
return new FieldInfoGenerator(this, fields);
}
final FieldInternal[] fieldArray() {
return fields;
}
/**
* Returns a list of names for all interfaces this class implements. This list may be empty, but never null.
*
* Note that this information is also available on the Type
instances returned by
* {@link #interfaceTypes}
*
* @return the list of names implemented by this class
*/
public final List interfaceNames() {
return new AbstractList() {
@Override
public DotName get(int i) {
return interfaceTypes[i].name();
}
@Override
public int size() {
return interfaceTypes.length;
}
};
}
/**
* Returns the list of types in the implements clause of this class. These types may be generic types.
* This list may be empty, but never null
*
* @return the list of types declared in the implements clause of this class
*/
public final List interfaceTypes() {
return Collections.unmodifiableList(Arrays.asList(interfaceTypes));
}
final Type[] interfaceTypeArray() {
return interfaceTypes;
}
final Type[] copyInterfaceTypes() {
return interfaceTypes.clone();
}
/**
* Returns a super type represented by the extends clause of this class. This type might be a generic type.
*
* @return the super class type definition in the extends clause
*/
public final Type superClassType() {
return superClassType;
}
/**
* Returns the generic type parameters of this class, if any. These will be returned as resolved type variables,
* so if a parameter has a bound on another parameter, that information will be available.
*
* @return the generic type parameters of this class
*/
public final List typeParameters() {
@SuppressWarnings("unchecked")
List list = (List) Arrays.asList(typeParameters);
return Collections.unmodifiableList(list);
}
final Type[] typeParameterArray() {
return typeParameters;
}
/**
* Returns a boolean indicating the presence of a no-arg constructor, if supported by the underlying index store.
* This information is available in indexes produced by Jandex 1.2.0 and later.
*
* @return true
in case of the Java class has a no-copyParameters constructor, false
* if it does not, or it is not known
* @since 1.2.0
*/
public final boolean hasNoArgsConstructor() {
return hasNoArgsConstructor;
}
/**
* Returns the nesting type of this class, which could either be a standard top level class, an inner class,
* an anonymous class, or a local class.
*
* For historical reasons, static nested classes are returned as INNER
. You can differentiate
* between a non-static nested class (inner class) and a static nested class by calling
* {@link java.lang.reflect.Modifier#isStatic(int)} on the return of {@link #flags()}
*
* @return the nesting type of this class
*/
public NestingType nestingType() {
if (nestingInfo == null) {
return NestingType.TOP_LEVEL;
} else if (nestingInfo.enclosingClass != null) {
return NestingType.INNER;
} else if (nestingInfo.simpleName != null) {
return NestingType.LOCAL;
}
return NestingType.ANONYMOUS;
}
/**
* Returns the source declared name of this class if it is an inner class, or a local class. Otherwise
* this method will return null.
*
* @return the simple name of a top-level, local, or inner class, or null if this is an anonymous class
*/
public String simpleName() {
return nestingInfo != null ? nestingInfo.simpleName : name.local();
}
String nestingSimpleName() {
return nestingInfo != null ? nestingInfo.simpleName : null;
}
/**
* Returns the enclosing class if this is an inner class, or null if this is an anonymous, a local, or
* a top level class.
*
* @return the enclosing class if this class is an inner class
*/
public DotName enclosingClass() {
return nestingInfo != null ? nestingInfo.enclosingClass : null;
}
/**
* Returns the enclosing method of this class if it is a local, or anonymous class, and it is declared
* within the body of a method or constructor. It will return null if this class is a top level, or an inner class.
* It will also return null if the local or anonymous class is on an initializer.
*
* @return the enclosing method/constructor, if this class is local or anonymous, and it is within a
* method/constructor
*/
public EnclosingMethodInfo enclosingMethod() {
return nestingInfo != null ? nestingInfo.enclosingMethod : null;
}
@Override
public ClassInfo asClass() {
return this;
}
@Override
public FieldInfo asField() {
throw new IllegalArgumentException("Not a field");
}
@Override
public MethodInfo asMethod() {
throw new IllegalArgumentException("Not a method");
}
@Override
public MethodParameterInfo asMethodParameter() {
throw new IllegalArgumentException("Not a method parameter");
}
@Override
public TypeTarget asType() {
throw new IllegalArgumentException("Not a type");
}
void setHasNoArgsConstructor(boolean hasNoArgsConstructor) {
this.hasNoArgsConstructor = hasNoArgsConstructor;
}
void setFields(List fields, NameTable names) {
if (fields.size() == 0) {
this.fields = FieldInternal.EMPTY_ARRAY;
return;
}
this.fields = new FieldInternal[fields.size()];
for (int i = 0; i < fields.size(); i++) {
FieldInfo fieldInfo = fields.get(i);
FieldInternal internal = names.intern(fieldInfo.fieldInternal());
fieldInfo.setFieldInternal(internal);
this.fields[i] = internal;
}
Arrays.sort(this.fields, FieldInternal.NAME_COMPARATOR);
}
void setFieldArray(FieldInternal[] fields) {
this.fields = fields;
}
void setMethodArray(MethodInternal[] methods) {
this.methods = methods;
}
void setMethods(List methods, NameTable names) {
if (methods.size() == 0) {
this.methods = MethodInternal.EMPTY_ARRAY;
return;
}
this.methods = new MethodInternal[methods.size()];
for (int i = 0; i < methods.size(); i++) {
MethodInfo methodInfo = methods.get(i);
MethodInternal internal = names.intern(methodInfo.methodInternal());
methodInfo.setMethodInternal(internal);
this.methods[i] = internal;
}
Arrays.sort(this.methods, MethodInternal.NAME_AND_PARAMETER_COMPONENT_COMPARATOR);
}
void setSuperClassType(Type superClassType) {
this.superClassType = superClassType;
}
void setInterfaceTypes(Type[] interfaceTypes) {
this.interfaceTypes = interfaceTypes.length == 0 ? Type.EMPTY_ARRAY : interfaceTypes;
}
void setTypeParameters(Type[] typeParameters) {
this.typeParameters = typeParameters.length == 0 ? Type.EMPTY_ARRAY : typeParameters;
}
void setInnerClassInfo(DotName enclosingClass, String simpleName, boolean knownInnerClass) {
boolean setValues = enclosingClass != null || simpleName != null;
// Always init known inner types since we might have an anonymous type
// with a method-less encloser (static block).
if (nestingInfo == null && (knownInnerClass || setValues)) {
nestingInfo = new NestingInfo();
}
if (! setValues){
return;
}
nestingInfo.enclosingClass = enclosingClass;
nestingInfo.simpleName = simpleName;
}
void setEnclosingMethod(EnclosingMethodInfo enclosingMethod) {
if (enclosingMethod == null) {
return;
}
if (nestingInfo == null) {
nestingInfo = new NestingInfo();
}
nestingInfo.enclosingMethod = enclosingMethod;
}
void setFlags(short flags) {
this.flags = flags;
}
}