Add some alternate options for primary group calculation

This commit is contained in:
Luck 2017-03-17 22:54:18 +00:00
parent e91c2ca427
commit 2e75bc3f3b
No known key found for this signature in database
GPG Key ID: EFA9B3EC5FD90F8B
32 changed files with 311 additions and 50 deletions

View File

@ -238,7 +238,7 @@ public class MigrationGroupManager extends SubCommand<Object> {
String primaryGroup = primaryGroups.get(e.getKey()); String primaryGroup = primaryGroups.get(e.getKey());
if (primaryGroup != null) { if (primaryGroup != null) {
user.setPermissionUnchecked(NodeFactory.make("group." + primaryGroup)); user.setPermissionUnchecked(NodeFactory.make("group." + primaryGroup));
user.setPrimaryGroup(primaryGroup); user.getPrimaryGroup().setStoredValue(primaryGroup);
user.unsetPermissionUnchecked(NodeFactory.make("group.default")); user.unsetPermissionUnchecked(NodeFactory.make("group.default"));
} }

View File

@ -221,7 +221,7 @@ public class MigrationPermissionsEx extends SubCommand<Object> {
if (primary != null && !primary.equalsIgnoreCase("default")) { if (primary != null && !primary.equalsIgnoreCase("default")) {
lpUser.setPermissionUnchecked(NodeFactory.make("group." + primary.toLowerCase())); lpUser.setPermissionUnchecked(NodeFactory.make("group." + primary.toLowerCase()));
lpUser.setPrimaryGroup(primary); lpUser.getPrimaryGroup().setStoredValue(primary);
lpUser.unsetPermissionUnchecked(NodeFactory.make("group.default")); lpUser.unsetPermissionUnchecked(NodeFactory.make("group.default"));
} }

View File

@ -262,7 +262,7 @@ public class MigrationPowerfulPerms extends SubCommand<Object> {
String primary = joinFuture(pm.getPlayerPrimaryGroup(uuid)).getName().toLowerCase(); String primary = joinFuture(pm.getPlayerPrimaryGroup(uuid)).getName().toLowerCase();
if (!primary.equals("default")) { if (!primary.equals("default")) {
user.setPermissionUnchecked(NodeFactory.make("group." + primary)); user.setPermissionUnchecked(NodeFactory.make("group." + primary));
user.setPrimaryGroup(primary); user.getPrimaryGroup().setStoredValue(primary);
} }
plugin.getUserManager().cleanup(user); plugin.getUserManager().cleanup(user);

View File

@ -132,7 +132,7 @@ public class MigrationZPermissions extends SubCommand<Object> {
plugin.getStorage().loadUser(u, username).join(); plugin.getStorage().loadUser(u, username).join();
User user = plugin.getUserManager().get(u); User user = plugin.getUserManager().get(u);
migrateEntity(user, entity, internalService.getGroups(u)); migrateEntity(user, entity, internalService.getGroups(u));
user.setPrimaryGroup(MigrationUtils.standardizeName(service.getPlayerPrimaryGroup(u))); user.getPrimaryGroup().setStoredValue(MigrationUtils.standardizeName(service.getPlayerPrimaryGroup(u)));
plugin.getUserManager().cleanup(user); plugin.getUserManager().cleanup(user);
plugin.getStorage().saveUser(user); plugin.getStorage().saveUser(user);

View File

@ -323,7 +323,7 @@ public class VaultPermissionHook extends Permission {
// nothing special, just return the value. // nothing special, just return the value.
if (!isPgo()) { if (!isPgo()) {
String g = user.getPrimaryGroup(); String g = user.getPrimaryGroup().getValue();
return plugin.getConfiguration().get(ConfigKeys.GROUP_NAME_REWRITES).getOrDefault(g, g); return plugin.getConfiguration().get(ConfigKeys.GROUP_NAME_REWRITES).getOrDefault(g, g);
} }
@ -376,7 +376,7 @@ public class VaultPermissionHook extends Permission {
} }
// Fallback // Fallback
String g = user.getPrimaryGroup(); String g = user.getPrimaryGroup().getValue();
return plugin.getConfiguration().get(ConfigKeys.GROUP_NAME_REWRITES).getOrDefault(g, g); return plugin.getConfiguration().get(ConfigKeys.GROUP_NAME_REWRITES).getOrDefault(g, g);
} }

View File

@ -56,6 +56,14 @@ log-notify: true
# If "deny": the command will just fail if you try to add another node with the same expiry # If "deny": the command will just fail if you try to add another node with the same expiry
temporary-add-behaviour: deny temporary-add-behaviour: deny
# How should LuckPerms determine a users "primary" group.
#
# Available Options:
# -> stored use the value stored against the users record in the file/database
# -> parents-by-weight just use the users most highly weighted parent
# -> all-parents-by-weight same as above, but calculates based upon all parents inherits from both directly and indirectly
primary-group-calculation: stored

View File

@ -66,6 +66,14 @@ group-name-rewrite:
# If "deny": the command will just fail if you try to add another node with the same expiry # If "deny": the command will just fail if you try to add another node with the same expiry
temporary-add-behaviour: deny temporary-add-behaviour: deny
# How should LuckPerms determine a users "primary" group.
#
# Available Options:
# -> stored use the value stored against the users record in the file/database
# -> parents-by-weight just use the users most highly weighted parent
# -> all-parents-by-weight same as above, but calculates based upon all parents inherits from both directly and indirectly
primary-group-calculation: stored

View File

@ -63,7 +63,7 @@ public final class UserDelegate extends PermissionHolderDelegate implements User
@Override @Override
public String getPrimaryGroup() { public String getPrimaryGroup() {
return master.getPrimaryGroup(); return master.getPrimaryGroup().getValue();
} }
@Override @Override
@ -76,7 +76,7 @@ public final class UserDelegate extends PermissionHolderDelegate implements User
throw new IllegalStateException("User is not a member of that group."); throw new IllegalStateException("User is not a member of that group.");
} }
master.setPrimaryGroup(s.toLowerCase()); master.getPrimaryGroup().setStoredValue(s.toLowerCase());
} }
@Override @Override

View File

@ -29,6 +29,7 @@ import me.lucko.luckperms.common.commands.abstraction.SharedSubCommand;
import me.lucko.luckperms.common.commands.sender.Sender; import me.lucko.luckperms.common.commands.sender.Sender;
import me.lucko.luckperms.common.commands.utils.ArgumentUtils; import me.lucko.luckperms.common.commands.utils.ArgumentUtils;
import me.lucko.luckperms.common.commands.utils.ContextHelper; import me.lucko.luckperms.common.commands.utils.ContextHelper;
import me.lucko.luckperms.common.config.ConfigKeys;
import me.lucko.luckperms.common.constants.Message; import me.lucko.luckperms.common.constants.Message;
import me.lucko.luckperms.common.constants.Permission; import me.lucko.luckperms.common.constants.Permission;
import me.lucko.luckperms.common.core.model.PermissionHolder; import me.lucko.luckperms.common.core.model.PermissionHolder;
@ -68,7 +69,8 @@ public class ParentRemove extends SharedSubCommand {
boolean shouldPrevent = (context == ContextHelper.CommandContext.NONE || boolean shouldPrevent = (context == ContextHelper.CommandContext.NONE ||
(context == ContextHelper.CommandContext.SERVER && server.equalsIgnoreCase("global"))) && (context == ContextHelper.CommandContext.SERVER && server.equalsIgnoreCase("global"))) &&
user.getPrimaryGroup().equalsIgnoreCase(groupName); plugin.getConfiguration().get(ConfigKeys.PRIMARY_GROUP_CALCULATION_METHOD).equals("stored") &&
user.getPrimaryGroup().getStoredValue().equalsIgnoreCase(groupName);
if (shouldPrevent) { if (shouldPrevent) {
Message.USER_REMOVEGROUP_ERROR_PRIMARY.send(sender); Message.USER_REMOVEGROUP_ERROR_PRIMARY.send(sender);

View File

@ -82,7 +82,7 @@ public class ParentSet extends SharedSubCommand {
} catch (ObjectAlreadyHasException ignored) {} } catch (ObjectAlreadyHasException ignored) {}
if (holder instanceof User) { if (holder instanceof User) {
((User) holder).setPrimaryGroup(group.getName()); ((User) holder).getPrimaryGroup().setStoredValue(group.getName());
} }
Message.SET_PARENT_SUCCESS.send(sender, holder.getFriendlyName(), group.getDisplayName()); Message.SET_PARENT_SUCCESS.send(sender, holder.getFriendlyName(), group.getDisplayName());
@ -96,7 +96,7 @@ public class ParentSet extends SharedSubCommand {
} }
if (server.equalsIgnoreCase("global") && holder instanceof User) { if (server.equalsIgnoreCase("global") && holder instanceof User) {
((User) holder).setPrimaryGroup(group.getName()); ((User) holder).getPrimaryGroup().setStoredValue(group.getName());
} }
Message.SET_PARENT_SERVER_SUCCESS.send(sender, holder.getFriendlyName(), group.getDisplayName(), server); Message.SET_PARENT_SERVER_SUCCESS.send(sender, holder.getFriendlyName(), group.getDisplayName(), server);

View File

@ -170,8 +170,8 @@ public class UserDemote extends SubCommand<User> {
user.setPermission(NodeFactory.newBuilder("group." + previousGroup.getName()).setServer(server).setWorld(world).build()); user.setPermission(NodeFactory.newBuilder("group." + previousGroup.getName()).setServer(server).setWorld(world).build());
} catch (ObjectAlreadyHasException ignored) {} } catch (ObjectAlreadyHasException ignored) {}
if (server == null && world == null && user.getPrimaryGroup().equalsIgnoreCase(old)) { if (server == null && world == null && user.getPrimaryGroup().getStoredValue().equalsIgnoreCase(old)) {
user.setPrimaryGroup(previousGroup.getName()); user.getPrimaryGroup().setStoredValue(previousGroup.getName());
} }
switch (ContextHelper.determine(server, world)) { switch (ContextHelper.determine(server, world)) {

View File

@ -54,7 +54,7 @@ public class UserInfo extends SubCommand<User> {
user.getName(), user.getName(),
user.getUuid(), user.getUuid(),
plugin.getPlayerStatus(user.getUuid()), plugin.getPlayerStatus(user.getUuid()),
user.getPrimaryGroup(), user.getPrimaryGroup().getValue(),
user.getPermanentNodes().size(), user.getPermanentNodes().size(),
user.getTemporaryNodes().size(), user.getTemporaryNodes().size(),
user.getPrefixNodes().size(), user.getPrefixNodes().size(),

View File

@ -187,8 +187,8 @@ public class UserPromote extends SubCommand<User> {
user.setPermission(NodeFactory.newBuilder("group." + nextGroup.getName()).setServer(server).setWorld(world).build()); user.setPermission(NodeFactory.newBuilder("group." + nextGroup.getName()).setServer(server).setWorld(world).build());
} catch (ObjectAlreadyHasException ignored) {} } catch (ObjectAlreadyHasException ignored) {}
if (server == null && world == null && user.getPrimaryGroup().equalsIgnoreCase(old)) { if (server == null && world == null && user.getPrimaryGroup().getStoredValue().equalsIgnoreCase(old)) {
user.setPrimaryGroup(nextGroup.getName()); user.getPrimaryGroup().setStoredValue(nextGroup.getName());
} }
switch (ContextHelper.determine(server, world)) { switch (ContextHelper.determine(server, world)) {

View File

@ -27,6 +27,7 @@ import me.lucko.luckperms.common.commands.CommandException;
import me.lucko.luckperms.common.commands.CommandResult; import me.lucko.luckperms.common.commands.CommandResult;
import me.lucko.luckperms.common.commands.abstraction.SubCommand; import me.lucko.luckperms.common.commands.abstraction.SubCommand;
import me.lucko.luckperms.common.commands.sender.Sender; import me.lucko.luckperms.common.commands.sender.Sender;
import me.lucko.luckperms.common.config.ConfigKeys;
import me.lucko.luckperms.common.constants.Message; import me.lucko.luckperms.common.constants.Message;
import me.lucko.luckperms.common.constants.Permission; import me.lucko.luckperms.common.constants.Permission;
import me.lucko.luckperms.common.core.model.Group; import me.lucko.luckperms.common.core.model.Group;
@ -47,13 +48,18 @@ public class UserSwitchPrimaryGroup extends SubCommand<User> {
@Override @Override
public CommandResult execute(LuckPermsPlugin plugin, Sender sender, User user, List<String> args, String label) throws CommandException { public CommandResult execute(LuckPermsPlugin plugin, Sender sender, User user, List<String> args, String label) throws CommandException {
String opt = plugin.getConfiguration().get(ConfigKeys.PRIMARY_GROUP_CALCULATION_METHOD);
if (!opt.equals("stored")) {
Message.USER_PRIMARYGROUP_WARN_OPTION.send(sender, opt);
}
Group group = plugin.getGroupManager().getIfLoaded(args.get(0).toLowerCase()); Group group = plugin.getGroupManager().getIfLoaded(args.get(0).toLowerCase());
if (group == null) { if (group == null) {
Message.GROUP_DOES_NOT_EXIST.send(sender); Message.GROUP_DOES_NOT_EXIST.send(sender);
return CommandResult.INVALID_ARGS; return CommandResult.INVALID_ARGS;
} }
if (user.getPrimaryGroup().equalsIgnoreCase(group.getName())) { if (user.getPrimaryGroup().getStoredValue().equalsIgnoreCase(group.getName())) {
Message.USER_PRIMARYGROUP_ERROR_ALREADYHAS.send(sender); Message.USER_PRIMARYGROUP_ERROR_ALREADYHAS.send(sender);
return CommandResult.STATE_ERROR; return CommandResult.STATE_ERROR;
} }
@ -66,7 +72,7 @@ public class UserSwitchPrimaryGroup extends SubCommand<User> {
} }
} }
user.setPrimaryGroup(group.getName()); user.getPrimaryGroup().setStoredValue(group.getName());
Message.USER_PRIMARYGROUP_SUCCESS.send(sender, user.getName(), group.getDisplayName()); Message.USER_PRIMARYGROUP_SUCCESS.send(sender, user.getName(), group.getDisplayName());
LogEntry.build().actor(sender).acted(user) LogEntry.build().actor(sender).acted(user)
.action("setprimarygroup " + group.getName()) .action("setprimarygroup " + group.getName())

View File

@ -91,7 +91,7 @@ public class BulkEditGroup extends SubCommand<Storage> {
continue; continue;
} }
if (element.getGroupName().equals(user.getPrimaryGroup())) { if (element.getGroupName().equals(user.getPrimaryGroup().getStoredValue())) {
if (!element.isServerSpecific() && !element.isWorldSpecific() && !element.isTemporary()) { if (!element.isServerSpecific() && !element.isWorldSpecific() && !element.isTemporary()) {
continue; continue;
} }
@ -117,7 +117,7 @@ public class BulkEditGroup extends SubCommand<Storage> {
continue; continue;
} }
if (element.getGroupName().equals(user.getPrimaryGroup())) { if (element.getGroupName().equals(user.getPrimaryGroup().getStoredValue())) {
continue; continue;
} }

View File

@ -37,7 +37,12 @@ import me.lucko.luckperms.common.config.keys.MapKey;
import me.lucko.luckperms.common.config.keys.StaticKey; import me.lucko.luckperms.common.config.keys.StaticKey;
import me.lucko.luckperms.common.config.keys.StringKey; import me.lucko.luckperms.common.config.keys.StringKey;
import me.lucko.luckperms.common.core.TemporaryModifier; import me.lucko.luckperms.common.core.TemporaryModifier;
import me.lucko.luckperms.common.core.model.User;
import me.lucko.luckperms.common.defaults.Rule; import me.lucko.luckperms.common.defaults.Rule;
import me.lucko.luckperms.common.primarygroup.AllParentsByWeightHolder;
import me.lucko.luckperms.common.primarygroup.ParentsByWeightHolder;
import me.lucko.luckperms.common.primarygroup.PrimaryGroupHolder;
import me.lucko.luckperms.common.primarygroup.StoredHolder;
import me.lucko.luckperms.common.storage.DatastoreConfiguration; import me.lucko.luckperms.common.storage.DatastoreConfiguration;
import me.lucko.luckperms.common.utils.ImmutableCollectors; import me.lucko.luckperms.common.utils.ImmutableCollectors;
@ -46,6 +51,7 @@ import java.lang.reflect.Modifier;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.function.Function;
@UtilityClass @UtilityClass
public class ConfigKeys { public class ConfigKeys {
@ -70,6 +76,25 @@ public class ConfigKeys {
return TemporaryModifier.valueOf(option.toUpperCase()); return TemporaryModifier.valueOf(option.toUpperCase());
}); });
public static final ConfigKey<String> PRIMARY_GROUP_CALCULATION_METHOD = EnduringKey.wrap(AbstractKey.of(c -> {
String option = c.getString("primary-group-calculation", "stored").toLowerCase();
if (!option.equals("stored") && !option.equals("parents-by-weight") && !option.equals("all-parents-by-weight")) {
option = "stored";
}
return option;
}));
public static final ConfigKey<Function<User, PrimaryGroupHolder>> PRIMARY_GROUP_CALCULATION = EnduringKey.wrap(AbstractKey.of(c -> {
String option = PRIMARY_GROUP_CALCULATION_METHOD.get(c);
switch (option) {
case "stored":
return (Function<User, PrimaryGroupHolder>) StoredHolder::new;
case "parents-by-weight":
return (Function<User, PrimaryGroupHolder>) ParentsByWeightHolder::new;
default:
return (Function<User, PrimaryGroupHolder>) AllParentsByWeightHolder::new;
}
}));
public static final ConfigKey<Boolean> APPLYING_WILDCARDS = EnduringKey.wrap(BooleanKey.of("apply-wildcards", true)); public static final ConfigKey<Boolean> APPLYING_WILDCARDS = EnduringKey.wrap(BooleanKey.of("apply-wildcards", true));
public static final ConfigKey<Boolean> APPLYING_REGEX = EnduringKey.wrap(BooleanKey.of("apply-regex", true)); public static final ConfigKey<Boolean> APPLYING_REGEX = EnduringKey.wrap(BooleanKey.of("apply-regex", true));
public static final ConfigKey<Boolean> APPLYING_SHORTHAND = EnduringKey.wrap(BooleanKey.of("apply-shorthand", true)); public static final ConfigKey<Boolean> APPLYING_SHORTHAND = EnduringKey.wrap(BooleanKey.of("apply-shorthand", true));

View File

@ -330,6 +330,7 @@ public enum Message {
USER_GETUUID("&bThe UUID of &b{0}&b is &b{1}&b.", true), USER_GETUUID("&bThe UUID of &b{0}&b is &b{1}&b.", true),
USER_REMOVEGROUP_ERROR_PRIMARY("You cannot remove a user from their primary group.", true), USER_REMOVEGROUP_ERROR_PRIMARY("You cannot remove a user from their primary group.", true),
USER_PRIMARYGROUP_SUCCESS("&b{0}&a's primary group was set to &b{1}&a.", true), USER_PRIMARYGROUP_SUCCESS("&b{0}&a's primary group was set to &b{1}&a.", true),
USER_PRIMARYGROUP_WARN_OPTION("&cWarning: The primary group calculation method being used by this server &7({0}) &cmay not reflect this change.", true),
USER_PRIMARYGROUP_ERROR_ALREADYHAS("The user already has this group set as their primary group.", true), USER_PRIMARYGROUP_ERROR_ALREADYHAS("The user already has this group set as their primary group.", true),
USER_PRIMARYGROUP_ERROR_NOTMEMBER("&b{0}&a was not already a member of &b{1}&a, adding them now.", true), USER_PRIMARYGROUP_ERROR_NOTMEMBER("&b{0}&a was not already a member of &b{1}&a, adding them now.", true),
USER_TRACK_ERROR_NOT_CONTAIN_GROUP("The user specified isn't already in any groups on this track.", true), USER_TRACK_ERROR_NOT_CONTAIN_GROUP("The user specified isn't already in any groups on this track.", true),

View File

@ -95,7 +95,7 @@ public abstract class PermissionHolder {
/** /**
* Reference to the main plugin instance * Reference to the main plugin instance
*/ */
@Getter(AccessLevel.PROTECTED) @Getter
private final LuckPermsPlugin plugin; private final LuckPermsPlugin plugin;
/** /**
@ -168,10 +168,6 @@ public abstract class PermissionHolder {
} }
}); });
/* Caching apply methods. Are just called by the caching instances to gather data about the instance. */
protected void forceCleanup() { protected void forceCleanup() {
getAllNodesCache.cleanUp(); getAllNodesCache.cleanUp();
getAllNodesFilteredCache.cleanUp(); getAllNodesFilteredCache.cleanUp();
@ -221,6 +217,8 @@ public abstract class PermissionHolder {
declareState(); declareState();
} }
/* Caching apply methods. Are just called by the caching instances to gather data about the instance. */
private ImmutableSortedSet<LocalizedNode> cacheApply(boolean mergeTemp) { private ImmutableSortedSet<LocalizedNode> cacheApply(boolean mergeTemp) {
TreeSet<LocalizedNode> combined = new TreeSet<>(PriorityComparator.reverse()); TreeSet<LocalizedNode> combined = new TreeSet<>(PriorityComparator.reverse());
Set<Node> enduring = getNodes(); Set<Node> enduring = getNodes();

View File

@ -32,8 +32,10 @@ import me.lucko.luckperms.common.api.delegates.UserDelegate;
import me.lucko.luckperms.common.caching.UserCache; import me.lucko.luckperms.common.caching.UserCache;
import me.lucko.luckperms.common.caching.handlers.HolderReference; import me.lucko.luckperms.common.caching.handlers.HolderReference;
import me.lucko.luckperms.common.caching.handlers.UserReference; import me.lucko.luckperms.common.caching.handlers.UserReference;
import me.lucko.luckperms.common.config.ConfigKeys;
import me.lucko.luckperms.common.core.UserIdentifier; import me.lucko.luckperms.common.core.UserIdentifier;
import me.lucko.luckperms.common.plugin.LuckPermsPlugin; import me.lucko.luckperms.common.plugin.LuckPermsPlugin;
import me.lucko.luckperms.common.primarygroup.PrimaryGroupHolder;
import me.lucko.luckperms.common.utils.BufferedRequest; import me.lucko.luckperms.common.utils.BufferedRequest;
import me.lucko.luckperms.common.utils.Identifiable; import me.lucko.luckperms.common.utils.Identifiable;
@ -60,8 +62,7 @@ public class User extends PermissionHolder implements Identifiable<UserIdentifie
* The users primary group * The users primary group
*/ */
@Getter @Getter
@Setter private final PrimaryGroupHolder primaryGroup;
private String primaryGroup = null;
/** /**
* The users data cache instance, if present. * The users data cache instance, if present.
@ -85,12 +86,14 @@ public class User extends PermissionHolder implements Identifiable<UserIdentifie
super(uuid.toString(), plugin); super(uuid.toString(), plugin);
this.uuid = uuid; this.uuid = uuid;
this.name = null; this.name = null;
this.primaryGroup = plugin.getConfiguration().get(ConfigKeys.PRIMARY_GROUP_CALCULATION).apply(this);
} }
public User(UUID uuid, String name, LuckPermsPlugin plugin) { public User(UUID uuid, String name, LuckPermsPlugin plugin) {
super(uuid.toString(), plugin); super(uuid.toString(), plugin);
this.uuid = uuid; this.uuid = uuid;
this.name = name; this.name = name;
this.primaryGroup = plugin.getConfiguration().get(ConfigKeys.PRIMARY_GROUP_CALCULATION).apply(this);
} }
@Override @Override

View File

@ -226,7 +226,7 @@ public class Exporter implements Runnable {
output.add(NodeFactory.nodeAsCommand(node, user.getUuid().toString(), false)); output.add(NodeFactory.nodeAsCommand(node, user.getUuid().toString(), false));
} }
if (!user.getPrimaryGroup().equalsIgnoreCase("default")) { if (!user.getPrimaryGroup().getStoredValue().equalsIgnoreCase("default")) {
output.add("/luckperms user " + user.getUuid().toString() + " switchprimarygroup " + user.getPrimaryGroup()); output.add("/luckperms user " + user.getUuid().toString() + " switchprimarygroup " + user.getPrimaryGroup());
} }

View File

@ -97,7 +97,7 @@ public class Rule {
} }
if (setPrimaryGroup != null) { if (setPrimaryGroup != null) {
user.setPrimaryGroup(setPrimaryGroup); user.getPrimaryGroup().setStoredValue(setPrimaryGroup);
} }
return true; return true;

View File

@ -25,12 +25,12 @@ package me.lucko.luckperms.common.managers.impl;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import me.lucko.luckperms.api.Node; import me.lucko.luckperms.api.Node;
import me.lucko.luckperms.common.core.NodeFactory;
import me.lucko.luckperms.common.core.UserIdentifier; import me.lucko.luckperms.common.core.UserIdentifier;
import me.lucko.luckperms.common.core.model.User; import me.lucko.luckperms.common.core.model.User;
import me.lucko.luckperms.common.managers.AbstractManager; import me.lucko.luckperms.common.managers.AbstractManager;
import me.lucko.luckperms.common.managers.UserManager; import me.lucko.luckperms.common.managers.UserManager;
import me.lucko.luckperms.common.plugin.LuckPermsPlugin; import me.lucko.luckperms.common.plugin.LuckPermsPlugin;
import me.lucko.luckperms.exceptions.ObjectAlreadyHasException;
import java.util.Set; import java.util.Set;
import java.util.UUID; import java.util.UUID;
@ -40,7 +40,7 @@ public class GenericUserManager extends AbstractManager<UserIdentifier, User> im
public static boolean giveDefaultIfNeeded(User user, boolean save, LuckPermsPlugin plugin) { public static boolean giveDefaultIfNeeded(User user, boolean save, LuckPermsPlugin plugin) {
boolean hasGroup = false; boolean hasGroup = false;
if (user.getPrimaryGroup() != null && !user.getPrimaryGroup().isEmpty()) { if (user.getPrimaryGroup().getStoredValue() != null && !user.getPrimaryGroup().getStoredValue().isEmpty()) {
for (Node node : user.getPermissions(false)) { for (Node node : user.getPermissions(false)) {
if (node.isServerSpecific() || node.isWorldSpecific()) { if (node.isServerSpecific() || node.isWorldSpecific()) {
continue; continue;
@ -57,10 +57,8 @@ public class GenericUserManager extends AbstractManager<UserIdentifier, User> im
return false; return false;
} }
user.setPrimaryGroup("default"); user.getPrimaryGroup().setStoredValue("default");
try { user.setPermissionUnchecked(NodeFactory.make("group.default"));
user.setPermission("group.default", true);
} catch (ObjectAlreadyHasException ignored) {}
if (save) { if (save) {
plugin.getStorage().saveUser(user); plugin.getStorage().saveUser(user);
@ -97,7 +95,7 @@ public class GenericUserManager extends AbstractManager<UserIdentifier, User> im
} }
// Not in the default primary group // Not in the default primary group
return !user.getPrimaryGroup().equalsIgnoreCase("default"); return !user.getPrimaryGroup().getStoredValue().equalsIgnoreCase("default");
} }
private final LuckPermsPlugin plugin; private final LuckPermsPlugin plugin;

View File

@ -0,0 +1,64 @@
/*
* Copyright (c) 2016 Lucko (Luck) <luck@lucko.me>
*
* 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.primarygroup;
import me.lucko.luckperms.api.Contexts;
import me.lucko.luckperms.api.Node;
import me.lucko.luckperms.common.core.model.Group;
import me.lucko.luckperms.common.core.model.User;
import me.lucko.luckperms.common.utils.ExtractedContexts;
import java.util.Comparator;
import java.util.Optional;
public class AllParentsByWeightHolder extends StoredHolder {
private String cachedValue = null;
private boolean useCached = false;
public AllParentsByWeightHolder(User user) {
super(user);
user.getStateListeners().add(() -> useCached = false);
}
@Override
public String getValue() {
if (useCached) {
return cachedValue;
}
cachedValue = user.getAllNodes(null, ExtractedContexts.generate(Contexts.allowAll())).stream()
.filter(Node::isGroupNode)
.filter(Node::getValue)
.map(n -> Optional.ofNullable(user.getPlugin().getGroupManager().getIfLoaded(n.getGroupName())))
.filter(Optional::isPresent)
.map(Optional::get)
.sorted(Comparator.comparingInt(o -> o.getWeight().orElse(0)))
.findFirst()
.map(Group::getName)
.orElse(null);
useCached = true;
return cachedValue;
}
}

View File

@ -0,0 +1,62 @@
/*
* Copyright (c) 2016 Lucko (Luck) <luck@lucko.me>
*
* 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.primarygroup;
import me.lucko.luckperms.api.Node;
import me.lucko.luckperms.common.core.model.Group;
import me.lucko.luckperms.common.core.model.User;
import java.util.Comparator;
import java.util.Optional;
public class ParentsByWeightHolder extends StoredHolder {
private String cachedValue = null;
private boolean useCached = false;
public ParentsByWeightHolder(User user) {
super(user);
user.getStateListeners().add(() -> useCached = false);
}
@Override
public String getValue() {
if (useCached) {
return cachedValue;
}
cachedValue = user.getPermissions(true).stream()
.filter(Node::isGroupNode)
.filter(Node::getValue)
.map(n -> Optional.ofNullable(user.getPlugin().getGroupManager().getIfLoaded(n.getGroupName())))
.filter(Optional::isPresent)
.map(Optional::get)
.sorted(Comparator.comparingInt(o -> o.getWeight().orElse(0)))
.findFirst()
.map(Group::getName)
.orElse(null);
useCached = true;
return cachedValue;
}
}

View File

@ -0,0 +1,33 @@
/*
* Copyright (c) 2016 Lucko (Luck) <luck@lucko.me>
*
* 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.primarygroup;
public interface PrimaryGroupHolder {
String getValue();
String getStoredValue();
void setStoredValue(String storedValue);
}

View File

@ -0,0 +1,44 @@
/*
* Copyright (c) 2016 Lucko (Luck) <luck@lucko.me>
*
* 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.primarygroup;
import lombok.Getter;
import lombok.RequiredArgsConstructor;
import lombok.Setter;
import me.lucko.luckperms.common.core.model.User;
@RequiredArgsConstructor
public class StoredHolder implements PrimaryGroupHolder {
protected final User user;
@Getter
@Setter
private String storedValue = null;
public String getValue() {
return storedValue;
}
}

View File

@ -118,7 +118,7 @@ public class JSONBacking extends FlatfileBacking {
reader.nextName(); // name record reader.nextName(); // name record
String name = reader.nextString(); // name String name = reader.nextString(); // name
reader.nextName(); // primaryGroup record reader.nextName(); // primaryGroup record
user.setPrimaryGroup(reader.nextString()); // primaryGroup user.getPrimaryGroup().setStoredValue(reader.nextString()); // primaryGroup
reader.nextName(); // perms reader.nextName(); // perms
reader.beginObject(); reader.beginObject();
Map<String, Boolean> map = new HashMap<>(); Map<String, Boolean> map = new HashMap<>();
@ -146,7 +146,7 @@ public class JSONBacking extends FlatfileBacking {
writer.beginObject(); writer.beginObject();
writer.name("uuid").value(user.getUuid().toString()); writer.name("uuid").value(user.getUuid().toString());
writer.name("name").value(user.getName()); writer.name("name").value(user.getName());
writer.name("primaryGroup").value(user.getPrimaryGroup()); writer.name("primaryGroup").value(user.getPrimaryGroup().getStoredValue());
writer.name("perms"); writer.name("perms");
writer.beginObject(); writer.beginObject();
for (Map.Entry<String, Boolean> e : exportToLegacy(user.getNodes()).entrySet()) { for (Map.Entry<String, Boolean> e : exportToLegacy(user.getNodes()).entrySet()) {
@ -162,7 +162,7 @@ public class JSONBacking extends FlatfileBacking {
} else { } else {
if (GenericUserManager.shouldSave(user)) { if (GenericUserManager.shouldSave(user)) {
user.clearNodes(); user.clearNodes();
user.setPrimaryGroup(null); user.getPrimaryGroup().setStoredValue(null);
plugin.getUserManager().giveDefaultIfNeeded(user, false); plugin.getUserManager().giveDefaultIfNeeded(user, false);
} }
return true; return true;
@ -202,7 +202,7 @@ public class JSONBacking extends FlatfileBacking {
writer.beginObject(); writer.beginObject();
writer.name("uuid").value(user.getUuid().toString()); writer.name("uuid").value(user.getUuid().toString());
writer.name("name").value(user.getName()); writer.name("name").value(user.getName());
writer.name("primaryGroup").value(user.getPrimaryGroup()); writer.name("primaryGroup").value(user.getPrimaryGroup().getStoredValue());
writer.name("perms"); writer.name("perms");
writer.beginObject(); writer.beginObject();
for (Map.Entry<String, Boolean> e : exportToLegacy(user.getNodes()).entrySet()) { for (Map.Entry<String, Boolean> e : exportToLegacy(user.getNodes()).entrySet()) {

View File

@ -94,7 +94,7 @@ public class MongoDBBacking extends AbstractBacking {
private static Document fromUser(User user) { private static Document fromUser(User user) {
Document main = new Document("_id", user.getUuid()) Document main = new Document("_id", user.getUuid())
.append("name", user.getName()) .append("name", user.getName())
.append("primaryGroup", user.getPrimaryGroup()); .append("primaryGroup", user.getPrimaryGroup().getStoredValue());
Document perms = new Document(); Document perms = new Document();
for (Map.Entry<String, Boolean> e : convert(exportToLegacy(user.getNodes())).entrySet()) { for (Map.Entry<String, Boolean> e : convert(exportToLegacy(user.getNodes())).entrySet()) {
@ -238,7 +238,7 @@ public class MongoDBBacking extends AbstractBacking {
// User exists, let's load. // User exists, let's load.
Document d = cursor.next(); Document d = cursor.next();
user.setNodes(revert((Map<String, Boolean>) d.get("perms"))); user.setNodes(revert((Map<String, Boolean>) d.get("perms")));
user.setPrimaryGroup(d.getString("primaryGroup")); user.getPrimaryGroup().setStoredValue(d.getString("primaryGroup"));
boolean save = plugin.getUserManager().giveDefaultIfNeeded(user, false); boolean save = plugin.getUserManager().giveDefaultIfNeeded(user, false);
@ -256,7 +256,7 @@ public class MongoDBBacking extends AbstractBacking {
} else { } else {
if (GenericUserManager.shouldSave(user)) { if (GenericUserManager.shouldSave(user)) {
user.clearNodes(); user.clearNodes();
user.setPrimaryGroup(null); user.getPrimaryGroup().setStoredValue(null);
plugin.getUserManager().giveDefaultIfNeeded(user, false); plugin.getUserManager().giveDefaultIfNeeded(user, false);
} }
} }

View File

@ -301,7 +301,7 @@ public class SQLBacking extends AbstractBacking {
if (pg == null) { if (pg == null) {
pg = "default"; pg = "default";
} }
user.setPrimaryGroup(pg); user.getPrimaryGroup().setStoredValue(pg);
String name = userName.get(); String name = userName.get();
if (name == null) { if (name == null) {
@ -328,7 +328,7 @@ public class SQLBacking extends AbstractBacking {
// User has no data in storage. // User has no data in storage.
if (GenericUserManager.shouldSave(user)) { if (GenericUserManager.shouldSave(user)) {
user.clearNodes(); user.clearNodes();
user.setPrimaryGroup(null); user.getPrimaryGroup().setStoredValue(null);
plugin.getUserManager().giveDefaultIfNeeded(user, false); plugin.getUserManager().giveDefaultIfNeeded(user, false);
} }
} }
@ -436,7 +436,7 @@ public class SQLBacking extends AbstractBacking {
try (Connection c = provider.getConnection()) { try (Connection c = provider.getConnection()) {
try (PreparedStatement ps = c.prepareStatement(prefix.apply(PLAYER_UPDATE_PRIMARY_GROUP))) { try (PreparedStatement ps = c.prepareStatement(prefix.apply(PLAYER_UPDATE_PRIMARY_GROUP))) {
ps.setString(1, user.getPrimaryGroup() == null ? "default" : user.getPrimaryGroup()); ps.setString(1, user.getPrimaryGroup().getStoredValue() == null ? "default" : user.getPrimaryGroup().getStoredValue());
ps.setString(2, user.getUuid().toString()); ps.setString(2, user.getUuid().toString());
ps.execute(); ps.execute();
} }

View File

@ -115,7 +115,7 @@ public class YAMLBacking extends FlatfileBacking {
return readMapFromFile(userFile, values -> { return readMapFromFile(userFile, values -> {
// User exists, let's load. // User exists, let's load.
String name = (String) values.get("name"); String name = (String) values.get("name");
user.setPrimaryGroup((String) values.get("primary-group")); user.getPrimaryGroup().setStoredValue((String) values.get("primary-group"));
Map<String, Boolean> perms = (Map<String, Boolean>) values.get("perms"); Map<String, Boolean> perms = (Map<String, Boolean>) values.get("perms");
user.setNodes(perms); user.setNodes(perms);
@ -133,7 +133,7 @@ public class YAMLBacking extends FlatfileBacking {
Map<String, Object> data = new HashMap<>(); Map<String, Object> data = new HashMap<>();
data.put("uuid", user.getUuid().toString()); data.put("uuid", user.getUuid().toString());
data.put("name", user.getName()); data.put("name", user.getName());
data.put("primary-group", user.getPrimaryGroup()); data.put("primary-group", user.getPrimaryGroup().getStoredValue());
data.put("perms", exportToLegacy(user.getNodes())); data.put("perms", exportToLegacy(user.getNodes()));
writeMapToFile(userFile, data); writeMapToFile(userFile, data);
} }
@ -142,7 +142,7 @@ public class YAMLBacking extends FlatfileBacking {
} else { } else {
if (GenericUserManager.shouldSave(user)) { if (GenericUserManager.shouldSave(user)) {
user.clearNodes(); user.clearNodes();
user.setPrimaryGroup(null); user.getPrimaryGroup().setStoredValue(null);
plugin.getUserManager().giveDefaultIfNeeded(user, false); plugin.getUserManager().giveDefaultIfNeeded(user, false);
} }
return true; return true;
@ -180,7 +180,7 @@ public class YAMLBacking extends FlatfileBacking {
Map<String, Object> values = new HashMap<>(); Map<String, Object> values = new HashMap<>();
values.put("uuid", user.getUuid().toString()); values.put("uuid", user.getUuid().toString());
values.put("name", user.getName()); values.put("name", user.getName());
values.put("primary-group", user.getPrimaryGroup()); values.put("primary-group", user.getPrimaryGroup().getStoredValue());
values.put("perms", exportToLegacy(user.getNodes())); values.put("perms", exportToLegacy(user.getNodes()));
return writeMapToFile(userFile, values); return writeMapToFile(userFile, values);
}, false); }, false);

View File

@ -252,6 +252,7 @@ info-temp-parent-header: "&f- &aTemporary Parent Groups:"
user-getuuid: "&bThe UUID of &b{0}&b is &b{1}&b." user-getuuid: "&bThe UUID of &b{0}&b is &b{1}&b."
user-removegroup-error-primary: "You cannot remove a user from their primary group." user-removegroup-error-primary: "You cannot remove a user from their primary group."
user-primarygroup-success: "&b{0}&a's primary group was set to &b{1}&a." user-primarygroup-success: "&b{0}&a's primary group was set to &b{1}&a."
user-primarygroup-warn-option: "&cWarning: The primary group calculation method being used by this server &7({0}) &cmay not reflect this change."
user-primarygroup-error-alreadyhas: "The user already has this group set as their primary group." user-primarygroup-error-alreadyhas: "The user already has this group set as their primary group."
user-primarygroup-error-notmember: "&b{0}&a was not already a member of &b{1}&a, adding them now." user-primarygroup-error-notmember: "&b{0}&a was not already a member of &b{1}&a, adding them now."
user-track-error-not-contain-group: "The user specified isn't already in any groups on this track." user-track-error-not-contain-group: "The user specified isn't already in any groups on this track."

View File

@ -67,6 +67,14 @@ group-name-rewrite {
# If "deny": the command will just fail if you try to add another node with the same expiry # If "deny": the command will just fail if you try to add another node with the same expiry
temporary-add-behaviour="deny" temporary-add-behaviour="deny"
# How should LuckPerms determine a users "primary" group.
#
# Available Options:
# -> stored use the value stored against the users record in the file/database
# -> parents-by-weight just use the users most highly weighted parent
# -> all-parents-by-weight same as above, but calculates based upon all parents inherits from both directly and indirectly
primary-group-calculation="parents-by-weight"