mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-09 11:44:19 +01:00
Queue tasks from secondary threads. Fixes BUKKIT-2546 and BUKKIT-2600
This change affects the old chat compatibility layer from an implementation only standpoint. It does not queue the 'event' to fire, but rather queues a runnable that allows the calling thread to wait for execution to finish. The other effect of this change is that rcon connects now have their commands queued to be run on next server tick using the same implementation. The internal implementation is in org.bukkit.craftbukkit.util.Waitable. It is very similar to a Future<T> task, but only contains minimal implementation with object.wait() and object.notify() calls under the hood of waitable.get() and waitable.run(). PlayerPreLoginEvent now properly implements thread-safe event execution by queuing the events similar to chat and rcon. This is still a poor way albeit proper way to implement thread-safety; PlayerPreLoginEvent will stay deprecated. By: Wesley Wolfe <weswolf@aol.com>
This commit is contained in:
parent
dd1816bda4
commit
730e3ceb1b
1 changed files with 46 additions and 0 deletions
|
@ -0,0 +1,46 @@
|
||||||
|
package org.bukkit.craftbukkit.util;
|
||||||
|
|
||||||
|
import java.util.concurrent.ExecutionException;
|
||||||
|
|
||||||
|
|
||||||
|
public abstract class Waitable<T> implements Runnable {
|
||||||
|
private enum Status {
|
||||||
|
WAITING,
|
||||||
|
RUNNING,
|
||||||
|
FINISHED,
|
||||||
|
}
|
||||||
|
Throwable t = null;
|
||||||
|
T value = null;
|
||||||
|
Status status = Status.WAITING;
|
||||||
|
|
||||||
|
public final void run() {
|
||||||
|
synchronized (this) {
|
||||||
|
if (status != Status.WAITING) {
|
||||||
|
throw new IllegalStateException("Invalid state " + status);
|
||||||
|
}
|
||||||
|
status = Status.RUNNING;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
value = evaluate();
|
||||||
|
} catch (Throwable t) {
|
||||||
|
this.t = t;
|
||||||
|
} finally {
|
||||||
|
synchronized (this) {
|
||||||
|
status = Status.FINISHED;
|
||||||
|
this.notifyAll();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract T evaluate();
|
||||||
|
|
||||||
|
public synchronized T get() throws InterruptedException, ExecutionException {
|
||||||
|
while (status != Status.FINISHED) {
|
||||||
|
this.wait();
|
||||||
|
}
|
||||||
|
if (t != null) {
|
||||||
|
throw new ExecutionException(t);
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue