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

org.apache.webbeans.container.BeanManagerImpl Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you 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.apache.webbeans.container;

import java.io.InputStream;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.CopyOnWriteArraySet;

import javax.el.ELResolver;
import javax.el.ExpressionFactory;
import javax.enterprise.context.ContextNotActiveException;
import javax.enterprise.context.Dependent;
import javax.enterprise.context.NormalScope;
import javax.enterprise.context.spi.Context;
import javax.enterprise.context.spi.Contextual;
import javax.enterprise.context.spi.CreationalContext;
import javax.enterprise.inject.Default;
import javax.enterprise.inject.Stereotype;
import javax.enterprise.inject.spi.AnnotatedType;
import javax.enterprise.inject.spi.Bean;
import javax.enterprise.inject.spi.BeanManager;
import javax.enterprise.inject.spi.Decorator;
import javax.enterprise.inject.spi.InjectionPoint;
import javax.enterprise.inject.spi.InjectionTarget;
import javax.enterprise.inject.spi.InterceptionType;
import javax.enterprise.inject.spi.Interceptor;
import javax.enterprise.inject.spi.ObserverMethod;
import javax.enterprise.inject.spi.PassivationCapable;
import javax.enterprise.inject.spi.Producer;
import javax.inject.Scope;
import javax.interceptor.InterceptorBinding;
import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.Referenceable;
import javax.naming.StringRefAddr;

import org.apache.webbeans.component.AbstractOwbBean;
import org.apache.webbeans.component.EnterpriseBeanMarker;
import org.apache.webbeans.component.JmsBeanMarker;
import org.apache.webbeans.component.NewBean;
import org.apache.webbeans.component.OwbBean;
import org.apache.webbeans.component.third.PassivationCapableThirdpartyBeanImpl;
import org.apache.webbeans.component.third.ThirdpartyBeanImpl;
import org.apache.webbeans.config.WebBeansContext;
import org.apache.webbeans.context.creational.CreationalContextImpl;
import org.apache.webbeans.decorator.DecoratorComparator;
import org.apache.webbeans.event.EventMetadata;
import org.apache.webbeans.event.EventMetadataImpl;
import org.apache.webbeans.event.NotificationManager;
import org.apache.webbeans.exception.WebBeansConfigurationException;
import org.apache.webbeans.exception.definition.DuplicateDefinitionException;
import org.apache.webbeans.exception.inject.DefinitionException;
import org.apache.webbeans.plugins.OpenWebBeansJmsPlugin;
import org.apache.webbeans.portable.AnnotatedElementFactory;
import org.apache.webbeans.portable.events.discovery.ErrorStack;
import org.apache.webbeans.spi.adaptor.ELAdaptor;
import org.apache.webbeans.spi.plugins.OpenWebBeansEjbPlugin;
import org.apache.webbeans.util.AnnotationUtil;
import org.apache.webbeans.util.Asserts;
import org.apache.webbeans.util.ClassUtil;
import org.apache.webbeans.util.GenericsUtil;
import org.apache.webbeans.util.WebBeansUtil;
import org.apache.webbeans.xml.WebBeansXMLConfigurator;

/**
 * Implementation of the {@link BeanManager} contract of the web beans
 * container.
 * 
 * 

* It is written as thread-safe. *

* * @version $Rev$ $Date$ * @see BeanManager */ @SuppressWarnings("unchecked") public class BeanManagerImpl extends AbstractBeanManager implements BeanManager, Referenceable { private static final long serialVersionUID = 2L; /** * Holds the non-standard contexts with key = scope type * This will get used if more than 1 scope exists. * Since the contexts will only get added through the * {@link org.apache.webbeans.portable.events.discovery.AfterBeanDiscoveryImpl} * we don't even need a ConcurrentHashMap. * @see #singleContextMap */ private Map, List> contextMap = new HashMap, List>(); /** * This will hold non-standard contexts where only one Context implementation got registered * for the given key = scope type * Since the contexts will only get added through the * {@link org.apache.webbeans.portable.events.discovery.AfterBeanDiscoveryImpl} * we don't even need a ConcurrentHashMap. * @see #contextMap */ private Map, Context> singleContextMap = new HashMap, Context>(); /**Deployment archive beans*/ private Set> deploymentBeans = new CopyOnWriteArraySet>(); /**Normal scoped cache proxies*/ private Map, Object> cacheProxies = new ConcurrentHashMap, Object>(); /**Event notification manager instance*/ private NotificationManager notificationManager = null; /**Injection resolver instance*/ private InjectionResolver injectionResolver = null; /**XML configurator instance*/ private WebBeansXMLConfigurator xmlConfigurator = null; /** * This list contains additional qualifiers which got set via the * {@link javax.enterprise.inject.spi.BeforeBeanDiscovery#addQualifier(Class)} * event function. */ private List> additionalQualifiers = new ArrayList>(); /** * This list contains additional interceptor bindings which got set via the * {@link javax.enterprise.inject.spi.BeforeBeanDiscovery#addInterceptorBinding(Class, java.lang.annotation.Annotation...)} * event function. */ private List> additionalInterceptorBindings = new ArrayList>(); /** * This list contains additional scopes which got set via the * {@link javax.enterprise.inject.spi.BeforeBeanDiscovery#addScope(Class, boolean, boolean)} event function. */ private List additionalScopes = new ArrayList(); private ConcurrentMap, ConcurrentMap>> additionalAnnotatedTypes = new ConcurrentHashMap, ConcurrentMap>>(); private ErrorStack errorStack = new ErrorStack(); /** * This map stores all beans along with their unique {@link javax.enterprise.inject.spi.PassivationCapable} id. * This is used as a reference for serialization. */ private ConcurrentMap> passivationBeans = new ConcurrentHashMap>(); /**InjectionTargets for Java EE component instances that supports injections*/ private Map, Producer> producersForJavaEeComponents = new ConcurrentHashMap, Producer>(); private AnnotatedElementFactory annotatedElementFactory; private final WebBeansContext webBeansContext; /** * This flag will get set to true if a custom bean * (all non-internal beans like {@link org.apache.webbeans.component.BeanManagerBean;} etc) * gets set. */ private boolean inUse = false; /** * we cache results of calls to {@link #isNormalScope(Class)} because * this doesn't change at runtime. * We don't need to take special care about classloader * hierarchies, because each cl has other classes. */ private static Map, Boolean> isScopeTypeNormalCache = new ConcurrentHashMap, Boolean>(); /** * Creates a new {@link BeanManager} instance. * Called by the system. Do not use outside of the * system. */ public BeanManagerImpl(WebBeansContext webBeansContext) { this.webBeansContext = webBeansContext; injectionResolver = new InjectionResolver(webBeansContext); notificationManager = new NotificationManager(webBeansContext); annotatedElementFactory = webBeansContext.getAnnotatedElementFactory(); } @Override public WebBeansContext getWebBeansContext() { return webBeansContext; } public void putProducerForJavaEeComponent(Class javaEeComponentClass, Producer wrapper) { Asserts.assertNotNull(javaEeComponentClass); Asserts.assertNotNull(wrapper); producersForJavaEeComponents.put(javaEeComponentClass, wrapper); } public Producer getProducerForJavaEeComponent(Class javaEeComponentClass) { Asserts.assertNotNull(javaEeComponentClass); return (Producer) producersForJavaEeComponents.get(javaEeComponentClass); } public ErrorStack getErrorStack() { return errorStack; } /** * Return manager notification manager. * * @return notification manager */ public NotificationManager getNotificationManager() { return notificationManager; } /** * Gets injection resolver. * * @return injection resolver */ public InjectionResolver getInjectionResolver() { return injectionResolver; } /** * Sets the xml configurator instance. * * @param xmlConfigurator set xml configurator instance. * @see WebBeansXMLConfigurator */ public synchronized void setXMLConfigurator(WebBeansXMLConfigurator xmlConfigurator) { if(this.xmlConfigurator != null) { throw new IllegalStateException("WebBeansXMLConfigurator is already defined!"); } this.xmlConfigurator = xmlConfigurator; } /** * Gets the active context for the given scope type. * * @param scopeType scope type of the context * @throws ContextNotActiveException if no active context * @throws IllegalStateException if more than one active context */ @Override public Context getContext(Class scopeType) { Asserts.assertNotNull(scopeType, "scopeType parameter can not be null"); Context standardContext = webBeansContext.getContextsService().getCurrentContext(scopeType); if(standardContext != null && standardContext.isActive()) { return standardContext; } // this is by far the most case Context singleContext = singleContextMap.get(scopeType); if (singleContext != null) { if (!singleContext.isActive()) { throw new ContextNotActiveException("WebBeans context with scope type annotation @" + scopeType.getSimpleName() + " does not exist within current thread"); } return singleContext; } // the spec also allows for multiple contexts existing for the same scope type // but in this case only one must be active at a time (for the current thread) List others = contextMap.get(scopeType); Context found = null; if(others != null) { for(Context otherContext : others) { if(otherContext.isActive()) { if (found != null) { throw new IllegalStateException("More than one active context exists with scope type annotation @" + scopeType.getSimpleName()); } found = otherContext; } } } if (found == null) { throw new ContextNotActiveException("WebBeans context with scope type annotation @" + scopeType.getSimpleName() + " does not exist within current thread"); } return found; } /** * Add new bean to the BeanManager. * This will also set OWBs {@link #inUse} status. * * @param newBean new bean instance * @return the this manager */ public BeanManager addBean(Bean newBean) { inUse = true; return addInternalBean(newBean); } /** * This method is reserved for adding 'internal beans' * like e.g. a BeanManagerBean, * @param newBean * @return */ public BeanManager addInternalBean(Bean newBean) { if(newBean instanceof AbstractOwbBean) { addPassivationInfo((OwbBean)newBean); deploymentBeans.add(newBean); } else { ThirdpartyBeanImpl bean; if (!PassivationCapable.class.isInstance(newBean)) { bean = new ThirdpartyBeanImpl(webBeansContext, newBean); } else { bean = new PassivationCapableThirdpartyBeanImpl(webBeansContext, newBean); } addPassivationInfo(bean); deploymentBeans.add(bean); } return this; } /** * Check if the bean is has a passivation id and add it to the id store. * * @param bean * @throws DefinitionException if the id is not unique. */ public void addPassivationInfo(Bean bean) throws DefinitionException { String id = null; if (bean instanceof OwbBean) { id = ((OwbBean) bean).getId(); } if (id == null && bean instanceof PassivationCapable) { id = ((PassivationCapable) bean).getId(); } if(id != null) { Bean oldBean = passivationBeans.putIfAbsent(id, bean); if (oldBean != null) { throw new DuplicateDefinitionException("PassivationCapable bean id is not unique: " + id + " bean:" + bean); } } } public BeanManager addContext(Context context) { addContext(context.getScope(), webBeansContext.getContextFactory().getCustomContext(context)); return this; } /** * {@inheritDoc} */ @Override public void fireEvent(Object event, Annotation... bindings) { fireEvent(event, new EventMetadataImpl(event.getClass(), null, bindings), false); } /** * Like {@link #fireEvent(Object, java.lang.annotation.Annotation...)} but intended for * internal CDI Container lifecycle events. The difference is that those * events must only be delivered to CDI Extensions and not to normal beans. */ public void fireLifecycleEvent(Object event, Annotation... bindings) { fireEvent(event, new EventMetadataImpl(event.getClass(), null, bindings), true); } public void fireEvent(Object event, EventMetadata metadata, boolean isLifecycleEvent) { if (ClassUtil.isDefinitionContainsTypeVariables(event.getClass())) { throw new IllegalArgumentException("Event class : " + event.getClass().getName() + " can not be defined as generic type"); } notificationManager.fireEvent(event, metadata, isLifecycleEvent); } public Set> getComponents() { return deploymentBeans; } /** * {@inheritDoc} */ @Override public List> resolveDecorators(Set types, Annotation... bindingTypes) { webBeansContext.getAnnotationManager().checkDecoratorResolverParams(types, bindingTypes); Set> intsSet = webBeansContext.getDecoratorsManager().findDeployedWebBeansDecorator(types, bindingTypes); List> decoratorList = new ArrayList>(intsSet); Collections.sort(decoratorList, new DecoratorComparator(webBeansContext)); return decoratorList; } /** * {@inheritDoc} */ @Override public List> resolveInterceptors(InterceptionType type, Annotation... interceptorBindings) { webBeansContext.getAnnotationManager().checkInterceptorResolverParams(interceptorBindings); return webBeansContext.getInterceptorsManager().resolveInterceptors(type, interceptorBindings); } public Set> getBeans() { return deploymentBeans; } private void addContext(Class scopeType, javax.enterprise.context.spi.Context context) { Asserts.assertNotNull(scopeType, "scopeType parameter can not be null"); Asserts.assertNotNull(context, "context parameter can not be null"); List contextList = contextMap.get(scopeType); if(contextList == null) { Context singleContext = singleContextMap.get(scopeType); if (singleContext == null) { // first put them into the singleContextMap singleContextMap.put(scopeType, context); } else { // from the 2nd Context for this scopetype on, we need to maintain a List for them contextList = new ArrayList(); contextList.add(singleContext); contextList.add(context); contextMap.put(scopeType, contextList); singleContextMap.remove(scopeType); } } else { contextList.add(context); } } @Override public Reference getReference() throws NamingException { return new Reference(BeanManagerImpl.class.getName(), new StringRefAddr("ManagerImpl", "ManagerImpl"), ManagerObjectFactory.class.getName(), null); } /** * Parse the given XML input stream for adding XML defined artifacts. * * @param xmlStream beans xml definitions * @return {@link BeanManager} instance */ public BeanManager parse(InputStream xmlStream) { xmlConfigurator.configure(xmlStream); return this; } /** * {@inheritDoc} */ @Override public AnnotatedType createAnnotatedType(Class type) { AnnotatedType annotatedType = annotatedElementFactory.newAnnotatedType(type); return annotatedType; } /** * {@inheritDoc} */ @Override public CreationalContextImpl createCreationalContext(Contextual contextual) { if (contextual instanceof SerializableBean) { contextual = ((SerializableBean)contextual).getBean(); } return webBeansContext.getCreationalContextFactory().getCreationalContext(contextual); } /** * {@inheritDoc} */ @Override public Set> getBeans(Type beanType, Annotation... bindings) { if(ClassUtil.isTypeVariable(beanType)) { throw new IllegalArgumentException("Exception in getBeans method. Bean type can not be TypeVariable for bean type : " + beanType); } webBeansContext.getAnnotationManager().checkQualifierConditions(bindings); return injectionResolver.implResolveByType(false, beanType, bindings); } @Override public Set> getBeans(String name) { Asserts.assertNotNull(name, "name parameter can not be null"); return injectionResolver.implResolveByName(name); } @Override public ELResolver getELResolver() { ELAdaptor elAdaptor = webBeansContext.getService(ELAdaptor.class); return elAdaptor.getOwbELResolver(); } /** * {@inheritDoc} */ @Override public Object getInjectableReference(InjectionPoint injectionPoint, CreationalContext ownerCreationalContext) { Asserts.assertNotNull(injectionPoint, "injectionPoint parameter can not be null"); //Injected instance Object instance = null; //Injection point is null if(injectionPoint == null) { return null; } //Find the injection point Bean Bean injectedBean = (Bean)injectionResolver.getInjectionPointBean(injectionPoint); if(WebBeansUtil.isDependent(injectedBean)) { if (!(ownerCreationalContext instanceof CreationalContextImpl)) { ownerCreationalContext = webBeansContext.getCreationalContextFactory().wrappedCreationalContext(ownerCreationalContext, injectionPoint.getBean()); } ((CreationalContextImpl)ownerCreationalContext).putInjectionPoint(injectionPoint); //Using owner creational context //Dependents use parent creational context try { instance = getReference(injectedBean, injectionPoint.getType(), ownerCreationalContext); } finally { ((CreationalContextImpl)ownerCreationalContext).removeInjectionPoint(); } } else { //New creational context for normal scoped beans CreationalContextImpl injectedCreational = (CreationalContextImpl)createCreationalContext(injectedBean); injectedCreational.putInjectionPoint(injectionPoint); try { instance = getReference(injectedBean, injectionPoint.getType(), injectedCreational); } finally { injectedCreational.removeInjectionPoint(); } } return instance; } /** * {@inheritDoc} */ @Override public Set getInterceptorBindingDefinition(Class binding) { Annotation[] annotations = binding.getDeclaredAnnotations(); Set set = new HashSet(); if(binding.isAnnotationPresent(InterceptorBinding.class)) { Collections.addAll(set, annotations); } return set; } @Override public Bean getPassivationCapableBean(String id) { return passivationBeans.get(id); } /** * {@inheritDoc} */ @Override public Object getReference(Bean bean, Type beanType, CreationalContext creationalContext) { Asserts.assertNotNull(bean, "bean parameter can not be null"); Context context = null; Object instance = null; if (bean instanceof SerializableBean) { bean = ((SerializableBean)bean).getBean(); } if(!(creationalContext instanceof CreationalContextImpl)) { creationalContext = webBeansContext.getCreationalContextFactory().wrappedCreationalContext(creationalContext, bean); } //Check type if bean type is given if(beanType != null && beanType != Object.class) { if(!isBeanTypeAssignableToGivenType(bean.getTypes(), beanType, bean instanceof NewBean) && !GenericsUtil.satisfiesDependency(false, beanType, bean.getBeanClass())) { throw new IllegalArgumentException("Given bean type : " + beanType + " is not applicable for the bean instance : " + bean); } } else if (bean instanceof OwbBean) { // we cannot always use getBeanClass() as this will // return the containing class for producer methods and fields beanType = ((OwbBean) bean).getReturnType(); } else { beanType = bean.getBeanClass(); } //Scope is normal if (isNormalScope(bean.getScope())) { instance = getEjbOrJmsProxyReference(bean, beanType,creationalContext); if(instance != null) { return instance; } instance = cacheProxies.get(bean); if (instance == null) { //Create Managed Bean Proxy instance = webBeansContext.getNormalScopeProxyFactory().createNormalScopeProxy(bean); //Cached instance cacheProxies.put(bean, instance); } } //Create Pseudo-Scope Bean Instance else { //Get bean context context = getContext(bean.getScope()); //Get instance for ejb or jms instance = getEjbOrJmsProxyReference(bean, beanType, creationalContext); if(instance != null) { return instance; } //Get dependent from DependentContex that create contextual instance instance = context.get((Bean)bean, (CreationalContext)creationalContext); } return instance; } private boolean isBeanTypeAssignableToGivenType(Set beanTypes, Type givenType, boolean newBean) { Iterator itBeanApiTypes = beanTypes.iterator(); while (itBeanApiTypes.hasNext()) { Type beanApiType = itBeanApiTypes.next(); if(GenericsUtil.satisfiesDependency(false, givenType, beanApiType)) { return true; } else { //Check for @New if(newBean && ClassUtil.isParametrizedType(givenType)) { Class requiredType = ClassUtil.getClass(givenType); if(ClassUtil.isClassAssignable(requiredType, ClassUtil.getClass(beanApiType))) { return true; } } } } return false; } private Object getEjbOrJmsProxyReference(Bean bean,Type beanType, CreationalContext creationalContext) { //Create session bean proxy if(bean instanceof EnterpriseBeanMarker) { if(isNormalScope(bean.getScope())) { //Maybe it is cached if(cacheProxies.containsKey(bean)) { return cacheProxies.get(bean); } } OpenWebBeansEjbPlugin ejbPlugin = webBeansContext.getPluginLoader().getEjbPlugin(); if(ejbPlugin == null) { throw new IllegalStateException("There is no EJB plugin provider. Injection is failed for bean : " + bean); } return ejbPlugin.getSessionBeanProxy(bean,ClassUtil.getClazz(beanType), creationalContext); } //Create JMS Proxy else if(bean instanceof JmsBeanMarker) { OpenWebBeansJmsPlugin jmsPlugin = webBeansContext.getPluginLoader().getJmsPlugin(); if(jmsPlugin == null) { throw new IllegalStateException("There is no JMS plugin provider. Injection is failed for bean : " + bean); } return jmsPlugin.getJmsBeanProxy(bean, ClassUtil.getClass(beanType)); } return null; } @Override public Set getStereotypeDefinition(Class stereotype) { Annotation[] annotations = stereotype.getDeclaredAnnotations(); Set set = new HashSet(); if(stereotype.isAnnotationPresent(Stereotype.class)) { Collections.addAll(set, annotations); } return set; } @Override public boolean isQualifier(Class annotationType) { return webBeansContext.getAnnotationManager().isQualifierAnnotation(annotationType); } @Override public boolean isInterceptorBinding(Class annotationType) { return webBeansContext.getAnnotationManager().isInterceptorBindingAnnotation(annotationType); } @Override public boolean isScope(Class annotationType) { if(AnnotationUtil.hasAnnotation(annotationType.getDeclaredAnnotations(), Scope.class) || AnnotationUtil.hasAnnotation(annotationType.getDeclaredAnnotations(), NormalScope.class)) { return true; } for(ExternalScope ext : additionalScopes) { if(ext.getScope().equals(annotationType)) { return true; } } return false; } @Override public boolean isNormalScope(Class scopeType) { Boolean isNormal = isScopeTypeNormalCache.get(scopeType); if (isNormal != null) { return isNormal; } for(ExternalScope extScope : additionalScopes) { if (extScope.getScope().equals(scopeType)) { isScopeTypeNormalCache.put(scopeType, extScope.isNormal()); return extScope.isNormal(); } } isNormal = scopeType.getAnnotation(NormalScope.class) != null; isScopeTypeNormalCache.put(scopeType, isNormal); return isNormal; } @Override public boolean isPassivatingScope(Class annotationType) { for(ExternalScope extScope : additionalScopes) { if (extScope.getScope().equals(annotationType)) { return extScope.isPassivating(); } } NormalScope scope = annotationType.getAnnotation(NormalScope.class); if(scope != null) { return scope.passivating(); } return false; } @Override public boolean isStereotype(Class annotationType) { return AnnotationUtil.hasAnnotation(annotationType.getDeclaredAnnotations(), Stereotype.class); } @Override public Bean resolve(Set> beans) { return injectionResolver.resolve(beans, null); } /** * {@inheritDoc} */ @Override public void validate(InjectionPoint injectionPoint) { Bean bean = injectionPoint.getBean(); //Check for correct injection type injectionResolver.checkInjectionPointType(injectionPoint); Class rawType = ClassUtil.getRawTypeForInjectionPoint(injectionPoint); // check for InjectionPoint injection if (rawType.equals(InjectionPoint.class)) { if (AnnotationUtil.hasAnnotation(AnnotationUtil.asArray(injectionPoint.getQualifiers()), Default.class)) { if (!bean.getScope().equals(Dependent.class)) { throw new WebBeansConfigurationException("Bean " + bean.getBeanClass() + " scope can not define other scope except @Dependent to inject InjectionPoint"); } } } else { injectionResolver.checkInjectionPoint(injectionPoint); } } /** * {@inheritDoc} */ @Override public InjectionTarget createInjectionTarget(AnnotatedType type) { InjectionTargetFactoryImpl factory = new InjectionTargetFactoryImpl(type, webBeansContext); return factory.createInjectionTarget(); } @Override public Set> resolveObserverMethods(T event, Annotation... qualifiers) { return resolveObserverMethods(event, new EventMetadataImpl(event.getClass(), null, qualifiers)); } public Set> resolveObserverMethods(T event, EventMetadata metadata) { if(ClassUtil.isDefinitionContainsTypeVariables(ClassUtil.getClass(metadata.getType()))) { throw new IllegalArgumentException("Event type can not contain type variables. Event class is : " + event.getClass()); } return notificationManager.resolveObservers(event, metadata); } @Override public ExpressionFactory wrapExpressionFactory(ExpressionFactory expressionFactory) { ELAdaptor elAdaptor = webBeansContext.getService(ELAdaptor.class); return elAdaptor.getOwbWrappedExpressionFactory(expressionFactory); } public void addAdditionalQualifier(Class qualifier) { if (!additionalQualifiers.contains(qualifier)) { additionalQualifiers.add(qualifier); } } public void addAdditionalInterceptorBindings(Class interceptorBinding) { if (!additionalInterceptorBindings.contains(interceptorBinding)) { additionalInterceptorBindings.add(interceptorBinding); } } public void addAdditionalAnnotatedType(AnnotatedType annotatedType) { addAdditionalAnnotatedType(annotatedType, AnnotatedElementFactory.OWB_DEFAULT_KEY); } public void addAdditionalAnnotatedType(AnnotatedType annotatedType, String id) { webBeansContext.getAnnotatedElementFactory().setAnnotatedType(annotatedType, id); ConcurrentMap> annotatedTypes = additionalAnnotatedTypes.get(annotatedType.getJavaClass()); if (annotatedTypes == null) { annotatedTypes = new ConcurrentHashMap>(); ConcurrentMap> oldAnnotatedTypes = additionalAnnotatedTypes.putIfAbsent(annotatedType.getJavaClass(), annotatedTypes); if (oldAnnotatedTypes != null) { annotatedTypes = oldAnnotatedTypes; } } annotatedTypes.put(id, annotatedType); } public void removeAdditionalAnnotatedType(AnnotatedType annotatedType) { removeAdditionalAnnotatedType(annotatedType, AnnotatedElementFactory.OWB_DEFAULT_KEY); } public void removeAdditionalAnnotatedType(AnnotatedType annotatedType, String id) { ConcurrentMap> annotatedTypes = additionalAnnotatedTypes.get(annotatedType.getJavaClass()); if (annotatedTypes == null) { return; } if (annotatedType.equals(annotatedTypes.get(id))) { annotatedTypes.remove(id); } } public List> getAdditionalQualifiers() { return additionalQualifiers; } public void addAdditionalScope(ExternalScope additionalScope) { if (!additionalScopes.contains(additionalScope)) { additionalScopes.add(additionalScope); } } public List getAdditionalScopes() { return additionalScopes; } public Collection> getAdditionalAnnotatedTypes() { Collection> annotatedTypes = new ArrayList>(); for (ConcurrentMap> types: additionalAnnotatedTypes.values()) { annotatedTypes.addAll(types.values()); } return annotatedTypes; } public AnnotatedType getAdditionalAnnotatedType(Class type, String id) { ConcurrentMap> annotatedTypes = additionalAnnotatedTypes.get(type); if (annotatedTypes == null) { return null; } return (AnnotatedType)annotatedTypes.get(id); } public void clear() { additionalAnnotatedTypes.clear(); additionalQualifiers.clear(); additionalScopes.clear(); clearCacheProxies(); singleContextMap.clear(); contextMap.clear(); deploymentBeans.clear(); errorStack.clear(); producersForJavaEeComponents.clear(); passivationBeans.clear(); webBeansContext.getInterceptorsManager().clear(); webBeansContext.getDecoratorsManager().clear(); } public void clearCacheProxies() { cacheProxies.clear(); } public boolean isInUse() { return inUse; } }