From eefb99e7b8afaeb17f8579c7decb502c2d93219d Mon Sep 17 00:00:00 2001 From: Frank van der Heijden Date: Thu, 31 Dec 2020 19:52:14 +0100 Subject: [PATCH] Make CommandFilter more extendable + fix AsyncTeleport not applying command cooldown --- .../earth2me/essentials/AsyncTeleport.java | 4 +- .../earth2me/essentials/CommandFilter.java | 21 ++++----- .../earth2me/essentials/CommandFilters.java | 46 ++++++++----------- .../earth2me/essentials/EssCommandFilter.java | 18 ++++++++ .../essentials/EssentialsPlayerListener.java | 8 ++-- .../essentials/RegexCommandFilter.java | 11 +++++ .../java/com/earth2me/essentials/Trade.java | 27 +++++++---- 7 files changed, 82 insertions(+), 53 deletions(-) create mode 100644 Essentials/src/main/java/com/earth2me/essentials/EssCommandFilter.java create mode 100644 Essentials/src/main/java/com/earth2me/essentials/RegexCommandFilter.java diff --git a/Essentials/src/main/java/com/earth2me/essentials/AsyncTeleport.java b/Essentials/src/main/java/com/earth2me/essentials/AsyncTeleport.java index 5c1f0648c2f..abdec520782 100644 --- a/Essentials/src/main/java/com/earth2me/essentials/AsyncTeleport.java +++ b/Essentials/src/main/java/com/earth2me/essentials/AsyncTeleport.java @@ -281,7 +281,7 @@ private void teleport(final IUser teleportee, final ITarget target, final Trade } nowAsync(teleportee, target, cause, future); if (cashCharge != null) { - cashCharge.charge(teleportOwner, future); + cashCharge.charge(teleportOwner, chargeFor.getCommand(), future); if (future.isCompletedExceptionally()) { return; } @@ -332,7 +332,7 @@ private void teleportOther(final IUser teleporter, final IUser teleportee, final nowAsync(teleportee, target, cause, future); if (teleporter != null && cashCharge != null) { - cashCharge.charge(teleporter, future); + cashCharge.charge(teleporter, chargeFor.getCommand(), future); if (future.isCompletedExceptionally()) { return; } diff --git a/Essentials/src/main/java/com/earth2me/essentials/CommandFilter.java b/Essentials/src/main/java/com/earth2me/essentials/CommandFilter.java index b64b2a0f9b4..7466abd04f2 100644 --- a/Essentials/src/main/java/com/earth2me/essentials/CommandFilter.java +++ b/Essentials/src/main/java/com/earth2me/essentials/CommandFilter.java @@ -1,23 +1,28 @@ package com.earth2me.essentials; +import com.google.common.base.Preconditions; import net.ess3.api.IUser; import java.math.BigDecimal; import java.util.Date; import java.util.regex.Pattern; -public class CommandFilter { +public abstract class CommandFilter { + + public enum Type { + REGEX, + ESS + } private final String name; - private final String command; private final Pattern pattern; private final Integer cooldown; private final boolean persistentCooldown; private final BigDecimal cost; - public CommandFilter(String name, String command, Pattern pattern, Integer cooldown, boolean persistentCooldown, BigDecimal cost) { + public CommandFilter(String name, Pattern pattern, Integer cooldown, boolean persistentCooldown, BigDecimal cost) { + Preconditions.checkNotNull(pattern); this.name = name; - this.command = command; this.pattern = pattern; this.cooldown = cooldown; this.persistentCooldown = persistentCooldown; @@ -28,14 +33,6 @@ public String getName() { return name; } - public String getCommand() { - return command; - } - - public boolean hasCommand() { - return command != null; - } - public Pattern getPattern() { return pattern; } diff --git a/Essentials/src/main/java/com/earth2me/essentials/CommandFilters.java b/Essentials/src/main/java/com/earth2me/essentials/CommandFilters.java index 96bedc09bb6..89843876564 100644 --- a/Essentials/src/main/java/com/earth2me/essentials/CommandFilters.java +++ b/Essentials/src/main/java/com/earth2me/essentials/CommandFilters.java @@ -6,7 +6,9 @@ import java.io.File; import java.math.BigDecimal; -import java.util.HashMap; +import java.util.ArrayList; +import java.util.EnumMap; +import java.util.List; import java.util.Locale; import java.util.Map; import java.util.Objects; @@ -19,7 +21,7 @@ public class CommandFilters implements IConf { private final IEssentials essentials; private final EssentialsConf config; private ConfigurationSection filters; - private Map commandFilters; + private Map> commandFilters; public CommandFilters(final IEssentials essentials) { this.essentials = essentials; @@ -50,8 +52,8 @@ private ConfigurationSection _getCommandFilterSection() { return null; } - private Map _getCommandFilters() { - final Map commandFilters = new HashMap<>(); + private Map> _getCommandFilters() { + final Map> commandFilters = new EnumMap<>(CommandFilter.Type.class); for (final String name : filters.getKeys(false)) { if (!filters.isConfigurationSection(name)) { EssentialsConf.LOGGER.warning("Invalid command filter '" + name + "'"); @@ -59,7 +61,7 @@ private Map _getCommandFilters() { } final ConfigurationSection section = Objects.requireNonNull(filters.getConfigurationSection(name)); - Pattern pattern = section.isString("pattern") ? compileRegex(section.getString("pattern")) : null; + final Pattern pattern = section.isString("pattern") ? compileRegex(section.getString("pattern")) : null; final String command = section.getString("command"); if (pattern == null && command == null) { @@ -72,11 +74,6 @@ private Map _getCommandFilters() { continue; } - // Compile the command as a regex if the pattern hasn't been set, so pattern is always available. - if (pattern == null) { - pattern = compileRegex(command); - } - Integer cooldown = section.getInt("cooldown", -1); if (cooldown < 0) { cooldown = null; @@ -88,7 +85,12 @@ private Map _getCommandFilters() { final BigDecimal cost = EssentialsConf.toBigDecimal(section.getString("cost"), null); final String lowerName = name.toLowerCase(Locale.ENGLISH); - commandFilters.put(lowerName, new CommandFilter(lowerName, command, pattern, cooldown, persistentCooldown, cost)); + + if (pattern == null) { + commandFilters.computeIfAbsent(CommandFilter.Type.ESS, k -> new ArrayList<>()).add(new EssCommandFilter(lowerName, command, compileRegex(command), cooldown, persistentCooldown, cost)); + } else { + commandFilters.computeIfAbsent(CommandFilter.Type.REGEX, k -> new ArrayList<>()).add(new RegexCommandFilter(lowerName, pattern, cooldown, persistentCooldown, cost)); + } } config.save(); return commandFilters; @@ -116,28 +118,18 @@ public EssentialsConf getConfig() { return config; } - public CommandFilter getFilterByName(final String name) { - return commandFilters.get(name.toLowerCase(Locale.ENGLISH)); - } - - public CommandFilter getCommandCooldown(final IUser user, final String label, boolean essCommand) { + public CommandFilter getCommandCooldown(final IUser user, final String label, CommandFilter.Type type) { if (user.isAuthorized("essentials.commandcooldowns.bypass")) return null; - return getFilter(label, essCommand, filter -> filter.hasCooldown() && !user.isAuthorized("essentials.commandcooldowns.bypass." + filter.getName())); + return getFilter(label, type, filter -> filter.hasCooldown() && !user.isAuthorized("essentials.commandcooldowns.bypass." + filter.getName())); } - public CommandFilter getCommandCost(final IUser user, final String label, boolean essCommand) { + public CommandFilter getCommandCost(final IUser user, final String label, CommandFilter.Type type) { if (user.isAuthorized("essentials.nocommandcost.all")) return null; - return getFilter(label, essCommand, filter -> filter.hasCost() && !user.isAuthorized("essentials.nocommandcost." + filter.getName())); + return getFilter(label, type, filter -> filter.hasCost() && !user.isAuthorized("essentials.nocommandcost." + filter.getName())); } - private CommandFilter getFilter(final String label, boolean essCommand, Predicate filterPredicate) { - for (CommandFilter filter : commandFilters.values()) { - // When the label is an ess command, the filter must define a command entry. - if (essCommand && !filter.hasCommand()) continue; - - // Same vice versa. - if (!essCommand && filter.hasCommand()) continue; - + private CommandFilter getFilter(final String label, CommandFilter.Type type, Predicate filterPredicate) { + for (CommandFilter filter : commandFilters.get(type)) { if (!filterPredicate.test(filter)) continue; final boolean matches = filter.getPattern().matcher(label).matches(); diff --git a/Essentials/src/main/java/com/earth2me/essentials/EssCommandFilter.java b/Essentials/src/main/java/com/earth2me/essentials/EssCommandFilter.java new file mode 100644 index 00000000000..b2e2e943b4f --- /dev/null +++ b/Essentials/src/main/java/com/earth2me/essentials/EssCommandFilter.java @@ -0,0 +1,18 @@ +package com.earth2me.essentials; + +import java.math.BigDecimal; +import java.util.regex.Pattern; + +public class EssCommandFilter extends CommandFilter { + + private final String command; + + public EssCommandFilter(String name, String command, Pattern pattern, Integer cooldown, boolean persistentCooldown, BigDecimal cost) { + super(name, pattern, cooldown, persistentCooldown, cost); + this.command = command; + } + + public String getCommand() { + return command; + } +} diff --git a/Essentials/src/main/java/com/earth2me/essentials/EssentialsPlayerListener.java b/Essentials/src/main/java/com/earth2me/essentials/EssentialsPlayerListener.java index 5eb8a68bd3f..91bad18b4c4 100644 --- a/Essentials/src/main/java/com/earth2me/essentials/EssentialsPlayerListener.java +++ b/Essentials/src/main/java/com/earth2me/essentials/EssentialsPlayerListener.java @@ -603,18 +603,18 @@ public void onPlayerCommandPreprocess(final PlayerCommandPreprocessEvent event) } if (!cooldownFound) { - final CommandFilter cooldownFilter = ess.getCommandFilters().getCommandCooldown(user, fullCommand, false); + final CommandFilter cooldownFilter = ess.getCommandFilters().getCommandCooldown(user, fullCommand, CommandFilter.Type.REGEX); if (cooldownFilter != null) { if (ess.getSettings().isDebug()) { - ess.getLogger().info("Applying " + cooldownFilter.getCooldown() + "ms cooldown on /" + fullCommand + " for" + user.getName() + "."); + ess.getLogger().info("Applying " + cooldownFilter.getCooldown() + "ms cooldown on /" + fullCommand + " for " + user.getName() + "."); } cooldownFilter.applyCooldownTo(user); } - final CommandFilter costFilter = ess.getCommandFilters().getCommandCost(user, fullCommand, false); + final CommandFilter costFilter = ess.getCommandFilters().getCommandCost(user, fullCommand, CommandFilter.Type.REGEX); if (costFilter != null) { if (ess.getSettings().isDebug()) { - ess.getLogger().info("Applying a cost of " + costFilter.getCost() + " on /" + fullCommand + " for" + user.getName() + "."); + ess.getLogger().info("Applying a cost of " + costFilter.getCost() + " on /" + fullCommand + " for " + user.getName() + "."); } final BigDecimal cost = costFilter.getCost(); diff --git a/Essentials/src/main/java/com/earth2me/essentials/RegexCommandFilter.java b/Essentials/src/main/java/com/earth2me/essentials/RegexCommandFilter.java new file mode 100644 index 00000000000..81674a87dce --- /dev/null +++ b/Essentials/src/main/java/com/earth2me/essentials/RegexCommandFilter.java @@ -0,0 +1,11 @@ +package com.earth2me.essentials; + +import java.math.BigDecimal; +import java.util.regex.Pattern; + +public class RegexCommandFilter extends CommandFilter { + + public RegexCommandFilter(String name, Pattern pattern, Integer cooldown, boolean persistentCooldown, BigDecimal cost) { + super(name, pattern, cooldown, persistentCooldown, cost); + } +} diff --git a/Essentials/src/main/java/com/earth2me/essentials/Trade.java b/Essentials/src/main/java/com/earth2me/essentials/Trade.java index 9991256605b..920c088be77 100644 --- a/Essentials/src/main/java/com/earth2me/essentials/Trade.java +++ b/Essentials/src/main/java/com/earth2me/essentials/Trade.java @@ -283,17 +283,14 @@ public void charge(final IUser user) throws ChargeException { } catch (final ExecutionException e) { throw (ChargeException) e.getCause(); } - } else { - if (command != null && !command.isEmpty()) { - final CommandFilter cooldownFilter = ess.getCommandFilters().getCommandCooldown(user, command, true); - if (cooldownFilter != null) { - cooldownFilter.applyCooldownTo(user); - } - } } } public void charge(final IUser user, final CompletableFuture future) { + charge(user, this.command, future); + } + + public void charge(final IUser user, final String cooldownCommand, final CompletableFuture future) { if (ess.getSettings().isDebug()) { ess.getLogger().log(Level.INFO, "attempting to charge user " + user.getName()); } @@ -340,6 +337,20 @@ public void charge(final IUser user, final CompletableFuture future) { if (ess.getSettings().isDebug()) { ess.getLogger().log(Level.INFO, "charge user " + user.getName() + " completed"); } + + if (cooldownCommand != null && !cooldownCommand.isEmpty()) { + final CommandFilter cooldownFilter = ess.getCommandFilters().getCommandCooldown(user, cooldownCommand, CommandFilter.Type.ESS); + if (cooldownFilter != null) { + if (ess.getSettings().isDebug()) { + ess.getLogger().info("Applying " + cooldownFilter.getCooldown() + "ms cooldown on /" + cooldownCommand + " for " + user.getName() + "."); + } + cooldownFilter.applyCooldownTo(user); + } + } + } + + public String getCommand() { + return command; } public BigDecimal getMoney() { @@ -369,7 +380,7 @@ public TradeType getType() { public BigDecimal getCommandCost(final IUser user) { BigDecimal cost = BigDecimal.ZERO; if (command != null && !command.isEmpty()) { - final CommandFilter filter = ess.getCommandFilters().getCommandCost(user, command.charAt(0) == '/' ? command.substring(1) : command, true); + final CommandFilter filter = ess.getCommandFilters().getCommandCost(user, command.charAt(0) == '/' ? command.substring(1) : command, CommandFilter.Type.ESS); if (filter != null && filter.getCost().signum() != 0) { cost = filter.getCost(); } else if (fallbackTrade != null) {