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

org.infinispan.factories.components.ComponentMetadata Maven / Gradle / Ivy

package org.infinispan.factories.components;

import java.io.Serializable;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.infinispan.factories.ComponentRegistry;
import org.infinispan.factories.annotations.ComponentName;
import org.infinispan.factories.annotations.DefaultFactoryFor;
import org.infinispan.factories.annotations.Inject;
import org.infinispan.factories.annotations.Start;
import org.infinispan.factories.annotations.Stop;
import org.infinispan.factories.annotations.SurvivesRestarts;
import org.infinispan.factories.scopes.Scope;

/**
 * This class contains all of the metadata and implications expressed via the {@link Scope}, {@link SurvivesRestarts},
 * {@link DefaultFactoryFor}, {@link ComponentName}, {@link Inject}, {@link Start} and {@link Stop} annotations.  Instead
 * of scanning for these annotations and working out dependency chains at runtime "on-demand", since Infinispan 5.1, this
 * process now happens offline, at build-time.
 * 

* When compiling Infinispan, components and their dependency chains are inspected and the information expressed by the * annotations above are denormalized and a series of {@link ComponentMetadata} objects are created and persisted in the * Infinispan jar. *

* This metadata is then read in by the {@link ComponentMetadataRepo} at runtime, and used by the {@link ComponentRegistry} * and other factory-like classes to bootstrap an Infinispan node. *

* Also see {@link ManageableComponentMetadata} for components that also expose JMX information. * @author Manik Surtani * @since 5.1 * @see ManageableComponentMetadata * @see ComponentMetadataRepo */ public class ComponentMetadata implements Serializable { public static final InjectMetadata[] EMPTY_INJECT_METHODS = {}; public static final PrioritizedMethodMetadata[] EMPTY_PRIORITIZED_METHODS = {}; private String name; private transient Map dependencies; private InjectMetadata[] injectMetadata; private PrioritizedMethodMetadata[] startMethods; private PrioritizedMethodMetadata[] stopMethods; private boolean globalScope = false; private boolean survivesRestarts = false; private transient Class clazz; ComponentMetadata() { globalScope = false; survivesRestarts = true; } public ComponentMetadata(Class component, List injectMethods, List startMethods, List stopMethods, boolean global, boolean survivesRestarts) { clazz = component; name = component.getName(); globalScope = global; this.survivesRestarts = survivesRestarts; if (startMethods != null && !startMethods.isEmpty()) { this.startMethods = new PrioritizedMethodMetadata[startMethods.size()]; int i=0; for (Method m : startMethods) { Start s = m.getAnnotation(Start.class); this.startMethods[i++] = new PrioritizedMethodMetadata(m.getName(), s.priority()); } } if (stopMethods != null && !stopMethods.isEmpty()) { this.stopMethods = new PrioritizedMethodMetadata[stopMethods.size()]; int i=0; for (Method m : stopMethods) { Stop s = m.getAnnotation(Stop.class); this.stopMethods[i++] = new PrioritizedMethodMetadata(m.getName(), s.priority()); } } if (injectMethods != null && !injectMethods.isEmpty()) { this.injectMetadata = new InjectMetadata[injectMethods.size()]; this.dependencies = new HashMap(injectMethods.size() * 2); int j=0; for (Method m : injectMethods) { InjectMetadata injectMetadata = new InjectMetadata(m.getName()); Class[] parameterTypes = m.getParameterTypes(); // Need to use arrays instead of Map due to JDK bug see ISPN-3611 String[] params = new String[parameterTypes.length]; String[] paramNames = new String[parameterTypes.length]; // Add this to our dependencies map Annotation[][] annotations = m.getParameterAnnotations(); for (int i=0; i position) { Annotation[] paramAnnotations = annotations[position]; if (paramAnnotations != null) { for (Annotation a: paramAnnotations) { if (a instanceof ComponentName) { return ((ComponentName) a).value(); } } } } return null; } public String getName() { return name; } public Map getDependencies() { return dependencies; } public InjectMetadata[] getInjectMethods() { if (injectMetadata == null) return EMPTY_INJECT_METHODS; return injectMetadata; } public PrioritizedMethodMetadata[] getStartMethods() { if (startMethods == null) return EMPTY_PRIORITIZED_METHODS; return startMethods; } public PrioritizedMethodMetadata[] getStopMethods() { if (stopMethods == null) return EMPTY_PRIORITIZED_METHODS; return stopMethods; } public boolean isGlobalScope() { return globalScope; } public boolean isSurvivesRestarts() { return survivesRestarts; } public boolean isManageable() { return false; } public Class getClazz() { return clazz; } public ManageableComponentMetadata toManageableComponentMetadata() { throw new UnsupportedOperationException("This component is not manageable!"); } @Override public String toString() { return "ComponentMetadata{" + "name='" + name + '\'' + ", dependencies=" + dependencies + ", injectMetadata=" + Arrays.toString(injectMetadata) + ", startMethods=" + Arrays.toString(startMethods) + ", stopMethods=" + Arrays.toString(stopMethods) + ", globalScope=" + globalScope + ", survivesRestarts=" + survivesRestarts + '}'; } /** * This class encapsulates metadata on a prioritized method, such as one annotated with {@link Start} or {@link @Stop} */ public static class PrioritizedMethodMetadata implements Serializable { String methodName; transient Method method; int priority; public PrioritizedMethodMetadata(String methodName, int priority) { this.methodName = methodName; this.priority = priority; } public String getMethodName() { return methodName; } public Method getMethod() { return method; } public void setMethod(Method method) { this.method = method; } public int getPriority() { return priority; } @Override public String toString() { return method.getName() + "(priority=" + priority + ")"; } } /** * This class encapsulates metadata on an inject method, such as one annotated with {@link Inject} */ public static class InjectMetadata implements Serializable { //To avoid mismatches during development like as created by Maven vs IDE compiled classes: private static final long serialVersionUID = 4848856551345751894L; String methodName; transient Method method; String[] parameters; transient Class[] parameterClasses; String[] parameterNames; private InjectMetadata(String methodName) { this.methodName = methodName; } public String getMethodName() { return methodName; } public String[] getParameters() { return parameters; } public String getParameterName(int subscript) { String name = parameterNames == null ? null : parameterNames[subscript]; return name == null ? parameters[subscript] : name; } public boolean isParameterNameSet(int subscript) { return parameterNames != null && parameterNames[subscript] != null; } public synchronized Method getMethod() { return method; } public synchronized void setMethod(Method method) { this.method = method; } public synchronized Class[] getParameterClasses() { return parameterClasses; } public synchronized void setParameterClasses(Class[] parameterClasses) { this.parameterClasses = parameterClasses; } @Override public String toString() { return methodName + "(" + String.join(", ", parameters) + ")"; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy