brooklyn.entity.proxying.InternalEntityFactory Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of brooklyn-core Show documentation
Show all versions of brooklyn-core Show documentation
Entity implementation classes, events, and other core elements
package brooklyn.entity.proxying;
import static com.google.common.base.Preconditions.checkNotNull;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.Map;
import java.util.Set;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import brooklyn.config.ConfigKey;
import brooklyn.entity.Entity;
import brooklyn.entity.basic.AbstractEntity;
import brooklyn.entity.basic.EntityInternal;
import brooklyn.entity.basic.EntityLocal;
import brooklyn.management.ManagementContext;
import brooklyn.management.internal.ManagementContextInternal;
import brooklyn.policy.Policy;
import brooklyn.policy.basic.AbstractPolicy;
import brooklyn.util.collections.MutableMap;
import brooklyn.util.collections.MutableSet;
import brooklyn.util.exceptions.Exceptions;
import brooklyn.util.javalang.Reflections;
import com.google.common.collect.Maps;
/**
* Creates entities (and proxies) of required types, given the
*
* This is an internal class for use by core-brooklyn. End-users are strongly discouraged from
* using this class directly.
*
* @author aled
*/
public class InternalEntityFactory {
private static final Logger log = LoggerFactory.getLogger(InternalEntityFactory.class);
private final ManagementContextInternal managementContext;
private final EntityTypeRegistry entityTypeRegistry;
/**
* For tracking if AbstractEntity constructor has been called by framework, or in legacy way (i.e. directly).
*
* To be deleted once we delete support for constructing entities directly (and expecting configure() to be
* called inside the constructor, etc).
*
* @author aled
*/
public static class FactoryConstructionTracker {
private static ThreadLocal constructing = new ThreadLocal();
public static boolean isConstructing() {
return (constructing.get() == Boolean.TRUE);
}
static void reset() {
constructing.set(Boolean.FALSE);
}
static void setConstructing() {
constructing.set(Boolean.TRUE);
}
}
/**
* Returns true if this is a "new-style" entity (i.e. where not expected to call the constructor to instantiate it).
* That means it is an entity with a no-arg constructor, and where there is a mapped for an entity type interface.
* @param managementContext
* @param clazz
*/
@SuppressWarnings({ "unchecked", "rawtypes" })
public static boolean isNewStyleEntity(ManagementContext managementContext, Class> clazz) {
try {
return isNewStyleEntity(clazz) && managementContext.getEntityManager().getEntityTypeRegistry().getEntityTypeOf((Class)clazz) != null;
} catch (IllegalArgumentException e) {
return false;
}
}
public static boolean isNewStyleEntity(Class> clazz) {
if (!Entity.class.isAssignableFrom(clazz)) {
throw new IllegalArgumentException("Class "+clazz+" is not an entity");
}
try {
clazz.getConstructor(new Class[0]);
return true;
} catch (NoSuchMethodException e) {
return false;
}
}
public InternalEntityFactory(ManagementContextInternal managementContext, EntityTypeRegistry entityTypeRegistry) {
this.managementContext = checkNotNull(managementContext, "managementContext");
this.entityTypeRegistry = checkNotNull(entityTypeRegistry, "entityTypeRegistry");
}
@SuppressWarnings("unchecked")
public T createEntityProxy(EntitySpec spec, T entity) {
// TODO Don't want the proxy to have to implement EntityLocal, but required by how
// AbstractEntity.parent is used (e.g. parent.getAllConfig)
ClassLoader classloader = (spec.getImplementation() != null ? spec.getImplementation() : spec.getType()).getClassLoader();
MutableSet.Builder> builder = MutableSet.>builder()
.addAll(EntityProxy.class, Entity.class, EntityLocal.class, EntityInternal.class);
if (spec.getType().isInterface()) {
builder.add(spec.getType());
} else {
log.warn("EntitySpec declared in terms of concrete type "+spec.getType()+"; should be supplied in terms of interface");
}
builder.addAll(spec.getAdditionalInterfaces());
Set> interfaces = builder.build();
return (T) java.lang.reflect.Proxy.newProxyInstance(
classloader,
interfaces.toArray(new Class[interfaces.size()]),
new EntityProxyImpl(entity));
}
@SuppressWarnings({ "unchecked", "rawtypes" })
public T createEntity(EntitySpec spec) {
if (spec.getFlags().containsKey("parent") || spec.getFlags().containsKey("owner")) {
throw new IllegalArgumentException("Spec's flags must not contain parent or owner; use spec.parent() instead for "+spec);
}
try {
Class extends T> clazz = getImplementedBy(spec);
FactoryConstructionTracker.setConstructing();
T entity;
try {
entity = construct(clazz, spec);
} finally {
FactoryConstructionTracker.reset();
}
if (spec.getDisplayName()!=null)
((AbstractEntity)entity).setDisplayName(spec.getDisplayName());
if (isNewStyleEntity(clazz)) {
((AbstractEntity)entity).setManagementContext(managementContext);
((AbstractEntity)entity).setProxy(createEntityProxy(spec, entity));
((AbstractEntity)entity).configure(MutableMap.copyOf(spec.getFlags()));
}
for (Map.Entry, Object> entry : spec.getConfig().entrySet()) {
((EntityLocal)entity).setConfig((ConfigKey)entry.getKey(), entry.getValue());
}
((AbstractEntity)entity).init();
for (Policy policy : spec.getPolicies()) {
entity.addPolicy((AbstractPolicy)policy);
}
Entity parent = spec.getParent();
if (parent != null) {
parent = (parent instanceof AbstractEntity) ? ((AbstractEntity)parent).getProxyIfAvailable() : parent;
entity.setParent(parent);
}
return entity;
} catch (Exception e) {
throw Exceptions.propagate(e);
}
}
private Class extends T> getImplementedBy(EntitySpec spec) {
if (spec.getImplementation() != null) {
return spec.getImplementation();
} else {
return entityTypeRegistry.getImplementedBy(spec.getType());
}
}
private T construct(Class extends T> clazz, EntitySpec spec) throws InstantiationException, IllegalAccessException, InvocationTargetException {
if (isNewStyleEntity(clazz)) {
return clazz.newInstance();
} else {
return constructOldStyle(clazz, MutableMap.copyOf(spec.getFlags()));
}
}
private T constructOldStyle(Class extends T> clazz, Map flags) throws InstantiationException, IllegalAccessException, InvocationTargetException {
if (flags.containsKey("parent") || flags.containsKey("owner")) {
throw new IllegalArgumentException("Spec's flags must not contain parent or owner; use spec.parent() instead for "+clazz);
}
Constructor extends T> c2 = Reflections.findCallabaleConstructor(clazz, new Object[] {flags});
if (c2 != null) {
return c2.newInstance(Maps.newLinkedHashMap(flags));
} else {
throw new IllegalStateException("No valid constructor defined for "+clazz+" (expected no-arg or single java.util.Map argument)");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy