
org.jboss.weld.event.EventImpl 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.io.ObjectInputStream;
import java.io.ObjectStreamException;
import java.io.Serializable;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import javax.enterprise.event.Event;
import javax.enterprise.inject.spi.InjectionPoint;
import javax.enterprise.util.TypeLiteral;
import org.jboss.weld.bean.builtin.AbstractFacade;
import org.jboss.weld.bean.builtin.FacadeInjectionPoint;
import org.jboss.weld.exceptions.InvalidObjectException;
import org.jboss.weld.logging.EventLogger;
import org.jboss.weld.manager.BeanManagerImpl;
import org.jboss.weld.resolution.Resolvable;
import org.jboss.weld.util.Preconditions;
import org.jboss.weld.util.Types;
import org.jboss.weld.util.reflection.EventObjectTypeResolverBuilder;
import org.jboss.weld.util.reflection.Formats;
import org.jboss.weld.util.reflection.HierarchyDiscovery;
import org.jboss.weld.util.reflection.TypeResolver;
/**
* Implementation of the Event interface
*
* @param The type of event being wrapped
* @author David Allen
* @see javax.enterprise.event.Event
*/
@edu.umd.cs.findbugs.annotations.SuppressWarnings(value = "SE_NO_SUITABLE_CONSTRUCTOR", justification = "Uses SerializationProxy")
public class EventImpl extends AbstractFacade> implements Event, Serializable {
private static final String SUBTYPE_ARGUMENT_NAME = "subtype";
private static final long serialVersionUID = 656782657242515455L;
public static EventImpl of(InjectionPoint injectionPoint, BeanManagerImpl beanManager) {
return new EventImpl(injectionPoint, beanManager);
}
private final transient HierarchyDiscovery injectionPointTypeHierarchy;
private transient volatile CachedResolvable cachedResolvable;
private EventImpl(InjectionPoint injectionPoint, BeanManagerImpl beanManager) {
super(injectionPoint, null, beanManager);
this.injectionPointTypeHierarchy = new HierarchyDiscovery(getType());
}
/**
* Gets a string representation
*
* @return A string representation
*/
@Override
public String toString() {
return Formats.formatAnnotations(getQualifiers()) + " Event<" + Formats.formatType(getType()) + ">";
}
@Override
public void fire(T event) {
Preconditions.checkArgumentNotNull(event, "event");
CachedResolvable resolvable = getEventResolvable(event);
EventPacket packet = EventPacket.of(event, resolvable.type, resolvable.resolvable, getQualifiers(), getInjectionPoint());
getBeanManager().getGlobalStrictObserverNotifier().fireEvent(packet);
}
private CachedResolvable getEventResolvable(T event) {
CachedResolvable cachedResolvable = this.cachedResolvable;
if (cachedResolvable != null) {
if (cachedResolvable.rawType.equals(event.getClass())) {
return cachedResolvable;
}
}
Type eventType = getEventType(event);
Resolvable resolvable = getBeanManager().getGlobalStrictObserverNotifier().buildEventResolvable(eventType, getQualifiers());
cachedResolvable = new CachedResolvable(event.getClass(), eventType, resolvable);
this.cachedResolvable = cachedResolvable;
return cachedResolvable;
}
@Override
public Event select(Annotation... qualifiers) {
return selectEvent(this.getType(), qualifiers);
}
@Override
public Event select(Class subtype, Annotation... qualifiers) {
Preconditions.checkArgumentNotNull(subtype, SUBTYPE_ARGUMENT_NAME);
return selectEvent(subtype, qualifiers);
}
@Override
public Event select(TypeLiteral subtype, Annotation... qualifiers) {
Preconditions.checkArgumentNotNull(subtype, SUBTYPE_ARGUMENT_NAME);
return selectEvent(subtype.getType(), qualifiers);
}
private Event selectEvent(Type subtype, Annotation[] newQualifiers) {
getBeanManager().getGlobalStrictObserverNotifier().checkEventObjectType(subtype);
return new EventImpl(new FacadeInjectionPoint(getBeanManager(), getInjectionPoint(), subtype, getQualifiers(), newQualifiers),
getBeanManager());
}
protected Type getEventType(T event) {
Type resolvedType = event.getClass();
if (Types.containsUnresolvedTypeVariableOrWildcard(resolvedType)) {
/*
* If the container is unable to resolve the parameterized type of the event object, it uses the specified type to infer the parameterized type of the event types.
*/
resolvedType = injectionPointTypeHierarchy.resolveType(resolvedType);
}
if (Types.containsUnresolvedTypeVariableOrWildcard(resolvedType)) {
/*
* Examining the hierarchy of the specified type did not help. This may still be one of the cases when combining the
* event type and the specified type reveals the actual values for type variables. Let's try that.
*/
TypeResolver objectTypeResolver = new EventObjectTypeResolverBuilder(injectionPointTypeHierarchy.getResolver()
.getResolvedTypeVariables(), new HierarchyDiscovery(event.getClass()).getResolver()
.getResolvedTypeVariables()).build();
resolvedType = objectTypeResolver.resolveType(Types.getCanonicalType(event.getClass()));
}
return resolvedType;
}
// Serialization
private Object writeReplace() throws ObjectStreamException {
return new SerializationProxy(this);
}
private void readObject(ObjectInputStream stream) throws InvalidObjectException {
throw EventLogger.LOG.proxyRequired();
}
private static class SerializationProxy extends AbstractFacadeSerializationProxy> {
private static final long serialVersionUID = 9181171328831559650L;
public SerializationProxy(EventImpl event) {
super(event);
}
private Object readResolve() throws ObjectStreamException {
return EventImpl.of(getInjectionPoint(), getBeanManager());
}
}
private static class CachedResolvable {
private final Class> rawType;
private final Type type;
private final Resolvable resolvable;
public CachedResolvable(Class> rawType, Type type, Resolvable resolvable) {
this.rawType = rawType;
this.type = type;
this.resolvable = resolvable;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy