New ping checker
All checks were successful
/ deploy (push) Successful in 5m3s

This commit is contained in:
Minecon724 2024-11-30 20:13:22 +01:00
parent 47e347c4a2
commit b23f773198
Signed by: Minecon724
GPG key ID: 3CCC4D267742C8E8
13 changed files with 149 additions and 174 deletions

View file

@ -53,4 +53,21 @@ Sleeping doesn't skip night, but speeds it up. The more players, the faster it g
### Instant sleep
One can instantly skip, but only a part of the night. \
There's 5 players on the server. A night is 10 minutes long. \
Each player can instantly skip 2 minutes of the night at any time, even if others aren't sleeping
Each player can instantly skip 2 minutes of the night at any time, even if others aren't sleeping
# Commands
### /chat
Changes chatroom
### /chatmanage
`tweaks724.chatmanage` \
Manages chatroom, like create, delete etc.
### /ping
Displays your ping. \
**Ping is calculated by the plugin**. \
That allows for more precision (decimal places) and to get the ping immediately after a player join
### /pomodoro
`tweaks724.pomodoro` \
Manage your pomodoro. `/pom start` to start, `stop` to stop, no arguments to forward

View file

@ -10,8 +10,6 @@ import org.bukkit.configuration.file.FileConfiguration;
import org.bukkit.plugin.Plugin;
public record TweaksConfig(
boolean isProtocolLib,
boolean worldborderHide,
boolean brandEnabled,
@ -56,8 +54,6 @@ public record TweaksConfig(
plugin.saveDefaultConfig();
FileConfiguration config = plugin.getConfig();
boolean isProtocolLib = plugin.getServer().getPluginManager().getPlugin("ProtocolLib") != null;
int configVersion = config.getInt("magic number don't modify this", 0);
RuntimeException exception = new RuntimeException("Config version is %d, expected %d".formatted(configVersion, CONFIG_VERSION));
if (configVersion == 0) {
@ -102,7 +98,6 @@ public record TweaksConfig(
boolean sleepInstant = config.getBoolean("sleep.instant");
TweaksConfig.config = new TweaksConfig(
isProtocolLib,
hideWorldBorder,
brandEnabled, brandText, brandShowPing, brandShowMspt,
doorEnabled, doorDoubleOpen, doorKnocking,

View file

@ -52,7 +52,6 @@ public class TweaksPlugin extends JavaPlugin {
new PingChecker(this).init();
Objects.requireNonNull(getCommand("ping")).setExecutor(new PingCommands());
Objects.requireNonNull(getCommand("dkick")).setExecutor(new PingCommands());
/*if (getServer().getPluginManager().getPlugin("voicechat") != null) {
new MusicPlayer(this).init();

View file

@ -78,14 +78,14 @@ public class F3NameListener {
String sep = text.isEmpty() ? "" : " | ";
if (showMspt) {
double mspt = PlayerPing.getMillisPerTick();
double mspt = PlayerPingStorage.getMillisPerTick();
if (mspt > 50.05) { // mspt is at least like 50.01 because of some measuring overheads (I think)
_text += sep + "%.2f mspt".formatted(mspt);
}
}
if (showPing) {
_text += sep + "%.2f ms".formatted(PlayerPing.getPingMillis(player));
_text += sep + "%.2f ms".formatted(PlayerPingStorage.getPingMillis(player));
}
return _text;

View file

@ -0,0 +1,69 @@
/*
* Copyright (C) 2024 Minecon724
* Tweaks724 is licensed under the GNU General Public License. See the LICENSE.md file
* in the project root for the full license text.
*/
package eu.m724.tweaks.ping;
import com.comphenix.protocol.PacketType;
import com.comphenix.protocol.ProtocolLibrary;
import com.comphenix.protocol.events.ListenerPriority;
import com.comphenix.protocol.events.PacketAdapter;
import com.comphenix.protocol.events.PacketContainer;
import com.comphenix.protocol.events.PacketEvent;
import org.bukkit.entity.Player;
import org.bukkit.plugin.Plugin;
import org.bukkit.scheduler.BukkitRunnable;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ThreadLocalRandom;
public class KeepAlivePingChecker extends BukkitRunnable {
// ID, timestamp
private final Map<Long, Long> keepAliveIDs = new HashMap<>();
private final Plugin plugin;
KeepAlivePingChecker(Plugin plugin) {
this.plugin = plugin;
}
public void start() {
ProtocolLibrary.getProtocolManager().addPacketListener(new PacketAdapter(
plugin,
ListenerPriority.NORMAL,
PacketType.Play.Client.KEEP_ALIVE
) {
@Override
public void onPacketReceiving(PacketEvent event) {
long id = event.getPacket().getLongs().read(0);
Long start = keepAliveIDs.remove(id);
if (start != null) {
Player player = event.getPlayer();
PlayerPingStorage.submitPing(player, System.nanoTime() - start);
// server will kick if we don't cancel
event.setCancelled(true);
}
}
});
this.runTaskTimerAsynchronously(plugin, 0, 40); // 2 secs
}
@Override
public void run() {
plugin.getServer().getOnlinePlayers().forEach(player -> {
long id = ThreadLocalRandom.current().nextLong();
keepAliveIDs.put(id, System.nanoTime());
PacketContainer packet = new PacketContainer(PacketType.Play.Server.KEEP_ALIVE);
packet.getLongs().write(0, id);
ProtocolLibrary.getProtocolManager().sendServerPacket(player, packet);
});
}
}

View file

@ -15,11 +15,11 @@ public class MsptChecker extends BukkitRunnable {
@Override
public void run() {
long now = System.nanoTime();
PlayerPing.nspt = (now - lastLoop) / 100;
PlayerPingStorage.submitNspt((now - lastLoop) / 40);
lastLoop = now;
}
public void init(Plugin plugin) {
this.runTaskTimer(plugin, 0, 100); // 5 secs
this.runTaskTimer(plugin, 0, 40); // 2 secs
}
}

View file

@ -6,7 +6,6 @@
package eu.m724.tweaks.ping;
import eu.m724.tweaks.TweaksConfig;
import org.bukkit.plugin.Plugin;
public class PingChecker {
@ -17,14 +16,7 @@ public class PingChecker {
}
public void init() {
if (TweaksConfig.getConfig().isProtocolLib()) {
plugin.getLogger().info("Using ProtocolLib for checking ping");
new ProtocolPingChecker(plugin).start();
} else {
plugin.getLogger().info("Using Spigot for checking ping");
}
new KeepAlivePingChecker(plugin).start();
new MsptChecker().init(plugin); // TODO should this be here
}
}

View file

@ -6,11 +6,9 @@
package eu.m724.tweaks.ping;
import eu.m724.tweaks.TweaksConfig;
import net.md_5.bungee.api.ChatColor;
import net.md_5.bungee.api.chat.BaseComponent;
import net.md_5.bungee.api.chat.ComponentBuilder;
import org.bukkit.Bukkit;
import org.bukkit.command.Command;
import org.bukkit.command.CommandExecutor;
import org.bukkit.command.CommandSender;
@ -23,33 +21,9 @@ public class PingCommands implements CommandExecutor {
if (command.getName().equals("ping")) {
Player player = (Player) sender;
BaseComponent[] component = new ComponentBuilder("Ping: ").color(ChatColor.GOLD)
.append("%.2fms".formatted(PlayerPing.getPingMillis(player))).color(ChatColor.AQUA)
.append("%.2fms".formatted(PlayerPingStorage.getPingMillis(player))).color(ChatColor.AQUA)
.create();
player.spigot().sendMessage(component);
} else if (command.getName().equals("dkick")) {
if (!TweaksConfig.getConfig().isProtocolLib()) {
sender.sendMessage("This feature is not available");
return true;
}
if (args.length == 0) {
sender.sendMessage("Include one or more player names");
} else {
for (String name : args) {
Player player = Bukkit.getPlayer(name);
if (player == null) {
sender.sendMessage("Player %s is not online".formatted(name));
}
if (PlayerPing.kick(player)) {
if (player.getName().equals(sender.getName())) {
sender.sendMessage("Kicking yourself? Very well");
}
sender.sendMessage("Player %s will be kicked shortly".formatted(name));
} else {
sender.sendMessage("Player %s is already queued".formatted(name));
}
}
}
}
return true;
}

View file

@ -1,51 +0,0 @@
/*
* Copyright (C) 2024 Minecon724
* Tweaks724 is licensed under the GNU General Public License. See the LICENSE.md file
* in the project root for the full license text.
*/
package eu.m724.tweaks.ping;
import eu.m724.tweaks.TweaksConfig;
import org.bukkit.entity.Player;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
public class PlayerPing {
// if protocol lib is installed we use our own checker
private static final boolean external = TweaksConfig.getConfig().isProtocolLib();
// TODO remove this feature because why
static final Set<Player> kickQueue = ConcurrentHashMap.newKeySet();
// current player latencies, in nanos
static final Map<Player, Long> pings = new ConcurrentHashMap<>();
// nanoseconds per tick
static volatile long nspt = 0L;
public static long getPingNanos(Player player) {
if (!external) {
return (long) player.getPing() * 1000000;
} else {
return pings.getOrDefault(player, -1L);
}
}
public static double getPingMillis(Player player) {
return getPingNanos(player) / 1000000.0; // a mil ns in ms
}
public static long getNanosPerTick() {
return nspt;
}
public static double getMillisPerTick() {
return nspt / 1000000.0; // a mil ns in ms
}
public static boolean kick(Player player) {
return kickQueue.add(player);
}
}

View file

@ -0,0 +1,55 @@
/*
* Copyright (C) 2024 Minecon724
* Tweaks724 is licensed under the GNU General Public License. See the LICENSE.md file
* in the project root for the full license text.
*/
package eu.m724.tweaks.ping;
import org.bukkit.entity.Player;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
public class PlayerPingStorage {
// in nanos
private static final Map<Player, Long> pings = new ConcurrentHashMap<>();
private static final Map<Player, Integer> pingsCount = new ConcurrentHashMap<>();
// nanoseconds per tick
private static volatile long nspt = 0L;
public static long getPingNanos(Player player) {
return pings.getOrDefault(player, 0L);
}
public static double getPingMillis(Player player) {
return getPingNanos(player) / 1000000.0; // a mil ns in ms
}
public static long getNanosPerTick() {
return nspt;
}
public static double getMillisPerTick() {
return nspt / 1000000.0; // a mil ns in ms
}
static void submitPing(Player player, long currentPing) {
int count = pingsCount.getOrDefault(player, 0);
long average = pings.getOrDefault(player, 0L);
if (count < 5) {
average = ((average * count) + currentPing) / (count + 1);
pingsCount.put(player, count + 1);
} else {
average += (currentPing - average) / 5;
}
pings.put(player, average);
}
static void submitNspt(long currentNspt) {
nspt = currentNspt;
}
}

View file

@ -1,72 +0,0 @@
/*
* Copyright (C) 2024 Minecon724
* Tweaks724 is licensed under the GNU General Public License. See the LICENSE.md file
* in the project root for the full license text.
*/
package eu.m724.tweaks.ping;
import com.comphenix.protocol.PacketType;
import com.comphenix.protocol.ProtocolLibrary;
import com.comphenix.protocol.events.ListenerPriority;
import com.comphenix.protocol.events.PacketAdapter;
import com.comphenix.protocol.events.PacketContainer;
import com.comphenix.protocol.events.PacketEvent;
import com.comphenix.protocol.wrappers.MinecraftKey;
import org.bukkit.entity.Player;
import org.bukkit.plugin.Plugin;
import org.bukkit.scheduler.BukkitRunnable;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
public class ProtocolPingChecker extends BukkitRunnable {
// this is in nanoseconds
private final Map<Player, Long> pending = new ConcurrentHashMap<>();
private final Plugin plugin;
ProtocolPingChecker(Plugin plugin) {
this.plugin = plugin;
}
public void start() {
ProtocolLibrary.getProtocolManager().addPacketListener(new PacketAdapter(
plugin,
ListenerPriority.NORMAL,
PacketType.Play.Client.COOKIE_RESPONSE
) {
@Override
public void onPacketReceiving(PacketEvent event) {
// below line checks, whether the first (and sole) identifier field in the packet is minecraft:chk_ping
if (event.getPacket().getMinecraftKeys().read(0).getKey().equals("chk_ping")) {
Player player = event.getPlayer();
long start = pending.remove(player);
PlayerPing.pings.put(player, System.nanoTime() - start);
// gotta cancel because the server will kick
if (!PlayerPing.kickQueue.contains(player)) {
event.setCancelled(true);
} else {
PlayerPing.kickQueue.remove(player);
}
}
}
});
this.runTaskTimerAsynchronously(plugin, 0, 200); // 10 secs
}
@Override
public void run() {
plugin.getServer().getOnlinePlayers().forEach(player -> {
// dropped packets happen so timing out a request after 30 seconds
if (System.nanoTime() - pending.getOrDefault(player, 0L) < 30000000000L) return;
pending.put(player, System.nanoTime()); // here or at the bottom? probably doesn't matter
PacketContainer packet = new PacketContainer(PacketType.Play.Server.COOKIE_REQUEST);
packet.getMinecraftKeys().write(0, new MinecraftKey("chk_ping"));
ProtocolLibrary.getProtocolManager().sendServerPacket(player, packet);
});
}
}

View file

@ -42,8 +42,8 @@ chat:
# Name of the default / global chatroom
defaultName: "global"
# Compass shown in a text form on the actionbar
compass:
# Compass will be shown in a text form
enabled: true
# How much points (each point is separated with a space)
# I suggest making this uneven to make center... in center

View file

@ -19,9 +19,6 @@ commands:
aliases: [cm, crm]
ping:
description: Your ping
dkick:
description: Kick a player discreetly
permission: tweaks724.dkick
pomodoro:
description: Pomodoro management
permission: tweaks724.pomodoro