
org.jboss.weld.event.ObserverMethodImpl Maven / Gradle / Ivy
/*
* 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.event;
import static org.jboss.weld.util.collections.WeldCollections.immutableSet;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import javax.enterprise.context.ContextNotActiveException;
import javax.enterprise.context.Dependent;
import javax.enterprise.context.spi.CreationalContext;
import javax.enterprise.event.ObserverException;
import javax.enterprise.event.Observes;
import javax.enterprise.event.Reception;
import javax.enterprise.event.TransactionPhase;
import javax.enterprise.inject.Disposes;
import javax.enterprise.inject.New;
import javax.enterprise.inject.Produces;
import javax.enterprise.inject.spi.BeanManager;
import javax.enterprise.inject.spi.ObserverMethod;
import javax.enterprise.inject.spi.WithAnnotations;
import javax.inject.Inject;
import javax.inject.Qualifier;
import org.jboss.weld.annotated.enhanced.EnhancedAnnotatedMethod;
import org.jboss.weld.annotated.enhanced.EnhancedAnnotatedParameter;
import org.jboss.weld.bean.AbstractClassBean;
import org.jboss.weld.bean.RIBean;
import org.jboss.weld.injection.InjectionPointFactory;
import org.jboss.weld.injection.MethodInjectionPoint;
import org.jboss.weld.injection.ParameterInjectionPoint;
import org.jboss.weld.injection.attributes.SpecialParameterInjectionPoint;
import org.jboss.weld.injection.attributes.WeldInjectionPointAttributes;
import org.jboss.weld.logging.EventLogger;
import org.jboss.weld.manager.BeanManagerImpl;
import org.jboss.weld.resources.SharedObjectCache;
import org.jboss.weld.util.Observers;
import org.jboss.weld.util.reflection.HierarchyDiscovery;
/**
*
* Reference implementation for the ObserverMethod interface, which represents
* an observer method. Each observer method has an event type which is the class
* of the event object being observed, and event binding types that are
* annotations applied to the event parameter to narrow the event notifications
* delivered.
*
*
* @author David Allen
* @author Jozef Hartinger
* @author Marko Luksa
*/
public class ObserverMethodImpl implements ObserverMethod {
public static final String ID_PREFIX = ObserverMethodImpl.class.getPackage().getName();
public static final String ID_SEPARATOR = "-";
private final Set bindings;
private final Type eventType;
protected final BeanManagerImpl beanManager;
private final Reception reception;
protected final RIBean declaringBean;
protected final MethodInjectionPoint observerMethod;
protected TransactionPhase transactionPhase;
private final String id;
private final Set> injectionPoints;
private final Set> newInjectionPoints;
/**
* Creates an Observer which describes and encapsulates an observer method
* (8.5).
*
* @param observer The observer
* @param declaringBean The observer bean
* @param manager The Bean manager
*/
protected ObserverMethodImpl(final EnhancedAnnotatedMethod observer, final RIBean declaringBean, final BeanManagerImpl manager) {
this.beanManager = manager;
this.declaringBean = declaringBean;
this.observerMethod = initMethodInjectionPoint(observer, declaringBean, manager);
EnhancedAnnotatedParameter, ? super X> eventParameter = observer.getEnhancedParameters(Observes.class).get(0);
this.eventType = new HierarchyDiscovery(declaringBean.getBeanClass()).resolveType(eventParameter.getBaseType());
this.id = createId(observer, declaringBean);
this.bindings = manager.getServices().get(SharedObjectCache.class).getSharedSet(observer.getEnhancedParameters(Observes.class).get(0).getMetaAnnotations(Qualifier.class));
Observes observesAnnotation = observer.getEnhancedParameters(Observes.class).get(0).getAnnotation(Observes.class);
this.reception = observesAnnotation.notifyObserver();
transactionPhase = ObserverFactory.getTransactionalPhase(observer);
Set> injectionPoints = new HashSet>();
Set> newInjectionPoints = new HashSet>();
for (ParameterInjectionPoint, ?> injectionPoint : observerMethod.getParameterInjectionPoints()) {
if (injectionPoint instanceof SpecialParameterInjectionPoint) {
continue;
}
if (injectionPoint.getQualifier(New.class) != null) {
newInjectionPoints.add(injectionPoint);
}
injectionPoints.add(injectionPoint);
}
this.injectionPoints = immutableSet(injectionPoints);
this.newInjectionPoints = immutableSet(newInjectionPoints);
}
protected static String createId(final EnhancedAnnotatedMethod, ? > observer, final RIBean> declaringBean) {
String typeId = null;
if (declaringBean instanceof AbstractClassBean>) {
AbstractClassBean> classBean = (AbstractClassBean>) declaringBean;
typeId = classBean.getAnnotated().getIdentifier().asString();
} else {
typeId = declaringBean.getBeanClass().getName();
}
return new StringBuilder().append(ID_PREFIX).append(ID_SEPARATOR).append(ObserverMethod.class.getSimpleName()).append(ID_SEPARATOR).append(typeId).append(".").append(observer.getSignature()).toString();
}
protected MethodInjectionPoint initMethodInjectionPoint(EnhancedAnnotatedMethod observer, RIBean declaringBean, BeanManagerImpl manager) {
return InjectionPointFactory.instance().createMethodInjectionPoint(observer, declaringBean, declaringBean.getBeanClass(), true, manager);
}
public Set> getInjectionPoints() {
return Collections.unmodifiableSet(injectionPoints);
}
public Set> getNewInjectionPoints() {
return Collections.unmodifiableSet(newInjectionPoints);
}
/**
* Performs validation of the observer method for compliance with the
* specifications.
*/
private void checkObserverMethod(EnhancedAnnotatedMethod annotated) {
// Make sure exactly one and only one parameter is annotated with Observes
List> eventObjects = annotated.getEnhancedParameters(Observes.class);
if (this.reception.equals(Reception.IF_EXISTS) && declaringBean.getScope().equals(Dependent.class)) {
throw EventLogger.LOG.invalidScopedConditionalObserver(this);
}
if (eventObjects.size() > 1) {
throw EventLogger.LOG.multipleEventParameters(this);
}
EnhancedAnnotatedParameter, Y> eventParameter = eventObjects.iterator().next();
checkRequiredTypeAnnotations(eventParameter);
// Check for parameters annotated with @Disposes
List> disposeParams = annotated.getEnhancedParameters(Disposes.class);
if (disposeParams.size() > 0) {
throw EventLogger.LOG.invalidDisposesParameter(this);
}
// Check annotations on the method to make sure this is not a producer
// method, initializer method, or destructor method.
if (this.observerMethod.getAnnotated().isAnnotationPresent(Produces.class)) {
throw EventLogger.LOG.invalidProducer(this);
}
if (this.observerMethod.getAnnotated().isAnnotationPresent(Inject.class)) {
throw EventLogger.LOG.invalidInitializer(this);
}
boolean containerLifecycleObserverMethod = Observers.isContainerLifecycleObserverMethod(this);
for (EnhancedAnnotatedParameter, ?> parameter : annotated.getEnhancedParameters()) {
// if this is an observer method for container lifecycle event, it must not inject anything besides BeanManager
if (containerLifecycleObserverMethod && !parameter.isAnnotationPresent(Observes.class) && !BeanManager.class.equals(parameter.getBaseType())) {
throw EventLogger.LOG.invalidInjectionPoint(this);
}
}
}
protected void checkRequiredTypeAnnotations(EnhancedAnnotatedParameter, ?> eventParameter) {
if (eventParameter.isAnnotationPresent(WithAnnotations.class)) {
throw EventLogger.LOG.invalidWithAnnotations(this);
}
}
public Class getBeanClass() {
return declaringBean.getType();
}
public RIBean getDeclaringBean() {
return declaringBean;
}
public Reception getReception() {
return reception;
}
public Set getObservedQualifiers() {
return bindings;
}
public Type getObservedType() {
return eventType;
}
public TransactionPhase getTransactionPhase() {
return transactionPhase;
}
/**
* @return the observerMethod
*/
public MethodInjectionPoint getMethod() {
return observerMethod;
}
/**
* Completes initialization of the observer and allows derived types to
* override behavior.
*/
public void initialize(EnhancedAnnotatedMethod annotated) {
checkObserverMethod(annotated);
}
@Override
public void notify(final T event) {
sendEvent(event);
}
/**
* Invokes the observer method immediately passing the event.
*
* @param event The event to notify observer with
*/
protected void sendEvent(final T event) {
if (observerMethod.getAnnotated().isStatic()) {
sendEvent(event, null, beanManager.createCreationalContext(declaringBean));
} else {
CreationalContext> creationalContext;
if (reception.equals(Reception.IF_EXISTS)) {
creationalContext = null;
} else {
creationalContext = beanManager.createCreationalContext(declaringBean);
}
Object receiver = getReceiverIfExists(creationalContext);
if (receiver != null) {
sendEvent(event, receiver, creationalContext);
}
}
}
protected void sendEvent(T event, Object receiver, CreationalContext> creationalContext) {
try {
preNotify(event, receiver);
if (receiver == null) {
observerMethod.invokeWithSpecialValue(null, Observes.class, event, beanManager, creationalContext, ObserverException.class);
} else {
// As we are working with the contextual instance, we may not have the
// actual object, but a container proxy (e.g. EJB)
observerMethod.invokeOnInstanceWithSpecialValue(receiver, Observes.class, event, beanManager, creationalContext, ObserverException.class);
}
} finally {
postNotify(event, receiver);
if (creationalContext != null) {
creationalContext.release();
}
}
}
/**
* Hooks allowing subclasses to perform additional logic just before and just after an event is delivered to an observer method.
*/
protected void preNotify(T event, Object receiver) {
}
protected void postNotify(T event, Object receiver) {
}
private Object getReceiverIfExists(CreationalContext> creationalContext) {
try {
return getReceiver(creationalContext);
} catch (ContextNotActiveException e) {
return null;
}
}
protected Object getReceiver(CreationalContext> ctx) {
return beanManager.getReference(declaringBean, null, ctx, true);
}
@Override
public String toString() {
return observerMethod.toString();
}
public String getId() {
return id;
}
@Override
public boolean equals(Object obj) {
if (obj instanceof ObserverMethodImpl, ?>) {
ObserverMethodImpl, ?> that = (ObserverMethodImpl, ?>) obj;
return this.getId().equals(that.getId());
} else {
return false;
}
}
@Override
public int hashCode() {
return getId().hashCode();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy