Fix shutdown hang (#881)
This commit is contained in:
parent
0e69c8902d
commit
cbeaaca7af
@ -116,15 +116,6 @@ public class BukkitSchedulerAdapter implements SchedulerAdapter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public ExecutorService asyncFallback() {
|
|
||||||
return this.asyncFallback;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Executor asyncBukkit() {
|
|
||||||
return this.asyncBukkit;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Executor sync() {
|
public Executor sync() {
|
||||||
return this.sync;
|
return this.sync;
|
||||||
@ -135,6 +126,11 @@ public class BukkitSchedulerAdapter implements SchedulerAdapter {
|
|||||||
return this.async;
|
return this.async;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Executor platformAsync() {
|
||||||
|
return this.asyncBukkit;
|
||||||
|
}
|
||||||
|
|
||||||
public void setUseFallback(boolean useFallback) {
|
public void setUseFallback(boolean useFallback) {
|
||||||
this.useFallback = useFallback;
|
this.useFallback = useFallback;
|
||||||
}
|
}
|
||||||
|
@ -102,8 +102,8 @@ public abstract class AbstractLuckPermsPlugin implements LuckPermsPlugin {
|
|||||||
displayBanner(getConsoleSender());
|
displayBanner(getConsoleSender());
|
||||||
|
|
||||||
// load some utilities early
|
// load some utilities early
|
||||||
this.verboseHandler = new VerboseHandler(getBootstrap().getScheduler().platformAsync());
|
this.verboseHandler = new VerboseHandler();
|
||||||
this.permissionVault = new PermissionVault(getBootstrap().getScheduler().platformAsync());
|
this.permissionVault = new PermissionVault();
|
||||||
this.logDispatcher = new LogDispatcher(this);
|
this.logDispatcher = new LogDispatcher(this);
|
||||||
|
|
||||||
// load configuration
|
// load configuration
|
||||||
@ -194,8 +194,8 @@ public abstract class AbstractLuckPermsPlugin implements LuckPermsPlugin {
|
|||||||
performEarlyDisableTasks();
|
performEarlyDisableTasks();
|
||||||
|
|
||||||
// shutdown permission vault and verbose handler tasks
|
// shutdown permission vault and verbose handler tasks
|
||||||
this.permissionVault.shutdown();
|
this.permissionVault.stop();
|
||||||
this.verboseHandler.shutdown();
|
this.verboseHandler.stop();
|
||||||
|
|
||||||
// remove any hooks into the platform
|
// remove any hooks into the platform
|
||||||
removePlatformHooks();
|
removePlatformHooks();
|
||||||
|
@ -28,6 +28,8 @@ package me.lucko.luckperms.common.treeview;
|
|||||||
import com.google.common.base.Splitter;
|
import com.google.common.base.Splitter;
|
||||||
import com.google.common.collect.ImmutableSet;
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
|
||||||
|
import me.lucko.luckperms.common.utils.RepeatingTask;
|
||||||
|
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
@ -35,13 +37,13 @@ import java.util.Queue;
|
|||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.concurrent.ConcurrentLinkedQueue;
|
import java.util.concurrent.ConcurrentLinkedQueue;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stores a collection of all permissions known to the platform.
|
* Stores a collection of all permissions known to the platform.
|
||||||
*/
|
*/
|
||||||
public class PermissionVault implements Runnable {
|
public class PermissionVault extends RepeatingTask {
|
||||||
private static final Splitter DOT_SPLIT = Splitter.on('.').omitEmptyStrings();
|
private static final Splitter DOT_SPLIT = Splitter.on('.').omitEmptyStrings();
|
||||||
|
|
||||||
// the root node in the tree
|
// the root node in the tree
|
||||||
@ -53,15 +55,11 @@ public class PermissionVault implements Runnable {
|
|||||||
// a queue of permission strings to be processed by the tree
|
// a queue of permission strings to be processed by the tree
|
||||||
private final Queue<String> queue;
|
private final Queue<String> queue;
|
||||||
|
|
||||||
// if the handler should shutdown
|
public PermissionVault() {
|
||||||
private boolean shutdown = false;
|
super(1000, TimeUnit.MILLISECONDS, "luckperms-permission-vault");
|
||||||
|
|
||||||
public PermissionVault(Executor executor) {
|
|
||||||
this.rootNode = new TreeNode();
|
this.rootNode = new TreeNode();
|
||||||
this.knownPermissions = ConcurrentHashMap.newKeySet(3000);
|
this.knownPermissions = ConcurrentHashMap.newKeySet(3000);
|
||||||
this.queue = new ConcurrentLinkedQueue<>();
|
this.queue = new ConcurrentLinkedQueue<>();
|
||||||
|
|
||||||
executor.execute(this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public TreeNode getRootNode() {
|
public TreeNode getRootNode() {
|
||||||
@ -69,27 +67,17 @@ public class PermissionVault implements Runnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
protected void tick() {
|
||||||
while (true) {
|
for (String e; (e = this.queue.poll()) != null; ) {
|
||||||
for (String e; (e = this.queue.poll()) != null; ) {
|
|
||||||
try {
|
|
||||||
String s = e.toLowerCase();
|
|
||||||
// only attempt an insert if we're not seen this permission before
|
|
||||||
if (this.knownPermissions.add(s)) {
|
|
||||||
insert(s);
|
|
||||||
}
|
|
||||||
} catch (Exception ex) {
|
|
||||||
ex.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.shutdown) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Thread.sleep(5000);
|
String s = e.toLowerCase();
|
||||||
} catch (InterruptedException ignored) {}
|
// only attempt an insert if we're not seen this permission before
|
||||||
|
if (this.knownPermissions.add(s)) {
|
||||||
|
insert(s);
|
||||||
|
}
|
||||||
|
} catch (Exception ex) {
|
||||||
|
ex.printStackTrace();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -127,8 +115,4 @@ public class PermissionVault implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void shutdown() {
|
|
||||||
this.shutdown = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,54 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of LuckPerms, licensed under the MIT License.
|
||||||
|
*
|
||||||
|
* Copyright (c) lucko (Luck) <luck@lucko.me>
|
||||||
|
* Copyright (c) contributors
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package me.lucko.luckperms.common.utils;
|
||||||
|
|
||||||
|
import com.google.common.util.concurrent.ThreadFactoryBuilder;
|
||||||
|
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
import java.util.concurrent.ScheduledExecutorService;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
public abstract class RepeatingTask {
|
||||||
|
|
||||||
|
// the executor thread
|
||||||
|
private final ScheduledExecutorService executor;
|
||||||
|
|
||||||
|
protected RepeatingTask(long time, TimeUnit unit, String nameFormat) {
|
||||||
|
this.executor = Executors.newSingleThreadScheduledExecutor(new ThreadFactoryBuilder().setNameFormat(nameFormat).build());
|
||||||
|
this.executor.scheduleAtFixedRate(this::tick, time, time, unit);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract void tick();
|
||||||
|
|
||||||
|
public void stop() {
|
||||||
|
this.executor.shutdown();
|
||||||
|
try {
|
||||||
|
this.executor.awaitTermination(10, TimeUnit.SECONDS);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -28,18 +28,19 @@ package me.lucko.luckperms.common.verbose;
|
|||||||
import me.lucko.luckperms.api.Tristate;
|
import me.lucko.luckperms.api.Tristate;
|
||||||
import me.lucko.luckperms.api.context.ContextSet;
|
import me.lucko.luckperms.api.context.ContextSet;
|
||||||
import me.lucko.luckperms.common.sender.Sender;
|
import me.lucko.luckperms.common.sender.Sender;
|
||||||
|
import me.lucko.luckperms.common.utils.RepeatingTask;
|
||||||
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Queue;
|
import java.util.Queue;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.concurrent.ConcurrentLinkedQueue;
|
import java.util.concurrent.ConcurrentLinkedQueue;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Accepts {@link CheckData} and passes it onto registered {@link VerboseListener}s.
|
* Accepts {@link CheckData} and passes it onto registered {@link VerboseListener}s.
|
||||||
*/
|
*/
|
||||||
public class VerboseHandler implements Runnable {
|
public class VerboseHandler extends RepeatingTask {
|
||||||
|
|
||||||
// the listeners currently registered
|
// the listeners currently registered
|
||||||
private final Map<UUID, VerboseListener> listeners;
|
private final Map<UUID, VerboseListener> listeners;
|
||||||
@ -50,14 +51,10 @@ public class VerboseHandler implements Runnable {
|
|||||||
// if there are any listeners currently registered
|
// if there are any listeners currently registered
|
||||||
private boolean listening = false;
|
private boolean listening = false;
|
||||||
|
|
||||||
// if the handler should shutdown
|
public VerboseHandler() {
|
||||||
private boolean shutdown = false;
|
super(100, TimeUnit.MILLISECONDS, "luckperms-verbose");
|
||||||
|
|
||||||
public VerboseHandler(Executor executor) {
|
|
||||||
this.listeners = new ConcurrentHashMap<>();
|
this.listeners = new ConcurrentHashMap<>();
|
||||||
this.queue = new ConcurrentLinkedQueue<>();
|
this.queue = new ConcurrentLinkedQueue<>();
|
||||||
|
|
||||||
executor.execute(this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -111,27 +108,15 @@ public class VerboseHandler implements Runnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
protected void tick() {
|
||||||
while (true) {
|
// remove listeners where the sender is no longer valid
|
||||||
|
this.listeners.values().removeIf(l -> !l.getNotifiedSender().isValid());
|
||||||
|
|
||||||
// remove listeners where the sender is no longer valid
|
// handle all checks in the queue
|
||||||
this.listeners.values().removeIf(l -> !l.getNotifiedSender().isValid());
|
flush();
|
||||||
|
|
||||||
// handle all checks in the queue
|
// update listening state
|
||||||
flush();
|
this.listening = !this.listeners.isEmpty();
|
||||||
|
|
||||||
// break the loop if the handler has been shutdown
|
|
||||||
if (this.shutdown) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// update listening state
|
|
||||||
this.listening = !this.listeners.isEmpty();
|
|
||||||
|
|
||||||
try {
|
|
||||||
Thread.sleep(100);
|
|
||||||
} catch (InterruptedException ignored) {}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -144,8 +129,4 @@ public class VerboseHandler implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void shutdown() {
|
|
||||||
this.shutdown = true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -116,15 +116,6 @@ public class NukkitSchedulerAdapter implements SchedulerAdapter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public ExecutorService asyncFallback() {
|
|
||||||
return this.asyncFallback;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Executor asyncNukkit() {
|
|
||||||
return this.asyncNukkit;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Executor sync() {
|
public Executor sync() {
|
||||||
return this.sync;
|
return this.sync;
|
||||||
@ -135,6 +126,11 @@ public class NukkitSchedulerAdapter implements SchedulerAdapter {
|
|||||||
return this.async;
|
return this.async;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Executor platformAsync() {
|
||||||
|
return this.asyncNukkit;
|
||||||
|
}
|
||||||
|
|
||||||
public void setUseFallback(boolean useFallback) {
|
public void setUseFallback(boolean useFallback) {
|
||||||
this.useFallback = useFallback;
|
this.useFallback = useFallback;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user