mirror of
https://github.com/NoCheatPlus/NoCheatPlus.git
synced 2025-01-04 23:07:44 +01:00
Even more documentation
This commit is contained in:
parent
b742875ead
commit
32d09868d9
@ -82,7 +82,7 @@ public class DirectionCheck extends BlockPlaceCheck {
|
|||||||
|
|
||||||
// If the player is still in penalty time, cancel the event anyway
|
// If the player is still in penalty time, cancel the event anyway
|
||||||
if(data.directionLastViolationTime + cc.directionPenaltyTime > time) {
|
if(data.directionLastViolationTime + cc.directionPenaltyTime > time) {
|
||||||
// A saveguard to avoid people getting stuck in penalty time
|
// A safeguard to avoid people getting stuck in penalty time
|
||||||
// indefinitely in case the system time of the server gets changed
|
// indefinitely in case the system time of the server gets changed
|
||||||
if(data.directionLastViolationTime > time) {
|
if(data.directionLastViolationTime > time) {
|
||||||
data.directionLastViolationTime = 0;
|
data.directionLastViolationTime = 0;
|
||||||
|
@ -11,6 +11,11 @@ import cc.co.evenprime.bukkit.nocheat.checks.CheckUtil;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The DirectionCheck will find out if a player tried to interact with something
|
||||||
|
* that's not in his field of view.
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class DirectionCheck extends FightCheck {
|
public class DirectionCheck extends FightCheck {
|
||||||
|
|
||||||
public DirectionCheck(NoCheat plugin) {
|
public DirectionCheck(NoCheat plugin) {
|
||||||
@ -23,7 +28,7 @@ public class DirectionCheck extends FightCheck {
|
|||||||
|
|
||||||
final long time = System.currentTimeMillis();
|
final long time = System.currentTimeMillis();
|
||||||
|
|
||||||
// Get the width of the damagee
|
// Get the damagee (entity that got hit)
|
||||||
Entity entity = data.damagee;
|
Entity entity = data.damagee;
|
||||||
|
|
||||||
// Safeguard, if entity is complex, this check will fail
|
// Safeguard, if entity is complex, this check will fail
|
||||||
@ -32,40 +37,51 @@ public class DirectionCheck extends FightCheck {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Find out how wide the entity is
|
||||||
final float width = entity.length > entity.width ? entity.length : entity.width;
|
final float width = entity.length > entity.width ? entity.length : entity.width;
|
||||||
// entity.height is broken and will always be 0, therefore calculate height instead
|
// entity.height is broken and will always be 0, therefore
|
||||||
|
// calculate height instead based on boundingBox
|
||||||
final double height = entity.boundingBox.e - entity.boundingBox.b;
|
final double height = entity.boundingBox.e - entity.boundingBox.b;
|
||||||
|
|
||||||
|
// How far "off" is the player with his aim. We calculate from the
|
||||||
|
// players eye location and view direction to the center of the target
|
||||||
|
// entity. If the line of sight is more too far off, "off" will be
|
||||||
|
// bigger than 0
|
||||||
final double off = CheckUtil.directionCheck(player, entity.locX, entity.locY + (height / 2D), entity.locZ, width, height, cc.directionPrecision);
|
final double off = CheckUtil.directionCheck(player, entity.locX, entity.locY + (height / 2D), entity.locZ, width, height, cc.directionPrecision);
|
||||||
|
|
||||||
if(off < 0.1D) {
|
if(off < 0.1D) {
|
||||||
// Player did probably nothing wrong
|
// Player did probably nothing wrong
|
||||||
// reduce violation counter
|
// reduce violation counter to reward him
|
||||||
data.directionVL *= 0.80D;
|
data.directionVL *= 0.80D;
|
||||||
} else {
|
} else {
|
||||||
// Player failed the check
|
// Player failed the check
|
||||||
// Increment violation counter
|
// Increment violation counter and statistics, but only if there
|
||||||
|
// wasn't serious lag
|
||||||
if(!plugin.skipCheck()) {
|
if(!plugin.skipCheck()) {
|
||||||
double sqrt = Math.sqrt(off);
|
double sqrt = Math.sqrt(off);
|
||||||
data.directionVL += sqrt;
|
data.directionVL += sqrt;
|
||||||
incrementStatistics(player, Id.FI_DIRECTION, sqrt);
|
incrementStatistics(player, Id.FI_DIRECTION, sqrt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Execute whatever actions are associated with this check and the
|
||||||
|
// violation level and find out if we should cancel the event
|
||||||
cancel = executeActions(player, cc.directionActions, data.directionVL);
|
cancel = executeActions(player, cc.directionActions, data.directionVL);
|
||||||
|
|
||||||
if(cancel) {
|
if(cancel) {
|
||||||
// Needed to calculate penalty times
|
// if we should cancel, remember the current time too
|
||||||
data.directionLastViolationTime = time;
|
data.directionLastViolationTime = time;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the player is still in penalty time, cancel the event anyway
|
// If the player is still in penalty time, cancel the event anyway
|
||||||
if(data.directionLastViolationTime + cc.directionPenaltyTime > time) {
|
if(data.directionLastViolationTime + cc.directionPenaltyTime > time) {
|
||||||
|
// A safeguard to avoid people getting stuck in penalty time
|
||||||
|
// indefinitely in case the system time of the server gets changed
|
||||||
if(data.directionLastViolationTime > time) {
|
if(data.directionLastViolationTime > time) {
|
||||||
System.out.println("Nocheat noted that your time ran backwards for " + (data.directionLastViolationTime - time) + " ms");
|
|
||||||
// Security check for server time changed situations
|
|
||||||
data.directionLastViolationTime = 0;
|
data.directionLastViolationTime = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// He is in penalty time, therefore request cancelling of the event
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -81,7 +97,7 @@ public class DirectionCheck extends FightCheck {
|
|||||||
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
||||||
|
|
||||||
if(wildcard == ParameterName.VIOLATIONS)
|
if(wildcard == ParameterName.VIOLATIONS)
|
||||||
return String.format(Locale.US, "%d", (int) getData(player.getDataStore()).directionVL);
|
return String.format(Locale.US, "%d", (int) getData(player).directionVL);
|
||||||
else
|
else
|
||||||
return super.getParameter(wildcard, player);
|
return super.getParameter(wildcard, player);
|
||||||
}
|
}
|
||||||
|
@ -24,7 +24,15 @@ public abstract class FightCheck extends Check {
|
|||||||
|
|
||||||
public abstract boolean isEnabled(FightConfig cc);
|
public abstract boolean isEnabled(FightConfig cc);
|
||||||
|
|
||||||
public static FightData getData(DataStore base) {
|
/**
|
||||||
|
* Get the "FightData" object that belongs to the player. Will ensure
|
||||||
|
* that such a object exists and if not, create one
|
||||||
|
*
|
||||||
|
* @param player
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public static FightData getData(NoCheatPlayer player) {
|
||||||
|
DataStore base = player.getDataStore();
|
||||||
FightData data = base.get(id);
|
FightData data = base.get(id);
|
||||||
if(data == null) {
|
if(data == null) {
|
||||||
data = new FightData();
|
data = new FightData();
|
||||||
@ -33,6 +41,17 @@ public abstract class FightCheck extends Check {
|
|||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the FightConfig object that belongs to the world that the player
|
||||||
|
* currently resides in.
|
||||||
|
*
|
||||||
|
* @param player
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public static FightConfig getConfig(NoCheatPlayer player) {
|
||||||
|
return getConfig(player.getConfigurationStore());
|
||||||
|
}
|
||||||
|
|
||||||
public static FightConfig getConfig(ConfigurationCacheStore cache) {
|
public static FightConfig getConfig(ConfigurationCacheStore cache) {
|
||||||
FightConfig config = cache.get(id);
|
FightConfig config = cache.get(id);
|
||||||
if(config == null) {
|
if(config == null) {
|
||||||
|
@ -19,8 +19,12 @@ import cc.co.evenprime.bukkit.nocheat.EventManager;
|
|||||||
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.config.ConfigurationCacheStore;
|
import cc.co.evenprime.bukkit.nocheat.config.ConfigurationCacheStore;
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Central location to listen to events that are
|
||||||
|
* relevant for the fight checks
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class FightCheckListener implements Listener, EventManager {
|
public class FightCheckListener implements Listener, EventManager {
|
||||||
|
|
||||||
private final List<FightCheck> checks;
|
private final List<FightCheck> checks;
|
||||||
@ -30,7 +34,9 @@ public class FightCheckListener implements Listener, EventManager {
|
|||||||
|
|
||||||
public FightCheckListener(NoCheat plugin) {
|
public FightCheckListener(NoCheat plugin) {
|
||||||
|
|
||||||
this.checks = new ArrayList<FightCheck>(3);
|
this.checks = new ArrayList<FightCheck>(4);
|
||||||
|
|
||||||
|
// Keep these in a list, because they can be executed in a bundle
|
||||||
this.checks.add(new SpeedCheck(plugin));
|
this.checks.add(new SpeedCheck(plugin));
|
||||||
this.checks.add(new NoswingCheck(plugin));
|
this.checks.add(new NoswingCheck(plugin));
|
||||||
this.checks.add(new DirectionCheck(plugin));
|
this.checks.add(new DirectionCheck(plugin));
|
||||||
@ -41,9 +47,14 @@ public class FightCheckListener implements Listener, EventManager {
|
|||||||
this.plugin = plugin;
|
this.plugin = plugin;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We listen to EntityDamage events for obvious reasons
|
||||||
|
* @param event The EntityDamage Event
|
||||||
|
*/
|
||||||
@EventHandler(priority = EventPriority.LOWEST)
|
@EventHandler(priority = EventPriority.LOWEST)
|
||||||
public void entityDamage(final EntityDamageEvent event) {
|
public void entityDamage(final EntityDamageEvent event) {
|
||||||
|
|
||||||
|
// Filter some unwanted events right now
|
||||||
if(event.isCancelled() || !(event instanceof EntityDamageByEntityEvent))
|
if(event.isCancelled() || !(event instanceof EntityDamageByEntityEvent))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -59,44 +70,55 @@ public class FightCheckListener implements Listener, EventManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We listen to EntityDamage events (again) for obvious reasons
|
||||||
|
* @param event The EntityDamage Event
|
||||||
|
*/
|
||||||
@EventHandler(priority = EventPriority.LOW)
|
@EventHandler(priority = EventPriority.LOW)
|
||||||
public void entityDamageForGodmodeCheck(final EntityDamageEvent event) {
|
public void entityDamageForGodmodeCheck(final EntityDamageEvent event) {
|
||||||
|
|
||||||
if(event.isCancelled())
|
if(event.isCancelled())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
// Filter unwanted events right here
|
||||||
final Entity entity = event.getEntity();
|
final Entity entity = event.getEntity();
|
||||||
if(!(entity instanceof Player) || entity.isDead()) {
|
if(!(entity instanceof Player) || entity.isDead()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
NoCheatPlayer player = plugin.getPlayer((Player) entity);
|
NoCheatPlayer player = plugin.getPlayer((Player) entity);
|
||||||
FightConfig cc = FightCheck.getConfig(player.getConfigurationStore());
|
FightConfig cc = FightCheck.getConfig(player);
|
||||||
|
|
||||||
if(!godmodeCheck.isEnabled(cc) || player.hasPermission(godmodeCheck.permission)) {
|
if(!godmodeCheck.isEnabled(cc) || player.hasPermission(godmodeCheck.permission)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
FightData data = FightCheck.getData(player.getDataStore());
|
FightData data = FightCheck.getData(player);
|
||||||
|
|
||||||
|
// Run the godmode check on the attacked player
|
||||||
boolean cancelled = godmodeCheck.check(plugin.getPlayer((Player) entity), data, cc);
|
boolean cancelled = godmodeCheck.check(plugin.getPlayer((Player) entity), data, cc);
|
||||||
|
|
||||||
|
// It requested to "cancel" the players invulnerability, so set his
|
||||||
|
// noDamageTicks to 0
|
||||||
if(cancelled) {
|
if(cancelled) {
|
||||||
// Remove the invulnerability from the player
|
// Remove the invulnerability from the player
|
||||||
player.getPlayer().setNoDamageTicks(0);
|
player.getPlayer().setNoDamageTicks(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A player attacked something with DamageCause ENTITY_ATTACK. That's most
|
||||||
|
* likely what we want to really check.
|
||||||
|
*
|
||||||
|
* @param event The EntityDamageByEntityEvent
|
||||||
|
*/
|
||||||
private void normalDamage(final EntityDamageByEntityEvent event) {
|
private void normalDamage(final EntityDamageByEntityEvent event) {
|
||||||
|
|
||||||
final Player damager = (Player) event.getDamager();
|
final Player damager = (Player) event.getDamager();
|
||||||
|
|
||||||
final NoCheatPlayer player = plugin.getPlayer(damager);
|
final NoCheatPlayer player = plugin.getPlayer(damager);
|
||||||
final FightConfig cc = FightCheck.getConfig(player.getConfigurationStore());
|
final FightConfig cc = FightCheck.getConfig(player);
|
||||||
|
final FightData data = FightCheck.getData(player);
|
||||||
if(!cc.damageChecks || player.hasPermission(Permissions.FIGHT)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final FightData data = FightCheck.getData(player.getDataStore());
|
|
||||||
|
|
||||||
// For some reason we decided to skip this event anyway
|
// For some reason we decided to skip this event anyway
|
||||||
if(data.skipNext) {
|
if(data.skipNext) {
|
||||||
@ -106,9 +128,10 @@ public class FightCheckListener implements Listener, EventManager {
|
|||||||
|
|
||||||
boolean cancelled = false;
|
boolean cancelled = false;
|
||||||
|
|
||||||
// Get the attacked entity
|
// Get the attacked entity and remember it
|
||||||
data.damagee = ((CraftEntity) event.getEntity()).getHandle();
|
data.damagee = ((CraftEntity) event.getEntity()).getHandle();
|
||||||
|
|
||||||
|
// Run through the four main checks
|
||||||
for(FightCheck check : checks) {
|
for(FightCheck check : checks) {
|
||||||
// If it should be executed, do it
|
// If it should be executed, do it
|
||||||
if(!cancelled && check.isEnabled(cc) && !player.hasPermission(check.permission)) {
|
if(!cancelled && check.isEnabled(cc) && !player.hasPermission(check.permission)) {
|
||||||
@ -116,18 +139,27 @@ public class FightCheckListener implements Listener, EventManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Forget the attacked entity (to allow garbage collecting etc.
|
||||||
data.damagee = null;
|
data.damagee = null;
|
||||||
|
|
||||||
|
// One of the checks requested the event to be cancelled, so do it
|
||||||
if(cancelled)
|
if(cancelled)
|
||||||
event.setCancelled(cancelled);
|
event.setCancelled(cancelled);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* There is an unofficial agreement that if a plugin wants an attack to
|
||||||
|
* not get checked by NoCheat, it either has to use a Damage type different
|
||||||
|
* from ENTITY_ATTACK or fire an event with damage type CUSTOM and damage
|
||||||
|
* 0 directly before the to-be-ignored event.
|
||||||
|
* @param event The EntityDamageByEntityEvent
|
||||||
|
*/
|
||||||
private void customDamage(final EntityDamageByEntityEvent event) {
|
private void customDamage(final EntityDamageByEntityEvent event) {
|
||||||
|
|
||||||
final Player damager = (Player) event.getDamager();
|
final Player damager = (Player) event.getDamager();
|
||||||
final NoCheatPlayer player = plugin.getPlayer(damager);
|
final NoCheatPlayer player = plugin.getPlayer(damager);
|
||||||
|
|
||||||
final FightData data = FightCheck.getData(player.getDataStore());
|
final FightData data = FightCheck.getData(player);
|
||||||
|
|
||||||
// Skip the next damage event, because it is with high probability
|
// Skip the next damage event, because it is with high probability
|
||||||
// something from the Heroes plugin
|
// something from the Heroes plugin
|
||||||
@ -136,8 +168,15 @@ public class FightCheckListener implements Listener, EventManager {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We listen to death events to prevent a very specific method of doing
|
||||||
|
* godmode.
|
||||||
|
*
|
||||||
|
* @param event The EntityDeathEvent
|
||||||
|
*/
|
||||||
@EventHandler(priority = EventPriority.MONITOR)
|
@EventHandler(priority = EventPriority.MONITOR)
|
||||||
protected void death(final EntityDeathEvent event) {
|
protected void death(final EntityDeathEvent event) {
|
||||||
|
// Only interested in dying players
|
||||||
if(!(event.getEntity() instanceof CraftPlayer)) {
|
if(!(event.getEntity() instanceof CraftPlayer)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -145,9 +184,14 @@ public class FightCheckListener implements Listener, EventManager {
|
|||||||
godmodeCheck.death((CraftPlayer) event.getEntity());
|
godmodeCheck.death((CraftPlayer) event.getEntity());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We listen to PlayerAnimationEvent because it is used for arm swinging
|
||||||
|
* @param event The PlayerAnimationEvent
|
||||||
|
*/
|
||||||
@EventHandler(priority = EventPriority.MONITOR)
|
@EventHandler(priority = EventPriority.MONITOR)
|
||||||
protected void armSwing(final PlayerAnimationEvent event) {
|
protected void armSwing(final PlayerAnimationEvent event) {
|
||||||
FightCheck.getData(plugin.getPlayer(event.getPlayer()).getDataStore()).armswung = true;
|
// Set a flag telling us that the arm has been swung
|
||||||
|
FightCheck.getData(plugin.getPlayer(event.getPlayer())).armswung = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<String> getActiveChecks(ConfigurationCacheStore cc) {
|
public List<String> getActiveChecks(ConfigurationCacheStore cc) {
|
||||||
|
@ -6,26 +6,34 @@ import cc.co.evenprime.bukkit.nocheat.config.ConfPaths;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.config.NoCheatConfiguration;
|
import cc.co.evenprime.bukkit.nocheat.config.NoCheatConfiguration;
|
||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configurations specific for the "Fight" checks
|
||||||
|
* Every world gets one of these assigned to it, or if a world doesn't get
|
||||||
|
* it's own, it will use the "global" version
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class FightConfig implements ConfigItem {
|
public class FightConfig implements ConfigItem {
|
||||||
|
|
||||||
public final boolean directionCheck;
|
public final boolean directionCheck;
|
||||||
public final double directionPrecision;
|
public final double directionPrecision;
|
||||||
public final ActionList directionActions;
|
public final ActionList directionActions;
|
||||||
public final long directionPenaltyTime;
|
public final long directionPenaltyTime;
|
||||||
|
|
||||||
public final boolean noswingCheck;
|
public final boolean noswingCheck;
|
||||||
public final ActionList noswingActions;
|
public final ActionList noswingActions;
|
||||||
|
|
||||||
public final boolean reachCheck;
|
public final boolean reachCheck;
|
||||||
public final double reachLimit;
|
public final double reachLimit;
|
||||||
public final long reachPenaltyTime;
|
public final long reachPenaltyTime;
|
||||||
public final ActionList reachActions;
|
public final ActionList reachActions;
|
||||||
|
|
||||||
public final int speedAttackLimit;
|
public final int speedAttackLimit;
|
||||||
public final ActionList speedActions;
|
public final ActionList speedActions;
|
||||||
public final boolean speedCheck;
|
public final boolean speedCheck;
|
||||||
|
|
||||||
public final boolean godmodeCheck;
|
public final boolean godmodeCheck;
|
||||||
public final ActionList godmodeActions;
|
public final ActionList godmodeActions;
|
||||||
|
|
||||||
public final boolean damageChecks;
|
|
||||||
|
|
||||||
public FightConfig(NoCheatConfiguration data) {
|
public FightConfig(NoCheatConfiguration data) {
|
||||||
|
|
||||||
directionCheck = data.getBoolean(ConfPaths.FIGHT_DIRECTION_CHECK);
|
directionCheck = data.getBoolean(ConfPaths.FIGHT_DIRECTION_CHECK);
|
||||||
@ -44,7 +52,5 @@ public class FightConfig implements ConfigItem {
|
|||||||
|
|
||||||
godmodeCheck = data.getBoolean(ConfPaths.FIGHT_GODMODE_CHECK);
|
godmodeCheck = data.getBoolean(ConfPaths.FIGHT_GODMODE_CHECK);
|
||||||
godmodeActions = data.getActionList(ConfPaths.FIGHT_GODMODE_ACTIONS, Permissions.FIGHT_GODMODE);
|
godmodeActions = data.getActionList(ConfPaths.FIGHT_GODMODE_ACTIONS, Permissions.FIGHT_GODMODE);
|
||||||
|
|
||||||
damageChecks = directionCheck || noswingCheck || reachCheck || speedCheck;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,24 +3,38 @@ package cc.co.evenprime.bukkit.nocheat.checks.fight;
|
|||||||
import net.minecraft.server.Entity;
|
import net.minecraft.server.Entity;
|
||||||
import cc.co.evenprime.bukkit.nocheat.DataItem;
|
import cc.co.evenprime.bukkit.nocheat.DataItem;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Player specific data for the fight checks
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class FightData implements DataItem {
|
public class FightData implements DataItem {
|
||||||
|
|
||||||
|
// Keep track of the violation levels of the checks
|
||||||
public double directionVL;
|
public double directionVL;
|
||||||
public double noswingVL;
|
public double noswingVL;
|
||||||
public double reachVL;
|
public double reachVL;
|
||||||
public int speedVL;
|
public int speedVL;
|
||||||
public double godmodeVL;
|
public double godmodeVL;
|
||||||
|
|
||||||
|
// For checks that have penalty time
|
||||||
public long directionLastViolationTime;
|
public long directionLastViolationTime;
|
||||||
public long reachLastViolationTime;
|
public long reachLastViolationTime;
|
||||||
|
|
||||||
|
// godmode check needs to know these
|
||||||
public long godmodeLastDamageTime;
|
public long godmodeLastDamageTime;
|
||||||
public int godmodeLastAge;
|
public int godmodeLastAge;
|
||||||
public int godmodeBuffer = 40;
|
public int godmodeBuffer = 40;
|
||||||
|
|
||||||
|
// While handling an event, use this to keep the attacked entity
|
||||||
public Entity damagee;
|
public Entity damagee;
|
||||||
|
|
||||||
|
// The player swung his arm
|
||||||
public boolean armswung = true;
|
public boolean armswung = true;
|
||||||
|
|
||||||
|
// For some reason the next event should be ignored
|
||||||
public boolean skipNext = false;
|
public boolean skipNext = false;
|
||||||
|
|
||||||
|
// Keep track of time and amount of attacks
|
||||||
public long speedTime;
|
public long speedTime;
|
||||||
public int speedAttackCount;
|
public int speedAttackCount;
|
||||||
|
|
||||||
|
@ -10,6 +10,11 @@ import cc.co.evenprime.bukkit.nocheat.actions.ParameterName;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.Statistics;
|
import cc.co.evenprime.bukkit.nocheat.data.Statistics;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The Godmode Check will find out if a player tried to stay invulnerable after
|
||||||
|
* being hit or after dying
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class GodmodeCheck extends FightCheck {
|
public class GodmodeCheck extends FightCheck {
|
||||||
|
|
||||||
public GodmodeCheck(NoCheat plugin) {
|
public GodmodeCheck(NoCheat plugin) {
|
||||||
@ -22,6 +27,7 @@ public class GodmodeCheck extends FightCheck {
|
|||||||
boolean cancelled = false;
|
boolean cancelled = false;
|
||||||
|
|
||||||
long time = System.currentTimeMillis();
|
long time = System.currentTimeMillis();
|
||||||
|
|
||||||
// Check at most once a second
|
// Check at most once a second
|
||||||
if(data.godmodeLastDamageTime + 1000L < time) {
|
if(data.godmodeLastDamageTime + 1000L < time) {
|
||||||
data.godmodeLastDamageTime = time;
|
data.godmodeLastDamageTime = time;
|
||||||
@ -34,14 +40,18 @@ public class GodmodeCheck extends FightCheck {
|
|||||||
int nodamageTicks = player.getPlayer().getNoDamageTicks();
|
int nodamageTicks = player.getPlayer().getNoDamageTicks();
|
||||||
|
|
||||||
if(nodamageTicks > 0 && ageDiff < 15) {
|
if(nodamageTicks > 0 && ageDiff < 15) {
|
||||||
// He is invulnerable and didn't age fast enough, that costs some points
|
// He is invulnerable and didn't age fast enough, that costs
|
||||||
|
// some points
|
||||||
data.godmodeBuffer -= (15 - ageDiff);
|
data.godmodeBuffer -= (15 - ageDiff);
|
||||||
|
|
||||||
// Still points left?
|
// Still points left?
|
||||||
if(data.godmodeBuffer <= 0) {
|
if(data.godmodeBuffer <= 0) {
|
||||||
// No
|
// No, that means VL and statistics increased
|
||||||
data.godmodeVL -= data.godmodeBuffer;
|
data.godmodeVL -= data.godmodeBuffer;
|
||||||
incrementStatistics(player, Statistics.Id.FI_GODMODE, -data.godmodeBuffer);
|
incrementStatistics(player, Statistics.Id.FI_GODMODE, -data.godmodeBuffer);
|
||||||
|
|
||||||
|
// Execute whatever actions are associated with this check and the
|
||||||
|
// violation level and find out if we should cancel the event
|
||||||
cancelled = executeActions(player, cc.godmodeActions, data.godmodeVL);
|
cancelled = executeActions(player, cc.godmodeActions, data.godmodeVL);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -51,8 +61,10 @@ public class GodmodeCheck extends FightCheck {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(data.godmodeBuffer < 0) {
|
if(data.godmodeBuffer < 0) {
|
||||||
|
// Can't have less than 0
|
||||||
data.godmodeBuffer = 0;
|
data.godmodeBuffer = 0;
|
||||||
} else if(data.godmodeBuffer > 30) {
|
} else if(data.godmodeBuffer > 30) {
|
||||||
|
// And 30 is enough for simple lag situations
|
||||||
data.godmodeBuffer = 30;
|
data.godmodeBuffer = 30;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -72,27 +84,33 @@ public class GodmodeCheck extends FightCheck {
|
|||||||
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
||||||
|
|
||||||
if(wildcard == ParameterName.VIOLATIONS)
|
if(wildcard == ParameterName.VIOLATIONS)
|
||||||
return String.format(Locale.US, "%d", (int) getData(player.getDataStore()).godmodeVL);
|
return String.format(Locale.US, "%d", (int) getData(player).godmodeVL);
|
||||||
else
|
else
|
||||||
return super.getParameter(wildcard, player);
|
return super.getParameter(wildcard, player);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If a player apparently died, make sure he really dies after some time
|
* If a player apparently died, make sure he really dies after some time
|
||||||
* if he didn't already.
|
* if he didn't already, by setting up a Bukkit task
|
||||||
*
|
*
|
||||||
* @param player
|
* @param player The player
|
||||||
*/
|
*/
|
||||||
public void death(CraftPlayer player) {
|
public void death(CraftPlayer player) {
|
||||||
|
// First check if the player is really dead (e.g. another plugin could
|
||||||
|
// have just fired an artificial event)
|
||||||
if(player.getHealth() <= 0 && player.isDead()) {
|
if(player.getHealth() <= 0 && player.isDead()) {
|
||||||
try {
|
try {
|
||||||
final EntityPlayer entity = player.getHandle();
|
final EntityPlayer entity = player.getHandle();
|
||||||
|
|
||||||
|
// Schedule a task to be executed in roughly 1.5 seconds
|
||||||
Bukkit.getScheduler().scheduleSyncDelayedTask(plugin, new Runnable() {
|
Bukkit.getScheduler().scheduleSyncDelayedTask(plugin, new Runnable() {
|
||||||
|
|
||||||
public void run() {
|
public void run() {
|
||||||
try {
|
try {
|
||||||
|
// Check again if the player should be dead, and
|
||||||
|
// if the game didn't mark him as dead
|
||||||
if(entity.getHealth() <= 0 && !entity.dead) {
|
if(entity.getHealth() <= 0 && !entity.dead) {
|
||||||
|
// Artifically "kill" him
|
||||||
entity.deathTicks = 19;
|
entity.deathTicks = 19;
|
||||||
entity.a(true);
|
entity.a(true);
|
||||||
}
|
}
|
||||||
|
@ -7,6 +7,11 @@ import cc.co.evenprime.bukkit.nocheat.actions.ParameterName;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We require that the player moves his arm between attacks, this is
|
||||||
|
* what gets checked here.
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class NoswingCheck extends FightCheck {
|
public class NoswingCheck extends FightCheck {
|
||||||
|
|
||||||
public NoswingCheck(NoCheat plugin) {
|
public NoswingCheck(NoCheat plugin) {
|
||||||
@ -19,12 +24,16 @@ public class NoswingCheck extends FightCheck {
|
|||||||
|
|
||||||
// did he swing his arm before?
|
// did he swing his arm before?
|
||||||
if(data.armswung) {
|
if(data.armswung) {
|
||||||
|
// Yes, reward him with reduction of his vl
|
||||||
data.armswung = false;
|
data.armswung = false;
|
||||||
data.noswingVL *= 0.90D;
|
data.noswingVL *= 0.90D;
|
||||||
} else {
|
} else {
|
||||||
|
// No, increase vl and statistics
|
||||||
data.noswingVL += 1;
|
data.noswingVL += 1;
|
||||||
incrementStatistics(player, Id.FI_NOSWING, 1);
|
incrementStatistics(player, Id.FI_NOSWING, 1);
|
||||||
|
|
||||||
|
// Execute whatever actions are associated with this check and the
|
||||||
|
// violation level and find out if we should cancel the event
|
||||||
cancel = executeActions(player, cc.noswingActions, data.noswingVL);
|
cancel = executeActions(player, cc.noswingActions, data.noswingVL);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -36,10 +45,11 @@ public class NoswingCheck extends FightCheck {
|
|||||||
return cc.noswingCheck;
|
return cc.noswingCheck;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
||||||
|
|
||||||
if(wildcard == ParameterName.VIOLATIONS)
|
if(wildcard == ParameterName.VIOLATIONS)
|
||||||
return String.format(Locale.US, "%d", (int) getData(player.getDataStore()).noswingVL);
|
return String.format(Locale.US, "%d", (int) getData(player).noswingVL);
|
||||||
else
|
else
|
||||||
return super.getParameter(wildcard, player);
|
return super.getParameter(wildcard, player);
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,6 @@ import java.util.Locale;
|
|||||||
import net.minecraft.server.Entity;
|
import net.minecraft.server.Entity;
|
||||||
import net.minecraft.server.EntityComplex;
|
import net.minecraft.server.EntityComplex;
|
||||||
import net.minecraft.server.EntityComplexPart;
|
import net.minecraft.server.EntityComplexPart;
|
||||||
import net.minecraft.server.EntityGiantZombie;
|
|
||||||
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.actions.ParameterName;
|
import cc.co.evenprime.bukkit.nocheat.actions.ParameterName;
|
||||||
@ -12,6 +11,11 @@ import cc.co.evenprime.bukkit.nocheat.checks.CheckUtil;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The reach check will find out if a player interacts with something that's
|
||||||
|
* too far away
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class ReachCheck extends FightCheck {
|
public class ReachCheck extends FightCheck {
|
||||||
|
|
||||||
public ReachCheck(NoCheat plugin) {
|
public ReachCheck(NoCheat plugin) {
|
||||||
@ -29,45 +33,48 @@ public class ReachCheck extends FightCheck {
|
|||||||
|
|
||||||
// Safeguard, if entity is Giant or Ender Dragon, this check will fail
|
// Safeguard, if entity is Giant or Ender Dragon, this check will fail
|
||||||
// due to giant and hard to define hitboxes
|
// due to giant and hard to define hitboxes
|
||||||
if(entity instanceof EntityComplex || entity instanceof EntityComplexPart || entity instanceof EntityGiantZombie) {
|
if(entity instanceof EntityComplex || entity instanceof EntityComplexPart) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// height = 2.0D as minecraft doesn't store the height of entities,
|
// Distance is calculated from eye location to center of targeted
|
||||||
// and that should be enough. Because entityLocations are always set
|
// If the player is further away from his target than allowed, the
|
||||||
// to center bottom of the hitbox, increase "y" location by 1/2
|
// difference will be assigned to "distance"
|
||||||
// height to get the "center" of the hitbox
|
|
||||||
final double off = CheckUtil.reachCheck(player, entity.locX, entity.locY + 1.0D, entity.locZ, cc.reachLimit);
|
final double off = CheckUtil.reachCheck(player, entity.locX, entity.locY + 1.0D, entity.locZ, cc.reachLimit);
|
||||||
|
|
||||||
if(off < 0.1D) {
|
if(off < 0.1D) {
|
||||||
// Player did probably nothing wrong
|
// Player did probably nothing wrong
|
||||||
// reduce violation counter
|
// reduce violation counter to reward him
|
||||||
data.reachVL *= 0.80D;
|
data.reachVL *= 0.80D;
|
||||||
} else {
|
} else {
|
||||||
// Player failed the check
|
// Player failed the check
|
||||||
// Increment violation counter
|
// Increment violation counter and statistics
|
||||||
// This is influenced by lag, so don't do it if there was server lag
|
// This is influenced by lag, so don't do it if there was lag
|
||||||
if(!plugin.skipCheck()) {
|
if(!plugin.skipCheck()) {
|
||||||
double sqrt = Math.sqrt(off);
|
double sqrt = Math.sqrt(off);
|
||||||
data.reachVL += sqrt;
|
data.reachVL += sqrt;
|
||||||
incrementStatistics(player, Id.FI_REACH, sqrt);
|
incrementStatistics(player, Id.FI_REACH, sqrt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Execute whatever actions are associated with this check and the
|
||||||
|
// violation level and find out if we should cancel the event
|
||||||
cancel = executeActions(player, cc.reachActions, data.reachVL);
|
cancel = executeActions(player, cc.reachActions, data.reachVL);
|
||||||
|
|
||||||
if(cancel) {
|
if(cancel) {
|
||||||
// Needed to calculate penalty times
|
// if we should cancel, remember the current time too
|
||||||
data.reachLastViolationTime = time;
|
data.reachLastViolationTime = time;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the player is still in penalty time, cancel the event anyway
|
// If the player is still in penalty time, cancel the event anyway
|
||||||
if(data.reachLastViolationTime + cc.reachPenaltyTime > time) {
|
if(data.reachLastViolationTime + cc.reachPenaltyTime > time) {
|
||||||
|
// A safeguard to avoid people getting stuck in penalty time
|
||||||
|
// indefinitely in case the system time of the server gets changed
|
||||||
if(data.reachLastViolationTime > time) {
|
if(data.reachLastViolationTime > time) {
|
||||||
System.out.println("Nocheat noted that your time ran backwards for " + (data.reachLastViolationTime - time) + " ms");
|
|
||||||
// Security check for server time changed situations
|
|
||||||
data.reachLastViolationTime = 0;
|
data.reachLastViolationTime = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// He is in penalty time, therefore request cancelling of the event
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -79,10 +86,11 @@ public class ReachCheck extends FightCheck {
|
|||||||
return cc.reachCheck;
|
return cc.reachCheck;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
||||||
|
|
||||||
if(wildcard == ParameterName.VIOLATIONS)
|
if(wildcard == ParameterName.VIOLATIONS)
|
||||||
return String.format(Locale.US, "%d", (int) getData(player.getDataStore()).reachVL);
|
return String.format(Locale.US, "%d", (int) getData(player).reachVL);
|
||||||
else
|
else
|
||||||
return super.getParameter(wildcard, player);
|
return super.getParameter(wildcard, player);
|
||||||
}
|
}
|
||||||
|
@ -7,6 +7,11 @@ import cc.co.evenprime.bukkit.nocheat.actions.ParameterName;
|
|||||||
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
import cc.co.evenprime.bukkit.nocheat.config.Permissions;
|
||||||
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
import cc.co.evenprime.bukkit.nocheat.data.Statistics.Id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The speed check will find out if a player interacts with something that's
|
||||||
|
* too far away
|
||||||
|
*
|
||||||
|
*/
|
||||||
public class SpeedCheck extends FightCheck {
|
public class SpeedCheck extends FightCheck {
|
||||||
|
|
||||||
public SpeedCheck(NoCheat plugin) {
|
public SpeedCheck(NoCheat plugin) {
|
||||||
@ -19,20 +24,26 @@ public class SpeedCheck extends FightCheck {
|
|||||||
|
|
||||||
final long time = System.currentTimeMillis();
|
final long time = System.currentTimeMillis();
|
||||||
|
|
||||||
if(data.speedTime + 1000 <= time) {
|
// Check if one second has passed and reset counters and vl in that case
|
||||||
|
if(data.speedTime + 1000L <= time) {
|
||||||
data.speedTime = time;
|
data.speedTime = time;
|
||||||
data.speedAttackCount = 0;
|
data.speedAttackCount = 0;
|
||||||
data.speedVL = 0;
|
data.speedVL = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// count the attack
|
||||||
data.speedAttackCount++;
|
data.speedAttackCount++;
|
||||||
|
|
||||||
|
// too many attacks
|
||||||
if(data.speedAttackCount > cc.speedAttackLimit) {
|
if(data.speedAttackCount > cc.speedAttackLimit) {
|
||||||
|
// if there was lag, don't count it towards statistics and vl
|
||||||
if(!plugin.skipCheck()) {
|
if(!plugin.skipCheck()) {
|
||||||
data.speedVL += 1;
|
data.speedVL += 1;
|
||||||
incrementStatistics(player, Id.FI_SPEED, 1);
|
incrementStatistics(player, Id.FI_SPEED, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Execute whatever actions are associated with this check and the
|
||||||
|
// violation level and find out if we should cancel the event
|
||||||
cancel = executeActions(player, cc.speedActions, data.speedVL);
|
cancel = executeActions(player, cc.speedActions, data.speedVL);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -44,10 +55,11 @@ public class SpeedCheck extends FightCheck {
|
|||||||
return cc.speedCheck;
|
return cc.speedCheck;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
public String getParameter(ParameterName wildcard, NoCheatPlayer player) {
|
||||||
|
|
||||||
if(wildcard == ParameterName.VIOLATIONS)
|
if(wildcard == ParameterName.VIOLATIONS)
|
||||||
return String.format(Locale.US, "%d", (int) getData(player.getDataStore()).speedVL);
|
return String.format(Locale.US, "%d", (int) getData(player).speedVL);
|
||||||
else if(wildcard == ParameterName.LIMIT)
|
else if(wildcard == ParameterName.LIMIT)
|
||||||
return String.format(Locale.US, "%d", (int) getConfig(player.getConfigurationStore()).speedAttackLimit);
|
return String.format(Locale.US, "%d", (int) getConfig(player.getConfigurationStore()).speedAttackLimit);
|
||||||
else
|
else
|
||||||
|
Loading…
Reference in New Issue
Block a user