org.jboss.weld.context.unbound.DependentContextImpl Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright 2009 Sun Microsystems, Inc. All rights reserved.
*
* Use is subject to license terms.
*
* 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.context.unbound;
import java.lang.annotation.Annotation;
import javax.enterprise.context.ContextNotActiveException;
import javax.enterprise.context.Dependent;
import javax.enterprise.context.spi.Contextual;
import javax.enterprise.context.spi.CreationalContext;
import javax.enterprise.inject.spi.Decorator;
import javax.enterprise.inject.spi.Interceptor;
import org.jboss.weld.bean.AbstractProducerBean;
import org.jboss.weld.bean.ManagedBean;
import org.jboss.weld.bean.builtin.AbstractBuiltInBean;
import org.jboss.weld.context.DependentContext;
import org.jboss.weld.context.SerializableContextualInstanceImpl;
import org.jboss.weld.context.WeldCreationalContext;
import org.jboss.weld.context.api.ContextualInstance;
import org.jboss.weld.exceptions.UnsupportedOperationException;
import org.jboss.weld.injection.producer.AbstractMemberProducer;
import org.jboss.weld.injection.producer.BasicInjectionTarget;
import org.jboss.weld.serialization.spi.ContextualStore;
/**
* The dependent context
*
* @author Nicklas Karlsson
*/
public class DependentContextImpl implements DependentContext {
private final ContextualStore contextualStore;
public DependentContextImpl(ContextualStore contextualStore) {
this.contextualStore = contextualStore;
}
/**
* Overridden method always creating a new instance
*
* @param contextual The bean to create
* @param creationalContext The creation context
*/
public T get(Contextual contextual, CreationalContext creationalContext) {
if (!isActive()) {
throw new ContextNotActiveException();
}
if (creationalContext != null) {
T instance = contextual.create(creationalContext);
if (creationalContext instanceof WeldCreationalContext>) {
addDependentInstance(instance, contextual, (WeldCreationalContext) creationalContext);
}
return instance;
} else {
return null;
}
}
protected void addDependentInstance(T instance, Contextual contextual, WeldCreationalContext creationalContext) {
// by this we are making sure that the dependent instance has no transitive dependency with @PreDestroy / disposal method
if (creationalContext.getDependentInstances().isEmpty()) {
if (contextual instanceof ManagedBean> && !isInterceptorOrDecorator(contextual)) {
ManagedBean> managedBean = (ManagedBean>) contextual;
if (managedBean.getProducer() instanceof BasicInjectionTarget>) {
BasicInjectionTarget> injectionTarget = (BasicInjectionTarget>) managedBean.getProducer();
if (!injectionTarget.getLifecycleCallbackInvoker().hasPreDestroyMethods() && !injectionTarget.hasInterceptors()) {
// there is no @PreDestroy callback to call when destroying this dependent instance
// therefore, we do not need to keep the reference
return;
}
}
}
if (contextual instanceof AbstractProducerBean, ?, ?>) {
AbstractProducerBean, ?, ?> producerBean = (AbstractProducerBean, ?, ?>) contextual;
if (producerBean.getProducer() instanceof AbstractMemberProducer, ?>) {
AbstractMemberProducer, ?> producer = (AbstractMemberProducer, ?>) producerBean.getProducer();
if (producer.getDisposalMethod() == null) {
// there is no disposal method to call when destroying this dependent instance
// therefore, we do not need to keep the reference
return;
}
}
}
if (isOptimizableBuiltInBean(contextual)) {
// Most built-in dependent beans do not have to be stored
return;
}
}
// Only add the dependent instance if none of the conditions above is met
ContextualInstance beanInstance = new SerializableContextualInstanceImpl, T>(contextual, instance, creationalContext, contextualStore);
creationalContext.addDependentInstance(beanInstance);
}
private boolean isInterceptorOrDecorator(Contextual> contextual) {
return contextual instanceof Interceptor> || contextual instanceof Decorator>;
}
public T get(Contextual contextual) {
return get(contextual, null);
}
public boolean isActive() {
return true;
}
public Class extends Annotation> getScope() {
return Dependent.class;
}
@Override
public void destroy(Contextual> contextual) {
throw new UnsupportedOperationException();
}
private boolean isOptimizableBuiltInBean(Contextual> contextual) {
if (contextual instanceof AbstractBuiltInBean>) {
AbstractBuiltInBean> abstractBuiltInBean = (AbstractBuiltInBean>) contextual;
return abstractBuiltInBean.isDependentContextOptimizationAllowed();
}
return false;
}
}