diff --git a/src/main/java/net/minestom/server/extras/bungee/BungeeCordProxy.java b/src/main/java/net/minestom/server/extras/bungee/BungeeCordProxy.java
index c4f9f577e..82884558d 100644
--- a/src/main/java/net/minestom/server/extras/bungee/BungeeCordProxy.java
+++ b/src/main/java/net/minestom/server/extras/bungee/BungeeCordProxy.java
@@ -1,12 +1,22 @@
package net.minestom.server.extras.bungee;
+import net.kyori.adventure.text.Component;
+import net.kyori.adventure.text.format.NamedTextColor;
+import org.jetbrains.annotations.NotNull;
+import org.jetbrains.annotations.Nullable;
+
+import java.util.Set;
+
/**
- * BungeeCord forwarding support. This does not count as a security feature, and you will still be required to manage your firewall.
+ * BungeeCord forwarding support. Enabling BungeeGuard support with {@link #setBungeeGuardTokens(Set)} helps to secure the server,
+ * but managing your firewall is still recommended.
*
* Please consider using {@link net.minestom.server.extras.velocity.VelocityProxy} instead.
*/
public final class BungeeCordProxy {
+ private static Set bungeeGuardTokens = null;
+
private static volatile boolean enabled;
/**
@@ -24,4 +34,35 @@ public final class BungeeCordProxy {
public static boolean isEnabled() {
return enabled;
}
+
+ /**
+ * Sets the tokens used by BungeeGuard authentication.
+ * Setting the tokens to a not-null value enables BungeeGuard authentication,
+ * and setting it to a null value disables BungeeGuard authentication.
+ *
+ * @param tokens The new BungeeGuard authentication tokens
+ */
+ public static void setBungeeGuardTokens(@Nullable Set tokens) {
+ bungeeGuardTokens = tokens;
+ }
+
+ /**
+ * Checks whether BungeeGuard authentication is enabled.
+ *
+ * @return Whether BungeeGuard authentication is enabled
+ */
+ public static boolean isBungeeGuardEnabled() {
+ return bungeeGuardTokens != null;
+ }
+
+ /**
+ * Checks whether a token is one of the valid BungeeGuard tokens
+ *
+ * @param token The token to test
+ * @return Whether the token is a valid BungeeGuard token
+ */
+ public static boolean isValidBungeeGuardToken(@NotNull String token) {
+ return isBungeeGuardEnabled() && bungeeGuardTokens.contains(token);
+ }
+
}
diff --git a/src/main/java/net/minestom/server/network/packet/client/handshake/HandshakePacket.java b/src/main/java/net/minestom/server/network/packet/client/handshake/HandshakePacket.java
index e24d2a8f0..e8922654e 100644
--- a/src/main/java/net/minestom/server/network/packet/client/handshake/HandshakePacket.java
+++ b/src/main/java/net/minestom/server/network/packet/client/handshake/HandshakePacket.java
@@ -16,6 +16,8 @@ import net.minestom.server.network.player.GameProfile;
import net.minestom.server.network.player.PlayerConnection;
import net.minestom.server.network.player.PlayerSocketConnection;
import org.jetbrains.annotations.NotNull;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
import java.net.SocketAddress;
import java.util.ArrayList;
@@ -27,14 +29,20 @@ import static net.minestom.server.network.NetworkBuffer.*;
public record HandshakePacket(int protocolVersion, @NotNull String serverAddress,
int serverPort, int nextState) implements ClientPreplayPacket {
+ private final static Logger LOGGER = LoggerFactory.getLogger(HandshakePacket.class);
+
/**
* Text sent if a player tries to connect with an invalid version of the client
*/
private static final Component INVALID_VERSION_TEXT = Component.text("Invalid Version, please use " + MinecraftServer.VERSION_NAME, NamedTextColor.RED);
- private static final Component INVALID_BUNGEE_FORWARDING = Component.text("If you wish to use IP forwarding, please enable it in your BungeeCord config as well!", NamedTextColor.RED);
+
+ /**
+ * Indicates that a BungeeGuard authentication was invalid due to missing, multiple, or invalid tokens.
+ */
+ private static final Component INVALID_BUNGEE_FORWARDING = Component.text("Invalid connection, please connect through the BungeeCord proxy. If you believe this is an error, contact a server administrator.", NamedTextColor.RED);
public HandshakePacket {
- if (serverAddress.length() > (BungeeCordProxy.isEnabled() ? Short.MAX_VALUE : 255)) {
+ if (serverAddress.length() > getMaxHandshakeLength()) {
throw new IllegalArgumentException("Server address too long: " + serverAddress.length());
}
}
@@ -47,7 +55,7 @@ public record HandshakePacket(int protocolVersion, @NotNull String serverAddress
@Override
public void write(@NotNull NetworkBuffer writer) {
writer.write(VAR_INT, protocolVersion);
- int maxLength = BungeeCordProxy.isEnabled() ? Short.MAX_VALUE : 255;
+ int maxLength = getMaxHandshakeLength();
if (serverAddress.length() > maxLength) {
throw new IllegalArgumentException("serverAddress is " + serverAddress.length() + " characters long, maximum allowed is " + maxLength);
}
@@ -64,6 +72,12 @@ public record HandshakePacket(int protocolVersion, @NotNull String serverAddress
final String[] split = address.split("\00");
if (split.length == 3 || split.length == 4) {
+ boolean hasProperties = split.length == 4;
+ if (BungeeCordProxy.isBungeeGuardEnabled() && !hasProperties) {
+ bungeeDisconnect(socketConnection);
+ return;
+ }
+
address = split[0];
final SocketAddress socketAddress = new java.net.InetSocketAddress(split[1],
@@ -78,7 +92,8 @@ public record HandshakePacket(int protocolVersion, @NotNull String serverAddress
);
List properties = new ArrayList<>();
- if (split.length == 4) {
+ if (hasProperties) {
+ boolean foundBungeeGuardToken = false;
final String rawPropertyJson = split[3];
final JsonArray propertyJson = JsonParser.parseString(rawPropertyJson).getAsJsonArray();
for (JsonElement element : propertyJson) {
@@ -92,15 +107,28 @@ public record HandshakePacket(int protocolVersion, @NotNull String serverAddress
final String valueString = value.getAsString();
final String signatureString = signature == null ? null : signature.getAsString();
+ if (BungeeCordProxy.isBungeeGuardEnabled() && nameString.equals("bungeeguard-token")) {
+ if (foundBungeeGuardToken || !BungeeCordProxy.isValidBungeeGuardToken(valueString)) {
+ bungeeDisconnect(socketConnection);
+ return;
+ }
+
+ foundBungeeGuardToken = true;
+ }
+
properties.add(new GameProfile.Property(nameString, valueString, signatureString));
}
+
+ if (BungeeCordProxy.isBungeeGuardEnabled() && !foundBungeeGuardToken) {
+ bungeeDisconnect(socketConnection);
+ return;
+ }
}
final GameProfile gameProfile = new GameProfile(playerUuid, "test", properties);
socketConnection.UNSAFE_setProfile(gameProfile);
} else {
- socketConnection.sendPacket(new LoginDisconnectPacket(INVALID_BUNGEE_FORWARDING));
- socketConnection.disconnect();
+ bungeeDisconnect(socketConnection);
return;
}
}
@@ -117,8 +145,7 @@ public record HandshakePacket(int protocolVersion, @NotNull String serverAddress
connection.setConnectionState(ConnectionState.LOGIN);
} else {
// Incorrect client version
- connection.sendPacket(new LoginDisconnectPacket(INVALID_VERSION_TEXT));
- connection.disconnect();
+ disconnect(connection, INVALID_VERSION_TEXT);
}
}
default -> {
@@ -126,4 +153,20 @@ public record HandshakePacket(int protocolVersion, @NotNull String serverAddress
}
}
}
+
+ private static int getMaxHandshakeLength() {
+ // BungeeGuard limits handshake length to 2500 characters, while vanilla limits it to 255
+ return BungeeCordProxy.isEnabled() ? (BungeeCordProxy.isBungeeGuardEnabled() ? 2500 : Short.MAX_VALUE) : 255;
+ }
+
+ private void disconnect(@NotNull PlayerConnection connection, @NotNull Component reason) {
+ connection.sendPacket(new LoginDisconnectPacket(reason));
+ connection.disconnect();
+ }
+
+ private void bungeeDisconnect(@NotNull PlayerConnection connection) {
+ LOGGER.warn("{} tried to log in without valid BungeeGuard forwarding information.", connection.getIdentifier());
+ disconnect(connection, INVALID_BUNGEE_FORWARDING);
+ }
+
}