Convert PermissionAttachment to use ConcurrentMap, and do refreshPermissions async
This commit is contained in:
parent
41fca40a36
commit
7a4135319f
@ -30,6 +30,9 @@ import org.bukkit.entity.Player;
|
|||||||
import org.bukkit.event.EventHandler;
|
import org.bukkit.event.EventHandler;
|
||||||
import org.bukkit.event.Listener;
|
import org.bukkit.event.Listener;
|
||||||
import org.bukkit.event.player.*;
|
import org.bukkit.event.player.*;
|
||||||
|
import org.bukkit.permissions.PermissionAttachment;
|
||||||
|
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
class BukkitListener extends AbstractListener implements Listener {
|
class BukkitListener extends AbstractListener implements Listener {
|
||||||
private final LPBukkitPlugin plugin;
|
private final LPBukkitPlugin plugin;
|
||||||
@ -62,7 +65,15 @@ class BukkitListener extends AbstractListener implements Listener {
|
|||||||
|
|
||||||
if (user instanceof BukkitUser) {
|
if (user instanceof BukkitUser) {
|
||||||
BukkitUser u = (BukkitUser) user;
|
BukkitUser u = (BukkitUser) user;
|
||||||
u.setAttachment(player.addAttachment(plugin));
|
|
||||||
|
PermissionAttachment attachment = player.addAttachment(plugin);
|
||||||
|
try {
|
||||||
|
BukkitUser.getPermissionsField().set(attachment, new ConcurrentHashMap<>());
|
||||||
|
} catch (Throwable t) {
|
||||||
|
t.printStackTrace();
|
||||||
|
}
|
||||||
|
|
||||||
|
u.setAttachment(attachment);
|
||||||
}
|
}
|
||||||
|
|
||||||
user.refreshPermissions();
|
user.refreshPermissions();
|
||||||
@ -71,16 +82,19 @@ class BukkitListener extends AbstractListener implements Listener {
|
|||||||
@EventHandler
|
@EventHandler
|
||||||
public void onPlayerJoin(PlayerJoinEvent e) {
|
public void onPlayerJoin(PlayerJoinEvent e) {
|
||||||
// Refresh permissions again
|
// Refresh permissions again
|
||||||
|
plugin.getUserManager().getWorldCache().put(e.getPlayer().getUniqueId(), e.getPlayer().getWorld().getName());
|
||||||
refreshPlayer(e.getPlayer().getUniqueId());
|
refreshPlayer(e.getPlayer().getUniqueId());
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
public void onPlayerChangedWorld(PlayerChangedWorldEvent e) {
|
public void onPlayerChangedWorld(PlayerChangedWorldEvent e) {
|
||||||
|
plugin.getUserManager().getWorldCache().put(e.getPlayer().getUniqueId(), e.getPlayer().getWorld().getName());
|
||||||
refreshPlayer(e.getPlayer().getUniqueId());
|
refreshPlayer(e.getPlayer().getUniqueId());
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
public void onPlayerQuit(PlayerQuitEvent e) {
|
public void onPlayerQuit(PlayerQuitEvent e) {
|
||||||
|
plugin.getUserManager().getWorldCache().remove(e.getPlayer().getUniqueId());
|
||||||
onLeave(e.getPlayer().getUniqueId());
|
onLeave(e.getPlayer().getUniqueId());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -39,7 +39,6 @@ import me.lucko.luckperms.storage.Datastore;
|
|||||||
import me.lucko.luckperms.storage.StorageFactory;
|
import me.lucko.luckperms.storage.StorageFactory;
|
||||||
import me.lucko.luckperms.tracks.TrackManager;
|
import me.lucko.luckperms.tracks.TrackManager;
|
||||||
import me.lucko.luckperms.users.BukkitUserManager;
|
import me.lucko.luckperms.users.BukkitUserManager;
|
||||||
import me.lucko.luckperms.users.UserManager;
|
|
||||||
import me.lucko.luckperms.utils.LogFactory;
|
import me.lucko.luckperms.utils.LogFactory;
|
||||||
import org.bukkit.command.PluginCommand;
|
import org.bukkit.command.PluginCommand;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
@ -49,13 +48,14 @@ import org.bukkit.plugin.java.JavaPlugin;
|
|||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
@Getter
|
@Getter
|
||||||
public class LPBukkitPlugin extends JavaPlugin implements LuckPermsPlugin {
|
public class LPBukkitPlugin extends JavaPlugin implements LuckPermsPlugin {
|
||||||
private final Set<UUID> ignoringLogs = new HashSet<>();
|
private final Set<UUID> ignoringLogs = ConcurrentHashMap.newKeySet();
|
||||||
private LPConfiguration configuration;
|
private LPConfiguration configuration;
|
||||||
private UserManager userManager;
|
private BukkitUserManager userManager;
|
||||||
private GroupManager groupManager;
|
private GroupManager groupManager;
|
||||||
private TrackManager trackManager;
|
private TrackManager trackManager;
|
||||||
private Datastore datastore;
|
private Datastore datastore;
|
||||||
|
@ -27,7 +27,6 @@ import lombok.Setter;
|
|||||||
import me.lucko.luckperms.LPBukkitPlugin;
|
import me.lucko.luckperms.LPBukkitPlugin;
|
||||||
import me.lucko.luckperms.api.event.events.UserPermissionRefreshEvent;
|
import me.lucko.luckperms.api.event.events.UserPermissionRefreshEvent;
|
||||||
import me.lucko.luckperms.api.implementation.internal.UserLink;
|
import me.lucko.luckperms.api.implementation.internal.UserLink;
|
||||||
import org.bukkit.entity.Player;
|
|
||||||
import org.bukkit.permissions.PermissionAttachment;
|
import org.bukkit.permissions.PermissionAttachment;
|
||||||
|
|
||||||
import java.lang.reflect.Field;
|
import java.lang.reflect.Field;
|
||||||
@ -36,7 +35,7 @@ import java.util.UUID;
|
|||||||
|
|
||||||
public class BukkitUser extends User {
|
public class BukkitUser extends User {
|
||||||
private static Field permissionsField = null;
|
private static Field permissionsField = null;
|
||||||
private static Field getPermissionsField() {
|
public static Field getPermissionsField() {
|
||||||
if (permissionsField == null) {
|
if (permissionsField == null) {
|
||||||
try {
|
try {
|
||||||
permissionsField = PermissionAttachment.class.getDeclaredField("permissions");
|
permissionsField = PermissionAttachment.class.getDeclaredField("permissions");
|
||||||
@ -68,46 +67,42 @@ public class BukkitUser extends User {
|
|||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
@Override
|
@Override
|
||||||
public void refreshPermissions() {
|
public void refreshPermissions() {
|
||||||
plugin.doSync(() -> {
|
if (attachment == null) {
|
||||||
final Player player = plugin.getServer().getPlayer(plugin.getUuidCache().getExternalUUID(getUuid()));
|
getPlugin().getLog().severe("User " + getName() + " does not have a permissions attachment defined.");
|
||||||
if (player == null) return;
|
}
|
||||||
|
|
||||||
if (attachment == null) {
|
// Calculate the permissions that should be applied
|
||||||
getPlugin().getLog().warn("User " + getName() + " does not have a permissions attachment defined.");
|
Map<String, Boolean> toApply = getLocalPermissions(getPlugin().getConfiguration().getServer(), plugin.getUserManager().getWorldCache().get(getUuid()), null);
|
||||||
setAttachment(player.addAttachment(plugin));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calculate the permissions that should be applied
|
try {
|
||||||
Map<String, Boolean> toApply = getLocalPermissions(getPlugin().getConfiguration().getServer(), player.getWorld().getName(), null);
|
// Existing is thread-safe, hopefully
|
||||||
|
Map<String, Boolean> existing = (Map<String, Boolean>) getPermissionsField().get(attachment);
|
||||||
|
|
||||||
try {
|
boolean different = false;
|
||||||
Map<String, Boolean> existing = (Map<String, Boolean>) getPermissionsField().get(attachment);
|
if (toApply.size() != existing.size()) {
|
||||||
|
different = true;
|
||||||
boolean different = false;
|
} else {
|
||||||
if (toApply.size() != existing.size()) {
|
for (Map.Entry<String, Boolean> e : existing.entrySet()) {
|
||||||
different = true;
|
if (toApply.containsKey(e.getKey()) && toApply.get(e.getKey()) == e.getValue()) {
|
||||||
} else {
|
continue;
|
||||||
for (Map.Entry<String, Boolean> e : existing.entrySet()) {
|
|
||||||
if (toApply.containsKey(e.getKey()) && toApply.get(e.getKey()) == e.getValue()) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
different = true;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
different = true;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!different) return;
|
|
||||||
|
|
||||||
// Faster than recalculating permissions after each PermissionAttachment#setPermission
|
|
||||||
existing.clear();
|
|
||||||
existing.putAll(toApply);
|
|
||||||
attachment.getPermissible().recalculatePermissions();
|
|
||||||
|
|
||||||
} catch (Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
plugin.getApiProvider().fireEventAsync(new UserPermissionRefreshEvent(new UserLink(this)));
|
if (!different) return;
|
||||||
});
|
|
||||||
|
// Faster than recalculating permissions after each PermissionAttachment#setPermission
|
||||||
|
existing.clear();
|
||||||
|
existing.putAll(toApply);
|
||||||
|
|
||||||
|
attachment.getPermissible().recalculatePermissions();
|
||||||
|
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
|
||||||
|
plugin.getApiProvider().fireEventAsync(new UserPermissionRefreshEvent(new UserLink(this)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -22,16 +22,22 @@
|
|||||||
|
|
||||||
package me.lucko.luckperms.users;
|
package me.lucko.luckperms.users;
|
||||||
|
|
||||||
|
import lombok.Getter;
|
||||||
import me.lucko.luckperms.LPBukkitPlugin;
|
import me.lucko.luckperms.LPBukkitPlugin;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
public class BukkitUserManager extends UserManager {
|
public class BukkitUserManager extends UserManager {
|
||||||
private final LPBukkitPlugin plugin;
|
private final LPBukkitPlugin plugin;
|
||||||
|
|
||||||
|
@Getter
|
||||||
|
private final Map<UUID, String> worldCache = new ConcurrentHashMap<>();
|
||||||
|
|
||||||
public BukkitUserManager(LPBukkitPlugin plugin) {
|
public BukkitUserManager(LPBukkitPlugin plugin) {
|
||||||
super(plugin);
|
super(plugin);
|
||||||
this.plugin = plugin;
|
this.plugin = plugin;
|
||||||
|
@ -44,13 +44,17 @@ import net.md_5.bungee.api.connection.ProxiedPlayer;
|
|||||||
import net.md_5.bungee.api.plugin.Plugin;
|
import net.md_5.bungee.api.plugin.Plugin;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.util.*;
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
@Getter
|
@Getter
|
||||||
public class LPBungeePlugin extends Plugin implements LuckPermsPlugin {
|
public class LPBungeePlugin extends Plugin implements LuckPermsPlugin {
|
||||||
private final Set<UUID> ignoringLogs = new HashSet<>();
|
private final Set<UUID> ignoringLogs = ConcurrentHashMap.newKeySet();
|
||||||
private LPConfiguration configuration;
|
private LPConfiguration configuration;
|
||||||
private UserManager userManager;
|
private UserManager userManager;
|
||||||
private GroupManager groupManager;
|
private GroupManager groupManager;
|
||||||
|
@ -60,6 +60,7 @@ import org.spongepowered.api.text.Text;
|
|||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
@ -79,7 +80,7 @@ public class LPSpongePlugin implements LuckPermsPlugin {
|
|||||||
|
|
||||||
private Scheduler scheduler = Sponge.getScheduler();
|
private Scheduler scheduler = Sponge.getScheduler();
|
||||||
|
|
||||||
private final Set<UUID> ignoringLogs = new HashSet<>();
|
private final Set<UUID> ignoringLogs = ConcurrentHashMap.newKeySet();
|
||||||
private LPConfiguration configuration;
|
private LPConfiguration configuration;
|
||||||
private UserManager userManager;
|
private UserManager userManager;
|
||||||
private GroupManager groupManager;
|
private GroupManager groupManager;
|
||||||
|
Loading…
Reference in New Issue
Block a user