mirror of
https://github.com/NoCheatPlus/NoCheatPlus.git
synced 2025-01-04 23:07:44 +01:00
Removed "godmode" check, as it is FUBAR in MC 1.0.0, may return later
in different form. Added "emptyChat" check, looking for empty chat messages.
This commit is contained in:
parent
5ce7bc04cd
commit
a3479b0f58
@ -70,6 +70,8 @@ permissions:
|
|||||||
children:
|
children:
|
||||||
nocheat.checks.chat.spam:
|
nocheat.checks.chat.spam:
|
||||||
description: Allow a player to send an infinite amount of chat messages
|
description: Allow a player to send an infinite amount of chat messages
|
||||||
|
nocheat.checks.chat.empty:
|
||||||
|
description: Allow a player to send empty messages (normally not possible)
|
||||||
nocheat.checks.fight:
|
nocheat.checks.fight:
|
||||||
description: Allow the player to bypass all fight checks
|
description: Allow the player to bypass all fight checks
|
||||||
children:
|
children:
|
||||||
@ -79,8 +81,3 @@ permissions:
|
|||||||
description: Allow a player to attack himself with close combat attacks (punching, swords, etc.)
|
description: Allow a player to attack himself with close combat attacks (punching, swords, etc.)
|
||||||
nocheat.checks.fight.noswing:
|
nocheat.checks.fight.noswing:
|
||||||
description: Allow a player to fight without swinging their arm
|
description: Allow a player to fight without swinging their arm
|
||||||
nocheat.checks.timed:
|
|
||||||
description: Allow the player to bypass all timed checks
|
|
||||||
children:
|
|
||||||
nocheat.checks.timed.godmode:
|
|
||||||
description: Allow a player to use the godmode hack to make himself invincible to damage
|
|
||||||
|
2
pom.xml
2
pom.xml
@ -3,7 +3,7 @@
|
|||||||
<modelVersion>4.0.0</modelVersion>
|
<modelVersion>4.0.0</modelVersion>
|
||||||
<groupId>cc.co.evenprime.bukkit</groupId>
|
<groupId>cc.co.evenprime.bukkit</groupId>
|
||||||
<artifactId>NoCheat</artifactId>
|
<artifactId>NoCheat</artifactId>
|
||||||
<version>2.17b</version>
|
<version>2.18</version>
|
||||||
<packaging>jar</packaging>
|
<packaging>jar</packaging>
|
||||||
<name>NoCheat</name>
|
<name>NoCheat</name>
|
||||||
<properties>
|
<properties>
|
||||||
|
@ -26,7 +26,6 @@ import cc.co.evenprime.bukkit.nocheat.events.ChatEventManager;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.events.EventManagerImpl;
|
import cc.co.evenprime.bukkit.nocheat.events.EventManagerImpl;
|
||||||
import cc.co.evenprime.bukkit.nocheat.events.FightEventManager;
|
import cc.co.evenprime.bukkit.nocheat.events.FightEventManager;
|
||||||
import cc.co.evenprime.bukkit.nocheat.events.MovingEventManager;
|
import cc.co.evenprime.bukkit.nocheat.events.MovingEventManager;
|
||||||
import cc.co.evenprime.bukkit.nocheat.events.TimedEventManager;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.events.WorkaroundsEventManager;
|
import cc.co.evenprime.bukkit.nocheat.events.WorkaroundsEventManager;
|
||||||
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;
|
||||||
@ -119,10 +118,6 @@ public class NoCheat extends JavaPlugin {
|
|||||||
eventManagers.add(new BlockPlaceEventManager(this));
|
eventManagers.add(new BlockPlaceEventManager(this));
|
||||||
eventManagers.add(new FightEventManager(this));
|
eventManagers.add(new FightEventManager(this));
|
||||||
|
|
||||||
TimedEventManager m = new TimedEventManager(this);
|
|
||||||
taskId = m.taskId; // There's a bukkit task, remember its id
|
|
||||||
eventManagers.add(m);
|
|
||||||
|
|
||||||
// Then set up a task to monitor server lag
|
// Then set up a task to monitor server lag
|
||||||
if(lagMeasureTask == null) {
|
if(lagMeasureTask == null) {
|
||||||
lagMeasureTask = new LagMeasureTask(this);
|
lagMeasureTask = new LagMeasureTask(this);
|
||||||
|
@ -21,8 +21,6 @@ public interface NoCheatPlayer {
|
|||||||
|
|
||||||
public int getTicksLived();
|
public int getTicksLived();
|
||||||
|
|
||||||
public void increaseAge(int ticks);
|
|
||||||
|
|
||||||
public ConfigurationCache getConfiguration();
|
public ConfigurationCache getConfiguration();
|
||||||
|
|
||||||
public float getSpeedAmplifier();
|
public float getSpeedAmplifier();
|
||||||
|
@ -1,23 +0,0 @@
|
|||||||
package cc.co.evenprime.bukkit.nocheat.checks;
|
|
||||||
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.cache.CCTimed;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.ExecutionHistory;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.TimedData;
|
|
||||||
|
|
||||||
public abstract class TimedCheck extends Check {
|
|
||||||
|
|
||||||
public TimedCheck(NoCheat plugin, String name, String permission) {
|
|
||||||
super(plugin, name, permission);
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract void check(final NoCheatPlayer player, TimedData data, CCTimed cc);
|
|
||||||
|
|
||||||
public abstract boolean isEnabled(CCTimed cc);
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected ExecutionHistory getHistory(NoCheatPlayer player) {
|
|
||||||
return player.getData().timed.history;
|
|
||||||
}
|
|
||||||
}
|
|
@ -0,0 +1,47 @@
|
|||||||
|
package cc.co.evenprime.bukkit.nocheat.checks.chat;
|
||||||
|
|
||||||
|
import java.util.Locale;
|
||||||
|
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.actions.types.ActionWithParameters.WildCard;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.checks.ChatCheck;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.config.cache.CCChat;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.data.ChatData;
|
||||||
|
|
||||||
|
public class EmptyCheck extends ChatCheck {
|
||||||
|
|
||||||
|
public EmptyCheck(NoCheat plugin) {
|
||||||
|
super(plugin, "chat.empty", Permissions.CHAT_EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean check(NoCheatPlayer player, ChatData data, CCChat cc) {
|
||||||
|
|
||||||
|
boolean cancel = false;
|
||||||
|
|
||||||
|
if(data.message.trim().length() == 0) {
|
||||||
|
|
||||||
|
data.emptyVL += 1;
|
||||||
|
cancel = executeActions(player, cc.emptyActions.getActions(data.emptyVL));
|
||||||
|
}
|
||||||
|
|
||||||
|
return cancel;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isEnabled(CCChat cc) {
|
||||||
|
return cc.emptyCheck;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getParameter(WildCard wildcard, NoCheatPlayer player) {
|
||||||
|
|
||||||
|
switch (wildcard) {
|
||||||
|
|
||||||
|
case VIOLATIONS:
|
||||||
|
return String.format(Locale.US, "%d", player.getData().chat.emptyVL);
|
||||||
|
default:
|
||||||
|
return super.getParameter(wildcard, player);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,112 +0,0 @@
|
|||||||
package cc.co.evenprime.bukkit.nocheat.checks.timed;
|
|
||||||
|
|
||||||
import java.util.Locale;
|
|
||||||
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.actions.types.ActionWithParameters.WildCard;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.checks.TimedCheck;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.cache.CCTimed;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.TimedData;
|
|
||||||
|
|
||||||
public class GodmodeCheck extends TimedCheck {
|
|
||||||
|
|
||||||
public GodmodeCheck(NoCheat plugin) {
|
|
||||||
super(plugin, "timed.godmode", Permissions.TIMED_GODMODE);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void check(NoCheatPlayer player, TimedData data, CCTimed cc) {
|
|
||||||
// server lag(ged), skip this, or player dead, therefore it's reasonable
|
|
||||||
// for him to not move :)
|
|
||||||
if(plugin.skipCheck() || player.isDead()) {
|
|
||||||
data.ticksBehind = 0;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final int ticksLived = player.getTicksLived();
|
|
||||||
|
|
||||||
// Haven't been checking before
|
|
||||||
if(data.ticksLived == 0) {
|
|
||||||
// setup data for next time
|
|
||||||
data.ticksLived = ticksLived;
|
|
||||||
|
|
||||||
// And give up already
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean cancel = false;
|
|
||||||
|
|
||||||
// How far behind is the player with his ticks
|
|
||||||
int behind = Math.min(10, (data.ticksLived + cc.tickTime) - ticksLived);
|
|
||||||
// difference should be >= tickTime for perfect synchronization
|
|
||||||
if(behind <= 1) {
|
|
||||||
// player as fast as expected, give him credit for that
|
|
||||||
data.ticksBehind -= cc.tickTime / 2;
|
|
||||||
// Reduce violation level over time
|
|
||||||
data.godmodeVL -= cc.tickTime / 2.0;
|
|
||||||
|
|
||||||
} else if(behind <= (cc.tickTime / 2) + 1) {
|
|
||||||
// close enough, let it pass
|
|
||||||
data.ticksBehind -= cc.tickTime / 4;
|
|
||||||
// Reduce violation level over time
|
|
||||||
data.godmodeVL -= cc.tickTime / 4.0;
|
|
||||||
} else {
|
|
||||||
// That's a bit suspicious, why is the player more than half the
|
|
||||||
// ticktime behind? Keep that in mind
|
|
||||||
data.ticksBehind += behind;
|
|
||||||
|
|
||||||
// Is he way too far behind, then correct that
|
|
||||||
if(data.ticksBehind > cc.godmodeTicksLimit) {
|
|
||||||
|
|
||||||
// Over the limit, start increasing VL for the player
|
|
||||||
data.godmodeVL += behind;
|
|
||||||
|
|
||||||
cancel = executeActions(player, cc.godmodeActions.getActions(data.godmodeVL));
|
|
||||||
|
|
||||||
if(cancel) {
|
|
||||||
// Catch up for at least some of the ticks
|
|
||||||
try {
|
|
||||||
player.increaseAge(cc.tickTime);
|
|
||||||
} catch(Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reduce the time the player is behind accordingly
|
|
||||||
data.ticksBehind -= cc.tickTime;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if(data.ticksBehind < 0) {
|
|
||||||
data.ticksBehind = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(data.godmodeVL < 0) {
|
|
||||||
data.godmodeVL = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// setup data for next time
|
|
||||||
data.ticksLived = player.getTicksLived();
|
|
||||||
|
|
||||||
return;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean isEnabled(CCTimed cc) {
|
|
||||||
return cc.godmodeCheck;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getParameter(WildCard wildcard, NoCheatPlayer player) {
|
|
||||||
|
|
||||||
switch (wildcard) {
|
|
||||||
|
|
||||||
case VIOLATIONS:
|
|
||||||
return String.format(Locale.US, "%d", (int) player.getData().timed.godmodeVL);
|
|
||||||
default:
|
|
||||||
return super.getParameter(wildcard, player);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -99,6 +99,10 @@ public abstract class Configuration {
|
|||||||
public final static OptionNode CHAT_SPAM_LIMIT = new OptionNode("limit", CHAT_SPAM, DataType.INTEGER);
|
public final static OptionNode CHAT_SPAM_LIMIT = new OptionNode("limit", CHAT_SPAM, DataType.INTEGER);
|
||||||
public final static OptionNode CHAT_SPAM_ACTIONS = new OptionNode("actions", CHAT_SPAM, DataType.ACTIONLIST);
|
public final static OptionNode CHAT_SPAM_ACTIONS = new OptionNode("actions", CHAT_SPAM, DataType.ACTIONLIST);
|
||||||
|
|
||||||
|
private final static OptionNode CHAT_EMPTY = new OptionNode("empty", CHAT, DataType.PARENT);
|
||||||
|
public static final OptionNode CHAT_EMPTY_CHECK = new OptionNode("check", CHAT_EMPTY, DataType.BOOLEAN);
|
||||||
|
public static final OptionNode CHAT_EMPTY_ACTIONS = new OptionNode("actions", CHAT_EMPTY, DataType.ACTIONLIST);
|
||||||
|
|
||||||
private final static OptionNode FIGHT = new OptionNode("fight", ROOT, DataType.PARENT);
|
private final static OptionNode FIGHT = new OptionNode("fight", ROOT, DataType.PARENT);
|
||||||
public final static OptionNode FIGHT_CHECK = new OptionNode("check", FIGHT, DataType.BOOLEAN);
|
public final static OptionNode FIGHT_CHECK = new OptionNode("check", FIGHT, DataType.BOOLEAN);
|
||||||
|
|
||||||
@ -116,14 +120,6 @@ public abstract class Configuration {
|
|||||||
public static final OptionNode FIGHT_NOSWING_CHECK = new OptionNode("check", FIGHT_NOSWING, DataType.BOOLEAN);
|
public static final OptionNode FIGHT_NOSWING_CHECK = new OptionNode("check", FIGHT_NOSWING, DataType.BOOLEAN);
|
||||||
public static final OptionNode FIGHT_NOSWING_ACTIONS = new OptionNode("actions", FIGHT_NOSWING, DataType.ACTIONLIST);
|
public static final OptionNode FIGHT_NOSWING_ACTIONS = new OptionNode("actions", FIGHT_NOSWING, 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;
|
||||||
|
|
||||||
|
@ -141,9 +141,15 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
setValue(CHAT_SPAM_LIMIT, 5);
|
setValue(CHAT_SPAM_LIMIT, 5);
|
||||||
|
|
||||||
ActionList spamActionList = new ActionList();
|
ActionList spamActionList = new ActionList();
|
||||||
spamActionList.setActions(0, action.getActions("spamLog spamCancel".split(" ")));
|
spamActionList.setActions(0, action.getActions("spamLog chatCancel".split(" ")));
|
||||||
spamActionList.setActions(50, action.getActions("spamLog spamCancel spamkick".split(" ")));
|
spamActionList.setActions(50, action.getActions("spamLog chatCancel spamkick".split(" ")));
|
||||||
setValue(CHAT_SPAM_ACTIONS, spamActionList);
|
setValue(CHAT_SPAM_ACTIONS, spamActionList);
|
||||||
|
|
||||||
|
setValue(CHAT_EMPTY_CHECK, true);
|
||||||
|
|
||||||
|
ActionList emptyActionList = new ActionList();
|
||||||
|
emptyActionList.setActions(0, action.getActions("emptyChatLog chatCancel emptyChatKick".split(" ")));
|
||||||
|
setValue(CHAT_EMPTY_ACTIONS, emptyActionList);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*** FIGHT ***/
|
/*** FIGHT ***/
|
||||||
@ -171,19 +177,6 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
noswingActionList.setActions(0, action.getActions("noswingLog fightCancel".split(" ")));
|
noswingActionList.setActions(0, action.getActions("noswingLog fightCancel".split(" ")));
|
||||||
setValue(FIGHT_NOSWING_ACTIONS, noswingActionList);
|
setValue(FIGHT_NOSWING_ACTIONS, noswingActionList);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*** 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) {
|
||||||
@ -259,11 +252,10 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
w(w, "# Some other log messages that are limited a bit by default, to avoid too extreme spam");
|
w(w, "# Some other log messages that are limited a bit by default, to avoid too extreme spam");
|
||||||
w(w, "log reachLog 0 5 med [player] failed [check]: tried to interact with a block over distance [reachdistance]. VL [violations]");
|
w(w, "log reachLog 0 5 med [player] failed [check]: tried to interact with a block over distance [reachdistance]. VL [violations]");
|
||||||
w(w, "log directionLog 2 5 med [player] failed [check]: tried to interact with a block out of line of sight. VL [violations]");
|
w(w, "log directionLog 2 5 med [player] failed [check]: tried to interact with a block out of line of sight. 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, "log noswingLog 2 5 med [player] failed [check]: Didn't swing arm. VL [violations]");
|
w(w, "log noswingLog 2 5 med [player] failed [check]: Didn't swing arm. VL [violations]");
|
||||||
|
w(w, "log emptyChatLog 0 5 med [player] failed [check]: Sent empty chat message. 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)");
|
||||||
@ -285,9 +277,9 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
w(w, "special blockbreakCancel 0 0");
|
w(w, "special blockbreakCancel 0 0");
|
||||||
w(w, "special blockplaceCancel 0 0");
|
w(w, "special blockplaceCancel 0 0");
|
||||||
w(w, "special spamCancel 0 0");
|
w(w, "special spamCancel 0 0");
|
||||||
|
w(w, "special chatCancel 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'");
|
||||||
@ -299,6 +291,7 @@ public class DefaultConfiguration extends Configuration {
|
|||||||
w(w, "# E.g. Kick a player");
|
w(w, "# E.g. Kick a player");
|
||||||
w(w, "consolecommand kick 0 1 kick [player]");
|
w(w, "consolecommand kick 0 1 kick [player]");
|
||||||
w(w, "consolecommand spamkick 0 1 kick [player]");
|
w(w, "consolecommand spamkick 0 1 kick [player]");
|
||||||
|
w(w, "consolecommand emptyChatKick 0 1 kick [player]");
|
||||||
w.flush();
|
w.flush();
|
||||||
w.close();
|
w.close();
|
||||||
} catch(IOException e) {
|
} catch(IOException e) {
|
||||||
|
@ -84,6 +84,9 @@ public class Explainations {
|
|||||||
set(Configuration.CHAT_SPAM_LIMIT, "How many messages per timeframe may the player send without it counting as spamming?");
|
set(Configuration.CHAT_SPAM_LIMIT, "How many messages per timeframe may the player send without it counting as spamming?");
|
||||||
set(Configuration.CHAT_SPAM_ACTIONS, "What should be done if a player is trying to spam the chat.\nUnit is number of chat messages above the limit you declared above.");
|
set(Configuration.CHAT_SPAM_ACTIONS, "What should be done if a player is trying to spam the chat.\nUnit is number of chat messages above the limit you declared above.");
|
||||||
|
|
||||||
|
set(Configuration.CHAT_EMPTY_CHECK, "If true, check if a player is sending an empty message.");
|
||||||
|
set(Configuration.CHAT_EMPTY_ACTIONS, "What should be done if a player sends an empty message.\nUnit is number of empty chat messages sent by the player.");
|
||||||
|
|
||||||
set(Configuration.FIGHT_CHECK, "If true, do various checks on Events related to fighting.");
|
set(Configuration.FIGHT_CHECK, "If true, do various checks on Events related to fighting.");
|
||||||
set(Configuration.FIGHT_DIRECTION_CHECK, "If true, check if a player is really looking at enemies that he attacks.");
|
set(Configuration.FIGHT_DIRECTION_CHECK, "If true, check if a player is really looking at enemies that he attacks.");
|
||||||
set(Configuration.FIGHT_DIRECTION_PRECISION, "Set how precise the check should be. If you experience the check to be too zealous, increase this value. \nIf you want to make it tighter, reduce this value. Default is 100.");
|
set(Configuration.FIGHT_DIRECTION_PRECISION, "Set how precise the check should be. If you experience the check to be too zealous, increase this value. \nIf you want to make it tighter, reduce this value. Default is 100.");
|
||||||
@ -96,11 +99,6 @@ public class Explainations {
|
|||||||
set(Configuration.FIGHT_NOSWING_CHECK, "If true, check if a player swung his arm before attacking, which he should have done.");
|
set(Configuration.FIGHT_NOSWING_CHECK, "If true, check if a player swung his arm before attacking, which he should have done.");
|
||||||
set(Configuration.FIGHT_NOSWING_ACTIONS, "What should be done if a player didn't swing his arm.\nUnit is number of attacks without armswinging.");
|
set(Configuration.FIGHT_NOSWING_ACTIONS, "What should be done if a player didn't swing his arm.\nUnit is number of attacks without armswinging.");
|
||||||
|
|
||||||
set(Configuration.TIMED_CHECK, "If true, do various checks on things related to server and client time.");
|
|
||||||
set(Configuration.TIMED_GODMODE_CHECK, "If true, check or prevent if a player made himself invulnerable by exploiting a time-related bug.\nThis 'godmode' exploit looks similar to a player with huge lag, so be careful when punishing people for it.");
|
|
||||||
set(Configuration.TIMED_GODMODE_TICKSLIMIT, "How many ticks may a player be behind the server time before NoCheat reacts. Default is 50.");
|
|
||||||
set(Configuration.TIMED_GODMODE_ACTIONS, "What should be done if a player is considered using 'godmode'.\nUnit is number of ticks of potential godmode usage.");
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void set(OptionNode id, String text) {
|
private static void set(OptionNode id, String text) {
|
||||||
|
@ -29,6 +29,7 @@ public class Permissions {
|
|||||||
|
|
||||||
public final static String CHAT = CHECKS + ".chat";
|
public final static String CHAT = CHECKS + ".chat";
|
||||||
public final static String CHAT_SPAM = CHAT + ".spam";
|
public final static String CHAT_SPAM = CHAT + ".spam";
|
||||||
|
public static final String CHAT_EMPTY = CHAT + ".empty";
|
||||||
|
|
||||||
public static final String FIGHT = CHECKS + ".fight";
|
public static final String FIGHT = CHECKS + ".fight";
|
||||||
public static final String FIGHT_DIRECTION = FIGHT + ".direction";
|
public static final String FIGHT_DIRECTION = FIGHT + ".direction";
|
||||||
|
@ -14,6 +14,8 @@ public class CCChat {
|
|||||||
public final int spamTimeframe;
|
public final int spamTimeframe;
|
||||||
public final int spamLimit;
|
public final int spamLimit;
|
||||||
public final ActionList spamActions;
|
public final ActionList spamActions;
|
||||||
|
public final boolean emptyCheck;
|
||||||
|
public final ActionList emptyActions;
|
||||||
|
|
||||||
public CCChat(Configuration data) {
|
public CCChat(Configuration data) {
|
||||||
|
|
||||||
@ -23,6 +25,8 @@ public class CCChat {
|
|||||||
spamTimeframe = data.getInteger(Configuration.CHAT_SPAM_TIMEFRAME);
|
spamTimeframe = data.getInteger(Configuration.CHAT_SPAM_TIMEFRAME);
|
||||||
spamLimit = data.getInteger(Configuration.CHAT_SPAM_LIMIT);
|
spamLimit = data.getInteger(Configuration.CHAT_SPAM_LIMIT);
|
||||||
spamActions = data.getActionList(Configuration.CHAT_SPAM_ACTIONS);
|
spamActions = data.getActionList(Configuration.CHAT_SPAM_ACTIONS);
|
||||||
|
emptyCheck = data.getBoolean(Configuration.CHAT_EMPTY_CHECK);
|
||||||
|
emptyActions = data.getActionList(Configuration.CHAT_EMPTY_ACTIONS);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
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 final int tickTime = 10;
|
|
||||||
|
|
||||||
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,7 +18,6 @@ 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
|
||||||
@ -35,7 +34,6 @@ 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);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -10,4 +10,5 @@ public class ChatData extends Data {
|
|||||||
public final ExecutionHistory history = new ExecutionHistory();
|
public final ExecutionHistory history = new ExecutionHistory();
|
||||||
public String message = "";
|
public String message = "";
|
||||||
public int spamVL;
|
public int spamVL;
|
||||||
|
public int emptyVL;
|
||||||
}
|
}
|
||||||
|
@ -11,7 +11,6 @@ import org.bukkit.entity.Player;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
||||||
import cc.co.evenprime.bukkit.nocheat.player.NoCheatPlayerImpl;
|
import cc.co.evenprime.bukkit.nocheat.player.NoCheatPlayerImpl;
|
||||||
import cc.co.evenprime.bukkit.nocheat.player.PlayerFactory;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provide secure access to player-specific data objects for various checks or
|
* Provide secure access to player-specific data objects for various checks or
|
||||||
@ -37,9 +36,7 @@ public class PlayerManager {
|
|||||||
NoCheatPlayerImpl p = this.players.get(player.getName());
|
NoCheatPlayerImpl p = this.players.get(player.getName());
|
||||||
|
|
||||||
if(p == null) {
|
if(p == null) {
|
||||||
// PlayerFactory may create different players based on the
|
p = new NoCheatPlayerImpl(player, plugin);
|
||||||
// MCVersion that we are running
|
|
||||||
p = PlayerFactory.createPlayer(player, plugin);
|
|
||||||
this.players.put(player.getName(), p);
|
this.players.put(player.getName(), p);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,6 +12,7 @@ import org.bukkit.event.player.PlayerCommandPreprocessEvent;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
||||||
import cc.co.evenprime.bukkit.nocheat.checks.ChatCheck;
|
import cc.co.evenprime.bukkit.nocheat.checks.ChatCheck;
|
||||||
|
import cc.co.evenprime.bukkit.nocheat.checks.chat.EmptyCheck;
|
||||||
import cc.co.evenprime.bukkit.nocheat.checks.chat.SpamCheck;
|
import cc.co.evenprime.bukkit.nocheat.checks.chat.SpamCheck;
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.cache.CCChat;
|
import cc.co.evenprime.bukkit.nocheat.config.cache.CCChat;
|
||||||
@ -27,7 +28,8 @@ public class ChatEventManager extends EventManagerImpl {
|
|||||||
|
|
||||||
super(plugin);
|
super(plugin);
|
||||||
|
|
||||||
this.checks = new ArrayList<ChatCheck>(1);
|
this.checks = new ArrayList<ChatCheck>(2);
|
||||||
|
this.checks.add(new EmptyCheck(plugin));
|
||||||
this.checks.add(new SpamCheck(plugin));
|
this.checks.add(new SpamCheck(plugin));
|
||||||
|
|
||||||
registerListener(Event.Type.PLAYER_CHAT, Priority.Lowest, true, plugin.getPerformance(Type.CHAT));
|
registerListener(Event.Type.PLAYER_CHAT, Priority.Lowest, true, plugin.getPerformance(Type.CHAT));
|
||||||
@ -72,6 +74,8 @@ public class ChatEventManager extends EventManagerImpl {
|
|||||||
|
|
||||||
if(cc.chat.check && cc.chat.spamCheck)
|
if(cc.chat.check && cc.chat.spamCheck)
|
||||||
s.add("chat.spam");
|
s.add("chat.spam");
|
||||||
|
if(cc.chat.check && cc.chat.emptyCheck)
|
||||||
|
s.add("chat.empty");
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,124 +0,0 @@
|
|||||||
package cc.co.evenprime.bukkit.nocheat.events;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.ConcurrentModificationException;
|
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import net.minecraft.server.EntityPlayer;
|
|
||||||
|
|
||||||
import org.bukkit.craftbukkit.CraftServer;
|
|
||||||
import org.bukkit.entity.Player;
|
|
||||||
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheatPlayer;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.checks.TimedCheck;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.checks.timed.GodmodeCheck;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.cache.CCTimed;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.cache.ConfigurationCache;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.TimedData;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.debug.Performance;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.debug.PerformanceManager.Type;
|
|
||||||
|
|
||||||
public class TimedEventManager extends EventManagerImpl {
|
|
||||||
|
|
||||||
private final List<TimedCheck> checks;
|
|
||||||
private final Performance timedPerformance;
|
|
||||||
public int taskId = -1;
|
|
||||||
|
|
||||||
public TimedEventManager(final NoCheat plugin) {
|
|
||||||
|
|
||||||
super(plugin);
|
|
||||||
|
|
||||||
checks = new ArrayList<TimedCheck>(1);
|
|
||||||
|
|
||||||
checks.add(new GodmodeCheck(plugin));
|
|
||||||
this.timedPerformance = plugin.getPerformance(Type.TIMED);
|
|
||||||
|
|
||||||
// "register a listener" for passed time
|
|
||||||
this.taskId = plugin.getServer().getScheduler().scheduleSyncRepeatingTask(plugin, new Runnable() {
|
|
||||||
|
|
||||||
private int executions = 0;
|
|
||||||
private int loopsize = 10;
|
|
||||||
|
|
||||||
private final List<EntityPlayer> entities = new ArrayList<EntityPlayer>(20);
|
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
|
||||||
public void run() {
|
|
||||||
|
|
||||||
executions++;
|
|
||||||
|
|
||||||
if(executions >= loopsize) {
|
|
||||||
executions = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// For performance reasons, we take some shortcuts here
|
|
||||||
CraftServer server = (CraftServer) plugin.getServer();
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Only collect the entities that we want to check this time
|
|
||||||
for(EntityPlayer p : (List<EntityPlayer>) server.getHandle().players) {
|
|
||||||
if(p.id % loopsize == executions) {
|
|
||||||
entities.add(p);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch(ConcurrentModificationException e) {
|
|
||||||
// Bad luck, better luck next time
|
|
||||||
} catch(Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Now initialize the checks one by one
|
|
||||||
for(EntityPlayer p : entities) {
|
|
||||||
try {
|
|
||||||
|
|
||||||
// Performance counter setup
|
|
||||||
long nanoTimeStart = 0;
|
|
||||||
final boolean performanceCheck = timedPerformance.isEnabled();
|
|
||||||
|
|
||||||
if(performanceCheck)
|
|
||||||
nanoTimeStart = System.nanoTime();
|
|
||||||
|
|
||||||
handleEvent(plugin.getPlayer((Player) p.getBukkitEntity()));
|
|
||||||
|
|
||||||
// store performance time
|
|
||||||
if(performanceCheck)
|
|
||||||
timedPerformance.addTime(System.nanoTime() - nanoTimeStart);
|
|
||||||
|
|
||||||
} catch(Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clear the list for next time
|
|
||||||
entities.clear();
|
|
||||||
}
|
|
||||||
}, 0, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void handleEvent(final NoCheatPlayer player) {
|
|
||||||
|
|
||||||
final TimedData data = player.getData().timed;
|
|
||||||
final CCTimed cc = player.getConfiguration().timed;
|
|
||||||
|
|
||||||
if(!cc.check || player.hasPermission(Permissions.TIMED)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for(TimedCheck check : checks) {
|
|
||||||
if(cc.check && !player.hasPermission(Permissions.TIMED)) {
|
|
||||||
check.check(player, data, cc);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -24,11 +24,11 @@ public class NoCheatPlayerImpl implements NoCheatPlayer {
|
|||||||
// The method that's used to artifically "fast-forward" the player
|
// The method that's used to artifically "fast-forward" the player
|
||||||
protected static Method incAge = null;
|
protected static Method incAge = null;
|
||||||
|
|
||||||
public NoCheatPlayerImpl(Player player, NoCheat plugin, BaseData data) {
|
public NoCheatPlayerImpl(Player player, NoCheat plugin) {
|
||||||
|
|
||||||
this.player = player;
|
this.player = player;
|
||||||
this.plugin = plugin;
|
this.plugin = plugin;
|
||||||
this.data = data;
|
this.data = new BaseData();
|
||||||
|
|
||||||
this.lastUsedTime = System.currentTimeMillis();
|
this.lastUsedTime = System.currentTimeMillis();
|
||||||
}
|
}
|
||||||
@ -40,6 +40,7 @@ public class NoCheatPlayerImpl implements NoCheatPlayer {
|
|||||||
public boolean isDead() {
|
public boolean isDead() {
|
||||||
return this.player.getHealth() <= 0 || this.player.isDead();
|
return this.player.getHealth() <= 0 || this.player.isDead();
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean hasPermission(String permission) {
|
public boolean hasPermission(String permission) {
|
||||||
if(permission == null) {
|
if(permission == null) {
|
||||||
System.out.println("NoCheat: Warning, asked for null permission");
|
System.out.println("NoCheat: Warning, asked for null permission");
|
||||||
@ -68,24 +69,6 @@ public class NoCheatPlayerImpl implements NoCheatPlayer {
|
|||||||
return player.getTicksLived();
|
return player.getTicksLived();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void increaseAge(int ticks) {
|
|
||||||
|
|
||||||
if(incAge == null) {
|
|
||||||
player.setTicksLived(player.getTicksLived() + ticks);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
EntityPlayer p = ((CraftPlayer) player).getHandle();
|
|
||||||
|
|
||||||
for(int i = 0; i < ticks; i++) {
|
|
||||||
try {
|
|
||||||
incAge.invoke(p, true);
|
|
||||||
} catch(Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public float getSpeedAmplifier() {
|
public float getSpeedAmplifier() {
|
||||||
EntityPlayer ep = ((CraftPlayer) player).getHandle();
|
EntityPlayer ep = ((CraftPlayer) player).getHandle();
|
||||||
if(ep.hasEffect(MobEffectList.FASTER_MOVEMENT)) {
|
if(ep.hasEffect(MobEffectList.FASTER_MOVEMENT)) {
|
||||||
|
@ -1,23 +0,0 @@
|
|||||||
package cc.co.evenprime.bukkit.nocheat.player;
|
|
||||||
|
|
||||||
import net.minecraft.server.EntityPlayer;
|
|
||||||
|
|
||||||
import org.bukkit.entity.Player;
|
|
||||||
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.BaseData;
|
|
||||||
|
|
||||||
public class NoCheatPlayer_100 extends NoCheatPlayerImpl {
|
|
||||||
|
|
||||||
static {
|
|
||||||
try {
|
|
||||||
incAge = EntityPlayer.class.getMethod("a", boolean.class);
|
|
||||||
} catch(Exception e) {
|
|
||||||
System.out.println("NoCheat couldn't initialize variable incAge");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public NoCheatPlayer_100(Player player, NoCheat plugin, BaseData data) {
|
|
||||||
super(player, plugin, data);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
package cc.co.evenprime.bukkit.nocheat.player;
|
|
||||||
|
|
||||||
import net.minecraft.server.EntityPlayer;
|
|
||||||
|
|
||||||
import org.bukkit.entity.Player;
|
|
||||||
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.BaseData;
|
|
||||||
|
|
||||||
public class NoCheatPlayer_181 extends NoCheatPlayerImpl {
|
|
||||||
|
|
||||||
public NoCheatPlayer_181(Player player, NoCheat plugin, BaseData data) {
|
|
||||||
super(player, plugin, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
static {
|
|
||||||
try {
|
|
||||||
incAge = EntityPlayer.class.getMethod("b", boolean.class);
|
|
||||||
} catch(Exception e) {
|
|
||||||
System.out.println("NoCheat couldn't initialize variable incAge");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,21 +0,0 @@
|
|||||||
package cc.co.evenprime.bukkit.nocheat.player;
|
|
||||||
|
|
||||||
import org.bukkit.entity.Player;
|
|
||||||
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.NoCheat;
|
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.BaseData;
|
|
||||||
|
|
||||||
public class PlayerFactory {
|
|
||||||
|
|
||||||
public static NoCheatPlayerImpl createPlayer(Player player, NoCheat plugin) {
|
|
||||||
|
|
||||||
switch (plugin.getMCVersion()) {
|
|
||||||
case MC100:
|
|
||||||
return new NoCheatPlayer_100(player, plugin, new BaseData());
|
|
||||||
case MC181:
|
|
||||||
return new NoCheatPlayer_181(player, plugin, new BaseData());
|
|
||||||
default:
|
|
||||||
return new NoCheatPlayerImpl(player, plugin, new BaseData());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user