/* * JBoss, Home of Professional Open Source * Copyright 2008, Red Hat, Inc., and individual contributors * by the @authors tag. See the copyright.txt in the distribution for a * full listing of individual contributors. * * Licensed 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.jboss.weld.bean; import static org.jboss.weld.bean.BeanIdentifiers.forManagedBean; import java.util.Set; import javax.enterprise.context.ApplicationScoped; import javax.enterprise.context.Dependent; import javax.enterprise.context.RequestScoped; import javax.enterprise.context.spi.CreationalContext; import javax.enterprise.inject.spi.Bean; import javax.enterprise.inject.spi.BeanAttributes; import javax.enterprise.inject.spi.Decorator; import javax.enterprise.inject.spi.InjectionTarget; import org.jboss.weld.annotated.enhanced.EnhancedAnnotatedField; import org.jboss.weld.annotated.enhanced.EnhancedAnnotatedType; import org.jboss.weld.bootstrap.BeanDeployerEnvironment; import org.jboss.weld.context.RequestContext; import org.jboss.weld.context.unbound.UnboundLiteral; import org.jboss.weld.contexts.CreationalContextImpl; import org.jboss.weld.injection.producer.BasicInjectionTarget; import org.jboss.weld.interceptor.spi.metadata.InterceptorClassMetadata; import org.jboss.weld.interceptor.spi.model.InterceptionModel; import org.jboss.weld.interceptor.spi.model.InterceptionType; import org.jboss.weld.logging.BeanLogger; import org.jboss.weld.manager.BeanManagerImpl; import org.jboss.weld.serialization.spi.BeanIdentifier; import org.jboss.weld.util.Decorators; import org.jboss.weld.util.Proxies; import org.jboss.weld.util.reflection.Formats; import org.jboss.weld.util.reflection.Reflections; /** * Represents a simple bean * * @param <T> The type (class) of the bean * @author Pete Muir * @author Marius Bogoevici * @author Ales Justin * @author Marko Luksa */ public class ManagedBean<T> extends AbstractClassBean<T> { private final boolean proxiable; private boolean passivationCapableBean; private boolean passivationCapableDependency; /* * tracks whether this bean has a @PostConstruct callbacks * if it does not, we can skip activating/deactivating @RequestScoped context during creation */ private boolean hasPostConstructCallback; /** * Creates a simple, annotation defined Web Bean * * @param <T> The type * @param clazz The class * @param beanManager the current manager * @return A Web Bean */ public static <T> ManagedBean<T> of(BeanAttributes<T> attributes, EnhancedAnnotatedType<T> clazz, BeanManagerImpl beanManager) { return new ManagedBean<T>(attributes, clazz, createId(attributes, clazz), beanManager); } private static BeanIdentifier createId(BeanAttributes<?> attributes, EnhancedAnnotatedType<?> clazz) { if (Dependent.class.equals(attributes.getScope()) || ApplicationScoped.class.equals(attributes.getScope())) { return new ManagedBeanIdentifier(clazz.slim().getIdentifier()); } else { return new StringBeanIdentifier(forManagedBean(clazz)); } } /** * Constructor * * @param type The type of the bean * @param beanManager The Bean manager */ protected ManagedBean(BeanAttributes<T> attributes, EnhancedAnnotatedType<T> type, BeanIdentifier identifier, BeanManagerImpl beanManager) { super(attributes, type, identifier, beanManager); this.proxiable = Proxies.isTypesProxyable(getTypes(), beanManager.getServices()); setProducer(beanManager.getLocalInjectionTargetFactory(getEnhancedAnnotated()).createInjectionTarget(getEnhancedAnnotated(), this, false)); } @Override public void internalInitialize(BeanDeployerEnvironment environment) { super.internalInitialize(environment); initPassivationCapable(); } private void initPassivationCapable() { this.passivationCapableBean = getEnhancedAnnotated().isSerializable(); this.passivationCapableDependency = isNormalScoped() || (isDependent() && passivationCapableBean); } @Override public void initializeAfterBeanDiscovery() { if (this.passivationCapableBean && hasDecorators() && !allDecoratorsArePassivationCapable()) { this.passivationCapableBean = false; } if (this.passivationCapableBean && hasInterceptors() && !allInterceptorsArePassivationCapable()) { this.passivationCapableBean = false; } super.initializeAfterBeanDiscovery(); } private boolean allDecoratorsArePassivationCapable() { return getFirstNonPassivationCapableDecorator() == null; } private Decorator<?> getFirstNonPassivationCapableDecorator() { for (Decorator<?> decorator : getDecorators()) { if (!Decorators.isPassivationCapable(decorator)) { return decorator; } } return null; } private boolean allInterceptorsArePassivationCapable() { return getFirstNonPassivationCapableInterceptor() == null; } private InterceptorClassMetadata<?> getFirstNonPassivationCapableInterceptor() { for (InterceptorClassMetadata<?> interceptorMetadata : getBeanManager().getInterceptorModelRegistry().get(getAnnotated()).getAllInterceptors()) { if (!Reflections.isSerializable(interceptorMetadata.getJavaClass())) { return interceptorMetadata; } } return null; } /** * Creates an instance of the bean * * @return The instance */ @Override public T create(CreationalContext<T> creationalContext) { T instance = getProducer().produce(creationalContext); getProducer().inject(instance, creationalContext); if (!hasPostConstructCallback || beanManager.isContextActive(RequestScoped.class)) { getProducer().postConstruct(instance); } else { /* * CDI-219 * The request scope is active during @PostConstruct callback of any bean. */ RequestContext context = getUnboundRequestContext(); try { context.activate(); beanManager.fireRequestContextInitialized(getId()); getProducer().postConstruct(instance); } finally { beanManager.fireRequestContextBeforeDestroyed(getId()); context.invalidate(); context.deactivate(); beanManager.fireRequestContextDestroyed(getId()); } } return instance; } /** * Destroys an instance of the bean * * @param instance The instance */ @Override public void destroy(T instance, CreationalContext<T> creationalContext) { super.destroy(instance, creationalContext); try { getProducer().preDestroy(instance); // WELD-1010 hack? if (creationalContext instanceof CreationalContextImpl) { ((CreationalContextImpl<T>) creationalContext).release(this, instance); } else { creationalContext.release(); } } catch (Exception e) { BeanLogger.LOG.errorDestroying(instance, this); BeanLogger.LOG.catchingDebug(e); } } /** * Validates the type */ @Override protected void checkType() { if (!isDependent() && getEnhancedAnnotated().isParameterizedType()) { throw BeanLogger.LOG.managedBeanWithParameterizedBeanClassMustBeDependent(type); } boolean passivating = beanManager.isPassivatingScope(getScope()); if (passivating && !isPassivationCapableBean()) { if (!getEnhancedAnnotated().isSerializable()) { throw BeanLogger.LOG.passivatingBeanNeedsSerializableImpl(this); } else if (hasDecorators() && !allDecoratorsArePassivationCapable()) { throw BeanLogger.LOG.passivatingBeanHasNonPassivationCapableDecorator(this, getFirstNonPassivationCapableDecorator()); } else if (hasInterceptors() && !allInterceptorsArePassivationCapable()) { throw BeanLogger.LOG.passivatingBeanHasNonPassivationCapableInterceptor(this, getFirstNonPassivationCapableInterceptor()); } } } @Override protected void checkBeanImplementation() { super.checkBeanImplementation(); if (isNormalScoped()) { for (EnhancedAnnotatedField<?, ?> field : getEnhancedAnnotated().getEnhancedFields()) { if (field.isPublic() && !field.isStatic()) { throw BeanLogger.LOG.publicFieldOnNormalScopedBeanNotAllowed(field); } } } } @Override protected void specialize() { Set<? extends AbstractBean<?, ?>> specializedBeans = getSpecializedBeans(); if (specializedBeans.isEmpty()) { throw BeanLogger.LOG.specializingBeanMustExtendABean(this); } for (AbstractBean<?, ?> specializedBean : specializedBeans) { if (!(specializedBean instanceof ManagedBean<?>)) { throw BeanLogger.LOG.specializingManagedBeanCanExtendOnlyManagedBeans(this, specializedBean); } } } @Override protected boolean isInterceptionCandidate() { return !((this instanceof InterceptorImpl<?>) || (this instanceof DecoratorImpl<?>)); } @Override public String toString() { return "Managed Bean [" + getBeanClass().toString() + "] with qualifiers [" + Formats.formatAnnotations(getQualifiers()) + "]"; } @Override public boolean isProxyable() { return proxiable; } @Override public boolean isPassivationCapableBean() { return passivationCapableBean; } @Override public boolean isPassivationCapableDependency() { return passivationCapableDependency; } private RequestContext getUnboundRequestContext() { final Bean<?> bean = beanManager.resolve(beanManager.getBeans(RequestContext.class, UnboundLiteral.INSTANCE)); final CreationalContext<?> ctx = beanManager.createCreationalContext(bean); return (RequestContext) beanManager.getReference(bean, RequestContext.class, ctx); } @Override public void setProducer(InjectionTarget<T> producer) { super.setProducer(producer); this.hasPostConstructCallback = initHasPostConstructCallback(producer); } private boolean initHasPostConstructCallback(InjectionTarget<T> producer) { if (producer instanceof BasicInjectionTarget<?>) { BasicInjectionTarget<?> weldProducer = (BasicInjectionTarget<?>) producer; final InterceptionModel interceptors = getInterceptors(); if (interceptors == null || interceptors.getInterceptors(InterceptionType.POST_CONSTRUCT, null).isEmpty()) { if (!weldProducer.getLifecycleCallbackInvoker().hasPostConstructCallback()) { return false; } } } // otherwise we assume there is a post construct callback, just to be safe return true; } }