diff --git a/README.md b/README.md index caa1e10..87da290 100644 --- a/README.md +++ b/README.md @@ -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 \ No newline at end of file +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 \ No newline at end of file diff --git a/src/main/java/eu/m724/tweaks/TweaksConfig.java b/src/main/java/eu/m724/tweaks/TweaksConfig.java index a8c1197..1bd5752 100644 --- a/src/main/java/eu/m724/tweaks/TweaksConfig.java +++ b/src/main/java/eu/m724/tweaks/TweaksConfig.java @@ -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, diff --git a/src/main/java/eu/m724/tweaks/TweaksPlugin.java b/src/main/java/eu/m724/tweaks/TweaksPlugin.java index 75aa267..4f92389 100644 --- a/src/main/java/eu/m724/tweaks/TweaksPlugin.java +++ b/src/main/java/eu/m724/tweaks/TweaksPlugin.java @@ -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(); diff --git a/src/main/java/eu/m724/tweaks/ping/F3NameListener.java b/src/main/java/eu/m724/tweaks/ping/F3NameListener.java index b2d2f5b..9577841 100644 --- a/src/main/java/eu/m724/tweaks/ping/F3NameListener.java +++ b/src/main/java/eu/m724/tweaks/ping/F3NameListener.java @@ -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; diff --git a/src/main/java/eu/m724/tweaks/ping/KeepAlivePingChecker.java b/src/main/java/eu/m724/tweaks/ping/KeepAlivePingChecker.java new file mode 100644 index 0000000..45fa7bd --- /dev/null +++ b/src/main/java/eu/m724/tweaks/ping/KeepAlivePingChecker.java @@ -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 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); + }); + } +} diff --git a/src/main/java/eu/m724/tweaks/ping/MsptChecker.java b/src/main/java/eu/m724/tweaks/ping/MsptChecker.java index c7fb4d7..dec46a0 100644 --- a/src/main/java/eu/m724/tweaks/ping/MsptChecker.java +++ b/src/main/java/eu/m724/tweaks/ping/MsptChecker.java @@ -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 } } diff --git a/src/main/java/eu/m724/tweaks/ping/PingChecker.java b/src/main/java/eu/m724/tweaks/ping/PingChecker.java index d271d4d..d5d982f 100644 --- a/src/main/java/eu/m724/tweaks/ping/PingChecker.java +++ b/src/main/java/eu/m724/tweaks/ping/PingChecker.java @@ -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 } } diff --git a/src/main/java/eu/m724/tweaks/ping/PingCommands.java b/src/main/java/eu/m724/tweaks/ping/PingCommands.java index 435f0a7..01d9a1c 100644 --- a/src/main/java/eu/m724/tweaks/ping/PingCommands.java +++ b/src/main/java/eu/m724/tweaks/ping/PingCommands.java @@ -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; } diff --git a/src/main/java/eu/m724/tweaks/ping/PlayerPing.java b/src/main/java/eu/m724/tweaks/ping/PlayerPing.java deleted file mode 100644 index 0903a22..0000000 --- a/src/main/java/eu/m724/tweaks/ping/PlayerPing.java +++ /dev/null @@ -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 kickQueue = ConcurrentHashMap.newKeySet(); - - // current player latencies, in nanos - static final Map 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); - } -} diff --git a/src/main/java/eu/m724/tweaks/ping/PlayerPingStorage.java b/src/main/java/eu/m724/tweaks/ping/PlayerPingStorage.java new file mode 100644 index 0000000..f41285f --- /dev/null +++ b/src/main/java/eu/m724/tweaks/ping/PlayerPingStorage.java @@ -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 pings = new ConcurrentHashMap<>(); + private static final Map 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; + } +} diff --git a/src/main/java/eu/m724/tweaks/ping/ProtocolPingChecker.java b/src/main/java/eu/m724/tweaks/ping/ProtocolPingChecker.java deleted file mode 100644 index 1db0b0b..0000000 --- a/src/main/java/eu/m724/tweaks/ping/ProtocolPingChecker.java +++ /dev/null @@ -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 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); - }); - } -} diff --git a/src/main/resources/config.yml b/src/main/resources/config.yml index 06a901f..94c1766 100644 --- a/src/main/resources/config.yml +++ b/src/main/resources/config.yml @@ -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 diff --git a/src/main/resources/plugin.yml b/src/main/resources/plugin.yml index e0b8f3f..2b366b2 100644 --- a/src/main/resources/plugin.yml +++ b/src/main/resources/plugin.yml @@ -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