geforkt von Mirrors/Velocity
Refactor VelocityConfiguration to better support for config upgrades
Dieser Commit ist enthalten in:
Ursprung
2c7dfaaaf9
Commit
64fadc436b
@ -21,7 +21,7 @@ import com.velocitypowered.proxy.config.VelocityConfiguration;
|
|||||||
import com.velocitypowered.proxy.connection.client.ConnectedPlayer;
|
import com.velocitypowered.proxy.connection.client.ConnectedPlayer;
|
||||||
import com.velocitypowered.proxy.connection.http.NettyHttpClient;
|
import com.velocitypowered.proxy.connection.http.NettyHttpClient;
|
||||||
import com.velocitypowered.proxy.command.VelocityCommandManager;
|
import com.velocitypowered.proxy.command.VelocityCommandManager;
|
||||||
import com.velocitypowered.proxy.config.AnnotationConfig;
|
import com.velocitypowered.proxy.config.AnnotatedConfig;
|
||||||
import com.velocitypowered.proxy.messages.VelocityChannelRegistrar;
|
import com.velocitypowered.proxy.messages.VelocityChannelRegistrar;
|
||||||
import com.velocitypowered.proxy.plugin.VelocityEventManager;
|
import com.velocitypowered.proxy.plugin.VelocityEventManager;
|
||||||
import com.velocitypowered.proxy.protocol.util.FaviconSerializer;
|
import com.velocitypowered.proxy.protocol.util.FaviconSerializer;
|
||||||
@ -115,9 +115,9 @@ public class VelocityServer implements ProxyServer {
|
|||||||
System.exit(1);
|
System.exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
AnnotationConfig.saveConfig(configuration.dumpConfig(), configPath); //Resave config to add new values
|
AnnotatedConfig.saveConfig(configuration.dumpConfig(), configPath); //Resave config to add new values
|
||||||
|
|
||||||
} catch (IOException | NullPointerException e) {
|
} catch (IOException | RuntimeException e) {
|
||||||
logger.error("Unable to load your velocity.toml. The server will shut down.", e);
|
logger.error("Unable to load your velocity.toml. The server will shut down.", e);
|
||||||
System.exit(1);
|
System.exit(1);
|
||||||
}
|
}
|
||||||
|
@ -25,9 +25,9 @@ import org.apache.logging.log4j.Logger;
|
|||||||
/**
|
/**
|
||||||
* Only for simple configs
|
* Only for simple configs
|
||||||
*/
|
*/
|
||||||
public class AnnotationConfig {
|
public class AnnotatedConfig {
|
||||||
|
|
||||||
private static final Logger logger = LogManager.getLogger(AnnotationConfig.class);
|
private static final Logger logger = LogManager.getLogger(AnnotatedConfig.class);
|
||||||
|
|
||||||
public static Logger getLogger() {
|
public static Logger getLogger() {
|
||||||
return logger;
|
return logger;
|
||||||
@ -37,7 +37,7 @@ public class AnnotationConfig {
|
|||||||
* Indicates that a field is a table
|
* Indicates that a field is a table
|
||||||
*/
|
*/
|
||||||
@Retention(RetentionPolicy.RUNTIME)
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
@Target({ElementType.FIELD})
|
@Target({ElementType.FIELD, ElementType.TYPE})
|
||||||
public @interface Table {
|
public @interface Table {
|
||||||
|
|
||||||
String value();
|
String value();
|
||||||
@ -64,11 +64,12 @@ public class AnnotationConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Indicates that a field is map and we need to save all data to config
|
* Indicates that a field is a map and we need to save all map data to
|
||||||
|
* config
|
||||||
*/
|
*/
|
||||||
@Retention(RetentionPolicy.RUNTIME)
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
@Target({ElementType.FIELD, ElementType.TYPE})
|
@Target({ElementType.FIELD, ElementType.TYPE})
|
||||||
public @interface AsMap {
|
public @interface IsMap {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -76,28 +77,36 @@ public class AnnotationConfig {
|
|||||||
*/
|
*/
|
||||||
@Retention(RetentionPolicy.RUNTIME)
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
@Target({ElementType.FIELD, ElementType.TYPE})
|
@Target({ElementType.FIELD, ElementType.TYPE})
|
||||||
public @interface AsBytes {
|
public @interface StringAsBytes {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Indicates that a field is a string converted to byte[]
|
* Indicates that a filed should be skiped
|
||||||
*/
|
*/
|
||||||
@Retention(RetentionPolicy.RUNTIME)
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
@Target({ElementType.FIELD, ElementType.TYPE})
|
@Target({ElementType.FIELD})
|
||||||
public @interface Ignore {
|
public @interface Ignore {
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<String> dumpConfig() {
|
public List<String> dumpConfig() {
|
||||||
List<String> lines = new ArrayList<>();
|
List<String> lines = new ArrayList<>();
|
||||||
dumpFields(getClass(), this, lines);
|
if (!dumpFields(this, lines)) {
|
||||||
|
throw new RuntimeException("can not dump config");
|
||||||
|
}
|
||||||
return lines;
|
return lines;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void dumpFields(Class root, Object caller, List<String> lines) {
|
/**
|
||||||
|
* Dump all field and they annotations to List
|
||||||
|
*
|
||||||
|
* @param toSave object those we need to dump
|
||||||
|
* @param lines a list where store dumped lines
|
||||||
|
*/
|
||||||
|
private boolean dumpFields(Object toSave, List<String> lines) {
|
||||||
|
|
||||||
try {
|
try {
|
||||||
for (Field field : root.getDeclaredFields()) {
|
for (Field field : toSave.getClass().getDeclaredFields()) {
|
||||||
if (field.getAnnotation(Ignore.class) != null) {
|
if (field.getAnnotation(Ignore.class) != null) { //Skip this field
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
Comment comment = field.getAnnotation(Comment.class);
|
Comment comment = field.getAnnotation(Comment.class);
|
||||||
@ -106,34 +115,36 @@ public class AnnotationConfig {
|
|||||||
lines.add("# " + line);
|
lines.add("# " + line);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
CfgKey key = field.getAnnotation(CfgKey.class);
|
CfgKey key = field.getAnnotation(CfgKey.class); //Get a key name for config
|
||||||
String name = key == null ? field.getName() : key.value();
|
String name = key == null ? field.getName() : key.value(); // Use a field name if name in annotation is not present
|
||||||
field.setAccessible(true);
|
field.setAccessible(true); // Make field accessible
|
||||||
Table table = field.getAnnotation(Table.class);
|
Table table = field.getAnnotation(Table.class);
|
||||||
if (table != null) {
|
if (table != null) { // Check if field is table.
|
||||||
lines.add(table.value()); // Write [name]
|
lines.add(table.value()); // Write [name]
|
||||||
dumpFields(field.getType(), field.get(caller), lines); // dump a table class
|
dumpFields(field.get(toSave), lines); // dump fields of table class
|
||||||
} else {
|
} else {
|
||||||
if (field.getAnnotation(AsMap.class) != null) {
|
if (field.getAnnotation(IsMap.class) != null) { // check if field is map
|
||||||
Map<String, ?> map = (Map<String, ?>) field.get(caller);
|
Map<String, ?> map = (Map<String, ?>) field.get(toSave);
|
||||||
for (Entry<String, ?> entry : map.entrySet()) {
|
for (Entry<String, ?> entry : map.entrySet()) {
|
||||||
lines.add(entry.getKey() + " = " + toString(entry.getValue()));
|
lines.add(entry.getKey() + " = " + toString(entry.getValue())); // Save a map data
|
||||||
}
|
}
|
||||||
lines.add("");
|
lines.add(""); //Add empty line
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
Object value = field.get(caller);
|
Object value = field.get(toSave);
|
||||||
if (field.getAnnotation(AsBytes.class) != null) {
|
if (field.getAnnotation(StringAsBytes.class) != null) { // Check if field is a byte[] representation of a string
|
||||||
value = new String((byte[]) value, StandardCharsets.UTF_8);
|
value = new String((byte[]) value, StandardCharsets.UTF_8);
|
||||||
}
|
}
|
||||||
lines.add(name + " = " + toString(value));
|
lines.add(name + " = " + toString(value)); // save field to config
|
||||||
lines.add("");
|
lines.add(""); // add empty line
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (IllegalAccessException | IllegalArgumentException | SecurityException e) {
|
} catch (IllegalAccessException | IllegalArgumentException | SecurityException e) {
|
||||||
logger.log(Level.ERROR, "Unexpected error while dumping fields", e);
|
logger.log(Level.ERROR, "Unexpected error while dumping fields", e);
|
||||||
lines.clear();
|
lines.clear();
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private String toString(Object value) {
|
private String toString(Object value) {
|
||||||
@ -163,6 +174,13 @@ public class AnnotationConfig {
|
|||||||
return value != null ? value.toString() : "null";
|
return value != null ? value.toString() : "null";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves lines to file
|
||||||
|
*
|
||||||
|
* @param lines Lines to save
|
||||||
|
* @param to A path of file where to save lines
|
||||||
|
* @throws IOException if lines is empty or was error during saving
|
||||||
|
*/
|
||||||
public static void saveConfig(List<String> lines, Path to) throws IOException {
|
public static void saveConfig(List<String> lines, Path to) throws IOException {
|
||||||
if (lines.isEmpty()) {
|
if (lines.isEmpty()) {
|
||||||
throw new IOException("Can not save config because list is empty");
|
throw new IOException("Can not save config because list is empty");
|
@ -22,19 +22,23 @@ import java.util.List;
|
|||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import org.apache.logging.log4j.Logger;
|
import org.apache.logging.log4j.Logger;
|
||||||
|
|
||||||
public class VelocityConfiguration extends AnnotationConfig {
|
public class VelocityConfiguration extends AnnotatedConfig {
|
||||||
|
|
||||||
|
@Comment("Config version. Do not change this")
|
||||||
|
@CfgKey("config-version")
|
||||||
|
private final String configVersion = "1.0";
|
||||||
|
|
||||||
@Comment("What port should the proxy be bound to? By default, we'll bind to all addresses on port 25577.")
|
@Comment("What port should the proxy be bound to? By default, we'll bind to all addresses on port 25577.")
|
||||||
private final String bind;
|
private String bind;
|
||||||
@Comment("What should be the MOTD? Legacy color codes and JSON are accepted.")
|
@Comment("What should be the MOTD? Legacy color codes and JSON are accepted.")
|
||||||
private final String motd;
|
private String motd;
|
||||||
@Comment({"What should we display for the maximum number of players? (Velocity does not support a cap",
|
@Comment({"What should we display for the maximum number of players? (Velocity does not support a cap",
|
||||||
"on the number of players online.)"})
|
"on the number of players online.)"})
|
||||||
@CfgKey("show-max-players")
|
@CfgKey("show-max-players")
|
||||||
private final int showMaxPlayers;
|
private int showMaxPlayers;
|
||||||
@Comment("Should we authenticate players with Mojang? By default, this is on.")
|
@Comment("Should we authenticate players with Mojang? By default, this is on.")
|
||||||
@CfgKey("online-mode")
|
@CfgKey("online-mode")
|
||||||
private final boolean onlineMode;
|
private boolean onlineMode;
|
||||||
@Comment({"Should we forward IP addresses and other data to backend servers?",
|
@Comment({"Should we forward IP addresses and other data to backend servers?",
|
||||||
"Available options:",
|
"Available options:",
|
||||||
"- \"none\": No forwarding will be done. All players will appear to be Should we forward IP addresses and other data to backend servers?connecting from the proxy",
|
"- \"none\": No forwarding will be done. All players will appear to be Should we forward IP addresses and other data to backend servers?connecting from the proxy",
|
||||||
@ -44,94 +48,21 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
"- \"modern\": Forward player IPs and UUIDs as part of the login process using Velocity's native",
|
"- \"modern\": Forward player IPs and UUIDs as part of the login process using Velocity's native",
|
||||||
" forwarding. Only applicable for Minecraft 1.13 or higher."})
|
" forwarding. Only applicable for Minecraft 1.13 or higher."})
|
||||||
@CfgKey("player-info-forwarding-mode")
|
@CfgKey("player-info-forwarding-mode")
|
||||||
private final PlayerInfoForwarding playerInfoForwardingMode;
|
private PlayerInfoForwarding playerInfoForwardingMode;
|
||||||
|
|
||||||
@AsBytes
|
@StringAsBytes
|
||||||
@Comment("If you are using modern IP forwarding, configure an unique secret here.")
|
@Comment("If you are using modern IP forwarding, configure an unique secret here.")
|
||||||
@CfgKey("forwarding-secret")
|
@CfgKey("forwarding-secret")
|
||||||
private final byte[] forwardingSecret;
|
private byte[] forwardingSecret;
|
||||||
|
|
||||||
@Table("[servers]")
|
@Table("[servers]")
|
||||||
private final Servers servers;
|
private final Servers servers;
|
||||||
|
|
||||||
private static class Servers {
|
|
||||||
|
|
||||||
@AsMap
|
|
||||||
@Comment("Configure your servers here.")
|
|
||||||
public final Map<String, String> servers;
|
|
||||||
|
|
||||||
@Comment("In what order we should try servers when a player logs in or is kicked from a server.")
|
|
||||||
@CfgKey("try")
|
|
||||||
public final List<String> attemptConnectionOrder;
|
|
||||||
|
|
||||||
public Servers(Map<String, String> servers, List<String> attemptConnectionOrder) {
|
|
||||||
this.servers = servers;
|
|
||||||
this.attemptConnectionOrder = attemptConnectionOrder;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return "Servers{" + "servers=" + servers + ", attemptConnectionOrder=" + attemptConnectionOrder + '}';
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Table("[advanced]")
|
@Table("[advanced]")
|
||||||
private final Advanced advanced;
|
private final Advanced advanced;
|
||||||
|
|
||||||
private static class Advanced {
|
|
||||||
|
|
||||||
@Comment({"How large a Minecraft packet has to be before we compress it. Setting this to zero will compress all packets, and",
|
|
||||||
"setting it to -1 will disable compression entirely."})
|
|
||||||
@CfgKey("compression-threshold")
|
|
||||||
public final int compressionThreshold;
|
|
||||||
@Comment("How much compression should be done (from 0-9). The default is -1, which uses zlib's default level of 6.")
|
|
||||||
@CfgKey("compression-level")
|
|
||||||
public final int compressionLevel;
|
|
||||||
@Comment({"How fast (in miliseconds) are clients allowed to connect after the last connection? Default: 3000",
|
|
||||||
"Disable by setting to 0"})
|
|
||||||
@CfgKey("login-ratelimit")
|
|
||||||
public final int loginRatelimit;
|
|
||||||
|
|
||||||
public Advanced(Toml toml) {
|
|
||||||
this.compressionThreshold = toml.getLong("compression-threshold", 1024L).intValue();
|
|
||||||
this.compressionLevel = toml.getLong("compression-level", -1L).intValue();
|
|
||||||
this.loginRatelimit = toml.getLong("login-ratelimit", 3000L).intValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return "Advanced{" + "compressionThreshold=" + compressionThreshold + ", compressionLevel=" + compressionLevel + ", loginRatelimit=" + loginRatelimit + '}';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Table("[query]")
|
@Table("[query]")
|
||||||
private final Query query;
|
private final Query query;
|
||||||
|
|
||||||
private static class Query {
|
|
||||||
|
|
||||||
@Comment("Whether to enable responding to GameSpy 4 query responses or not")
|
|
||||||
@CfgKey("enabled")
|
|
||||||
public final boolean queryEnabled;
|
|
||||||
@Comment("If query responding is enabled, on what port should query response listener listen on?")
|
|
||||||
@CfgKey("port")
|
|
||||||
public final int queryPort;
|
|
||||||
|
|
||||||
public Query(boolean queryEnabled, int queryPort) {
|
|
||||||
this.queryEnabled = queryEnabled;
|
|
||||||
this.queryPort = queryPort;
|
|
||||||
}
|
|
||||||
|
|
||||||
private Query(Toml toml) {
|
|
||||||
this.queryEnabled = toml.getBoolean("enabled", false);
|
|
||||||
this.queryPort = toml.getLong("port", 25577L).intValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return "Query{" + "queryEnabled=" + queryEnabled + ", queryPort=" + queryPort + '}';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@Ignore
|
@Ignore
|
||||||
private Component motdAsComponent;
|
private Component motdAsComponent;
|
||||||
@Ignore
|
@Ignore
|
||||||
@ -153,7 +84,7 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
|
|
||||||
public boolean validate() {
|
public boolean validate() {
|
||||||
boolean valid = true;
|
boolean valid = true;
|
||||||
Logger logger = AnnotationConfig.getLogger();
|
Logger logger = AnnotatedConfig.getLogger();
|
||||||
|
|
||||||
if (bind.isEmpty()) {
|
if (bind.isEmpty()) {
|
||||||
logger.error("'bind' option is empty.");
|
logger.error("'bind' option is empty.");
|
||||||
@ -183,16 +114,16 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (servers.servers.isEmpty()) {
|
if (servers.getServers().isEmpty()) {
|
||||||
logger.error("You have no servers configured. :(");
|
logger.error("You have no servers configured. :(");
|
||||||
valid = false;
|
valid = false;
|
||||||
} else {
|
} else {
|
||||||
if (servers.attemptConnectionOrder.isEmpty()) {
|
if (servers.getAttemptConnectionOrder().isEmpty()) {
|
||||||
logger.error("No fallback servers are configured!");
|
logger.error("No fallback servers are configured!");
|
||||||
valid = false;
|
valid = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (Map.Entry<String, String> entry : servers.servers.entrySet()) {
|
for (Map.Entry<String, String> entry : servers.getServers().entrySet()) {
|
||||||
try {
|
try {
|
||||||
AddressUtil.parseAddress(entry.getValue());
|
AddressUtil.parseAddress(entry.getValue());
|
||||||
} catch (IllegalArgumentException e) {
|
} catch (IllegalArgumentException e) {
|
||||||
@ -201,8 +132,8 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (String s : servers.attemptConnectionOrder) {
|
for (String s : servers.getAttemptConnectionOrder()) {
|
||||||
if (!servers.servers.containsKey(s)) {
|
if (!servers.getServers().containsKey(s)) {
|
||||||
logger.error("Fallback server " + s + " doesn't exist!");
|
logger.error("Fallback server " + s + " doesn't exist!");
|
||||||
valid = false;
|
valid = false;
|
||||||
}
|
}
|
||||||
@ -256,11 +187,11 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public boolean isQueryEnabled() {
|
public boolean isQueryEnabled() {
|
||||||
return query.queryEnabled;
|
return query.isQueryEnabled();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getQueryPort() {
|
public int getQueryPort() {
|
||||||
return query.queryPort;
|
return query.getQueryPort();
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getMotd() {
|
public String getMotd() {
|
||||||
@ -290,32 +221,64 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
return playerInfoForwardingMode;
|
return playerInfoForwardingMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public byte[] getForwardingSecret() {
|
||||||
|
return forwardingSecret;
|
||||||
|
}
|
||||||
|
|
||||||
public Map<String, String> getServers() {
|
public Map<String, String> getServers() {
|
||||||
return servers.servers;
|
return servers.getServers();
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<String> getAttemptConnectionOrder() {
|
public List<String> getAttemptConnectionOrder() {
|
||||||
return servers.attemptConnectionOrder;
|
return servers.getAttemptConnectionOrder();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getCompressionThreshold() {
|
public int getCompressionThreshold() {
|
||||||
return advanced.compressionThreshold;
|
return advanced.getCompressionThreshold();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getCompressionLevel() {
|
public int getCompressionLevel() {
|
||||||
return advanced.compressionLevel;
|
return advanced.getCompressionLevel();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getLoginRatelimit() {
|
public int getLoginRatelimit() {
|
||||||
return advanced.loginRatelimit;
|
return advanced.getLoginRatelimit();
|
||||||
}
|
}
|
||||||
|
|
||||||
public Favicon getFavicon() {
|
public Favicon getFavicon() {
|
||||||
return favicon;
|
return favicon;
|
||||||
}
|
}
|
||||||
|
|
||||||
public byte[] getForwardingSecret() {
|
private void setBind(String bind) {
|
||||||
return forwardingSecret;
|
this.bind = bind;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setMotd(String motd) {
|
||||||
|
this.motd = motd;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setShowMaxPlayers(int showMaxPlayers) {
|
||||||
|
this.showMaxPlayers = showMaxPlayers;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setOnlineMode(boolean onlineMode) {
|
||||||
|
this.onlineMode = onlineMode;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setPlayerInfoForwardingMode(PlayerInfoForwarding playerInfoForwardingMode) {
|
||||||
|
this.playerInfoForwardingMode = playerInfoForwardingMode;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setForwardingSecret(byte[] forwardingSecret) {
|
||||||
|
this.forwardingSecret = forwardingSecret;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setMotdAsComponent(Component motdAsComponent) {
|
||||||
|
this.motdAsComponent = motdAsComponent;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setFavicon(Favicon favicon) {
|
||||||
|
this.favicon = favicon;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -327,12 +290,12 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
+ ", showMaxPlayers=" + showMaxPlayers
|
+ ", showMaxPlayers=" + showMaxPlayers
|
||||||
+ ", onlineMode=" + onlineMode
|
+ ", onlineMode=" + onlineMode
|
||||||
+ ", playerInfoForwardingMode=" + playerInfoForwardingMode
|
+ ", playerInfoForwardingMode=" + playerInfoForwardingMode
|
||||||
|
+ ", forwardingSecret=" + ByteBufUtil.hexDump(forwardingSecret)
|
||||||
+ ", servers=" + servers
|
+ ", servers=" + servers
|
||||||
+ ", advanced=" + advanced
|
+ ", advanced=" + advanced
|
||||||
+ ", query=" + query
|
+ ", query=" + query
|
||||||
+ ", motdAsComponent=" + motdAsComponent
|
+ ", motdAsComponent=" + motdAsComponent
|
||||||
+ ", favicon=" + favicon
|
+ ", favicon=" + favicon
|
||||||
+ ", forwardingSecret=" + ByteBufUtil.hexDump(forwardingSecret)
|
|
||||||
+ '}';
|
+ '}';
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -347,7 +310,7 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Upgrdate old values to new, when config will be changed in future
|
// If config will be changed in future, do not forget to migrate old values if needed
|
||||||
Map<String, String> servers = new HashMap<>();
|
Map<String, String> servers = new HashMap<>();
|
||||||
for (Map.Entry<String, Object> entry : toml.getTable("servers").entrySet()) {
|
for (Map.Entry<String, Object> entry : toml.getTable("servers").entrySet()) {
|
||||||
if (entry.getValue() instanceof String) {
|
if (entry.getValue() instanceof String) {
|
||||||
@ -365,7 +328,7 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
byte[] forwardingSecret = toml.getString("player-info-forwarding-secret", "5up3r53cr3t")
|
byte[] forwardingSecret = toml.getString("player-info-forwarding-secret", "5up3r53cr3t")
|
||||||
.getBytes(StandardCharsets.UTF_8);
|
.getBytes(StandardCharsets.UTF_8);
|
||||||
|
|
||||||
return new VelocityConfiguration(
|
VelocityConfiguration configuration = new VelocityConfiguration(
|
||||||
toml.getString("bind", "0.0.0.0:25577"),
|
toml.getString("bind", "0.0.0.0:25577"),
|
||||||
toml.getString("motd", "&3A Velocity Server"),
|
toml.getString("motd", "&3A Velocity Server"),
|
||||||
toml.getLong("show-max-players", 500L).intValue(),
|
toml.getLong("show-max-players", 500L).intValue(),
|
||||||
@ -376,6 +339,152 @@ public class VelocityConfiguration extends AnnotationConfig {
|
|||||||
advanced,
|
advanced,
|
||||||
query
|
query
|
||||||
);
|
);
|
||||||
|
upgradeConfig(configuration, toml);
|
||||||
|
return configuration;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void upgradeConfig(VelocityConfiguration configuration, Toml toml) {
|
||||||
|
switch (toml.getString("config-version", configuration.configVersion)) {
|
||||||
|
case "1.0":
|
||||||
|
//TODO: Upgrade a 1.0 config to a new version. Maybe add a recursive support in future.
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class Servers {
|
||||||
|
|
||||||
|
@IsMap
|
||||||
|
@Comment("Configure your servers here.")
|
||||||
|
private Map<String, String> servers;
|
||||||
|
|
||||||
|
@Comment("In what order we should try servers when a player logs in or is kicked from a server.")
|
||||||
|
@CfgKey("try")
|
||||||
|
private List<String> attemptConnectionOrder;
|
||||||
|
|
||||||
|
private Servers(Map<String, String> servers, List<String> attemptConnectionOrder) {
|
||||||
|
this.servers = servers;
|
||||||
|
this.attemptConnectionOrder = attemptConnectionOrder;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Map<String, String> getServers() {
|
||||||
|
return servers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setServers(Map<String, String> servers) {
|
||||||
|
this.servers = servers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<String> getAttemptConnectionOrder() {
|
||||||
|
return attemptConnectionOrder;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAttemptConnectionOrder(List<String> attemptConnectionOrder) {
|
||||||
|
this.attemptConnectionOrder = attemptConnectionOrder;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "Servers{" + "servers=" + servers + ", attemptConnectionOrder=" + attemptConnectionOrder + '}';
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static class Advanced {
|
||||||
|
|
||||||
|
@Comment({"How large a Minecraft packet has to be before we compress it. Setting this to zero will compress all packets, and",
|
||||||
|
"setting it to -1 will disable compression entirely."})
|
||||||
|
@CfgKey("compression-threshold")
|
||||||
|
private int compressionThreshold;
|
||||||
|
@Comment("How much compression should be done (from 0-9). The default is -1, which uses zlib's default level of 6.")
|
||||||
|
@CfgKey("compression-level")
|
||||||
|
private int compressionLevel;
|
||||||
|
@Comment({"How fast (in miliseconds) are clients allowed to connect after the last connection? Default: 3000",
|
||||||
|
"Disable by setting to 0"})
|
||||||
|
@CfgKey("login-ratelimit")
|
||||||
|
private int loginRatelimit;
|
||||||
|
|
||||||
|
private Advanced(int compressionThreshold, int compressionLevel, int loginRatelimit) {
|
||||||
|
this.compressionThreshold = compressionThreshold;
|
||||||
|
this.compressionLevel = compressionLevel;
|
||||||
|
this.loginRatelimit = loginRatelimit;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Advanced(Toml toml) {
|
||||||
|
this.compressionThreshold = toml.getLong("compression-threshold", 1024L).intValue();
|
||||||
|
this.compressionLevel = toml.getLong("compression-level", -1L).intValue();
|
||||||
|
this.loginRatelimit = toml.getLong("login-ratelimit", 3000L).intValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getCompressionThreshold() {
|
||||||
|
return compressionThreshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCompressionThreshold(int compressionThreshold) {
|
||||||
|
this.compressionThreshold = compressionThreshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getCompressionLevel() {
|
||||||
|
return compressionLevel;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCompressionLevel(int compressionLevel) {
|
||||||
|
this.compressionLevel = compressionLevel;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getLoginRatelimit() {
|
||||||
|
return loginRatelimit;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setLoginRatelimit(int loginRatelimit) {
|
||||||
|
this.loginRatelimit = loginRatelimit;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "Advanced{" + "compressionThreshold=" + compressionThreshold + ", compressionLevel=" + compressionLevel + ", loginRatelimit=" + loginRatelimit + '}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class Query {
|
||||||
|
|
||||||
|
@Comment("Whether to enable responding to GameSpy 4 query responses or not")
|
||||||
|
@CfgKey("enabled")
|
||||||
|
private boolean queryEnabled;
|
||||||
|
@Comment("If query responding is enabled, on what port should query response listener listen on?")
|
||||||
|
@CfgKey("port")
|
||||||
|
private int queryPort;
|
||||||
|
|
||||||
|
private Query(boolean queryEnabled, int queryPort) {
|
||||||
|
this.queryEnabled = queryEnabled;
|
||||||
|
this.queryPort = queryPort;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Query(Toml toml) {
|
||||||
|
this.queryEnabled = toml.getBoolean("enabled", false);
|
||||||
|
this.queryPort = toml.getLong("port", 25577L).intValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isQueryEnabled() {
|
||||||
|
return queryEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setQueryEnabled(boolean queryEnabled) {
|
||||||
|
this.queryEnabled = queryEnabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getQueryPort() {
|
||||||
|
return queryPort;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setQueryPort(int queryPort) {
|
||||||
|
this.queryPort = queryPort;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "Query{" + "queryEnabled=" + queryEnabled + ", queryPort=" + queryPort + '}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Laden…
In neuem Issue referenzieren
Einen Benutzer sperren