From 06ba2f54e541f6463ec4973b47f2f88eba5e470b Mon Sep 17 00:00:00 2001 From: Outfluencer Date: Thu, 28 Nov 2024 15:39:18 +0100 Subject: [PATCH 1/2] Improve bungee command --- .../main/java/net/md_5/bungee/BungeeCord.java | 2 +- .../md_5/bungee/command/CommandBungee.java | 60 ++++++++++++++++++- .../bungee/util/BuildVersionRetriever.java | 52 ++++++++++++++++ proxy/src/main/resources/messages.properties | 1 + 4 files changed, 111 insertions(+), 4 deletions(-) create mode 100644 proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java diff --git a/proxy/src/main/java/net/md_5/bungee/BungeeCord.java b/proxy/src/main/java/net/md_5/bungee/BungeeCord.java index e7f3f09af9d..f674f5809a7 100644 --- a/proxy/src/main/java/net/md_5/bungee/BungeeCord.java +++ b/proxy/src/main/java/net/md_5/bungee/BungeeCord.java @@ -235,7 +235,7 @@ public BungeeCord() throws IOException getPluginManager().registerCommand( null, new CommandReload() ); getPluginManager().registerCommand( null, new CommandEnd() ); getPluginManager().registerCommand( null, new CommandIP() ); - getPluginManager().registerCommand( null, new CommandBungee() ); + getPluginManager().registerCommand( null, new CommandBungee( this ) ); getPluginManager().registerCommand( null, new CommandPerms() ); if ( !Boolean.getBoolean( "net.md_5.bungee.native.disable" ) ) diff --git a/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java b/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java index b07987915f5..20ba61b0a13 100644 --- a/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java +++ b/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java @@ -3,19 +3,73 @@ import net.md_5.bungee.api.ChatColor; import net.md_5.bungee.api.CommandSender; import net.md_5.bungee.api.ProxyServer; +import net.md_5.bungee.api.chat.ClickEvent; +import net.md_5.bungee.api.chat.ComponentBuilder; +import net.md_5.bungee.api.chat.HoverEvent; +import net.md_5.bungee.api.chat.TextComponent; +import net.md_5.bungee.api.chat.hover.content.Text; import net.md_5.bungee.api.plugin.Command; +import net.md_5.bungee.module.ModuleVersion; +import net.md_5.bungee.util.BuildVersionRetriever; public class CommandBungee extends Command { + private final BuildVersionRetriever versionRetriever = new BuildVersionRetriever(); + private final ModuleVersion moduleVersion; + private int currentVersion = -1; - public CommandBungee() + public CommandBungee(ProxyServer proxy) { super( "bungee" ); + moduleVersion = ModuleVersion.parse( proxy.getVersion() ); + try + { + if ( moduleVersion != null ) + { + currentVersion = Integer.parseInt( moduleVersion.getBuild() ); + } + } catch ( NumberFormatException ignored ) + { + } } - @Override public void execute(CommandSender sender, String[] args) { - sender.sendMessage( ChatColor.BLUE + "This server is running BungeeCord version " + ProxyServer.getInstance().getVersion() + " by md_5" ); + TextComponent component = new TextComponent( "This server is running BungeeCord version " ); + component.setColor( ChatColor.BLUE ); + TextComponent version = new TextComponent( ProxyServer.getInstance().getVersion() ); + version.setClickEvent( new ClickEvent( ClickEvent.Action.COPY_TO_CLIPBOARD, ProxyServer.getInstance().getVersion() ) ); + version.setHoverEvent( new HoverEvent( HoverEvent.Action.SHOW_TEXT, new Text( new TextComponent( ProxyServer.getInstance().getTranslation( "click_to_copy" ) ) ) ) ); + component.addExtra( version ); + component.addExtra( " by md_5" ); + sender.sendMessage( component ); + + versionRetriever.retrieveLatestBuild().thenAccept( latest -> + { + // custom build or spammed command while retrieving + if ( moduleVersion == null || currentVersion == -1 || latest == 0 ) + { + return; + } + + if ( currentVersion >= latest ) + { + sender.sendMessage( ChatColor.GREEN + "This server is running the latest build of BungeeCord" ); + return; + } + + int behind = latest - currentVersion; + sender.sendMessage( ChatColor.YELLOW + "This server is " + behind + " builds behind" ); + sender.sendMessage( new ComponentBuilder( "Click here to get the latest build" ) + .italic( true ) + .color( ChatColor.YELLOW ) + .event( new ClickEvent( ClickEvent.Action.OPEN_URL, "https://ci.md-5.net/job/BungeeCord/lastBuild/" ) ) + .build() ); + + } ).exceptionally( ex -> + { + sender.sendMessage( ChatColor.RED + "Error fetching latest successful build" ); + return null; + } ); } } diff --git a/proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java b/proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java new file mode 100644 index 00000000000..83c6e84eb53 --- /dev/null +++ b/proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java @@ -0,0 +1,52 @@ +package net.md_5.bungee.util; + +import com.google.gson.Gson; +import com.google.gson.GsonBuilder; +import com.google.gson.JsonObject; +import java.io.InputStreamReader; +import java.net.URL; +import java.net.URLConnection; +import java.time.Instant; +import java.util.concurrent.CompletableFuture; +import java.util.concurrent.Executor; +import java.util.concurrent.Executors; + +public class BuildVersionRetriever +{ + private static final Gson GSON = new GsonBuilder().create(); + private final Executor executor = Executors.newSingleThreadExecutor(); + + private int cachedLatestBuildVersion; + private Instant lastLatestBuildRetrieve; + + public CompletableFuture retrieveLatestBuild() + { + // cache result for 5 minutes + if ( lastLatestBuildRetrieve != null && Instant.now().isBefore( lastLatestBuildRetrieve.plusSeconds( 300 ) ) ) + { + return CompletableFuture.completedFuture( cachedLatestBuildVersion ); + } + + lastLatestBuildRetrieve = Instant.now(); + CompletableFuture completableFuture = new CompletableFuture<>(); + executor.execute( () -> + { + try + { + URL restApi = new URL( "https://ci.md-5.net/job/BungeeCord/api/json" ); + URLConnection connection = restApi.openConnection(); + + connection.setConnectTimeout( 15000 ); + connection.setReadTimeout( 15000 ); + + JsonObject jsonObject = GSON.fromJson( new InputStreamReader( connection.getInputStream() ), JsonObject.class ); + jsonObject = (JsonObject) jsonObject.get( "lastSuccessfulBuild" ); + completableFuture.complete( cachedLatestBuildVersion = jsonObject.get( "number" ).getAsInt() ); + } catch ( Exception exception ) + { + completableFuture.completeExceptionally( exception ); + } + } ); + return completableFuture; + } +} diff --git a/proxy/src/main/resources/messages.properties b/proxy/src/main/resources/messages.properties index 149c742f258..766f8f17d82 100644 --- a/proxy/src/main/resources/messages.properties +++ b/proxy/src/main/resources/messages.properties @@ -41,3 +41,4 @@ command_ip=\u00a79IP of {0} is {1} illegal_chat_characters=\u00a7cIllegal characters in chat ({0}) kick_message=\u00a7cYou have been kicked off the proxy. reject_transfer=\u00a7cYour transfer was rejected. +click_to_copy=Click to copy to clipboard From f483dcbd47525b24c6eec7a68e7a26e0a9b678c1 Mon Sep 17 00:00:00 2001 From: Outfluencer Date: Sun, 1 Dec 2024 14:40:03 +0100 Subject: [PATCH 2/2] cache the completeable future, don't retrieve custom builds, add retrieving info, change cachetime to 15min --- .../md_5/bungee/command/CommandBungee.java | 23 ++++++++----- .../java/net/md_5/bungee/conf/YamlConfig.java | 2 +- .../bungee/util/BuildVersionRetriever.java | 33 ++++++++++++++----- 3 files changed, 41 insertions(+), 17 deletions(-) diff --git a/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java b/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java index 20ba61b0a13..d3429a86ba3 100644 --- a/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java +++ b/proxy/src/main/java/net/md_5/bungee/command/CommandBungee.java @@ -1,5 +1,6 @@ package net.md_5.bungee.command; +import java.util.concurrent.CompletableFuture; import net.md_5.bungee.api.ChatColor; import net.md_5.bungee.api.CommandSender; import net.md_5.bungee.api.ProxyServer; @@ -15,13 +16,12 @@ public class CommandBungee extends Command { private final BuildVersionRetriever versionRetriever = new BuildVersionRetriever(); - private final ModuleVersion moduleVersion; private int currentVersion = -1; public CommandBungee(ProxyServer proxy) { super( "bungee" ); - moduleVersion = ModuleVersion.parse( proxy.getVersion() ); + ModuleVersion moduleVersion = ModuleVersion.parse( proxy.getVersion() ); try { if ( moduleVersion != null ) @@ -44,14 +44,21 @@ public void execute(CommandSender sender, String[] args) component.addExtra( " by md_5" ); sender.sendMessage( component ); - versionRetriever.retrieveLatestBuild().thenAccept( latest -> + // return if custom build or no permission + if ( currentVersion == -1 || !sender.hasPermission( "bungeecord.command.bungee.versioncheck" ) ) { - // custom build or spammed command while retrieving - if ( moduleVersion == null || currentVersion == -1 || latest == 0 ) - { - return; - } + return; + } + + CompletableFuture future = versionRetriever.retrieveLatestBuild(); + if ( !future.isDone() ) + { + sender.sendMessage( ChatColor.YELLOW + "Checking latest version..." ); + } + + future.thenAccept( latest -> + { if ( currentVersion >= latest ) { sender.sendMessage( ChatColor.GREEN + "This server is running the latest build of BungeeCord" ); diff --git a/proxy/src/main/java/net/md_5/bungee/conf/YamlConfig.java b/proxy/src/main/java/net/md_5/bungee/conf/YamlConfig.java index d659a138fb0..c5828c4a920 100644 --- a/proxy/src/main/java/net/md_5/bungee/conf/YamlConfig.java +++ b/proxy/src/main/java/net/md_5/bungee/conf/YamlConfig.java @@ -95,7 +95,7 @@ public void load() } ) ); set( "permissions.admin", Arrays.asList( new String[] { - "bungeecord.command.alert", "bungeecord.command.end", "bungeecord.command.ip", "bungeecord.command.reload", "bungeecord.command.kick", "bungeecord.command.send", "bungeecord.command.find" + "bungeecord.command.alert", "bungeecord.command.end", "bungeecord.command.ip", "bungeecord.command.reload", "bungeecord.command.kick", "bungeecord.command.send", "bungeecord.command.find", "bungeecord.command.bungee.versioncheck" } ) ); } diff --git a/proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java b/proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java index 83c6e84eb53..5ccc0664f7a 100644 --- a/proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java +++ b/proxy/src/main/java/net/md_5/bungee/util/BuildVersionRetriever.java @@ -6,31 +6,48 @@ import java.io.InputStreamReader; import java.net.URL; import java.net.URLConnection; -import java.time.Instant; import java.util.concurrent.CompletableFuture; import java.util.concurrent.Executor; import java.util.concurrent.Executors; +import java.util.concurrent.TimeUnit; public class BuildVersionRetriever { private static final Gson GSON = new GsonBuilder().create(); - private final Executor executor = Executors.newSingleThreadExecutor(); + private final Executor executor = Executors.newSingleThreadExecutor(); private int cachedLatestBuildVersion; - private Instant lastLatestBuildRetrieve; + private long lastLatestBuildRetrieve; + private CompletableFuture currentRetrievingFuture; - public CompletableFuture retrieveLatestBuild() + public synchronized CompletableFuture retrieveLatestBuild() { - // cache result for 5 minutes - if ( lastLatestBuildRetrieve != null && Instant.now().isBefore( lastLatestBuildRetrieve.plusSeconds( 300 ) ) ) + // return the awaiting retrieving future is there is any right now + if ( currentRetrievingFuture != null && !currentRetrievingFuture.isDone() ) + { + return currentRetrievingFuture; + } + + // cache result for 15 minutes + if ( System.currentTimeMillis() < lastLatestBuildRetrieve + TimeUnit.MINUTES.toMillis( 15 ) ) { + // if retrieving failed this is 0, so we try again. + if ( cachedLatestBuildVersion == 0 ) + { + return retrieveLatestBuild0(); + } return CompletableFuture.completedFuture( cachedLatestBuildVersion ); } - lastLatestBuildRetrieve = Instant.now(); - CompletableFuture completableFuture = new CompletableFuture<>(); + return retrieveLatestBuild0(); + } + + private CompletableFuture retrieveLatestBuild0() + { + CompletableFuture completableFuture = currentRetrievingFuture = new CompletableFuture<>(); executor.execute( () -> { + lastLatestBuildRetrieve = System.currentTimeMillis(); try { URL restApi = new URL( "https://ci.md-5.net/job/BungeeCord/api/json" );