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

com.github.themrmilchmann.mjl.events.EventBus Maven / Gradle / Ivy

/*
 * Copyright 2018 Leon Linhart
 *
 * 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 com.github.themrmilchmann.mjl.events;

import com.github.themrmilchmann.mjl.events.util.concurrent.MJLExecutors;

import java.lang.annotation.Annotation;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.CopyOnWriteArraySet;
import java.util.concurrent.Executor;
import java.util.stream.Collectors;

/**
 * An EventBus for publish-subscribe-style communication between components.
 *
 * 

Posting events

* *

To post an event, simply pass the event object to the {@link #post(Event)} method. The bus will dispatch the event * to all registered subscribers whose accepted type {@link Class#isAssignableFrom(Class) is assignable from} the * event's type.

* * *

Subscribers

* *

To receive events, a method, known as the subscriber, which accepts a single argument of the type of the * desired event must be marked with the bus' subscriber marker annotation and its enclosing class must be registered to * the bus.

* * *

Dead Events

* *

If a posted event cannot be received by any subscriber, it is considered "dead". Dead events are wrapped in an * instance of {@link DeadEvent} and posted again. Subscribing to these events can be useful for debugging purposes.

** *

A {@code DeadEvent} is dispatched only to subscribers who explicitly expect it. (A subscriber method which expects * the common supertype {@code Event} will not receive dead events.

* * *

Thread-safety

* *

All methods in this class are thread-safe by default. (Depending on the {@link EventDispatcher dispatcher} and * {@link Executor executor} implementations in use, the {@linkplain #post(Event) post} method might be unsafe. In this * case, it should be specified in the respective implementation's documentation explicitly.)

* * *

Memory management

* *

By default, it is not guaranteed that a EventBus removes all references to unused objects when * {@link #unregister(Object)} is called. Those references may be manually removed by calling {@link #clear()}.

* *

Alternatively it is possible to overwrite this behavior by creating a self-cleaning bus (via * {@link Builder#setSelfCleaning(boolean)}). A bus created with this option will always remove all unused references as * soon as possible at the cost of computation speed for registering/unregistering operations.

* * @since 1.0.0 * * @author Leon Linhart */ public final class EventBus { /** * Returns a builder for an {@code EventBus}. * * @return a builder instance * * @since 1.2.0 */ public static Builder builder() { return new Builder(); } private final ConcurrentMap, Set> subscribers = new ConcurrentHashMap<>(); private final EventDispatcher dispatcher; private final DispatchErrorHandler dispatchErrorHandler; private final Executor executor; private final Class subscriberMarker; private final boolean isSelfCleaning; private EventBus(EventDispatcher dispatcher, DispatchErrorHandler dispatchErrorHandler, Executor executor, Class subscriberMarker, boolean isSelfCleaning) { this.dispatcher = dispatcher; this.dispatchErrorHandler = dispatchErrorHandler; this.executor = executor; this.subscriberMarker = subscriberMarker; this.isSelfCleaning = isSelfCleaning; } /** * The annotation type that is used to identify methods that are potential subscribers to events on this bus. * * @return the annotation type that is used to identify methods that are potential subscribers to events on this * bus * * @see #register(Class, MethodHandles.Lookup) * @see #register(Object, MethodHandles.Lookup) * @see Builder#setSubscriberMarker(Class) * * @since 1.0.0 */ @SuppressWarnings({"unused", "WeakerAccess"}) public Class getSubscriberMarker() { return this.subscriberMarker; } /** * Registers all static methods of the given class that are visible to the given lookup and * {@link #getSubscriberMarker() marked} with this bus' marker. * *

Searches the given object's class for any non-synthetic instance method that is marked with this bus' * subscriber marker and performs several vanity checks on each of them and throws an * {@link IllegalArgumentException} if any of them:

*
    *
  • expects any number of parameters other than one, or {@link Event} is not assignable from * that parameter's type,
  • *
  • returns any other type than {@code void}, or
  • *
  • if the method is not visible to the given lookup.
  • *
* *

If all checks succeed, a {@link Subscriber} object for each subscriber method is registered to this bus.

* * @param cls the class, which's statically accessible methods should be subscribed to this bus (the passed * object is considered to be the subscribers origin) * @param lookup the lookup that is used to create subscriber objects * * @throws NullPointerException if the given class or the given lookup is {@code null} * @throws IllegalArgumentException if a method of the given class is statically accessible and marked as subscriber * but is not visible to the given lookup or expects unsupported parameters * * @see #register(Object, MethodHandles.Lookup) * @see #unregister(Object) * * @since 1.0.0 */ @SuppressWarnings({"unused", "unchecked"}) public void register(Class cls, MethodHandles.Lookup lookup) { Objects.requireNonNull(cls); Objects.requireNonNull(lookup); Map, Collection> eventSubscribers = new HashMap<>(); Method[] methods = cls.getDeclaredMethods(); for (Method method : methods) { int mods = method.getModifiers(); if (!method.isSynthetic() && method.isAnnotationPresent(subscriberMarker) && Modifier.isStatic(mods)) { if (method.getReturnType() != void.class) throw new IllegalArgumentException("Subscriber method must have void return type"); if (method.getParameterCount() != 1 || !Event.class.isAssignableFrom(method.getParameterTypes()[0])) throw new IllegalArgumentException("Subscriber method must only have one parameter of a type that Event can be assigned to"); try { MethodHandle methodHandle = lookup.unreflect(method); Class eventType = (Class) methodHandle.type().parameterList().get(0); Subscriber subscriber = new Subscriber(this, cls, method, methodHandle); eventSubscribers.computeIfAbsent(eventType, k -> new ArrayList()).add(subscriber); } catch (IllegalAccessException e) { throw new IllegalArgumentException("Unexpected exception during subscriber creation", e); } } } eventSubscribers.forEach((eventType, subscribers) -> this.subscribers.computeIfAbsent(eventType, k -> new CopyOnWriteArraySet()).addAll(subscribers)); } /** * Registers all instance methods of the given object's class that are visible to the given lookup and * {@link #getSubscriberMarker() marked} with this bus' marker. * *

Searches the given object's class for any non-synthetic instance method that is marked with this bus' * subscriber marker and performs several vanity checks on each of them and throws an * {@link IllegalArgumentException} if any of them:

*
    *
  • expects any number of parameters other than one, or {@link Event} is not assignable from * that parameter's type,
  • *
  • returns any other type than {@code void}, or
  • *
  • if the method is not visible to the given lookup.
  • *
* *

If all checks succeed, a {@link Subscriber} object for each subscriber method is registered to this bus.

* * @param object the object, which's instance accessible methods should be subscribed to this bus (the passed * object is considered to be the subscribers origin) * @param lookup the lookup that is used to create subscriber objects * * @throws NullPointerException if the given object or the given lookup is {@code null} * @throws IllegalArgumentException if a method of the given class is instance accessible and marked as subscriber * but is not visible to the given lookup or expects unsupported parameters * * @see #register(Class, MethodHandles.Lookup) * @see #unregister(Object) * * @since 1.0.0 */ @SuppressWarnings({"unused", "unchecked"}) public void register(Object object, MethodHandles.Lookup lookup) { Objects.requireNonNull(object); Objects.requireNonNull(lookup); Map, Collection> eventSubscribers = new HashMap<>(); Method[] methods = object.getClass().getDeclaredMethods(); for (Method method : methods) { int mods = method.getModifiers(); if (!method.isSynthetic() && method.isAnnotationPresent(subscriberMarker) && !Modifier.isStatic(mods)) { if (method.getReturnType() != void.class) throw new IllegalArgumentException("Subscriber method must have void return type"); if (method.getParameterCount() != 1 || !Event.class.isAssignableFrom(method.getParameterTypes()[0])) throw new IllegalArgumentException("Subscriber method must only have one parameter of a type that Event can be assigned to"); try { MethodHandle methodHandle = lookup.unreflect(method).bindTo(object); Class eventType = (Class) method.getParameterTypes()[0]; Subscriber subscriber = new Subscriber(this, object, method, methodHandle); eventSubscribers.computeIfAbsent(eventType, k -> new ArrayList()).add(subscriber); } catch (IllegalAccessException e) { throw new IllegalArgumentException("Unexpected exception during subscriber creation", e); } } } eventSubscribers.forEach((eventType, subscribers) -> this.subscribers.computeIfAbsent(eventType, k -> new CopyOnWriteArraySet()).addAll(subscribers)); } /** * Unregisters each subscriber, whose origin is equal to the given object, from this bus. * *

If this bus is not self-cleaning, this method might not remove all references to unneeded objects which may * result into memory leaks if not handled appropriately. These references may be removed manually by calling * {@link #clear()}.

* * @param object the origin of the subscribers that should be unregistered from this bus * * @throws NullPointerException if the given object is {@code null} * * @since 1.0.0 */ @SuppressWarnings({"unused"}) public void unregister(Object object) { Objects.requireNonNull(object); if (this.isSelfCleaning) { synchronized (this) { this.subscribers.entrySet().removeIf(entry -> entry.getValue().removeIf(subscriber -> subscriber.origin.equals(object))); } } else { this.subscribers.values().forEach(subscribers -> subscribers.removeIf(subscriber -> subscriber.origin.equals(object))); } } /** * Removes all references to empty or unused internal objects that are too expensive to remove otherwise. * *

If this bus is self-cleaning, this method does nothing instead.

* * @see EventBus.Builder#setSelfCleaning(boolean) * * @since 1.0.0 */ @SuppressWarnings({"unused", "WeakerAccess"}) public void clear() { if (!this.isSelfCleaning) { synchronized (this) { this.subscribers.entrySet().removeIf(entry -> entry.getValue().isEmpty()); } } } /** * Unregisters all subscribers of this bus and {@linkplain #clear() clears} the internal state, effectively * resetting the bus to its initial state. * * @since 1.0.0 */ @SuppressWarnings("unused") public void purge() { synchronized (this) { subscribers.clear(); } } /** * Posts the given {@linkplain Event} to this bus. * *

The event will be {@link EventDispatcher#dispatch(Event, Collection) dispatched} to all subscribers on this * bus whose accepted type {@link Class#isAssignableFrom(Class) is assignable from} the event's type.

* *

If a posted event cannot be received by any subscriber, it is considered "dead". Dead events are wrapped in an * instance of {@link DeadEvent} and posted again. Subscribing to these events can be useful for debugging purposes. * (Unlike any other event, a dead event will only be dispatched to subscribers that explicitly accept a * {@linkplain DeadEvent}.

* *

The subscribers are called in no particular order, some time in the future and no guarantees are made that the * events are passed to the subscribers in the order in which they are posted to the bus. This behaviour may be * changed at the discretion of this bus' {@link EventDispatcher} and {@link Executor} implementation respectively. * However, it is guaranteed, that an event will only be dispatched to the subscribers that were subscribed to the * event before the event was posted.

* * @param event the event to be dispatched to this bus' subscribers * * @throws NullPointerException if the given event is {@code null} * * @since 1.0.0 */ public void post(Event event) { Objects.requireNonNull(event); List subscribers; if (event instanceof DeadEvent) { subscribers = this.subscribers.entrySet().stream() .filter(entry -> DeadEvent.class.equals(entry.getKey())) .flatMap(entry -> entry.getValue().stream()) .collect(Collectors.toList()); if (!subscribers.isEmpty()) { this.dispatcher.dispatch(event, subscribers); } } else { subscribers = this.subscribers.entrySet().stream() .filter(e -> e.getKey().isAssignableFrom(event.getClass())) .flatMap(entry -> entry.getValue().stream()) .collect(Collectors.toList()); if (subscribers.isEmpty()) { post(new DeadEvent(event)); } else { this.dispatcher.dispatch(event, subscribers); } } } /** * A wrapper class for a subscriber method. * *

This class provides several methods that serve as proxies to the underlying subscriber method and may be used * by a {@link EventDispatcher dispatcher} to make decisions about the execution order.

* * @since 1.0.0 */ public static final class Subscriber { private final EventBus bus; private final Object origin; private final Method method; private final MethodHandle handle; private Subscriber(EventBus bus, Object origin, Method method, MethodHandle handle) { this.bus = bus; this.origin = origin; this.method = method; this.handle = handle; } /** * Dispatches the given {@link Event} to the wrapped subscriber. * *

The blocking behaviour of this method depends on the executor implementation of the {@link EventBus} that * the subscriber is registered to.

* * @param event the event to dispatch * * @throws NullPointerException if the given event is {@code null} * * @since 1.0.0 */ @SuppressWarnings("WeakerAccess") public void dispatch(Event event) { Objects.requireNonNull(event); this.bus.executor.execute(() -> { try { this.handle.invoke(event); } catch (Throwable t) { this.bus.dispatchErrorHandler.onDispatchError(event, this, t); } }); } /** * Serves as a proxy for {@link Method#getAnnotation(Class)}. * * @param the type of the annotation to query for and return if present * @param annotationClass the Class object corresponding to the annotation type * * @return this element's annotation for the specified annotation type if present on this element, else null * * @throws NullPointerException if the given annotation class is null * * @since 1.0.0 */ @SuppressWarnings("unused") public T getAnnotation(Class annotationClass) { return this.method.getAnnotation(annotationClass); } /** * Serves as a proxy for {@link Method#getAnnotationsByType(Class)}. * * @param the type of the annotation to query for and return if present * @param annotationClass the Class object corresponding to the annotation type * * @return all this element's annotations for the specified annotation type if associated with this element, * else an array of length zero * * @throws NullPointerException if the given annotation class is null * * @since 1.0.0 */ @SuppressWarnings("unused") public T[] getAnnotationsByType(Class annotationClass) { return this.method.getAnnotationsByType(annotationClass); } /** * Returns the {@link EventBus} that owns this subscriber. * * @return the bus that owns this subscriber * * @since 1.1.0 */ @SuppressWarnings("unused") public EventBus getBus() { return this.bus; } /** * Returns the origin of this subscriber. * * @return the origin of this subscriber * * @since 1.1.0 */ @SuppressWarnings("unused") public Object getOrigin() { return this.origin; } /** * Serves as a proxy for {@link Method#isAnnotationPresent(Class)}. * * @param annotationClass the Class object corresponding to the annotation type * * @return true if an annotation for the specified annotation type is present on this element, else false * * @throws NullPointerException if the given annotation class is null * * @since 1.0.0 */ @SuppressWarnings("unused") public boolean isAnnotationPresent(Class annotationClass) { return this.method.isAnnotationPresent(annotationClass); } } /** * A builder class for an {@link EventBus}. * *

A builder is reusable. Calling {@linkplain #build()} does not reset nor does it invalidate the builder.

* * @since 1.0.0 */ public static final class Builder { private EventDispatcher dispatcher; private DispatchErrorHandler dispatchErrorHandler; private Executor executor; private Class subscriberMarker; private boolean isSelfCleaning; /** * Creates a new builder instance. * * @deprecated Use {@link EventBus#builder()} instead. * * @since 1.0.0 */ @Deprecated // TODO remove in 2.0.0 public Builder() { this.dispatcher = EventDispatcher.perThreadDispatchQueue(); // Get rid of this the next time the major version is increased. (This shipped in 1.0.0, so I'll keep it for now) this.dispatchErrorHandler = (e, s, t) -> t.printStackTrace(); this.executor = MJLExecutors.directExecutor(); this.subscriberMarker = EventSubscriber.class; this.isSelfCleaning = false; } /** * Creates and returns a new bus with the settings currently defined in this builder. * * @return a new bus with the settings as currently specified in this builder * * @since 1.0.0 */ public EventBus build() { return new EventBus(this.dispatcher, this.dispatchErrorHandler, this.executor, this.subscriberMarker, this.isSelfCleaning); } /** * Sets the dispatcher implementation that will be used by the bus. * *

Consecutive calls overwrite the previously set value.

* * @param dispatcher the dispatcher for the bus * * @return this builder instance * * @throws NullPointerException if the given dispatcher is {@code null} * * @since 1.0.0 */ public Builder setDispatcher(EventDispatcher dispatcher) { this.dispatcher = Objects.requireNonNull(dispatcher); return this; } /** * Sets the {@link DispatchErrorHandler} that will be used by the bus. * *

Consecutive calls overwrite the previously set value.

* * @param handler the dispatch-error-handler for the bus * * @return this builder instance * * @since 1.1.0 */ @SuppressWarnings("WeakerAccess") public Builder setDispatchErrorHandler(DispatchErrorHandler handler) { this.dispatchErrorHandler = Objects.requireNonNull(handler); return this; } /** * Sets the executor implementation that will be used by the bus. * *

Consecutive calls overwrite the previously set value.

* * @param executor the executor for the bus * * @return this builder instance * * @throws NullPointerException if the given executor is {@code null} * * @since 1.0.0 */ public Builder setExecutor(Executor executor) { this.executor = Objects.requireNonNull(executor); return this; } /** * Sets whether or not the bus will be self-cleaning. * *

Consecutive calls overwrite the previously set value.

* * @param value whether or not the bus should be self-cleaning * * @return this builder instance * * @see EventBus#clear() * @see EventBus#unregister(Object) * * @since 1.0.0 */ @SuppressWarnings({"WeakerAccess", "unused"}) public Builder setSelfCleaning(boolean value) { this.isSelfCleaning = value; return this; } /** * Sets the subscriber marker that will be used by the bus. * *

Consecutive calls overwrite the previously set value.

* * @param type the marker type * * @return this builder instance * * @throws NullPointerException if the given type is {@code null} * @throws IllegalArgumentException if the given type is not visible at runtime * * @see EventBus#register(Class, MethodHandles.Lookup) * @see EventBus#register(Object, MethodHandles.Lookup) * * @since 1.0.0 */ public Builder setSubscriberMarker(Class type) { Retention retention = Objects.requireNonNull(type).getAnnotation(Retention.class); if (retention == null || retention.value() != RetentionPolicy.RUNTIME) throw new IllegalArgumentException(type + " is not visible at runtime"); this.subscriberMarker = Objects.requireNonNull(type); return this; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy