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

brooklyn.location.LocationSpec Maven / Gradle / Ivy

package brooklyn.location;

import static com.google.common.base.Preconditions.checkNotNull;

import java.io.Serializable;
import java.lang.reflect.Modifier;
import java.util.Collections;
import java.util.Map;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import brooklyn.config.ConfigKey;
import brooklyn.config.ConfigKey.HasConfigKey;
import brooklyn.management.Task;
import brooklyn.util.exceptions.Exceptions;

import com.google.common.base.Objects;
import com.google.common.collect.Maps;

/**
 * Gives details of a location to be created. It describes the location's configuration, and is
 * reusable to create multiple locations with the same configuration.
 * 
 * To create a LocationSpec, it is strongly encouraged to use {@code create(...)} methods.
 * 
 * @param  The type of location to be created
 * 
 * @author aled
 */
public class LocationSpec implements Serializable {

    // TODO Would like to add `configure(ConfigBag)`, but `ConfigBag` is in core rather than api
    
    private static final Logger log = LoggerFactory.getLogger(LocationSpec.class);

    private final static long serialVersionUID = 1L;


    /**
     * @deprecated since 0.6 (added 0.6.0-M1); use {@link #create(Class)}
     */
    @Deprecated
    public static  LocationSpec spec(Class type) {
        return create(type);
    }

    /**
     * @deprecated since 0.6 (added 0.6.0-M1); use {@link #create(Map, Class)}
     */
    @Deprecated
    public static  LocationSpec spec(Map config, Class type) {
        return create(config, type);
    }

    /**
     * Creates a new {@link LocationSpec} instance for a location of the given type. The returned 
     * {@link LocationSpec} can then be customized.
     * 
     * @param type A {@link Location} class
     */
    public static  LocationSpec create(Class type) {
        return new LocationSpec(type);
    }
    
    /**
     * Creates a new {@link LocationSpec} instance with the given config, for a location of the given type.
     * 
     * This is primarily for groovy code; equivalent to {@code LocationSpec.create(type).configure(config)}.
     * 
     * @param config The spec's configuration (see {@link LocationSpec#configure(Map)}).
     * @param type   A {@link Location} class
     */
    public static  LocationSpec create(Map config, Class type) {
        return LocationSpec.create(type).configure(config);
    }
    
    private final Class type;
    private String displayName;
    private Location parent;
    private final Map flags = Maps.newLinkedHashMap();
    private final Map, Object> config = Maps.newLinkedHashMap();
    private final Map, Object> extensions = Maps.newLinkedHashMap();

    protected LocationSpec(Class type) {
        checkIsImplementation(type);
        checkIsNewStyleImplementation(type);
        this.type = type;
    }
    
    public LocationSpec displayName(String val) {
        displayName = val;
        return this;
    }

    public LocationSpec parent(Location val) {
        parent = checkNotNull(val, "parent");
        return this;
    }

    public LocationSpec configure(Map val) {
        for (Map.Entry entry: val.entrySet()) {
            if (entry.getKey()==null) throw new NullPointerException("Null key not permitted");
            if (entry.getKey() instanceof CharSequence)
                flags.put(entry.getKey().toString(), entry.getValue());
            else if (entry.getKey() instanceof ConfigKey)
                config.put((ConfigKey)entry.getKey(), entry.getValue());
            else if (entry.getKey() instanceof HasConfigKey)
                config.put(((HasConfigKey)entry.getKey()).getConfigKey(), entry.getValue());
            else {
                log.warn("Spec "+this+" ignoring unknown config key "+entry.getKey());
            }
        }
        return this;
    }
    
    public LocationSpec configure(CharSequence key, Object val) {
        flags.put(checkNotNull(key, "key").toString(), val);
        return this;
    }
    
    public  LocationSpec configure(ConfigKey key, V val) {
        config.put(checkNotNull(key, "key"), val);
        return this;
    }

    public  LocationSpec configureIfNotNull(ConfigKey key, V val) {
        return (val != null) ? configure(key, val) : this;
    }

    public  LocationSpec configure(ConfigKey key, Task val) {
        config.put(checkNotNull(key, "key"), val);
        return this;
    }

    public  LocationSpec configure(HasConfigKey key, V val) {
        config.put(checkNotNull(key, "key").getConfigKey(), val);
        return this;
    }

    public  LocationSpec configure(HasConfigKey key, Task val) {
        config.put(checkNotNull(key, "key").getConfigKey(), val);
        return this;
    }

    public  LocationSpec extension(Class extensionType, E extension) {
        extensions.put(checkNotNull(extensionType, "extensionType"), checkNotNull(extension, "extension"));
        return this;
    }

    /**
     * @return The type of the location
     */
    public Class getType() {
        return type;
    }
    
    /**
     * @return The display name of the location
     */
    public String getDisplayName() {
        return displayName;
    }
    
    /**
     * @return The location's parent
     */
    public Location getParent() {
        return parent;
    }
    
    /**
     * @return Read-only construction flags
     * @see SetFromFlag declarations on the location type
     */
    public Map getFlags() {
        return Collections.unmodifiableMap(flags);
    }
    
    /**
     * @return Read-only configuration values
     */
    public Map, Object> getConfig() {
        return Collections.unmodifiableMap(config);
    }
        
    /**
     * @return Read-only extension values
     */
    public Map, Object> getExtensions() {
        return Collections.unmodifiableMap(extensions);
    }
        
    
    @Override
    public String toString() {
        return Objects.toStringHelper(this).add("type", type).toString();
    }
    
    // TODO Duplicates method in EntitySpec and BasicEntityTypeRegistry
    private void checkIsImplementation(Class val) {
        if (!Location.class.isAssignableFrom(val)) throw new IllegalStateException("Implementation "+val+" does not implement "+Location.class.getName());
        if (val.isInterface()) throw new IllegalStateException("Implementation "+val+" is an interface, but must be a non-abstract class");
        if (Modifier.isAbstract(val.getModifiers())) throw new IllegalStateException("Implementation "+val+" is abstract, but must be a non-abstract class");
    }

    // TODO Duplicates method in EntitySpec, BasicEntityTypeRegistry, and InternalEntityFactory.isNewStyleEntity
    private void checkIsNewStyleImplementation(Class implClazz) {
        try {
            implClazz.getConstructor(new Class[0]);
        } catch (NoSuchMethodException e) {
            throw new IllegalStateException("Implementation "+implClazz+" must have a no-argument constructor");
        } catch (SecurityException e) {
            throw Exceptions.propagate(e);
        }
        
        if (implClazz.isInterface()) throw new IllegalStateException("Implementation "+implClazz+" is an interface, but must be a non-abstract class");
        if (Modifier.isAbstract(implClazz.getModifiers())) throw new IllegalStateException("Implementation "+implClazz+" is abstract, but must be a non-abstract class");
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy