mirror of
https://github.com/NoCheatPlus/NoCheatPlus.git
synced 2025-01-20 06:21:23 +01:00
new check "timed.godmode" with config etc.
This commit is contained in:
parent
67b5505d04
commit
d31525ba3a
@ -31,6 +31,7 @@ import cc.co.evenprime.bukkit.nocheat.events.EventManager;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.events.PlayerChatEventManager;
|
import cc.co.evenprime.bukkit.nocheat.events.PlayerChatEventManager;
|
||||||
import cc.co.evenprime.bukkit.nocheat.events.PlayerMoveEventManager;
|
import cc.co.evenprime.bukkit.nocheat.events.PlayerMoveEventManager;
|
||||||
import cc.co.evenprime.bukkit.nocheat.events.PlayerTeleportEventManager;
|
import cc.co.evenprime.bukkit.nocheat.events.PlayerTeleportEventManager;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.events.TimedEventManager;
|
||||||
import cc.co.evenprime.bukkit.nocheat.log.LogLevel;
|
import cc.co.evenprime.bukkit.nocheat.log.LogLevel;
|
||||||
import cc.co.evenprime.bukkit.nocheat.log.LogManager;
|
import cc.co.evenprime.bukkit.nocheat.log.LogManager;
|
||||||
|
|
||||||
@ -101,6 +102,7 @@ public class NoCheat extends JavaPlugin {
|
|||||||
eventManagers.add(new BlockBreakEventManager(this));
|
eventManagers.add(new BlockBreakEventManager(this));
|
||||||
eventManagers.add(new BlockPlaceEventManager(this));
|
eventManagers.add(new BlockPlaceEventManager(this));
|
||||||
eventManagers.add(new EntityDamageEventManager(this));
|
eventManagers.add(new EntityDamageEventManager(this));
|
||||||
|
eventManagers.add(new TimedEventManager(this));
|
||||||
|
|
||||||
// Then set up a task to monitor server lag
|
// Then set up a task to monitor server lag
|
||||||
if(lagMeasureTask == null) {
|
if(lagMeasureTask == null) {
|
||||||
@ -188,7 +190,8 @@ public class NoCheat extends JavaPlugin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Call this periodically to walk over the stored data map and remove old/unused entries
|
* Call this periodically to walk over the stored data map and remove
|
||||||
|
* old/unused entries
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public void cleanDataMap() {
|
public void cleanDataMap() {
|
||||||
|
@ -0,0 +1,84 @@
|
|||||||
|
package cc.co.evenprime.bukkit.nocheat.checks.timed;
|
||||||
|
|
||||||
|
import net.minecraft.server.EntityPlayer;
|
||||||
|
|
||||||
|
import org.bukkit.craftbukkit.entity.CraftPlayer;
|
||||||
|
import org.bukkit.entity.Player;
|
||||||
|
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.config.cache.ConfigurationCache;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.data.BaseData;
|
||||||
|
|
||||||
|
public class TimedCheck {
|
||||||
|
|
||||||
|
private final NoCheat plugin;
|
||||||
|
|
||||||
|
public TimedCheck(NoCheat plugin) {
|
||||||
|
|
||||||
|
this.plugin = plugin;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void check(Player player, int tickTime, ConfigurationCache cc) {
|
||||||
|
|
||||||
|
// server lag(ged), skip this
|
||||||
|
if(plugin.skipCheck())
|
||||||
|
return;
|
||||||
|
|
||||||
|
boolean cancel = false;
|
||||||
|
|
||||||
|
BaseData data = plugin.getData(player.getName());
|
||||||
|
|
||||||
|
EntityPlayer p = ((CraftPlayer) player).getHandle();
|
||||||
|
|
||||||
|
// Compare ingame record of players ticks to our last observed value
|
||||||
|
int difference = p.ticksLived - data.timed.ticksLived;
|
||||||
|
|
||||||
|
// difference should be >= tickTime for perfect synchronization
|
||||||
|
if(difference > tickTime) {
|
||||||
|
// player was faster than expected, give him credit for the
|
||||||
|
// difference
|
||||||
|
data.timed.ticksBehind -= (difference - tickTime);
|
||||||
|
// Reduce violation level over time
|
||||||
|
data.timed.godmodeVL *= 0.90D;
|
||||||
|
|
||||||
|
} else if(difference >= tickTime / 2) {
|
||||||
|
// close enough, let it pass
|
||||||
|
|
||||||
|
// Reduce violation level over time
|
||||||
|
data.timed.godmodeVL *= 0.95D;
|
||||||
|
} else {
|
||||||
|
// That's a bit suspicious, why is the player more than half the
|
||||||
|
// ticktime behind? Keep that in mind
|
||||||
|
data.timed.ticksBehind += tickTime - difference;
|
||||||
|
|
||||||
|
// Is he way too far behind, then correct that
|
||||||
|
if(data.timed.ticksBehind > cc.timed.godmodeTicksLimit) {
|
||||||
|
|
||||||
|
data.timed.godmodeVL += tickTime - difference;
|
||||||
|
|
||||||
|
// Enough is enough
|
||||||
|
data.log.check = "timed.godmode";
|
||||||
|
data.log.godmodeTicksBehind = data.timed.ticksBehind;
|
||||||
|
|
||||||
|
cancel = plugin.execute(player, cc.timed.godmodeActions, (int) data.timed.godmodeVL, data.timed.history, cc);
|
||||||
|
|
||||||
|
// Reduce the time the player is behind accordingly
|
||||||
|
data.timed.ticksBehind -= tickTime;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(data.timed.ticksBehind < 0) {
|
||||||
|
data.timed.ticksBehind = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(cancel) {
|
||||||
|
// Catch up for at least some of the ticks
|
||||||
|
for(int i = 0; i < tickTime; i++) {
|
||||||
|
p.b(true); // Catch up with the server, one tick at a time
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// setup data for next time
|
||||||
|
data.timed.ticksLived = p.ticksLived;
|
||||||
|
}
|
||||||
|
}
|
@ -107,6 +107,14 @@ public abstract class Configuration {
|
|||||||
public static final OptionNode FIGHT_SELFHIT_CHECK = new OptionNode("check", FIGHT_SELFHIT, DataType.BOOLEAN);
|
public static final OptionNode FIGHT_SELFHIT_CHECK = new OptionNode("check", FIGHT_SELFHIT, DataType.BOOLEAN);
|
||||||
public final static OptionNode FIGHT_SELFHIT_ACTIONS = new OptionNode("actions", FIGHT_SELFHIT, DataType.ACTIONLIST);
|
public final static OptionNode FIGHT_SELFHIT_ACTIONS = new OptionNode("actions", FIGHT_SELFHIT, DataType.ACTIONLIST);
|
||||||
|
|
||||||
|
private static final OptionNode TIMED = new OptionNode("timed", ROOT, DataType.PARENT);
|
||||||
|
public static final OptionNode TIMED_CHECK = new OptionNode("check", TIMED, DataType.BOOLEAN);
|
||||||
|
|
||||||
|
private static final OptionNode TIMED_GODMODE = new OptionNode("godmode", TIMED, DataType.PARENT);
|
||||||
|
public static final OptionNode TIMED_GODMODE_CHECK = new OptionNode("check", TIMED_GODMODE, DataType.BOOLEAN);
|
||||||
|
public static final OptionNode TIMED_GODMODE_TICKSLIMIT = new OptionNode("tickslimit", TIMED_GODMODE, DataType.INTEGER);
|
||||||
|
public static final OptionNode TIMED_GODMODE_ACTIONS = new OptionNode("actions", TIMED_GODMODE, DataType.ACTIONLIST);
|
||||||
|
|
||||||
private final Map<OptionNode, Object> values;
|
private final Map<OptionNode, Object> values;
|
||||||
private final Configuration defaults;
|
private final Configuration defaults;
|
||||||
|
|
||||||
|
@ -164,6 +164,19 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
selfhitActionList.setActions(0, action.getActions("fightSelfhitLog fightCancel".split(" ")));
|
selfhitActionList.setActions(0, action.getActions("fightSelfhitLog fightCancel".split(" ")));
|
||||||
setValue(FIGHT_SELFHIT_ACTIONS, selfhitActionList);
|
setValue(FIGHT_SELFHIT_ACTIONS, selfhitActionList);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*** TIMED ***/
|
||||||
|
{
|
||||||
|
setValue(TIMED_CHECK, true);
|
||||||
|
|
||||||
|
setValue(TIMED_GODMODE_CHECK, true);
|
||||||
|
setValue(TIMED_GODMODE_TICKSLIMIT, 50);
|
||||||
|
|
||||||
|
ActionList directionActionList = new ActionList();
|
||||||
|
directionActionList.setActions(0, action.getActions("godmodeCancel".split(" ")));
|
||||||
|
directionActionList.setActions(100, action.getActions("godmodeLog godmodeCancel".split(" ")));
|
||||||
|
setValue(TIMED_GODMODE_ACTIONS, directionActionList);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void writeActionFile(File file) {
|
public static void writeActionFile(File file) {
|
||||||
@ -242,6 +255,7 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
w(w, "log onliquidLog 2 5 med [player] failed [check]: tried to place a [blocktype] block at [placelocation] against block at [placeagainst]. VL [violations]");
|
w(w, "log onliquidLog 2 5 med [player] failed [check]: tried to place a [blocktype] block at [placelocation] against block at [placeagainst]. VL [violations]");
|
||||||
w(w, "log spamLog 0 5 med [player] failed [check]: Last sent message \"[text]\". VL [violations]");
|
w(w, "log spamLog 0 5 med [player] failed [check]: Last sent message \"[text]\". VL [violations]");
|
||||||
w(w, "log nofallLog 0 5 med [player] failed [check]: tried to avoid fall damage for ~[falldistance] blocks. VL [violations]");
|
w(w, "log nofallLog 0 5 med [player] failed [check]: tried to avoid fall damage for ~[falldistance] blocks. VL [violations]");
|
||||||
|
w(w, "log godmodeLog 0 5 med [player] failed [check]: lagging or using godmode. VL [violations]");
|
||||||
w(w, "");
|
w(w, "");
|
||||||
w(w, "");
|
w(w, "");
|
||||||
w(w, "# Some log messages related to fighting, displaying the same text, but with different level (Info, Warning, Severe)");
|
w(w, "# Some log messages related to fighting, displaying the same text, but with different level (Info, Warning, Severe)");
|
||||||
@ -265,6 +279,7 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
w(w, "special spamCancel 0 0");
|
w(w, "special spamCancel 0 0");
|
||||||
w(w, "special nofallDamage 0 0");
|
w(w, "special nofallDamage 0 0");
|
||||||
w(w, "special fightCancel 0 0");
|
w(w, "special fightCancel 0 0");
|
||||||
|
w(w, "special godmodeCancel 0 0");
|
||||||
w(w, "");
|
w(w, "");
|
||||||
w(w, "# CONSOLECOMMAND Actions: They will execute a command as if it were typed into the console.");
|
w(w, "# CONSOLECOMMAND Actions: They will execute a command as if it were typed into the console.");
|
||||||
w(w, "# - They start with the word 'consolecommand'");
|
w(w, "# - They start with the word 'consolecommand'");
|
||||||
|
@ -33,6 +33,9 @@ public class Permissions {
|
|||||||
public static final String FIGHT_DIRECTION = FIGHT + ".direction";
|
public static final String FIGHT_DIRECTION = FIGHT + ".direction";
|
||||||
public static final String FIGHT_SELFHIT = FIGHT + ".selfhit";
|
public static final String FIGHT_SELFHIT = FIGHT + ".selfhit";
|
||||||
|
|
||||||
|
public static final String TIMED = CHECKS + ".timed";
|
||||||
|
public static final String TIMED_GODMODE = TIMED + ".godmode";
|
||||||
|
|
||||||
public final static String ADMIN_CHATLOG = ADMIN + ".chatlog";
|
public final static String ADMIN_CHATLOG = ADMIN + ".chatlog";
|
||||||
public static final String ADMIN_PERMLIST = ADMIN + ".permlist";
|
public static final String ADMIN_PERMLIST = ADMIN + ".permlist";
|
||||||
public static final String ADMIN_RELOAD = ADMIN + ".reload";
|
public static final String ADMIN_RELOAD = ADMIN + ".reload";
|
||||||
|
20
src/cc/co/evenprime/bukkit/nocheat/config/cache/CCTimed.java
vendored
Normal file
20
src/cc/co/evenprime/bukkit/nocheat/config/cache/CCTimed.java
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
package cc.co.evenprime.bukkit.nocheat.config.cache;
|
||||||
|
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.config.Configuration;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.config.util.ActionList;
|
||||||
|
|
||||||
|
public class CCTimed {
|
||||||
|
|
||||||
|
public final boolean check;
|
||||||
|
public final boolean godmodeCheck;
|
||||||
|
public final double godmodeTicksLimit;
|
||||||
|
public final ActionList godmodeActions;
|
||||||
|
|
||||||
|
public CCTimed(Configuration data) {
|
||||||
|
|
||||||
|
check = data.getBoolean(Configuration.TIMED_CHECK);
|
||||||
|
godmodeCheck = data.getBoolean(Configuration.TIMED_GODMODE_CHECK);
|
||||||
|
godmodeTicksLimit = data.getInteger(Configuration.TIMED_GODMODE_TICKSLIMIT);
|
||||||
|
godmodeActions = data.getActionList(Configuration.TIMED_GODMODE_ACTIONS);
|
||||||
|
}
|
||||||
|
}
|
@ -18,6 +18,7 @@ public class ConfigurationCache {
|
|||||||
public final CCChat chat;
|
public final CCChat chat;
|
||||||
public final CCDebug debug;
|
public final CCDebug debug;
|
||||||
public final CCFight fight;
|
public final CCFight fight;
|
||||||
|
public final CCTimed timed;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Instantiate a config cache and populate it with the data of a
|
* Instantiate a config cache and populate it with the data of a
|
||||||
@ -34,6 +35,7 @@ public class ConfigurationCache {
|
|||||||
logging = new CCLogging(data, worldSpecificFileLogger);
|
logging = new CCLogging(data, worldSpecificFileLogger);
|
||||||
debug = new CCDebug(data);
|
debug = new CCDebug(data);
|
||||||
fight = new CCFight(data);
|
fight = new CCFight(data);
|
||||||
|
timed = new CCTimed(data);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,7 @@ public class BaseData extends Data {
|
|||||||
public final LogData log;
|
public final LogData log;
|
||||||
public final MovingData moving;
|
public final MovingData moving;
|
||||||
public final FightData fight;
|
public final FightData fight;
|
||||||
|
public final TimedData timed;
|
||||||
|
|
||||||
private final Data[] data; // for convenience
|
private final Data[] data; // for convenience
|
||||||
|
|
||||||
@ -20,9 +21,10 @@ public class BaseData extends Data {
|
|||||||
this.log = new LogData();
|
this.log = new LogData();
|
||||||
this.moving = new MovingData();
|
this.moving = new MovingData();
|
||||||
this.fight = new FightData();
|
this.fight = new FightData();
|
||||||
|
this.timed = new TimedData();
|
||||||
|
|
||||||
data = new Data[] {this.blockbreak, this.blockplace, this.chat,
|
data = new Data[] {this.blockbreak, this.blockplace, this.chat,
|
||||||
this.log, this.moving, this.fight};
|
this.log, this.moving, this.fight, this.timed};
|
||||||
}
|
}
|
||||||
|
|
||||||
public void clearCriticalData() {
|
public void clearCriticalData() {
|
||||||
|
@ -18,5 +18,6 @@ public class LogData extends Data {
|
|||||||
public double reachdistance;
|
public double reachdistance;
|
||||||
public float falldistance;
|
public float falldistance;
|
||||||
public String playerName;
|
public String playerName;
|
||||||
|
public int godmodeTicksBehind;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
17
src/cc/co/evenprime/bukkit/nocheat/data/TimedData.java
Normal file
17
src/cc/co/evenprime/bukkit/nocheat/data/TimedData.java
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
package cc.co.evenprime.bukkit.nocheat.data;
|
||||||
|
|
||||||
|
public class TimedData extends Data {
|
||||||
|
|
||||||
|
public int ticksLived;
|
||||||
|
public int ticksBehind;
|
||||||
|
public double godmodeVL;
|
||||||
|
public final ExecutionHistory history = new ExecutionHistory(); ;
|
||||||
|
|
||||||
|
public TimedData() {}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void clearCriticalData() {
|
||||||
|
ticksBehind = 0;
|
||||||
|
ticksLived = 0;
|
||||||
|
}
|
||||||
|
}
|
@ -6,7 +6,7 @@ import java.util.Map;
|
|||||||
public class PerformanceManager {
|
public class PerformanceManager {
|
||||||
|
|
||||||
public enum Type {
|
public enum Type {
|
||||||
BLOCKBREAK, BLOCKDAMAGE, BLOCKPLACE, CHAT, MOVING, VELOCITY, FIGHT
|
BLOCKBREAK, BLOCKDAMAGE, BLOCKPLACE, CHAT, MOVING, VELOCITY, FIGHT, TIMED
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -0,0 +1,81 @@
|
|||||||
|
package cc.co.evenprime.bukkit.nocheat.events;
|
||||||
|
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.bukkit.entity.Player;
|
||||||
|
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.checks.timed.TimedCheck;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.config.cache.ConfigurationCache;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.debug.Performance;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.debug.PerformanceManager.Type;
|
||||||
|
|
||||||
|
public class TimedEventManager implements EventManager {
|
||||||
|
|
||||||
|
private final NoCheat plugin;
|
||||||
|
|
||||||
|
private final TimedCheck check;
|
||||||
|
|
||||||
|
private final Performance timedPerformance;
|
||||||
|
|
||||||
|
public TimedEventManager(final NoCheat plugin) {
|
||||||
|
this.plugin = plugin;
|
||||||
|
|
||||||
|
check = new TimedCheck(plugin);
|
||||||
|
|
||||||
|
this.timedPerformance = plugin.getPerformance(Type.TIMED);
|
||||||
|
|
||||||
|
// "register a listener" for passed time
|
||||||
|
plugin.getServer().getScheduler().scheduleSyncRepeatingTask(plugin, new Runnable() {
|
||||||
|
|
||||||
|
private int executions = 0;
|
||||||
|
private int loopsize = 10;
|
||||||
|
|
||||||
|
public void run() {
|
||||||
|
|
||||||
|
executions++;
|
||||||
|
|
||||||
|
if(executions >= loopsize) {
|
||||||
|
executions = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for(Player p : plugin.getServer().getOnlinePlayers()) {
|
||||||
|
if((p.hashCode() & 0x7FFFFFFF) % loopsize == executions) {
|
||||||
|
onTimedEvent(p, loopsize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, 0, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void onTimedEvent(Player player, int elapsedTicks) {
|
||||||
|
|
||||||
|
// Performance counter setup
|
||||||
|
long nanoTimeStart = 0;
|
||||||
|
final boolean performanceCheck = timedPerformance.isEnabled();
|
||||||
|
|
||||||
|
if(performanceCheck)
|
||||||
|
nanoTimeStart = System.nanoTime();
|
||||||
|
|
||||||
|
ConfigurationCache cc = plugin.getConfig(player);
|
||||||
|
|
||||||
|
if(cc.timed.check && !player.hasPermission(Permissions.TIMED)) {
|
||||||
|
check.check(player, elapsedTicks, cc);
|
||||||
|
}
|
||||||
|
|
||||||
|
// store performance time
|
||||||
|
if(performanceCheck)
|
||||||
|
timedPerformance.addTime(System.nanoTime() - nanoTimeStart);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<String> getActiveChecks(ConfigurationCache cc) {
|
||||||
|
LinkedList<String> s = new LinkedList<String>();
|
||||||
|
|
||||||
|
if(cc.timed.check && cc.timed.godmodeCheck)
|
||||||
|
s.add("timed.godmode");
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user