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

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 java.lang.annotation.Annotation;
import java.lang.reflect.Type;
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.Observes;
import javax.enterprise.event.Reception;
import javax.enterprise.event.TransactionPhase;
import javax.enterprise.inject.Disposes;
import javax.enterprise.inject.Instance;
import javax.enterprise.inject.Produces;
import javax.enterprise.inject.spi.BeanManager;
import javax.enterprise.inject.spi.EventMetadata;
import javax.enterprise.inject.spi.ObserverMethod;
import javax.enterprise.inject.spi.WithAnnotations;
import javax.enterprise.util.TypeLiteral;
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.ContextualInstance;
import org.jboss.weld.bean.RIBean;
import org.jboss.weld.context.CreationalContextImpl;
import org.jboss.weld.injection.InjectionPointFactory;
import org.jboss.weld.injection.MethodInjectionPoint;
import org.jboss.weld.injection.MethodInjectionPoint.MethodInjectionPointType;
import org.jboss.weld.injection.MethodInvocationStrategy;
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.collections.ImmutableSet;
import org.jboss.weld.util.reflection.Formats;
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 { @SuppressWarnings("serial") private static final Type EVENT_METADATA_INSTANCE_TYPE = new TypeLiteral>() { }.getType(); 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; // this turned out to be noticeable faster than observerMethod.getAnnotated().isStatic() private final boolean isStatic; private final boolean eventMetadataRequired; private final MethodInvocationStrategy notificationStrategy; /** * 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 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); ImmutableSet.Builder> injectionPoints = ImmutableSet.builder(); for (ParameterInjectionPoint injectionPoint : observerMethod.getParameterInjectionPoints()) { if (injectionPoint instanceof SpecialParameterInjectionPoint) { continue; } injectionPoints.add(injectionPoint); } this.injectionPoints = injectionPoints.build(); this.isStatic = observer.isStatic(); this.eventMetadataRequired = initMetadataRequired(this.injectionPoints); this.notificationStrategy = MethodInvocationStrategy.forObserver(observerMethod, beanManager); } private static boolean initMetadataRequired(Set> injectionPoints) { for (WeldInjectionPointAttributes ip : injectionPoints) { Type type = ip.getType(); if (EventMetadata.class.equals(type) || EVENT_METADATA_INSTANCE_TYPE.equals(type)) { return true; } } return false; } protected String createId(final EnhancedAnnotatedMethod observer, final RIBean declaringBean) { return new StringBuilder().append(ID_PREFIX).append(ID_SEPARATOR).append(ObserverMethod.class.getSimpleName()).append(ID_SEPARATOR) .append(createTypeId(declaringBean)).append(".").append(observer.getSignature()).toString(); } protected String createTypeId(RIBean declaringBean) { String typeId = null; if (declaringBean instanceof AbstractClassBean) { AbstractClassBean classBean = (AbstractClassBean) declaringBean; typeId = classBean.getAnnotated().getIdentifier().asString(); } else { typeId = declaringBean.getBeanClass().getName(); } return typeId; } protected MethodInjectionPoint initMethodInjectionPoint(EnhancedAnnotatedMethod observer, RIBean declaringBean, BeanManagerImpl manager) { return InjectionPointFactory.instance().createMethodInjectionPoint(MethodInjectionPointType.OBSERVER, observer, declaringBean, declaringBean.getBeanClass(), Observes.class, manager); } public Set> getInjectionPoints() { return injectionPoints; } /** * 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, Formats.formatAsStackTraceElement(annotated.getJavaMember())); } if (eventObjects.size() > 1) { throw EventLogger.LOG.multipleEventParameters(this, Formats.formatAsStackTraceElement(annotated.getJavaMember())); } EnhancedAnnotatedParameter 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, Formats.formatAsStackTraceElement(annotated.getJavaMember())); } // 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, Formats.formatAsStackTraceElement(annotated.getJavaMember())); } if (this.observerMethod.getAnnotated().isAnnotationPresent(Inject.class)) { throw EventLogger.LOG.invalidInitializer(this, Formats.formatAsStackTraceElement(annotated.getJavaMember())); } 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, Formats.formatAsStackTraceElement(annotated.getJavaMember())); } } } protected void checkRequiredTypeAnnotations(EnhancedAnnotatedParameter eventParameter) { if (eventParameter.isAnnotationPresent(WithAnnotations.class)) { throw EventLogger.LOG .invalidWithAnnotations(this, Formats.formatAsStackTraceElement(eventParameter.getDeclaringEnhancedCallable().getJavaMember())); } } @Override public Class getBeanClass() { return declaringBean.getType(); } public RIBean getDeclaringBean() { return declaringBean; } @Override public Reception getReception() { return reception; } @Override public Set getObservedQualifiers() { return bindings; } @Override public Type getObservedType() { return eventType; } @Override 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 (isStatic) { sendEvent(event, null, null); } else { CreationalContext creationalContext = null; try { Object receiver = getReceiverIfExists(null); if (receiver == null && reception != Reception.IF_EXISTS) { // creational context is created only if we need it for obtaining receiver // ObserverInvocationStrategy takes care of creating CC for parameters, if needed creationalContext = beanManager.createCreationalContext(declaringBean); receiver = getReceiverIfExists(creationalContext); } if (receiver != null) { sendEvent(event, receiver, creationalContext); } } finally { if (creationalContext != null) { creationalContext.release(); } } } } /** * Note that {@link CreationalContext#release()} is not invoked within this method. * * @param event * @param receiver * @param creationalContext */ protected void sendEvent(T event, Object receiver, CreationalContext creationalContext) { try { preNotify(event, receiver); // As we are working with the contextual instance, we may not have the // actual object, but a container proxy (e.g. EJB) notificationStrategy.invoke(receiver, observerMethod, event, beanManager, creationalContext); } finally { postNotify(event, receiver); } } /** * 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 creationalContext) { if (creationalContext != null) { if (creationalContext instanceof CreationalContextImpl) { // Create child creational context so that a dependent observer may be destroyed after the observer method completes creationalContext = ((CreationalContextImpl) creationalContext).getCreationalContext(declaringBean); } return ContextualInstance.get(declaringBean, beanManager, creationalContext); } // Conditional observer - no creational context is required return ContextualInstance.getIfExists(declaringBean, beanManager); } @Override public String toString() { return observerMethod.toString(); } public String getId() { return id; } @Override public boolean equals(Object obj) { if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } ObserverMethodImpl that = (ObserverMethodImpl) obj; return this.getId().equals(that.getId()); } @Override public int hashCode() { return getId().hashCode(); } public boolean isEventMetadataRequired() { return eventMetadataRequired; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy