mirror of
https://github.com/LuckPerms/LuckPerms.git
synced 2025-01-01 05:57:51 +01:00
Log verbose checks for the console, commandblocks & entities when running on Bukkit
This commit is contained in:
parent
b800db03fc
commit
5ae90f2a4b
@ -36,6 +36,7 @@ import me.lucko.luckperms.bukkit.listeners.BukkitPlatformListener;
|
|||||||
import me.lucko.luckperms.bukkit.messaging.BukkitMessagingFactory;
|
import me.lucko.luckperms.bukkit.messaging.BukkitMessagingFactory;
|
||||||
import me.lucko.luckperms.bukkit.model.LPPermissible;
|
import me.lucko.luckperms.bukkit.model.LPPermissible;
|
||||||
import me.lucko.luckperms.bukkit.model.PermissibleInjector;
|
import me.lucko.luckperms.bukkit.model.PermissibleInjector;
|
||||||
|
import me.lucko.luckperms.bukkit.model.PermissibleMonitoringInjector;
|
||||||
import me.lucko.luckperms.bukkit.model.SubscriptionMapInjector;
|
import me.lucko.luckperms.bukkit.model.SubscriptionMapInjector;
|
||||||
import me.lucko.luckperms.bukkit.processors.BukkitProcessorsSetupTask;
|
import me.lucko.luckperms.bukkit.processors.BukkitProcessorsSetupTask;
|
||||||
import me.lucko.luckperms.bukkit.processors.ChildPermissionProvider;
|
import me.lucko.luckperms.bukkit.processors.ChildPermissionProvider;
|
||||||
@ -256,6 +257,9 @@ public class LPBukkitPlugin extends JavaPlugin implements LuckPermsPlugin {
|
|||||||
// is replaced by some plugins :(
|
// is replaced by some plugins :(
|
||||||
this.scheduler.asyncLater(new SubscriptionMapInjector(this), 2L);
|
this.scheduler.asyncLater(new SubscriptionMapInjector(this), 2L);
|
||||||
|
|
||||||
|
// inject verbose handlers into internal bukkit objects
|
||||||
|
new PermissibleMonitoringInjector(this).run();
|
||||||
|
|
||||||
// Provide vault support
|
// Provide vault support
|
||||||
tryVaultHook(false);
|
tryVaultHook(false);
|
||||||
|
|
||||||
|
@ -45,7 +45,7 @@ public final class ReflectionUtil {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static String getServerVersion() {
|
private static String getServerVersion() {
|
||||||
return SERVER_VERSION;
|
return SERVER_VERSION;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,14 +31,52 @@ import org.bukkit.permissions.PermissionAttachment;
|
|||||||
import org.bukkit.permissions.PermissionAttachmentInfo;
|
import org.bukkit.permissions.PermissionAttachmentInfo;
|
||||||
import org.bukkit.plugin.Plugin;
|
import org.bukkit.plugin.Plugin;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
public class DummyPermissibleBase extends PermissibleBase {
|
public class DummyPermissibleBase extends PermissibleBase {
|
||||||
|
private static final Field ATTACHMENTS_FIELD;
|
||||||
|
private static final Field PERMISSIONS_FIELD;
|
||||||
|
|
||||||
|
static {
|
||||||
|
Field attachmentsField;
|
||||||
|
try {
|
||||||
|
attachmentsField = PermissibleBase.class.getDeclaredField("attachments");
|
||||||
|
attachmentsField.setAccessible(true);
|
||||||
|
} catch (NoSuchFieldException e) {
|
||||||
|
throw new ExceptionInInitializerError(e);
|
||||||
|
}
|
||||||
|
ATTACHMENTS_FIELD = attachmentsField;
|
||||||
|
|
||||||
|
Field permissionsField;
|
||||||
|
try {
|
||||||
|
permissionsField = PermissibleBase.class.getDeclaredField("permissions");
|
||||||
|
permissionsField.setAccessible(true);
|
||||||
|
} catch (NoSuchFieldException e) {
|
||||||
|
throw new ExceptionInInitializerError(e);
|
||||||
|
}
|
||||||
|
PERMISSIONS_FIELD = permissionsField;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void nullFields(PermissibleBase permissibleBase) {
|
||||||
|
try {
|
||||||
|
ATTACHMENTS_FIELD.set(permissibleBase, null);
|
||||||
|
} catch (Exception e) {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
PERMISSIONS_FIELD.set(permissibleBase, null);
|
||||||
|
} catch (Exception e) {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static final DummyPermissibleBase INSTANCE = new DummyPermissibleBase();
|
public static final DummyPermissibleBase INSTANCE = new DummyPermissibleBase();
|
||||||
|
|
||||||
private DummyPermissibleBase() {
|
private DummyPermissibleBase() {
|
||||||
super(null);
|
super(null);
|
||||||
|
nullFields(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public boolean isOp() { return false; }
|
@Override public boolean isOp() { return false; }
|
||||||
|
@ -64,7 +64,7 @@ public class LPPermissionAttachment extends PermissionAttachment {
|
|||||||
permissionAttachmentPermissionsField = PermissionAttachment.class.getDeclaredField("permissions");
|
permissionAttachmentPermissionsField = PermissionAttachment.class.getDeclaredField("permissions");
|
||||||
permissionAttachmentPermissionsField.setAccessible(true);
|
permissionAttachmentPermissionsField.setAccessible(true);
|
||||||
} catch (NoSuchFieldException e) {
|
} catch (NoSuchFieldException e) {
|
||||||
throw new RuntimeException(e);
|
throw new ExceptionInInitializerError(e);
|
||||||
}
|
}
|
||||||
PERMISSION_ATTACHMENT_PERMISSIONS_FIELD = permissionAttachmentPermissionsField;
|
PERMISSION_ATTACHMENT_PERMISSIONS_FIELD = permissionAttachmentPermissionsField;
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,143 @@
|
|||||||
|
/*
|
||||||
|
* 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.bukkit.model;
|
||||||
|
|
||||||
|
import me.lucko.luckperms.api.Tristate;
|
||||||
|
import me.lucko.luckperms.api.context.ContextSet;
|
||||||
|
import me.lucko.luckperms.common.verbose.CheckOrigin;
|
||||||
|
import me.lucko.luckperms.common.verbose.VerboseHandler;
|
||||||
|
|
||||||
|
import org.bukkit.permissions.PermissibleBase;
|
||||||
|
import org.bukkit.permissions.Permission;
|
||||||
|
import org.bukkit.permissions.PermissionAttachment;
|
||||||
|
import org.bukkit.permissions.PermissionAttachmentInfo;
|
||||||
|
import org.bukkit.plugin.Plugin;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A PermissibleBase extension which logs permission checks to the
|
||||||
|
* plugin's {@link VerboseHandler} facility.
|
||||||
|
*
|
||||||
|
* Method calls are forwarded to the delegate permissible.
|
||||||
|
*/
|
||||||
|
public class MonitoredPermissibleBase extends PermissibleBase {
|
||||||
|
private final VerboseHandler verboseHandler;
|
||||||
|
private final PermissibleBase delegate;
|
||||||
|
private final String name;
|
||||||
|
|
||||||
|
// remains false until the object has been constructed
|
||||||
|
// necessary to catch the superclass call to #recalculatePermissions on init
|
||||||
|
@SuppressWarnings("UnusedAssignment")
|
||||||
|
private boolean initialised = false;
|
||||||
|
|
||||||
|
public MonitoredPermissibleBase(VerboseHandler verboseHandler, PermissibleBase delegate, String name) {
|
||||||
|
super(null);
|
||||||
|
DummyPermissibleBase.nullFields(this);
|
||||||
|
|
||||||
|
this.verboseHandler = verboseHandler;
|
||||||
|
this.delegate = delegate;
|
||||||
|
this.name = name;
|
||||||
|
this.initialised = true;
|
||||||
|
|
||||||
|
// since we effectively cancel the execution of this call in the super
|
||||||
|
// constructor we need to call it again.
|
||||||
|
recalculatePermissions();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void logCheck(CheckOrigin origin, String permission, boolean result) {
|
||||||
|
this.verboseHandler.offerCheckData(origin, this.name, ContextSet.empty(), permission, Tristate.fromBoolean(result));
|
||||||
|
}
|
||||||
|
|
||||||
|
PermissibleBase getDelegate() {
|
||||||
|
return this.delegate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isPermissionSet(String permission) {
|
||||||
|
if (permission == null) {
|
||||||
|
throw new NullPointerException("permission");
|
||||||
|
}
|
||||||
|
|
||||||
|
final boolean result = this.delegate.isPermissionSet(permission);
|
||||||
|
logCheck(CheckOrigin.PLATFORM_LOOKUP_CHECK, permission, result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isPermissionSet(Permission permission) {
|
||||||
|
if (permission == null) {
|
||||||
|
throw new NullPointerException("permission");
|
||||||
|
}
|
||||||
|
|
||||||
|
final boolean result = this.delegate.isPermissionSet(permission);
|
||||||
|
logCheck(CheckOrigin.PLATFORM_LOOKUP_CHECK, permission.getName(), result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasPermission(String permission) {
|
||||||
|
if (permission == null) {
|
||||||
|
throw new NullPointerException("permission");
|
||||||
|
}
|
||||||
|
|
||||||
|
final boolean result = this.delegate.hasPermission(permission);
|
||||||
|
logCheck(CheckOrigin.PLATFORM_PERMISSION_CHECK, permission, result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasPermission(Permission permission) {
|
||||||
|
if (permission == null) {
|
||||||
|
throw new NullPointerException("permission");
|
||||||
|
}
|
||||||
|
|
||||||
|
final boolean result = this.delegate.hasPermission(permission);
|
||||||
|
logCheck(CheckOrigin.PLATFORM_PERMISSION_CHECK, permission.getName(), result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void recalculatePermissions() {
|
||||||
|
if (!this.initialised) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.delegate.recalculatePermissions();
|
||||||
|
}
|
||||||
|
|
||||||
|
// just forward calls to the delegate permissible
|
||||||
|
@Override public boolean isOp() { return this.delegate.isOp(); }
|
||||||
|
@Override public void setOp(boolean value) { this.delegate.setOp(value); }
|
||||||
|
@Override public PermissionAttachment addAttachment(Plugin plugin, String name, boolean value) { return this.delegate.addAttachment(plugin, name, value); }
|
||||||
|
@Override public PermissionAttachment addAttachment(Plugin plugin) { return this.delegate.addAttachment(plugin); }
|
||||||
|
@Override public void removeAttachment(PermissionAttachment attachment) { this.delegate.removeAttachment(attachment); }
|
||||||
|
@Override public void clearPermissions() { this.delegate.clearPermissions(); }
|
||||||
|
@Override public PermissionAttachment addAttachment(Plugin plugin, String name, boolean value, int ticks) { return this.delegate.addAttachment(plugin, name, value, ticks); }
|
||||||
|
@Override public PermissionAttachment addAttachment(Plugin plugin, int ticks) { return this.delegate.addAttachment(plugin, ticks); }
|
||||||
|
@Override public Set<PermissionAttachmentInfo> getEffectivePermissions() { return this.delegate.getEffectivePermissions(); }
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,162 @@
|
|||||||
|
/*
|
||||||
|
* 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.bukkit.model;
|
||||||
|
|
||||||
|
import me.lucko.luckperms.bukkit.LPBukkitPlugin;
|
||||||
|
import me.lucko.luckperms.bukkit.compat.ReflectionUtil;
|
||||||
|
|
||||||
|
import org.bukkit.command.ConsoleCommandSender;
|
||||||
|
import org.bukkit.permissions.PermissibleBase;
|
||||||
|
import org.bukkit.permissions.ServerOperator;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injects {@link MonitoredPermissibleBase}s into non-player permissibles on
|
||||||
|
* the server so their checks can be monitored by the verbose facility.
|
||||||
|
*/
|
||||||
|
public class PermissibleMonitoringInjector implements Runnable {
|
||||||
|
private final LPBukkitPlugin plugin;
|
||||||
|
|
||||||
|
public PermissibleMonitoringInjector(LPBukkitPlugin plugin) {
|
||||||
|
this.plugin = plugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
injectConsole();
|
||||||
|
} catch (Exception e) {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
injectCommandBlock();
|
||||||
|
} catch (Exception e) {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
injectEntity();
|
||||||
|
} catch (Exception e) {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private MonitoredPermissibleBase wrap(PermissibleBase permBase, String name) {
|
||||||
|
Objects.requireNonNull(permBase, "permBase");
|
||||||
|
|
||||||
|
// unwrap any previous injection
|
||||||
|
if (permBase instanceof MonitoredPermissibleBase) {
|
||||||
|
permBase = ((MonitoredPermissibleBase) permBase).getDelegate();
|
||||||
|
}
|
||||||
|
|
||||||
|
// create a monitored instance which delegates to the previous PermissibleBase
|
||||||
|
return new MonitoredPermissibleBase(this.plugin.getVerboseHandler(), permBase, name);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void injectConsole() throws Exception {
|
||||||
|
ConsoleCommandSender consoleSender = this.plugin.getServer().getConsoleSender();
|
||||||
|
|
||||||
|
// get the ServerCommandSender class
|
||||||
|
Class<?> serverCommandSenderClass = ReflectionUtil.obcClass("command.ServerCommandSender");
|
||||||
|
|
||||||
|
// get the perm field
|
||||||
|
Field permField = serverCommandSenderClass.getDeclaredField("perm");
|
||||||
|
permField.setAccessible(true);
|
||||||
|
|
||||||
|
// get the PermissibleBase instance
|
||||||
|
PermissibleBase permBase = (PermissibleBase) permField.get(consoleSender);
|
||||||
|
|
||||||
|
// create a monitored instance which delegates to the previous PermissibleBase
|
||||||
|
MonitoredPermissibleBase newPermBase = wrap(permBase, "internal/console");
|
||||||
|
|
||||||
|
// inject the monitored instance
|
||||||
|
permField.set(consoleSender, newPermBase);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void injectCommandBlock() throws Exception {
|
||||||
|
// get the ServerCommandSender class
|
||||||
|
Class<?> serverCommandSenderClass = ReflectionUtil.obcClass("command.ServerCommandSender");
|
||||||
|
|
||||||
|
// get the blockPermInst field
|
||||||
|
Field permField = serverCommandSenderClass.getDeclaredField("blockPermInst");
|
||||||
|
permField.setAccessible(true);
|
||||||
|
|
||||||
|
// get the PermissibleBase instance
|
||||||
|
PermissibleBase permBase = (PermissibleBase) permField.get(null);
|
||||||
|
|
||||||
|
// if no commandblock senders have been made yet, this field will be null
|
||||||
|
// we can just initialise one anyway
|
||||||
|
if (permBase == null) {
|
||||||
|
permBase = new PermissibleBase(new CommandBlockServerOperator());
|
||||||
|
}
|
||||||
|
|
||||||
|
// create a monitored instance which delegates to the previous PermissibleBase
|
||||||
|
MonitoredPermissibleBase newPermBase = wrap(permBase, "internal/commandblock");
|
||||||
|
|
||||||
|
// inject the monitored instance
|
||||||
|
permField.set(null, newPermBase);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void injectEntity() throws Exception {
|
||||||
|
// get the CraftEntity class
|
||||||
|
Class<?> entityClass = ReflectionUtil.obcClass("entity.CraftEntity");
|
||||||
|
|
||||||
|
// get the method used to obtain a PermissibleBase
|
||||||
|
// this method will initialise a new PB instance if one doesn't yet exist
|
||||||
|
Method getPermissibleBaseMethod = entityClass.getDeclaredMethod("getPermissibleBase");
|
||||||
|
getPermissibleBaseMethod.setAccessible(true);
|
||||||
|
|
||||||
|
// get the PermissibleBase instance
|
||||||
|
PermissibleBase permBase = (PermissibleBase) getPermissibleBaseMethod.invoke(null);
|
||||||
|
|
||||||
|
// get the perm field on CraftEntity
|
||||||
|
Field permField = entityClass.getDeclaredField("perm");
|
||||||
|
permField.setAccessible(true);
|
||||||
|
|
||||||
|
// create a monitored instance which delegates to the previous PermissibleBase
|
||||||
|
MonitoredPermissibleBase newPermBase = wrap(permBase, "internal/entity");
|
||||||
|
|
||||||
|
// inject the monitored instance
|
||||||
|
permField.set(null, newPermBase);
|
||||||
|
}
|
||||||
|
|
||||||
|
// behaviour copied from the implementation of obc.command.CraftBlockCommandSender
|
||||||
|
private static final class CommandBlockServerOperator implements ServerOperator {
|
||||||
|
@Override
|
||||||
|
public boolean isOp() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setOp(boolean value) {
|
||||||
|
throw new UnsupportedOperationException("Cannot change operator status of a block");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -44,7 +44,7 @@ public class SubscriptionMapInjector implements Runnable {
|
|||||||
permSubsField = SimplePluginManager.class.getDeclaredField("permSubs");
|
permSubsField = SimplePluginManager.class.getDeclaredField("permSubs");
|
||||||
permSubsField.setAccessible(true);
|
permSubsField.setAccessible(true);
|
||||||
} catch (NoSuchFieldException e) {
|
} catch (NoSuchFieldException e) {
|
||||||
throw new RuntimeException(e);
|
throw new ExceptionInInitializerError(e);
|
||||||
}
|
}
|
||||||
PERM_SUBS_FIELD = permSubsField;
|
PERM_SUBS_FIELD = permSubsField;
|
||||||
}
|
}
|
||||||
|
@ -34,16 +34,16 @@ public class ApiRegistrationUtil {
|
|||||||
private static final Method REGISTER;
|
private static final Method REGISTER;
|
||||||
private static final Method UNREGISTER;
|
private static final Method UNREGISTER;
|
||||||
static {
|
static {
|
||||||
Method register = null;
|
Method register;
|
||||||
Method unregister = null;
|
Method unregister;
|
||||||
try {
|
try {
|
||||||
register = LuckPerms.class.getDeclaredMethod("registerProvider", LuckPermsApi.class);
|
register = LuckPerms.class.getDeclaredMethod("registerProvider", LuckPermsApi.class);
|
||||||
register.setAccessible(true);
|
register.setAccessible(true);
|
||||||
|
|
||||||
unregister = LuckPerms.class.getDeclaredMethod("unregisterProvider");
|
unregister = LuckPerms.class.getDeclaredMethod("unregisterProvider");
|
||||||
unregister.setAccessible(true);
|
unregister.setAccessible(true);
|
||||||
} catch (Exception e) {
|
} catch (NoSuchMethodException e) {
|
||||||
e.printStackTrace();
|
throw new ExceptionInInitializerError(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
REGISTER = register;
|
REGISTER = register;
|
||||||
|
Loading…
Reference in New Issue
Block a user