Merge pull request #417 from Minestom/zero_cost_event

Free event calling
This commit is contained in:
TheMode 2021-08-22 05:04:21 +02:00 committed by GitHub
commit 3d1509aef0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 289 additions and 162 deletions

View File

@ -19,7 +19,7 @@ import java.util.function.Predicate;
*/ */
public interface EventListener<T extends Event> { public interface EventListener<T extends Event> {
@NotNull Class<T> getEventType(); @NotNull Class<T> eventType();
@NotNull Result run(@NotNull T event); @NotNull Result run(@NotNull T event);
@ -122,7 +122,7 @@ public interface EventListener<T extends Event> {
final var handler = this.handler; final var handler = this.handler;
return new EventListener<>() { return new EventListener<>() {
@Override @Override
public @NotNull Class<T> getEventType() { public @NotNull Class<T> eventType() {
return eventType; return eventType;
} }

View File

@ -182,15 +182,49 @@ public interface EventNode<T extends Event> {
} }
/** /**
* Executes the given event on this node. The event must pass all conditions before * Calls an event starting from this node.
* it will be forwarded to the listeners.
* <p>
* Calling an event on a node will execute all child nodes, however, an event may be
* called anywhere on the event graph and it will propagate down from there only.
* *
* @param event the event to execute * @param event the event to call
*/ */
void call(@NotNull T event); default void call(@NotNull T event) {
//noinspection unchecked
call(event, getHandle((Class<T>) event.getClass()));
}
/**
* Calls an event starting from this node.
* <p>
* The event handle can be retrieved using {@link #getHandle(Class)}
* and is useful to avoid map lookups for high-frequency events.
*
* @param event the event to call
* @param handle the event handle linked to this node
* @param <E> the event type
* @throws IllegalArgumentException if {@param handle}'s owner is not {@code this}
*/
<E extends T> void call(@NotNull E event, @NotNull ListenerHandle<E> handle);
/**
* Gets the handle of an event type.
*
* @param handleType the handle type
* @param <E> the event type
* @return the handle linked to {@param handleType}
*/
<E extends T> @NotNull ListenerHandle<E> getHandle(@NotNull Class<E> handleType);
/**
* Gets if any listener has been registered for the given handle.
* May trigger an update if the cached data is not correct.
* <p>
* Useful if you are able to avoid expensive computation in the case where
* the event is unused. Be aware that {@link #call(Event, ListenerHandle)}
* has similar optimization built-in.
*
* @param handle the listener handle
* @return true if the event has 1 or more listeners
*/
boolean hasListener(@NotNull ListenerHandle<? extends T> handle);
/** /**
* Execute a cancellable event with a callback to execute if the event is successful. * Execute a cancellable event with a callback to execute if the event is successful.
@ -318,9 +352,24 @@ public interface EventNode<T extends Event> {
@Contract(value = "_ -> this") @Contract(value = "_ -> this")
@NotNull EventNode<T> removeListener(@NotNull EventListener<? extends T> listener); @NotNull EventNode<T> removeListener(@NotNull EventListener<? extends T> listener);
/**
* Maps a specific object to a node.
* <p>
* Be aware that such structure have huge performance penalty as they will
* always require a map lookup. Use only at last resort.
*
* @param node the node to map
* @param value the mapped value
*/
@ApiStatus.Experimental @ApiStatus.Experimental
void map(@NotNull EventNode<? extends T> node, @NotNull Object value); void map(@NotNull EventNode<? extends T> node, @NotNull Object value);
/**
* Undo {@link #map(EventNode, Object)}
*
* @param value the value to unmap
* @return true if the value has been unmapped, false if nothing happened
*/
@ApiStatus.Experimental @ApiStatus.Experimental
boolean unmap(@NotNull Object value); boolean unmap(@NotNull Object value);

View File

@ -1,6 +1,7 @@
package net.minestom.server.event; package net.minestom.server.event;
import net.minestom.server.MinecraftServer; import net.minestom.server.MinecraftServer;
import net.minestom.server.event.trait.RecursiveEvent;
import net.minestom.server.utils.validate.Check; import net.minestom.server.utils.validate.Check;
import org.jetbrains.annotations.Contract; import org.jetbrains.annotations.Contract;
import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.NotNull;
@ -10,19 +11,17 @@ import java.util.*;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList; import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.CopyOnWriteArraySet; import java.util.concurrent.CopyOnWriteArraySet;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater; import java.util.function.BiConsumer;
import java.util.function.BiPredicate; import java.util.function.BiPredicate;
import java.util.function.Consumer; import java.util.function.Consumer;
import java.util.function.IntUnaryOperator;
import java.util.stream.Collectors;
class EventNodeImpl<T extends Event> implements EventNode<T> { class EventNodeImpl<T extends Event> implements EventNode<T> {
private static final Object GLOBAL_CHILD_LOCK = new Object(); private static final Object GLOBAL_CHILD_LOCK = new Object();
private final Object lock = new Object();
private final Map<Class<? extends T>, Handle<T>> handleMap = new ConcurrentHashMap<>();
private final Map<Class<? extends T>, ListenerEntry<T>> listenerMap = new ConcurrentHashMap<>(); private final Map<Class<? extends T>, ListenerEntry<T>> listenerMap = new ConcurrentHashMap<>();
private final Set<EventNode<T>> children = new CopyOnWriteArraySet<>(); private final Set<EventNodeImpl<T>> children = new CopyOnWriteArraySet<>();
private final Map<Object, ListenerEntry<T>> mappedNodeCache = new WeakHashMap<>(); private final Map<Object, EventNodeImpl<T>> mappedNodeCache = new WeakHashMap<>();
private final String name; private final String name;
private final EventFilter<T, ?> filter; private final EventFilter<T, ?> filter;
@ -31,9 +30,9 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
private volatile int priority; private volatile int priority;
private volatile EventNodeImpl<? super T> parent; private volatile EventNodeImpl<? super T> parent;
protected EventNodeImpl(@NotNull String name, EventNodeImpl(@NotNull String name,
@NotNull EventFilter<T, ?> filter, @NotNull EventFilter<T, ?> filter,
@Nullable BiPredicate<T, Object> predicate) { @Nullable BiPredicate<T, Object> predicate) {
this.name = name; this.name = name;
this.filter = filter; this.filter = filter;
this.predicate = predicate; this.predicate = predicate;
@ -41,31 +40,31 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
} }
@Override @Override
public void call(@NotNull T event) { public <E extends T> void call(@NotNull E event, @NotNull ListenerHandle<E> handle) {
final var eventClass = event.getClass(); final Handle<T> castedHandle = (Handle<T>) handle;
if (!eventType.isAssignableFrom(eventClass)) return; // Invalid event type Check.argCondition(castedHandle.node != this, "Invalid handle owner");
// Conditions if (!castedHandle.updated) castedHandle.update();
if (predicate != null) { final List<Consumer<T>> listeners = castedHandle.listeners;
try { if (listeners.isEmpty()) return;
final var value = filter.getHandler(event); for (Consumer<T> listener : listeners) {
if (!predicate.test(event, value)) return; listener.accept(event);
} catch (Exception e) {
MinecraftServer.getExceptionManager().handleException(e);
return;
}
}
// Process listeners list
final var entry = listenerMap.get(eventClass);
if (entry == null) return; // No listener nor children
entry.call(event);
// Process children
if (entry.childCount > 0) {
this.children.stream()
.sorted(Comparator.comparing(EventNode::getPriority))
.forEach(child -> child.call(event));
} }
} }
@Override
public <E extends T> @NotNull ListenerHandle<E> getHandle(@NotNull Class<E> handleType) {
//noinspection unchecked
return (ListenerHandle<E>) handleMap.computeIfAbsent(handleType,
aClass -> new Handle<>(this, (Class<T>) aClass));
}
@Override
public boolean hasListener(@NotNull ListenerHandle<? extends T> handle) {
final Handle<T> castedHandle = (Handle<T>) handle;
if (!castedHandle.updated) castedHandle.update();
return !castedHandle.listeners.isEmpty();
}
@Override @Override
public <E extends T> @NotNull List<EventNode<E>> findChildren(@NotNull String name, Class<E> eventType) { public <E extends T> @NotNull List<EventNode<E>> findChildren(@NotNull String name, Class<E> eventType) {
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
@ -126,12 +125,9 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
final var childImpl = (EventNodeImpl<? extends T>) child; final var childImpl = (EventNodeImpl<? extends T>) child;
Check.stateCondition(childImpl.parent != null, "Node already has a parent"); Check.stateCondition(childImpl.parent != null, "Node already has a parent");
Check.stateCondition(Objects.equals(parent, child), "Cannot have a child as parent"); Check.stateCondition(Objects.equals(parent, child), "Cannot have a child as parent");
final boolean result = this.children.add((EventNodeImpl<T>) childImpl); if (!children.add((EventNodeImpl<T>) childImpl)) return this; // Couldn't add the child (already present?)
if (result) { childImpl.parent = this;
childImpl.parent = this; childImpl.propagateEvents(this); // Propagate after setting the parent
// Increase listener count
propagateNode(childImpl, IntUnaryOperator.identity());
}
} }
return this; return this;
} }
@ -139,13 +135,11 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
@Override @Override
public @NotNull EventNode<T> removeChild(@NotNull EventNode<? extends T> child) { public @NotNull EventNode<T> removeChild(@NotNull EventNode<? extends T> child) {
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
final boolean result = this.children.remove(child); final var childImpl = (EventNodeImpl<? extends T>) child;
if (result) { final boolean result = this.children.remove(childImpl);
final var childImpl = (EventNodeImpl<? extends T>) child; if (!result) return this; // Child not found
childImpl.parent = null; childImpl.propagateEvents(parent); // Propagate before removing the parent
// Decrease listener count childImpl.parent = null;
propagateNode(childImpl, count -> -count);
}
} }
return this; return this;
} }
@ -153,10 +147,10 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
@Override @Override
public @NotNull EventNode<T> addListener(@NotNull EventListener<? extends T> listener) { public @NotNull EventNode<T> addListener(@NotNull EventListener<? extends T> listener) {
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
final var eventType = listener.getEventType(); final var eventType = listener.eventType();
var entry = getEntry(eventType); ListenerEntry<T> entry = getEntry(eventType);
entry.listeners.add((EventListener<T>) listener); entry.listeners.add((EventListener<T>) listener);
propagateToParent(eventType, 1); propagateEvent(parent, eventType);
} }
return this; return this;
} }
@ -164,50 +158,37 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
@Override @Override
public @NotNull EventNode<T> removeListener(@NotNull EventListener<? extends T> listener) { public @NotNull EventNode<T> removeListener(@NotNull EventListener<? extends T> listener) {
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
final var eventType = listener.getEventType(); final var eventType = listener.eventType();
var entry = listenerMap.get(eventType); ListenerEntry<T> entry = listenerMap.get(eventType);
if (entry == null) return this; if (entry == null) return this; // There is no listener with such type
var listeners = entry.listeners; var listeners = entry.listeners;
final boolean removed = listeners.remove(listener); if (listeners.remove(listener)) propagateEvent(parent, eventType);
if (removed) propagateToParent(eventType, -1);
} }
return this; return this;
} }
@Override @Override
public void map(@NotNull EventNode<? extends T> node, @NotNull Object value) { public void map(@NotNull EventNode<? extends T> node, @NotNull Object value) {
final var nodeImpl = (EventNodeImpl<? extends T>) node;
final var valueType = value.getClass();
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
nodeImpl.listenerMap.forEach((type, listenerEntry) -> { final var nodeImpl = (EventNodeImpl<? extends T>) node;
final var entry = getEntry(type); Check.stateCondition(nodeImpl.parent != null, "Node already has a parent");
final boolean correct = entry.filters.stream().anyMatch(eventFilter -> { Check.stateCondition(Objects.equals(parent, nodeImpl), "Cannot map to self");
final var handlerType = eventFilter.handlerType(); var previous = this.mappedNodeCache.put(value, (EventNodeImpl<T>) nodeImpl);
return handlerType != null && handlerType.isAssignableFrom(valueType); if (previous != null) previous.parent = null;
}); nodeImpl.parent = this;
Check.stateCondition(!correct, "The node filter {0} is not compatible with type {1}", nodeImpl.eventType, valueType); nodeImpl.propagateEvents(this); // Propagate after setting the parent
synchronized (mappedNodeCache) {
entry.mappedNode.put(value, (EventNode<T>) nodeImpl);
mappedNodeCache.put(value, entry);
// TODO propagate
}
});
} }
} }
@Override @Override
public boolean unmap(@NotNull Object value) { public boolean unmap(@NotNull Object value) {
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
synchronized (mappedNodeCache) { final var mappedNode = this.mappedNodeCache.remove(value);
var entry = mappedNodeCache.remove(value); if (mappedNode == null) return false; // Mapped node not found
if (entry == null) return false; final var childImpl = (EventNodeImpl<? extends T>) mappedNode;
final EventNode<T> previousNode = entry.mappedNode.remove(value); childImpl.propagateEvents(parent); // Propagate before removing the parent
if (previousNode != null) { childImpl.parent = null;
// TODO propagate return true;
return true;
}
return false;
}
} }
} }
@ -215,9 +196,9 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
public void register(@NotNull EventBinding<? extends T> binding) { public void register(@NotNull EventBinding<? extends T> binding) {
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
for (var eventType : binding.eventTypes()) { for (var eventType : binding.eventTypes()) {
var entry = getEntry((Class<? extends T>) eventType); ListenerEntry<T> entry = getEntry((Class<? extends T>) eventType);
final boolean added = entry.bindingConsumers.add((Consumer<T>) binding.consumer(eventType)); final boolean added = entry.bindingConsumers.add((Consumer<T>) binding.consumer(eventType));
if (added) propagateToParent((Class<? extends T>) eventType, 1); if (added) propagateEvent(parent, (Class<? extends T>) eventType);
} }
} }
} }
@ -226,10 +207,10 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
public void unregister(@NotNull EventBinding<? extends T> binding) { public void unregister(@NotNull EventBinding<? extends T> binding) {
synchronized (GLOBAL_CHILD_LOCK) { synchronized (GLOBAL_CHILD_LOCK) {
for (var eventType : binding.eventTypes()) { for (var eventType : binding.eventTypes()) {
var entry = listenerMap.get(eventType); ListenerEntry<T> entry = listenerMap.get(eventType);
if (entry == null) return; if (entry == null) return;
final boolean removed = entry.bindingConsumers.remove(binding.consumer(eventType)); final boolean removed = entry.bindingConsumers.remove(binding.consumer(eventType));
if (removed) propagateToParent((Class<? extends T>) eventType, -1); if (removed) propagateEvent(parent, (Class<? extends T>) eventType);
} }
} }
} }
@ -260,42 +241,22 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
return parent; return parent;
} }
private void propagateChildCountChange(Class<? extends T> eventClass, int count) { private void propagateEvents(EventNodeImpl<? super T> parent) {
var entry = getEntry(eventClass); this.listenerMap.keySet().forEach(aClass -> propagateEvent(parent, aClass));
final int result = ListenerEntry.CHILD_UPDATER.addAndGet(entry, count);
if (result == 0 && entry.listeners.isEmpty()) {
this.listenerMap.remove(eventClass);
} else if (result < 0) {
throw new IllegalStateException("Something wrong happened, listener count: " + result);
}
if (parent != null) {
parent.propagateChildCountChange(eventClass, count);
}
} }
private void propagateToParent(Class<? extends T> eventClass, int count) { private void propagateEvent(EventNodeImpl parent, Class<? extends T> eventClass) {
final var parent = this.parent; if (parent == null) return;
if (parent != null) { forTargetEvents(eventClass, type -> {
synchronized (parent.lock) { Handle<? super T> parentHandle = (Handle<? super T>) parent.handleMap.get(type);
parent.propagateChildCountChange(eventClass, count); if (parentHandle == null) return;
} parentHandle.updated = false;
} parent.propagateEvent(parent.parent, type);
} });
private void propagateNode(EventNodeImpl<? extends T> child, IntUnaryOperator operator) {
synchronized (lock) {
final var listeners = child.listenerMap;
listeners.forEach((eventClass, eventListeners) -> {
final var entry = listeners.get(eventClass);
if (entry == null) return;
final int childCount = entry.listeners.size() + entry.childCount;
propagateChildCountChange(eventClass, operator.applyAsInt(childCount));
});
}
} }
private ListenerEntry<T> getEntry(Class<? extends T> type) { private ListenerEntry<T> getEntry(Class<? extends T> type) {
return listenerMap.computeIfAbsent(type, aClass -> new ListenerEntry<>(this, (Class<T>) aClass)); return listenerMap.computeIfAbsent(type, aClass -> new ListenerEntry<>());
} }
private static boolean equals(EventNode<?> node, String name, Class<?> eventType) { private static boolean equals(EventNode<?> node, String name, Class<?> eventType) {
@ -304,59 +265,163 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
return nameCheck && typeCheck; return nameCheck && typeCheck;
} }
private static class ListenerEntry<T extends Event> { private static void forTargetEvents(Class<?> type, Consumer<Class<?>> consumer) {
private static final List<EventFilter<? extends Event, ?>> FILTERS = List.of( consumer.accept(type);
EventFilter.ENTITY, // Recursion
EventFilter.ITEM, EventFilter.INSTANCE, if (RecursiveEvent.class.isAssignableFrom(type)) {
EventFilter.INVENTORY, EventFilter.BLOCK); final Class<?> superclass = type.getSuperclass();
@SuppressWarnings("rawtypes") if (superclass != null && RecursiveEvent.class.isAssignableFrom(superclass)) {
private static final AtomicIntegerFieldUpdater<ListenerEntry> CHILD_UPDATER = forTargetEvents(superclass, consumer);
AtomicIntegerFieldUpdater.newUpdater(ListenerEntry.class, "childCount"); }
}
}
final EventNodeImpl<T> node; private static class ListenerEntry<T extends Event> {
final List<EventFilter<?, ?>> filters;
final List<EventListener<T>> listeners = new CopyOnWriteArrayList<>(); final List<EventListener<T>> listeners = new CopyOnWriteArrayList<>();
final Set<Consumer<T>> bindingConsumers = new CopyOnWriteArraySet<>(); final Set<Consumer<T>> bindingConsumers = new CopyOnWriteArraySet<>();
final Map<Object, EventNode<T>> mappedNode = new WeakHashMap<>(); }
volatile int childCount;
ListenerEntry(EventNodeImpl<T> node, Class<T> eventType) { private static final class Handle<E extends Event> implements ListenerHandle<E> {
private final EventNodeImpl<E> node;
private final Class<E> eventType;
private final List<Consumer<E>> listeners = new CopyOnWriteArrayList<>();
private volatile boolean updated;
Handle(EventNodeImpl<E> node, Class<E> eventType) {
this.node = node; this.node = node;
this.filters = FILTERS.stream().filter(eventFilter -> eventFilter.eventType().isAssignableFrom(eventType)).collect(Collectors.toList()); this.eventType = eventType;
} }
void call(T event) { void update() {
// Event interfaces synchronized (GLOBAL_CHILD_LOCK) {
if (!bindingConsumers.isEmpty()) { this.listeners.clear();
for (var consumer : bindingConsumers) { recursiveUpdate(node);
consumer.accept(event); this.updated = true;
}
}
private void recursiveUpdate(EventNodeImpl<E> targetNode) {
// Standalone listeners
forTargetEvents(eventType, type -> {
final ListenerEntry<E> entry = targetNode.listenerMap.get(type);
if (entry != null) appendEntries(entry, targetNode);
});
// Mapped nodes
handleMappedNode(targetNode);
// Add children
final Set<EventNodeImpl<E>> children = targetNode.children;
if (children.isEmpty()) return;
children.stream()
.filter(child -> child.eventType.isAssignableFrom(eventType)) // Invalid event type
.sorted(Comparator.comparing(EventNode::getPriority))
.forEach(this::recursiveUpdate);
}
/**
* Add the node's listeners from {@link EventNode#map(EventNode, Object)}.
* The goal is to limit the amount of map lookup.
*/
private void handleMappedNode(EventNodeImpl<E> targetNode) {
final var mappedNodeCache = targetNode.mappedNodeCache;
if (mappedNodeCache.isEmpty()) return;
Set<EventFilter<E, ?>> filters = new HashSet<>(mappedNodeCache.size());
Map<Object, Handle<E>> handlers = new HashMap<>(mappedNodeCache.size());
// Retrieve all filters used to retrieve potential handlers
for (var mappedEntry : mappedNodeCache.entrySet()) {
final EventNodeImpl<E> mappedNode = mappedEntry.getValue();
final Handle<E> handle = (Handle<E>) mappedNode.getHandle(eventType);
if (!mappedNode.hasListener(handle)) continue; // Implicit update
filters.add(mappedNode.filter);
handlers.put(mappedEntry.getKey(), handle);
}
// If at least one mapped node listen to this handle type,
// loop through them and forward to mapped node if there is a match
if (!filters.isEmpty()) {
final var filterList = List.copyOf(filters);
final int size = filterList.size();
final BiConsumer<EventFilter<E, ?>, E> mapper = (filter, event) -> {
final Object handler = filter.castHandler(event);
final Handle<E> handle = handlers.get(handler);
if (handle != null) { // Run the listeners of the mapped node
if (!handle.updated) handle.update();
for (Consumer<E> listener : handle.listeners) {
listener.accept(event);
}
}
};
if (size == 1) {
final var firstFilter = filterList.get(0);
// Common case where there is only one filter
this.listeners.add(event -> mapper.accept(firstFilter, event));
} else if (size == 2) {
final var firstFilter = filterList.get(0);
final var secondFilter = filterList.get(1);
this.listeners.add(event -> {
mapper.accept(firstFilter, event);
mapper.accept(secondFilter, event);
});
} else {
this.listeners.add(event -> {
for (var filter : filterList) {
mapper.accept(filter, event);
}
});
} }
} }
// Mapped listeners }
if (!mappedNode.isEmpty()) {
synchronized (node.mappedNodeCache) { /**
// Check mapped listeners for each individual event handler * Add listeners from {@link EventNode#addListener(EventListener)} and
for (var filter : filters) { * {@link EventNode#register(EventBinding)} to the handle list.
final var handler = filter.castHandler(event); * <p>
final var map = mappedNode.get(handler); * Most computation should ideally be done outside the consumers as a one-time cost.
if (map != null) map.call(event); */
} private void appendEntries(ListenerEntry<E> entry, EventNodeImpl<E> targetNode) {
} final var filter = targetNode.filter;
final var predicate = targetNode.predicate;
final boolean hasPredicate = predicate != null;
final List<EventListener<E>> listenersCopy = List.copyOf(entry.listeners);
final List<Consumer<E>> bindingsCopy = List.copyOf(entry.bindingConsumers);
if (!hasPredicate && listenersCopy.isEmpty() && bindingsCopy.isEmpty())
return; // Nothing to run
if (!hasPredicate && bindingsCopy.isEmpty() && listenersCopy.size() == 1) {
// Only one normal listener
final EventListener<E> listener = listenersCopy.get(0);
this.listeners.add(e -> callListener(targetNode, listener, e));
return;
} }
// Basic listeners
if (!listeners.isEmpty()) { // Worse case scenario, try to run everything
for (EventListener<T> listener : listeners) { this.listeners.add(e -> {
EventListener.Result result; if (hasPredicate) {
try { final Object value = filter.getHandler(e);
result = listener.run(event); if (!predicate.test(e, value)) return;
} catch (Exception e) { }
result = EventListener.Result.EXCEPTION; if (!listenersCopy.isEmpty()) {
MinecraftServer.getExceptionManager().handleException(e); for (EventListener<E> listener : listenersCopy) {
} callListener(targetNode, listener, e);
if (result == EventListener.Result.EXPIRED) {
listeners.remove(listener);
} }
} }
if (!bindingsCopy.isEmpty()) {
for (Consumer<E> eConsumer : bindingsCopy) {
eConsumer.accept(e);
}
}
});
}
static <E extends Event> void callListener(EventNodeImpl<E> targetNode, EventListener<E> listener, E event) {
EventListener.Result result;
try {
result = listener.run(event);
} catch (Exception e) {
result = EventListener.Result.EXCEPTION;
MinecraftServer.getExceptionManager().handleException(e);
}
if (result == EventListener.Result.EXPIRED) {
targetNode.removeListener(listener);
} }
} }
} }

View File

@ -0,0 +1,7 @@
package net.minestom.server.event;
import org.jetbrains.annotations.ApiStatus;
@ApiStatus.NonExtendable
public interface ListenerHandle<E extends Event> {
}

View File

@ -0,0 +1,6 @@
package net.minestom.server.event.trait;
import net.minestom.server.event.Event;
public interface RecursiveEvent extends Event {
}