mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-07 00:48:28 +01:00
Fix event node child/map leak
Signed-off-by: TheMode <themode@outlook.fr>
This commit is contained in:
parent
b4af87b8a4
commit
e11d15af0d
@ -1,5 +1,6 @@
|
|||||||
package net.minestom.server.event;
|
package net.minestom.server.event;
|
||||||
|
|
||||||
|
import com.github.benmanes.caffeine.cache.Caffeine;
|
||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.event.trait.RecursiveEvent;
|
import net.minestom.server.event.trait.RecursiveEvent;
|
||||||
import net.minestom.server.utils.validate.Check;
|
import net.minestom.server.utils.validate.Check;
|
||||||
@ -20,9 +21,12 @@ non-sealed class EventNodeImpl<T extends Event> implements EventNode<T> {
|
|||||||
|
|
||||||
private final Map<Class, Handle<T>> handleMap = new ConcurrentHashMap<>();
|
private final Map<Class, Handle<T>> handleMap = new ConcurrentHashMap<>();
|
||||||
final Map<Class<? extends T>, ListenerEntry<T>> listenerMap = new ConcurrentHashMap<>();
|
final Map<Class<? extends T>, ListenerEntry<T>> listenerMap = new ConcurrentHashMap<>();
|
||||||
final Set<EventNodeImpl<T>> children = new CopyOnWriteArraySet<>();
|
final Set<EventNodeImpl<T>> children = Collections.newSetFromMap(Caffeine.newBuilder()
|
||||||
final Map<Object, EventNodeImpl<T>> mappedNodeCache = new WeakHashMap<>();
|
.weakKeys().<EventNodeImpl<T>, Boolean>build().asMap());
|
||||||
final Map<Object, EventNodeImpl<T>> registeredMappedNode = new WeakHashMap<>();
|
final Map<Object, EventNodeImpl<T>> mappedNodeCache = Caffeine.newBuilder()
|
||||||
|
.weakKeys().weakValues().<Object, EventNodeImpl<T>>build().asMap();
|
||||||
|
final Map<Object, EventNodeImpl<T>> registeredMappedNode = Caffeine.newBuilder()
|
||||||
|
.weakKeys().weakValues().<Object, EventNodeImpl<T>>build().asMap();
|
||||||
|
|
||||||
final String name;
|
final String name;
|
||||||
final EventFilter<T, ?> filter;
|
final EventFilter<T, ?> filter;
|
||||||
@ -50,6 +54,7 @@ non-sealed class EventNodeImpl<T extends Event> implements EventNode<T> {
|
|||||||
@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) {
|
||||||
|
final Set<EventNode<T>> children = getChildren();
|
||||||
if (children.isEmpty()) return List.of();
|
if (children.isEmpty()) return List.of();
|
||||||
List<EventNode<E>> result = new ArrayList<>();
|
List<EventNode<E>> result = new ArrayList<>();
|
||||||
for (EventNode<T> child : children) {
|
for (EventNode<T> child : children) {
|
||||||
@ -70,6 +75,7 @@ non-sealed class EventNodeImpl<T extends Event> implements EventNode<T> {
|
|||||||
@Override
|
@Override
|
||||||
public <E extends T> void replaceChildren(@NotNull String name, @NotNull Class<E> eventType, @NotNull EventNode<E> eventNode) {
|
public <E extends T> void replaceChildren(@NotNull String name, @NotNull Class<E> eventType, @NotNull EventNode<E> eventNode) {
|
||||||
synchronized (GLOBAL_CHILD_LOCK) {
|
synchronized (GLOBAL_CHILD_LOCK) {
|
||||||
|
final Set<EventNode<T>> children = getChildren();
|
||||||
if (children.isEmpty()) return;
|
if (children.isEmpty()) return;
|
||||||
for (EventNode<T> child : children) {
|
for (EventNode<T> child : children) {
|
||||||
if (equals(child, name, eventType)) {
|
if (equals(child, name, eventType)) {
|
||||||
@ -85,6 +91,7 @@ non-sealed class EventNodeImpl<T extends Event> implements EventNode<T> {
|
|||||||
@Override
|
@Override
|
||||||
public void removeChildren(@NotNull String name, @NotNull Class<? extends T> eventType) {
|
public void removeChildren(@NotNull String name, @NotNull Class<? extends T> eventType) {
|
||||||
synchronized (GLOBAL_CHILD_LOCK) {
|
synchronized (GLOBAL_CHILD_LOCK) {
|
||||||
|
final Set<EventNode<T>> children = getChildren();
|
||||||
if (children.isEmpty()) return;
|
if (children.isEmpty()) return;
|
||||||
for (EventNode<T> child : children) {
|
for (EventNode<T> child : children) {
|
||||||
if (equals(child, name, eventType)) {
|
if (equals(child, name, eventType)) {
|
||||||
|
@ -6,6 +6,7 @@ import org.junit.jupiter.api.Test;
|
|||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
import static net.minestom.testing.TestUtils.assertEqualsIgnoreOrder;
|
||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
|
||||||
public class EventNodeQueryTest {
|
public class EventNodeQueryTest {
|
||||||
@ -23,12 +24,12 @@ public class EventNodeQueryTest {
|
|||||||
node.addChild(child2);
|
node.addChild(child2);
|
||||||
node.addChild(child3);
|
node.addChild(child3);
|
||||||
|
|
||||||
assertEquals(List.of(child1, child2), node.findChildren("test"));
|
assertEqualsIgnoreOrder(List.of(child1, child2), node.findChildren("test"));
|
||||||
assertEquals(List.of(child3), node.findChildren("test3"));
|
assertEqualsIgnoreOrder(List.of(child3), node.findChildren("test3"));
|
||||||
|
|
||||||
node.removeChild(child1);
|
node.removeChild(child1);
|
||||||
assertEquals(List.of(child2), node.findChildren("test"));
|
assertEqualsIgnoreOrder(List.of(child2), node.findChildren("test"));
|
||||||
assertEquals(List.of(child3), node.findChildren("test3"));
|
assertEqualsIgnoreOrder(List.of(child3), node.findChildren("test3"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -44,16 +45,16 @@ public class EventNodeQueryTest {
|
|||||||
node.addChild(child2);
|
node.addChild(child2);
|
||||||
node.addChild(child3);
|
node.addChild(child3);
|
||||||
|
|
||||||
assertEquals(List.of(child1, child2), node.findChildren("test", Event.class));
|
assertEqualsIgnoreOrder(List.of(child1, child2), node.findChildren("test", Event.class));
|
||||||
assertEquals(List.of(child1, child2), node.findChildren("test", EntityEvent.class));
|
assertEqualsIgnoreOrder(List.of(child1, child2), node.findChildren("test", EntityEvent.class));
|
||||||
assertEquals(List.of(child1), node.findChildren("test", PlayerEvent.class));
|
assertEqualsIgnoreOrder(List.of(child1), node.findChildren("test", PlayerEvent.class));
|
||||||
assertEquals(List.of(child3), node.findChildren("test3", EntityEvent.class));
|
assertEqualsIgnoreOrder(List.of(child3), node.findChildren("test3", EntityEvent.class));
|
||||||
|
|
||||||
node.removeChild(child1);
|
node.removeChild(child1);
|
||||||
assertEquals(List.of(child2), node.findChildren("test", Event.class));
|
assertEqualsIgnoreOrder(List.of(child2), node.findChildren("test", Event.class));
|
||||||
assertEquals(List.of(child2), node.findChildren("test", EntityEvent.class));
|
assertEqualsIgnoreOrder(List.of(child2), node.findChildren("test", EntityEvent.class));
|
||||||
assertEquals(List.of(), node.findChildren("test", PlayerEvent.class));
|
assertEqualsIgnoreOrder(List.of(), node.findChildren("test", PlayerEvent.class));
|
||||||
assertEquals(List.of(child3), node.findChildren("test3", EntityEvent.class));
|
assertEqualsIgnoreOrder(List.of(child3), node.findChildren("test3", EntityEvent.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -72,13 +73,13 @@ public class EventNodeQueryTest {
|
|||||||
var tmp2 = EventNode.all("tmp2");
|
var tmp2 = EventNode.all("tmp2");
|
||||||
|
|
||||||
node.replaceChildren("test", tmp1);
|
node.replaceChildren("test", tmp1);
|
||||||
assertEquals(List.of(child2), node.findChildren("test"));
|
assertEqualsIgnoreOrder(List.of(child2), node.findChildren("test"));
|
||||||
assertEquals(List.of(tmp1), node.findChildren("tmp1"));
|
assertEqualsIgnoreOrder(List.of(tmp1), node.findChildren("tmp1"));
|
||||||
|
|
||||||
node.replaceChildren("test3", tmp2);
|
node.replaceChildren("test3", tmp2);
|
||||||
assertEquals(List.of(child2), node.findChildren("test"));
|
assertEqualsIgnoreOrder(List.of(child2), node.findChildren("test"));
|
||||||
assertEquals(List.of(tmp1), node.findChildren("tmp1"));
|
assertEqualsIgnoreOrder(List.of(tmp1), node.findChildren("tmp1"));
|
||||||
assertEquals(List.of(), node.findChildren("test3"));
|
assertEqualsIgnoreOrder(List.of(), node.findChildren("test3"));
|
||||||
assertEquals(List.of(tmp2), node.findChildren("tmp2"));
|
assertEqualsIgnoreOrder(List.of(tmp2), node.findChildren("tmp2"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -227,6 +227,8 @@ public class EventNodeTest {
|
|||||||
public void nodeEmptyGC() {
|
public void nodeEmptyGC() {
|
||||||
var node = EventNode.all("main");
|
var node = EventNode.all("main");
|
||||||
var ref = new WeakReference<>(node);
|
var ref = new WeakReference<>(node);
|
||||||
|
|
||||||
|
//noinspection UnusedAssignment
|
||||||
node = null;
|
node = null;
|
||||||
waitUntilCleared(ref);
|
waitUntilCleared(ref);
|
||||||
}
|
}
|
||||||
@ -237,7 +239,39 @@ public class EventNodeTest {
|
|||||||
var ref = new WeakReference<>(node);
|
var ref = new WeakReference<>(node);
|
||||||
node.addListener(EventTest.class, event -> {
|
node.addListener(EventTest.class, event -> {
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//noinspection UnusedAssignment
|
||||||
node = null;
|
node = null;
|
||||||
waitUntilCleared(ref);
|
waitUntilCleared(ref);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void nodeChildGC() {
|
||||||
|
var node = EventNode.all("main");
|
||||||
|
|
||||||
|
var child = EventNode.all("child");
|
||||||
|
var ref = new WeakReference<>(child);
|
||||||
|
child.addListener(EventTest.class, event -> {
|
||||||
|
});
|
||||||
|
node.addChild(child);
|
||||||
|
|
||||||
|
//noinspection UnusedAssignment
|
||||||
|
child = null;
|
||||||
|
waitUntilCleared(ref);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void nodeMapGC() {
|
||||||
|
var node = EventNode.all("main");
|
||||||
|
|
||||||
|
var handler = ItemStack.AIR;
|
||||||
|
var mapped = node.map(handler, EventFilter.ITEM);
|
||||||
|
var ref = new WeakReference<>(mapped);
|
||||||
|
mapped.addListener(ItemTestEvent.class, event -> {
|
||||||
|
});
|
||||||
|
|
||||||
|
//noinspection UnusedAssignment
|
||||||
|
mapped = null;
|
||||||
|
waitUntilCleared(ref);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,10 @@
|
|||||||
package net.minestom.server.instance;
|
package net.minestom.server.instance;
|
||||||
|
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
|
import net.minestom.server.event.player.PlayerTickEvent;
|
||||||
import net.minestom.testing.Env;
|
import net.minestom.testing.Env;
|
||||||
import net.minestom.testing.EnvTest;
|
import net.minestom.testing.EnvTest;
|
||||||
import net.minestom.server.coordinate.Pos;
|
|
||||||
import net.minestom.server.event.player.PlayerTickEvent;
|
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
import java.lang.ref.WeakReference;
|
import java.lang.ref.WeakReference;
|
||||||
@ -46,6 +47,25 @@ public class InstanceUnregisterIntegrationTest {
|
|||||||
waitUntilCleared(ref);
|
waitUntilCleared(ref);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void instanceNodeGC(Env env) {
|
||||||
|
final class Game {
|
||||||
|
final Instance instance;
|
||||||
|
|
||||||
|
Game(Env env) {
|
||||||
|
instance = env.process().instance().createInstanceContainer();
|
||||||
|
instance.eventNode().addListener(PlayerMoveEvent.class, e -> System.out.println(instance));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var game = new Game(env);
|
||||||
|
var ref = new WeakReference<>(game);
|
||||||
|
env.process().instance().unregisterInstance(game.instance);
|
||||||
|
|
||||||
|
//noinspection UnusedAssignment
|
||||||
|
game = null;
|
||||||
|
waitUntilCleared(ref);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void chunkGC(Env env) {
|
public void chunkGC(Env env) {
|
||||||
// Ensure that unregistering an instance does release its chunks
|
// Ensure that unregistering an instance does release its chunks
|
||||||
|
@ -6,18 +6,31 @@ import org.jglrxavpok.hephaistos.parser.SNBTParser;
|
|||||||
|
|
||||||
import java.io.StringReader;
|
import java.io.StringReader;
|
||||||
import java.lang.ref.WeakReference;
|
import java.lang.ref.WeakReference;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
import static org.junit.jupiter.api.Assertions.*;
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
public class TestUtils {
|
public final class TestUtils {
|
||||||
public static void waitUntilCleared(WeakReference<?> ref) {
|
public static void waitUntilCleared(WeakReference<?> ref) {
|
||||||
while (ref.get() != null) {
|
final int maxTries = 100;
|
||||||
|
|
||||||
|
for (int i = 0; i < maxTries; i++) {
|
||||||
System.gc();
|
System.gc();
|
||||||
|
if (ref.get() == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
Thread.sleep(50);
|
Thread.sleep(10);
|
||||||
} catch (InterruptedException ignore) {
|
} catch (InterruptedException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
fail("Reference was not cleared");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T> void assertEqualsIgnoreOrder(Collection<T> expected, Collection<? extends T> actual) {
|
||||||
|
assertEquals(Set.copyOf(expected), Set.copyOf(actual));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void assertEqualsSNBT(String snbt, NBTCompound compound) {
|
public static void assertEqualsSNBT(String snbt, NBTCompound compound) {
|
||||||
|
Loading…
Reference in New Issue
Block a user