org.jvnet.hk2.component.Habitat Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2007-2011 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package org.jvnet.hk2.component;
import com.sun.hk2.component.CompanionSeed;
import static com.sun.hk2.component.CompanionSeed.Registerer.createCompanion;
import com.sun.hk2.component.ExistingSingletonInhabitant;
import com.sun.hk2.component.FactoryCreator;
import com.sun.hk2.component.InjectInjectionResolver;
import com.sun.hk2.component.InjectionResolver;
import com.sun.hk2.component.LeadInjectionResolver;
import static com.sun.hk2.component.InhabitantsFile.CAGE_BUILDER_KEY;
import com.sun.hk2.component.ScopeInstance;
import org.jvnet.hk2.annotations.Contract;
import org.jvnet.hk2.annotations.ContractProvided;
import org.jvnet.hk2.annotations.FactoryFor;
import org.jvnet.hk2.component.HabitatListener.EventType;
import org.jvnet.hk2.component.InhabitantTracker.Callback;
import org.jvnet.hk2.component.internal.runlevel.DefaultRunLevelService;
import java.lang.annotation.Annotation;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.Map.Entry;
import java.util.*;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.ThreadFactory;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* A set of templates that constitute a world of objects.
*
* @author Kohsuke Kawaguchi
* @author Jerome Dochez
*/
@SuppressWarnings("unchecked")
public class Habitat {
/**
* Name to use to programmatically store default instances of a particular service.
*/
public final String DEFAULT_NAME = "_HABITAT_DEFAULT";
/**
* Contract type FQCN to their corresponding inhabitants.
*
* Can't use {@link Class} as the key so that we can create index without
* loading contract types. Once populated upfront, it works as a read-only map.
*/
private final MultiMap byContract;
/**
* Index by {@link Inhabitant#type()}.
*/
private final MultiMap byType;
public final ScopeInstance singletonScope;
// TODO: toggle to use concurrency controls as the default (or after habitat is initially built)
// TODO: Jerome to review performance costs
static final boolean CONCURRENCY_CONTROLS_DEFAULT =
// "true".equals(System.getProperty("hk2.concurrency.controls", "true"));
Boolean.getBoolean("hk2.concurrency.controls");
private final boolean concurrencyControls;
// Listeners and trackers use this executor for notification of habitat
// changes. Many systems will require same-thread notifications. This is
// why OSGi defaults to synchronous service tracker customizers, etc.
// We also, therefore, will default to a synchronous model.
static boolean ASYNC_EXECUTOR =
Boolean.getBoolean("hk2.async.executor");
final ExecutorService exec;
private boolean initialized;
public Habitat() {
this(null, null);
}
Habitat(ExecutorService exec, Boolean concurrency_controls) {
this.concurrencyControls =
(null == concurrency_controls)
? CONCURRENCY_CONTROLS_DEFAULT : concurrency_controls;
this.byContract = new MultiMap(this.concurrencyControls);
this.byType = new MultiMap(this.concurrencyControls);
this.singletonScope = new ScopeInstance("singleton", new HashMap());
if (null == exec) {
if (ASYNC_EXECUTOR) {
exec = Executors.newCachedThreadPool(new ThreadFactory() {
@Override
public Thread newThread(Runnable runnable) {
Thread t = Executors.defaultThreadFactory().newThread(runnable);
t.setDaemon(true);
return t;
}
});
} else {
exec = new SameThreadExecutor();
}
}
this.exec = exec;
// make the listeners available very early in lifecycle
addHabitatListener(new SelfListener());
// add the set of injection resolvers
add(new ExistingSingletonInhabitant(InjectionResolver.class,
new InjectInjectionResolver(this)));
add(new ExistingSingletonInhabitant(InjectionResolver.class,
new LeadInjectionResolver(this)));
// make the habitat itself available
add(new ExistingSingletonInhabitant(Habitat.class,this));
add(new ExistingSingletonInhabitant(CompanionSeed.Registerer.class,
new CompanionSeed.Registerer(this)));
add(new ExistingSingletonInhabitant(CageBuilder.Registerer.class,
new CageBuilder.Registerer(this)));
// the default RunLevelService
DefaultRunLevelService rls = new DefaultRunLevelService(this);
ExistingSingletonInhabitant rlsI =
new ExistingSingletonInhabitant(RunLevelService.class, rls);
add(rlsI);
addIndex(rlsI, RunLevelService.class.getName(), DefaultRunLevelService.NAME);
}
/**
* Add a habitat listener with no contract-level filtering.
* This API is primarily intended for internal cases within
* Hk2.
*
* The listener with no contract-level filtering will be
* called for all change events within the habitat pertaining
* to inhabitants.
*
* @see {@link #addHabitatListener(HabitatListener, String...)}
* is recommended for most cases
*
* @param listener
* The habitat Listener to be added
*
* @since 3.1
*/
public void addHabitatListener(HabitatListener listener) {
addHabitatListener(listener, DEFAULT_NAME);
}
/**
* Add a habitat listener with contract-level filtering.
*
* The listener will be called based on the set of contract
* filters provided.
*
* @param listener
* The habitat Listener to be added
* @param typeNames
* The contracts to filter on; this should be non-null
*
* @since 3.1
*/
public void addHabitatListener(HabitatListener listener, String... typeNames) {
if (null == typeNames || 0 == typeNames.length) throw new IllegalArgumentException();
addHabitatListener(listener, new HashSet(Arrays.asList(typeNames)));
}
protected void addHabitatListener(HabitatListener listener, Set typeNames) {
ExistingSingletonInhabitant inhabitant =
new ExistingSingletonInhabitant(HabitatListener.class,
listener, metaData(typeNames));
add(inhabitant);
// track listeners by type - support of filtered notification
for (String contract : typeNames) {
ListenersByTypeInhabitant sameListeners;
synchronized (byContract) {
sameListeners = (ListenersByTypeInhabitant)
getInhabitantByContract(ListenersByTypeInhabitant.class.getName(),
contract);
if (null == sameListeners) {
sameListeners = new ListenersByTypeInhabitant(contract);
addIndex(sameListeners, ListenersByTypeInhabitant.class.getName(),
contract,
false); // don't send notifications for this type (no particular reason really)
}
}
sameListeners.add(listener);
}
}
/**
* Remove a habitat listener.
*
* @param listener
* The habitat Listener to be removed
* @param typeNames
* The contracts to filter on
*
* @return true; if the listener was indeed removed
*
* @since 3.1
*/
public boolean removeHabitatListener(HabitatListener listener) {
List list = byType._get(HabitatListener.class.getName());
List releaseList = new ArrayList();
Iterator iter = list.iterator();
while (iter.hasNext()) {
Inhabitant existing = iter.next();
if (existing.get() == listener) {
releaseList.add(existing);
if (concurrencyControls) {
list.remove(existing);
} else {
iter.remove();
}
}
}
// need to release the per-listener-contract entries too
for (Inhabitant released : releaseList) {
MultiMap metadata = released.metadata();
if (null != metadata) {
List filters = metadata.get(Constants.OBJECTCLASS);
for (String contract : filters) {
ListenersByTypeInhabitant sameListeners;
synchronized (byContract) {
sameListeners = (ListenersByTypeInhabitant)
getInhabitantByContract(ListenersByTypeInhabitant.class.getName(),
contract);
}
sameListeners.remove(listener);
}
}
released.release();
notify(released, EventType.INHABITANT_REMOVED, null, released);
}
return !releaseList.isEmpty();
}
/**
* Convert a set of type names into OBJECTCLASS keys within metadata
*/
private MultiMap metaData(Set typeNames) {
if (null == typeNames) {
return null;
}
MultiMap metadata = new MultiMap();
for (String typeName : typeNames) {
metadata.add(Constants.OBJECTCLASS, typeName);
}
return metadata;
}
/**
* Registers a dependency on the inhabitant with the given tracker context.
*
* Once the criteria is met, any callback provided is called. This callback may
* occur asynchronously from the thread initiating the event.
*
* @param itc
* The tracking criteria.
* @param callback
* Optionally the callback.
* @return
* The tracker
* @throws ComponentException
*
* @since 3.1
*/
public InhabitantTracker track(InhabitantTrackerContext itc,
Callback callback) throws ComponentException {
if (null == itc) throw new IllegalArgumentException();
return new InhabitantTrackerImpl(this, itc, callback);
}
/**
* Returns a future that can be checked asynchronously, and multiple times.
*
* Implementation Note: The Future that is returned does not behave in
* the traditional sense in that it is NOT directly submitted to an
* ExecutorService. Each call to get() or get(timeout) may result in a
* [re]submission to an internally held executor. This means that a
* call to get(...) may return a tracker, and a subsequent call to get(...) may
* return null, or vice versa. This is true until the underlying tracker is
* released at which point a tracker is no longer usable.
*
* @param itc
* The tracking criteria.
* @return
* The tracker
* @throws ComponentException
*
* @since 3.1
*/
public Future trackFuture(InhabitantTrackerContext itc) throws ComponentException {
if (null == itc) throw new IllegalArgumentException();
return new InhabitantTrackerJob(this, itc);
}
/*
Why initialize/processInhabitantDecorations didn't work:
when a new CageBuilder comes into the habitat, it needs to build cages for all existing components.
when a caged component comes into the habitat, it checks existing cage builders in the habitat.
Now, when a cage builder and a component are both initialized first and then processInhabitantDecorations
runs for both of them later, then you end up creating a cage twice, because the builder think it got into
habitat after than the component, and the component think it got into habitat after the builder.
*/
/**
* Removes all inhabitants for a particular type
*
* @param type of the component
*
* @return true if any inhabitants were removed
*/
public boolean removeAllByType(Class> type) {
boolean removed = false;
String name = type.getName();
// remove all existing inhabitants by type
List list = new ArrayList(byType.get(name));
for (Inhabitant existing : list) {
removed |= remove(existing);
}
return removed;
}
/**
* Adds a new inhabitant.
*
*
* See {@link Inhabitants} for typical ways to create {@link Inhabitant}s.
*/
public void add(final Inhabitant> i) {
String name = i.typeName();
byType.add(name,i);
// for each companion, create an inhabitant that goes with the lead and hook them up
List companions=null;
for(Inhabitant> c : getInhabitantsByAnnotation(CompanionSeed.class,name)) {
if (companions==null) {
companions = new ArrayList();
}
companions.add(createCompanion(this,i,c));
}
i.setCompanions(companions);
String cageBuilderName = i.metadata().getOne(CAGE_BUILDER_KEY);
if(cageBuilderName!=null) {
Inhabitant cageBuilder = byType.getOne(cageBuilderName);
if (cageBuilder!=null) {
((CageBuilder)cageBuilder.get()).onEntered(i);
}
// if cageBuilder==null, we can't cage this component now, but
// we'll do that when cageBuilder comes into the habitat.
// that happens because every CageBuilder implementations are caged by
// CageBuilder.Registerer.
}
notify(i, EventType.INHABITANT_ADDED, null, null);
}
/**
* Adds a new index to look up the given inhabitant.
*
* @param index
* Primary index name, such as contract FQCN.
* @param name
* Name that identifies the inhabitant among other inhabitants
* in the same index. Can be null for unnamed inhabitants.
*/
public void addIndex(Inhabitant> i, String index, String name) {
addIndex(i, index, name, true);
}
protected void addIndex(Inhabitant> i, String index, String name, boolean notify) {
byContract.add(index,new NamedInhabitant(name,i));
if (notify) {
notify(i, EventType.INHABITANT_INDEX_ADDED, index, name, null, null);
}
}
protected static Long getServiceRanking(Inhabitant> i, boolean wantNonNull) {
MultiMap meta = i.metadata();
String sr = meta.getOne(Constants.SERVICE_RANKING);
if (null == sr) {
return (wantNonNull) ? 0L : null;
}
return Long.valueOf(sr);
}
/**
* Removes an inhabitant
*
* @param inhabitant
* inhabitant to be removed
*/
public boolean remove(Inhabitant> inhabitant) {
String name = inhabitant.typeName();
if (byType.remove(name, inhabitant)) {
notify(inhabitant, EventType.INHABITANT_REMOVED, null, null);
inhabitant.release();
return true;
}
return false;
}
/**
* Removes a NamedInhabitant for a specific contract
*
* @param index contract name
* @param name instance name
* @return true if the removal was successful
*/
public boolean removeIndex(String index, String name) {
boolean removed = false;
if (byContract.containsKey(index)) {
List contracted = byContract._get(index);
Iterator iter = contracted.iterator();
while (iter.hasNext()) {
NamedInhabitant i = iter.next();
if ((i.name == null && name == null) ||
(i.name != null && i.name.equals(name))) {
if (concurrencyControls) {
removed = contracted.remove(i);
assert(removed);
} else {
iter.remove();
}
removed = true;
notify(i.inhabitant, EventType.INHABITANT_INDEX_REMOVED,
index, name, null, null);
// remember to remove the components stored under its type
remove(i.inhabitant);
}
}
}
return removed;
}
/**
* Removes a Contracted service
*
* @param index the contract name
* @param serviceOrInhabitant the service instance, or an Inhabitant instance
*/
public boolean removeIndex(String index, Object serviceOrInhabitant) {
boolean removed = false;
if (byContract.containsKey(index)) {
List contracted = byContract._get(index);
Iterator iter = contracted.iterator();
while (iter.hasNext()) {
NamedInhabitant i = iter.next();
if (matches(i.inhabitant, serviceOrInhabitant)) {
if (concurrencyControls) {
removed = contracted.remove(i);
assert(removed);
} else {
iter.remove();
}
removed = true;
notify(i.inhabitant, EventType.INHABITANT_INDEX_REMOVED,
index, null, service(serviceOrInhabitant), null);
// remember to remove the components stored under its type
remove(i.inhabitant);
}
}
}
return removed;
}
protected boolean matches(Inhabitant> inhabitant, Object serviceOrInhabitant) {
boolean matches;
if (serviceOrInhabitant instanceof Inhabitant) {
matches = (serviceOrInhabitant == inhabitant);
} else {
// call to isInstantiated is necessary to avoid loading of services that are not yet loaded.
matches = inhabitant.isInstantiated() && inhabitant.get()==serviceOrInhabitant;
}
return matches;
}
protected Object service(Object serviceOrInhabitant) {
return (serviceOrInhabitant instanceof Inhabitant)
? ((Inhabitant)serviceOrInhabitant).get() : serviceOrInhabitant;
}
protected static interface NotifyCall {
boolean inhabitantChanged(HabitatListener listener);
}
/**
* Trigger a notification that an inhabitant has changed.
*
* @param inhabitant the inhabitant that has changed
* @param contracts the contracts associated with the inhabitant
*/
public void notifyInhabitantChanged(Inhabitant> inhabitant,
String... contracts) {
if (null == contracts || 0 == contracts.length) {
// generic inhabitant modification (only general listeners will get notification)
notify(inhabitant, EventType.INHABITANT_MODIFIED, null, null);
} else {
// all scoped-listeners will get modification
for (String contract : contracts) {
notify(inhabitant, EventType.INHABITANT_MODIFIED, contract, null);
}
}
}
/**
* FOR INTERNAL USE ONLY
*/
public void initialized() {
if (initialized) throw new RuntimeException("already initialized");
initialized = true;
notify(null, EventType.HABITAT_INITIALIZED, null, null);
}
public boolean isInitialized() {
return initialized;
}
protected void notify(final Inhabitant> inhabitant,
final EventType event,
final String index,
final Inhabitant extraListenerToBeNotified) {
NotifyCall innerCall = new NotifyCall() {
public boolean inhabitantChanged(HabitatListener listener) {
return listener.inhabitantChanged(event, Habitat.this, inhabitant);
}
};
notify(innerCall, inhabitant, event, index, extraListenerToBeNotified);
}
protected void notify(final Inhabitant> inhabitant,
final EventType event,
final String index,
final String name,
final Object service,
final Inhabitant extraListenerToBeNotified) {
NotifyCall innerCall = new NotifyCall() {
public boolean inhabitantChanged(HabitatListener listener) {
return listener.inhabitantIndexChanged(event, Habitat.this, inhabitant,
index, name, service);
}
};
notify(innerCall, inhabitant, event, index, extraListenerToBeNotified);
}
protected void notify(final NotifyCall innerCall,
final Inhabitant> inhabitant,
final EventType event,
final String index,
final Inhabitant extraListenerToBeNotified) {
// do scoped listeners first
if (null != index) {
doNotify(innerCall, inhabitant, event, index, null);
}
// do general listeners next
doNotify(innerCall, inhabitant, event, DEFAULT_NAME, extraListenerToBeNotified);
}
private void doNotify(final NotifyCall innerCall,
final Inhabitant> inhabitant,
final EventType event,
final String index,
final Inhabitant extraListenerToBeNotified) {
final ListenersByTypeInhabitant sameListeners =
(ListenersByTypeInhabitant)getInhabitantByContract(
ListenersByTypeInhabitant.class.getName(),
index);
if (null != sameListeners && !sameListeners.listeners.isEmpty()) {
exec.execute(new Runnable() {
public void run() {
Iterator iter = sameListeners.listeners.iterator();
while (iter.hasNext()) {
Object entry = iter.next();
HabitatListener listener =
(entry instanceof HabitatListener) ?
(HabitatListener)entry :
(HabitatListener)((Inhabitant)entry).get();
try {
boolean keepMe = innerCall.inhabitantChanged(listener);
if (!keepMe) {
removeHabitatListener(listener);
}
} catch (Exception e){
// don't percolate the exception since it may negatively impact processing
Logger.getLogger(Habitat.class.getName()).
log(Level.WARNING, "exception caught from listener: ", e);
}
}
// we take the extraListenerToBeNotified because
// (a) we want to have all notifications in the executor, and
// (b) it might trigger an exception that we want to trap
if (null != extraListenerToBeNotified) {
extraListenerToBeNotified.get().inhabitantChanged(event, Habitat.this, extraListenerToBeNotified);
}
}
});
}
}
/**
* Checks if the given type is a contract interface that has some implementations in this {@link Habitat}.
*
*
* There are two ways for a type to be marked as a contract.
* Either it has {@link Contract}, or it's marked by {@link ContractProvided} from the implementation.
*
*
* Note that just having {@link Contract} is not enough to make this method return true.
* It can still return false if the contract has no implementation in this habitat.
*
*
* This method is useful during the injection to determine what lookup to perform,
* and it handles the case correctly when the type is marked as a contract by {@link ContractProvided}.
*/
public boolean isContract(Class> type) {
return byContract.containsKey(type.getName());
}
public boolean isContract(String fullyQualifiedClassName) {
return byContract.containsKey(fullyQualifiedClassName);
}
/**
* Gets all the inhabitants registered under the given {@link Contract}.
* This is an example of heterogeneous type-safe container.
*
* @return
* can be empty but never null.
*/
public Collection getAllByContract(Class contractType) {
return getAllByContract(contractType.getName());
}
public Collection getAllByContract(String contractType) {
final List l = byContract.get(contractType);
return new AbstractList() {
public T get(int index) {
return (T)l.get(index).inhabitant.get();
}
public int size() {
return l.size();
}
};
}
public Collection> getAllInhabitantsByContract(String contractType) {
final List l = byContract.get(contractType);
return new AbstractList>() {
public Inhabitant> get(int index) {
return l.get(index).inhabitant;
}
public int size() {
return l.size();
}
};
}
/**
* Gets the object of the given type.
*
* @return
* can be empty but never null.
*/
public Collection getAllByType(Class implType) {
final List l = byType.get(implType.getName());
return new AbstractList() {
public T get(int index) {
return (T)l.get(index).get();
}
public int size() {
return l.size();
}
};
}
/**
* Add an already instantiated component to this manager. The component has
* been instantiated by external code, however dependency injection, PostConstruct
* invocation and dependency extraction will be performed on this instance before
* it is store in the relevant scope's resource manager.
*
* @param name name of the component, could be default name
* @param component component instance
* @throws ComponentException if the passed object is not an HK2 component or
* injection/extraction failed.
*/
// TODO: mutating Habitat after it's created poses synchronization issue
public void addComponent(String name, T component) throws ComponentException {
add(new ExistingSingletonInhabitant(component));
}
/**
* Obtains a reference to the component inside the manager.
*
*
* This is the "new Foo()" equivalent in the IoC world.
*
*
* Depending on the {@link Scope} of the component, a new instance
* might be created, or an existing instance might be returned.
*
* @return
* non-null.
* @throws ComponentException
* If failed to obtain a requested instance.
* In practice, failure only happens when we try to create a
* new instance of the component.
*/
public T getComponent(Class clazz) throws ComponentException {
if(isContract(clazz))
return getByContract(clazz);
else
return getByType(clazz);
}
/**
* Loads a component that implements the given contract and has the given name.
*
* @param name
* can be null, in which case it'll only match to the unnamed component.
* @return
* null if no such servce exists.
*/
public T getComponent(final Class contract, String name) throws ComponentException {
if (name!=null && name.length()==0)
name=null;
final Inhabitant i = getInhabitant(contract, name);
if(i!=null)
try {
return contract.cast(i.get());
} catch (ClassCastException e) {
Logger.getAnonymousLogger().severe("ClassCastException between contract " + contract + " and service " + i.get());
AccessController.doPrivileged(new PrivilegedAction