Add pinpoint notification and getting players by permission support

(cached).

This adds the ability to keep track of players with a certain
permission with world-specific resolution (world changes, join, quit,
kick). Also adding a static access method to NoCheatPlus to notify all
with administration.notify permission.
This commit is contained in:
asofold 2012-11-09 14:29:51 +01:00
parent a7b45fc3d4
commit e0aca1a6eb
4 changed files with 276 additions and 60 deletions

View File

@ -8,17 +8,22 @@ import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Map.Entry; import java.util.Map.Entry;
import java.util.Set;
import org.bukkit.Bukkit; import org.bukkit.Bukkit;
import org.bukkit.ChatColor; import org.bukkit.ChatColor;
import org.bukkit.Server;
import org.bukkit.command.PluginCommand; import org.bukkit.command.PluginCommand;
import org.bukkit.entity.Player; import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler; import org.bukkit.event.EventHandler;
import org.bukkit.event.EventPriority; import org.bukkit.event.EventPriority;
import org.bukkit.event.Listener; import org.bukkit.event.Listener;
import org.bukkit.event.player.PlayerChangedWorldEvent;
import org.bukkit.event.player.PlayerJoinEvent; import org.bukkit.event.player.PlayerJoinEvent;
import org.bukkit.event.player.PlayerKickEvent;
import org.bukkit.event.player.PlayerLoginEvent; import org.bukkit.event.player.PlayerLoginEvent;
import org.bukkit.event.player.PlayerLoginEvent.Result; import org.bukkit.event.player.PlayerLoginEvent.Result;
import org.bukkit.event.player.PlayerQuitEvent;
import org.bukkit.plugin.PluginDescriptionFile; import org.bukkit.plugin.PluginDescriptionFile;
import org.bukkit.plugin.java.JavaPlugin; import org.bukkit.plugin.java.JavaPlugin;
@ -35,7 +40,9 @@ import fr.neatmonster.nocheatplus.command.CommandHandler;
import fr.neatmonster.nocheatplus.command.INotifyReload; import fr.neatmonster.nocheatplus.command.INotifyReload;
import fr.neatmonster.nocheatplus.components.ComponentWithName; import fr.neatmonster.nocheatplus.components.ComponentWithName;
import fr.neatmonster.nocheatplus.components.INeedConfig; import fr.neatmonster.nocheatplus.components.INeedConfig;
import fr.neatmonster.nocheatplus.components.NameSetPermState;
import fr.neatmonster.nocheatplus.components.NoCheatPlusAPI; import fr.neatmonster.nocheatplus.components.NoCheatPlusAPI;
import fr.neatmonster.nocheatplus.components.PermStateReceiver;
import fr.neatmonster.nocheatplus.config.ConfPaths; import fr.neatmonster.nocheatplus.config.ConfPaths;
import fr.neatmonster.nocheatplus.config.ConfigFile; import fr.neatmonster.nocheatplus.config.ConfigFile;
import fr.neatmonster.nocheatplus.config.ConfigManager; import fr.neatmonster.nocheatplus.config.ConfigManager;
@ -69,11 +76,14 @@ import fr.neatmonster.nocheatplus.utilities.Updates;
/** /**
* This is the main class of NoCheatPlus. The commands, events listeners and tasks are registered here. * This is the main class of NoCheatPlus. The commands, events listeners and tasks are registered here.
*/ */
public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI { public class NoCheatPlus extends JavaPlugin implements NoCheatPlusAPI {
/** Lower case player name to milliseconds point of time of release */ /** Lower case player name to milliseconds point of time of release */
private static final Map<String, Long> denyLoginNames = Collections.synchronizedMap(new HashMap<String, Long>()); private static final Map<String, Long> denyLoginNames = Collections.synchronizedMap(new HashMap<String, Long>());
/** Names of players with a certain permission. */
protected static final NameSetPermState nameSetPerms = new NameSetPermState(Permissions.ADMINISTRATION_NOTIFY);
/** /**
* Remove expired entries. * Remove expired entries.
*/ */
@ -156,6 +166,30 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
return (NoCheatPlusAPI) Bukkit.getPluginManager().getPlugin("NoCheatPlus"); return (NoCheatPlusAPI) Bukkit.getPluginManager().getPlugin("NoCheatPlus");
} }
/**
* Send all players with the nocheatplus.admin.notify permission a message.
* This is likely to be more efficient than iterating over all players and
* checking their permissions. However this only updates permissions with
* login and world changes currently.
*
* @param message
* @return Number of players messaged.
*/
public static int sendAdminNotifyMessage(final String message){
final Set<String> names = nameSetPerms.getPlayers(Permissions.ADMINISTRATION_NOTIFY);
if (names == null) return 0;
int done = 0;
final Server server = Bukkit.getServer();
for (final String name : names){
final Player player = server.getPlayerExact(name);
if (player != null){
player.sendMessage(message);
done ++;
}
}
return done;
}
/** The event listeners. */ /** The event listeners. */
private final List<Listener> listeners = new ArrayList<Listener>(); private final List<Listener> listeners = new ArrayList<Listener>();
@ -183,6 +217,8 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
private boolean manageListeners = true; private boolean manageListeners = true;
protected final List<PermStateReceiver> permStateReceivers = new ArrayList<PermStateReceiver>();
/** /**
* Interfaces checked for managed listeners: IHaveMethodOrder (method), ComponentWithName (tag)<br> * Interfaces checked for managed listeners: IHaveMethodOrder (method), ComponentWithName (tag)<br>
*/ */
@ -197,6 +233,10 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
((INeedConfig) obj).onReload(); ((INeedConfig) obj).onReload();
} }
} }
if (obj instanceof PermStateReceiver){
// No immediate update done.
permStateReceivers.add((PermStateReceiver) obj);
}
dataMan.addComponent(obj); dataMan.addComponent(obj);
} }
@ -204,7 +244,7 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
* Interfaces checked for managed listeners: IHaveMethodOrder (method), ComponentWithName (tag)<br> * Interfaces checked for managed listeners: IHaveMethodOrder (method), ComponentWithName (tag)<br>
* @param listener * @param listener
*/ */
private void addListener(final Listener listener) { protected void addListener(final Listener listener) {
if (manageListeners){ if (manageListeners){
String tag = "NoCheatPlus"; String tag = "NoCheatPlus";
if (listener instanceof ComponentWithName){ if (listener instanceof ComponentWithName){
@ -237,7 +277,12 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
listeners.remove(obj); listeners.remove(obj);
listenerManager.remove((Listener) obj); listenerManager.remove((Listener) obj);
} }
if (obj instanceof PermStateReceiver){
permStateReceivers.remove((PermStateReceiver) obj);
}
if (obj instanceof INotifyReload) {
notifyReload.remove(obj); notifyReload.remove(obj);
}
dataMan.removeComponent(obj); dataMan.removeComponent(obj);
} }
@ -270,15 +315,17 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
// Remove config listeners. // Remove config listeners.
notifyReload.clear(); notifyReload.clear();
permStateReceivers.clear();
// More cleanup. // More cleanup.
dataMan.onDisable(); dataMan.onDisable();
// Cleanup the configuration manager.
ConfigManager.cleanup();
// Restore changed commands. // Restore changed commands.
undoCommandChanges(); undoCommandChanges();
// Cleanup the configuration manager.
ConfigManager.cleanup();
// Remove listener references. // Remove listener references.
listenerManager.setRegisterDirectly(false); listenerManager.setRegisterDirectly(false);
listenerManager.clear(); listenerManager.clear();
@ -300,7 +347,7 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
} }
} }
private void setupCommandProtection() { protected void setupCommandProtection() {
final List<CommandProtectionEntry> changedCommands = PermissionUtil.protectCommands( final List<CommandProtectionEntry> changedCommands = PermissionUtil.protectCommands(
Arrays.asList("plugins", "version", "icanhasbukkit"), "nocheatplus.feature.command", false); Arrays.asList("plugins", "version", "icanhasbukkit"), "nocheatplus.feature.command", false);
if (this.changedCommands == null) this.changedCommands = changedCommands; if (this.changedCommands == null) this.changedCommands = changedCommands;
@ -338,7 +385,8 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
listenerManager.setRegisterDirectly(false); listenerManager.setRegisterDirectly(false);
listenerManager.clear(); listenerManager.clear();
} }
addListener(this); addComponent(nameSetPerms);
addListener(getCoreListener());
for (final Object obj : new Object[]{ for (final Object obj : new Object[]{
NCPExemptionManager.getListener(), NCPExemptionManager.getListener(),
dataMan, dataMan,
@ -347,7 +395,7 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
new BlockPlaceListener(), new BlockPlaceListener(),
new ChatListener(), new ChatListener(),
new CombinedListener(), new CombinedListener(),
// Do ming registration order: Combined must come before Fight. // Do mind registration order: Combined must come before Fight.
new FightListener(), new FightListener(),
new InventoryListener(), new InventoryListener(),
new MovingListener(), new MovingListener(),
@ -463,43 +511,12 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
// configFile.getBoolean(ConfPaths.MISCELLANEOUS_NOMOVEDTOOQUICKLY_USEPROXY, false)); // configFile.getBoolean(ConfPaths.MISCELLANEOUS_NOMOVEDTOOQUICKLY_USEPROXY, false));
// } // }
/**
* This event handler is used to send all the disabling messages to the client.
*
* @param event
* the event handled
*/
@EventHandler(
priority = EventPriority.MONITOR)
public void onPlayerJoinMonitor(final PlayerJoinEvent event) {
/*
* ____ _ _ _
* | _ \| | __ _ _ _ ___ _ __ | | ___ (_)_ __
* | |_) | |/ _` | | | |/ _ \ '__| _ | |/ _ \| | '_ \
* | __/| | (_| | |_| | __/ | | |_| | (_) | | | | |
* |_| |_|\__,_|\__, |\___|_| \___/ \___/|_|_| |_|
* |___/
*/
final Player player = event.getPlayer();
// Send a message to the player if a new update is available.
if (updateAvailable && player.hasPermission(Permissions.ADMINISTRATION_NOTIFY))
player.sendMessage(ChatColor.RED + "NCP: " + ChatColor.WHITE
+ "A new update of NoCheatPlus is available.\n" + "Download it at http://nocheatplus.org/update");
// Send a message to the player if the configuration is outdated.
if (configOutdated && player.hasPermission(Permissions.ADMINISTRATION_NOTIFY))
player.sendMessage(ChatColor.RED + "NCP: " + ChatColor.WHITE + "Your configuration might be outdated.\n"
+ "Some settings could have changed, you should regenerate it!");
checkModsMessage(player);
}
/** /**
* Send block codes to the player according to allowed or disallowed client-mods or client-mod features. * Send block codes to the player according to allowed or disallowed client-mods or client-mod features.
* @param player * @param player
*/ */
private void checkModsMessage(Player player) { protected void checkModsMessage(Player player) {
String message = ""; String message = "";
// Check if we allow all the client mods. // Check if we allow all the client mods.
@ -584,9 +601,17 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
player.sendMessage(message); player.sendMessage(message);
} }
/**
* Quick solution to hide the listener methods, expect refactoring.
* @return
*/
private Listener getCoreListener() {
return new Listener() {
@SuppressWarnings("unused")
@EventHandler(priority = EventPriority.HIGHEST) @EventHandler(priority = EventPriority.HIGHEST)
public void onPlayerLogin(final PlayerLoginEvent event) { public void onPlayerLogin(final PlayerLoginEvent event) {
// (HGHEST to give other plugins the possibility to add permissions or allow the player). // (HGHEST to give other plugins the possibility to add
// permissions or allow the player).
if (event.getResult() != Result.ALLOWED) return; if (event.getResult() != Result.ALLOWED) return;
final Player player = event.getPlayer(); final Player player = event.getPlayer();
// Check if login is denied: // Check if login is denied:
@ -600,4 +625,67 @@ public class NoCheatPlus extends JavaPlugin implements Listener, NoCheatPlusAPI
} }
} }
@SuppressWarnings("unused")
@EventHandler(priority = EventPriority.MONITOR)
public void onPlayerJoin(final PlayerJoinEvent event) {
final Player player = event.getPlayer();
updatePermStateReceivers(player);
if (nameSetPerms.hasPermission(player.getName(), Permissions.ADMINISTRATION_NOTIFY)){
// Login notifications...
// Update available.
if (updateAvailable) player.sendMessage(ChatColor.RED + "NCP: " + ChatColor.WHITE + "A new update of NoCheatPlus is available.\n" + "Download it at http://nocheatplus.org/update");
// Outdated config.
if (configOutdated) player.sendMessage(ChatColor.RED + "NCP: " + ChatColor.WHITE + "Your configuration might be outdated.\n" + "Some settings could have changed, you should regenerate it!");
}
checkModsMessage(player);
}
@SuppressWarnings("unused")
@EventHandler(priority = EventPriority.MONITOR)
public void onPlayerchangedWorld(final PlayerChangedWorldEvent event)
{
final Player player = event.getPlayer();
updatePermStateReceivers(player);
}
@SuppressWarnings("unused")
@EventHandler(priority = EventPriority.MONITOR, ignoreCancelled = true)
public void onPlayerKick(final PlayerKickEvent event) {
onLeave(event.getPlayer());
}
@SuppressWarnings("unused")
@EventHandler(priority = EventPriority.MONITOR)
public void onPlayerQuitMonitor(final PlayerQuitEvent event) {
onLeave(event.getPlayer());
}
};
}
protected void onLeave(final Player player) {
for (final PermStateReceiver pr : permStateReceivers) {
pr.removePlayer(player.getName());
}
}
protected void updatePermStateReceivers(final Player player) {
final Map<String, Boolean> checked = new HashMap<String, Boolean>(20);
final String name = player.getName();
for (final PermStateReceiver pr : permStateReceivers) {
for (final String permission : pr.getDefaultPermissions()) {
Boolean state = checked.get(permission);
if (state == null) {
state = player.hasPermission(permission);
checked.put(permission, state);
}
pr.setPermission(name, permission, state);
}
}
}
} }

View File

@ -0,0 +1,86 @@
package fr.neatmonster.nocheatplus.components;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.Map.Entry;
import java.util.Set;
/**
* Store set of player names by defaultPermissions they have to get players with a certain permission.<br>
* TODO: Might later detach to an interface (+SimpleNameSetPermStateHolder).
* @author mc_dev
*
*/
public class NameSetPermState implements PermStateReceiver{
/** Map permission to player names (all exact case). */
protected final HashMap<String, Set<String>> playerSets = new HashMap<String, Set<String>>();
protected String[] defaultPermissions;
public NameSetPermState(String... permissions){
this.defaultPermissions = permissions;
}
@Override
public String[] getDefaultPermissions() {
return defaultPermissions;
}
@Override
public boolean hasPermission(final String player, final String permission) {
final Set<String> names = playerSets.get(permission);
if (names == null) return false;
return (names.contains(player));
}
@Override
public void setPermission(final String player, final String permission, boolean state) {
Set<String> names = playerSets.get(permission);
if (names == null){
if (!state) return;
names = new LinkedHashSet<String>(20);
playerSets.put(permission, names);
}
if (state) names.add(player);
else names.remove(player);
}
@Override
public void removePlayer(final String player) {
// TODO: Something more efficient ? [mostly used with few defaultPermissions, though].
final Iterator<Entry<String, Set<String>>> it = playerSets.entrySet().iterator();
while (it.hasNext()){
final Entry<String, Set<String>> entry = it.next();
final Set<String> set = entry.getValue();
set.remove(player);
if (set.isEmpty()) it.remove();
}
}
/**
* Get a set with the players that hold the permission.
* @param permission
* @return
*/
public Set<String> getPlayers(final String permission){
return playerSets.get(permission);
}
public void addDefaultPermissions(Collection<String> permissions){
Collection<String> newDefaults = new HashSet<String>();
newDefaults.addAll(Arrays.asList(this.defaultPermissions));
newDefaults.addAll(permissions);
}
public void setDefaultPermissions(Collection<String> permissions){
defaultPermissions = new String[permissions.size()];
permissions.toArray(defaultPermissions);
}
}

View File

@ -0,0 +1,26 @@
package fr.neatmonster.nocheatplus.components;
/**
* Permission cache. Allow to query defaultPermissions (a defined set of defaultPermissions), to be registered and automatically be updated, according to registry.<br>
* The defaultPermissions are not updated in real time but on certain events, to be specified by the registry.
*
* @author mc_dev
*
*/
public interface PermStateHolder {
/**
* Get the defaultPermissions that are guaranteed to be held here.
* @return
*/
public String[] getDefaultPermissions();
/**
* Test a permission. If not available the result will be false, no updating of defaultPermissions is expected on calling this.
* @param player
* @param permission
* @return
*/
public boolean hasPermission(String player, String permission);
}

View File

@ -0,0 +1,16 @@
package fr.neatmonster.nocheatplus.components;
/**
* Receive permission changes for certain defaultPermissions.
* @author mc_dev
*
*/
public interface PermStateReceiver extends PermStateHolder{
public void setPermission(String player, String permission, boolean state);
public void removePlayer(String player);
}