mirror of
https://github.com/GravitLauncher/Launcher
synced 2025-01-09 09:09:46 +03:00
Обновление ServerWrapper
Фитча добавления classpath Фитча автозагрузки библиотек(libraries) Проксирование агента
This commit is contained in:
parent
d450a17d58
commit
01b37db0d2
3 changed files with 86 additions and 21 deletions
|
@ -1,9 +1,13 @@
|
||||||
package ru.gravit.launcher.server;
|
package ru.gravit.launcher.server;
|
||||||
|
|
||||||
|
import ru.gravit.utils.helper.IOHelper;
|
||||||
import ru.gravit.utils.helper.LogHelper;
|
import ru.gravit.utils.helper.LogHelper;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.lang.instrument.Instrumentation;
|
import java.lang.instrument.Instrumentation;
|
||||||
|
import java.lang.invoke.MethodHandle;
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.lang.invoke.MethodType;
|
||||||
import java.nio.file.*;
|
import java.nio.file.*;
|
||||||
import java.nio.file.attribute.BasicFileAttributes;
|
import java.nio.file.attribute.BasicFileAttributes;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
@ -38,14 +42,35 @@ public boolean isAgentStarted() {
|
||||||
public static long getObjSize(Object obj) {
|
public static long getObjSize(Object obj) {
|
||||||
return inst.getObjectSize(obj);
|
return inst.getObjectSize(obj);
|
||||||
}
|
}
|
||||||
|
public static Boolean isAutoloadLibraries = Boolean.getBoolean(System.getProperty("serverwrapper,agentlibrariesload","false"));
|
||||||
|
public static Boolean isAgentProxy = Boolean.getBoolean(System.getProperty("serverwrapper,agentproxy","false"));
|
||||||
public static void premain(String agentArgument, Instrumentation instrumentation) {
|
public static void premain(String agentArgument, Instrumentation instrumentation) {
|
||||||
LogHelper.debug("Server Agent");
|
LogHelper.debug("Server Agent");
|
||||||
inst = instrumentation;
|
inst = instrumentation;
|
||||||
isAgentStarted = true;
|
isAgentStarted = true;
|
||||||
|
if(isAutoloadLibraries)
|
||||||
|
{
|
||||||
|
Path libraries = Paths.get("libraries");
|
||||||
|
if(IOHelper.exists(libraries)) loadLibraries(libraries);
|
||||||
|
}
|
||||||
|
if(isAgentProxy)
|
||||||
|
{
|
||||||
|
String proxyClassName = System.getProperty("serverwrapper,agentproxyclass");
|
||||||
|
Class proxyClass;
|
||||||
|
try {
|
||||||
|
proxyClass = Class.forName(proxyClassName);
|
||||||
|
MethodHandle mainMethod = MethodHandles.publicLookup().findStatic(proxyClass, "premain", MethodType.methodType(void.class, String.class, Instrumentation.class));
|
||||||
|
Object[] args = {agentArgument,instrumentation};
|
||||||
|
mainMethod.invoke(args);
|
||||||
|
} catch (Throwable e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public static void loadLibraries(Path dir)
|
||||||
|
{
|
||||||
try {
|
try {
|
||||||
Files.walkFileTree(Paths.get("libraries"), Collections.singleton(FileVisitOption.FOLLOW_LINKS), Integer.MAX_VALUE, new StarterVisitor());
|
Files.walkFileTree(dir, Collections.singleton(FileVisitOption.FOLLOW_LINKS), Integer.MAX_VALUE, new StarterVisitor());
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
e.printStackTrace(System.err);
|
e.printStackTrace(System.err);
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,8 +7,10 @@
|
||||||
import java.lang.invoke.MethodHandle;
|
import java.lang.invoke.MethodHandle;
|
||||||
import java.lang.invoke.MethodHandles;
|
import java.lang.invoke.MethodHandles;
|
||||||
import java.lang.invoke.MethodType;
|
import java.lang.invoke.MethodType;
|
||||||
|
import java.net.URL;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.nio.file.Paths;
|
import java.nio.file.Paths;
|
||||||
|
import java.util.Arrays;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
|
||||||
import ru.gravit.launcher.Launcher;
|
import ru.gravit.launcher.Launcher;
|
||||||
|
@ -21,6 +23,7 @@
|
||||||
import ru.gravit.launcher.serialize.config.entry.BooleanConfigEntry;
|
import ru.gravit.launcher.serialize.config.entry.BooleanConfigEntry;
|
||||||
import ru.gravit.launcher.serialize.config.entry.IntegerConfigEntry;
|
import ru.gravit.launcher.serialize.config.entry.IntegerConfigEntry;
|
||||||
import ru.gravit.launcher.serialize.config.entry.StringConfigEntry;
|
import ru.gravit.launcher.serialize.config.entry.StringConfigEntry;
|
||||||
|
import ru.gravit.utils.PublicURLClassLoader;
|
||||||
import ru.gravit.utils.helper.CommonHelper;
|
import ru.gravit.utils.helper.CommonHelper;
|
||||||
import ru.gravit.utils.helper.IOHelper;
|
import ru.gravit.utils.helper.IOHelper;
|
||||||
import ru.gravit.utils.helper.LogHelper;
|
import ru.gravit.utils.helper.LogHelper;
|
||||||
|
@ -31,8 +34,13 @@
|
||||||
|
|
||||||
public class ServerWrapper {
|
public class ServerWrapper {
|
||||||
public static ModulesManager modulesManager;
|
public static ModulesManager modulesManager;
|
||||||
public static Path configFile;
|
|
||||||
public static Config config;
|
public static Config config;
|
||||||
|
public static PublicURLClassLoader ucp;
|
||||||
|
public static ClassLoader loader;
|
||||||
|
|
||||||
|
public static Path modulesDir = Paths.get(System.getProperty("serverwrapper.modulesDir","modules"));
|
||||||
|
public static Path configFile = Paths.get(System.getProperty("serverwrapper.configFile","ServerWrapper.cfg"));
|
||||||
|
public static Path publicKeyFile = Paths.get(System.getProperty("serverwrapper.publicKeyFile","public.key"));
|
||||||
|
|
||||||
public static boolean auth(ServerWrapper wrapper) {
|
public static boolean auth(ServerWrapper wrapper) {
|
||||||
try {
|
try {
|
||||||
|
@ -79,15 +87,14 @@ public static void main(String[] args) throws Throwable {
|
||||||
LogHelper.printVersion("ServerWrapper");
|
LogHelper.printVersion("ServerWrapper");
|
||||||
LogHelper.printLicense("ServerWrapper");
|
LogHelper.printLicense("ServerWrapper");
|
||||||
modulesManager = new ModulesManager(wrapper);
|
modulesManager = new ModulesManager(wrapper);
|
||||||
modulesManager.autoload(Paths.get("srv_modules")); //BungeeCord using modules dir
|
modulesManager.autoload(modulesDir);
|
||||||
Launcher.modulesManager = modulesManager;
|
Launcher.modulesManager = modulesManager;
|
||||||
configFile = Paths.get("ServerWrapper.cfg");
|
|
||||||
modulesManager.preInitModules();
|
modulesManager.preInitModules();
|
||||||
generateConfigIfNotExists();
|
generateConfigIfNotExists();
|
||||||
try (BufferedReader reader = IOHelper.newReader(configFile)) {
|
try (BufferedReader reader = IOHelper.newReader(configFile)) {
|
||||||
config = new Config(TextConfigReader.read(reader, true));
|
config = new Config(TextConfigReader.read(reader, true));
|
||||||
}
|
}
|
||||||
LauncherConfig cfg = new LauncherConfig(config.address, config.port, SecurityHelper.toPublicRSAKey(IOHelper.read(Paths.get("public.key"))), new HashMap<>(), config.projectname);
|
LauncherConfig cfg = new LauncherConfig(config.address, config.port, SecurityHelper.toPublicRSAKey(IOHelper.read(publicKeyFile)), new HashMap<>(), config.projectname);
|
||||||
Launcher.setConfig(cfg);
|
Launcher.setConfig(cfg);
|
||||||
if (config.syncAuth) auth(wrapper);
|
if (config.syncAuth) auth(wrapper);
|
||||||
else
|
else
|
||||||
|
@ -99,13 +106,36 @@ public static void main(String[] args) throws Throwable {
|
||||||
LogHelper.error("MainClass not found. Please set MainClass for ServerWrapper.cfg or first commandline argument");
|
LogHelper.error("MainClass not found. Please set MainClass for ServerWrapper.cfg or first commandline argument");
|
||||||
}
|
}
|
||||||
Class<?> mainClass;
|
Class<?> mainClass;
|
||||||
if (config.customClassLoader) {
|
if(config.customClassPath)
|
||||||
@SuppressWarnings("unchecked")
|
{
|
||||||
Class<ClassLoader> classloader_class = (Class<ClassLoader>) Class.forName(config.classloader);
|
String[] cp = config.classpath.split(":");
|
||||||
ClassLoader loader = classloader_class.getConstructor(ClassLoader.class).newInstance(ClassLoader.getSystemClassLoader());
|
if(ServerAgent.inst == null)
|
||||||
Thread.currentThread().setContextClassLoader(loader);
|
{
|
||||||
mainClass = Class.forName(classname, false, loader);
|
LogHelper.warning("JavaAgent not found. Using URLClassLoader");
|
||||||
} else mainClass = Class.forName(classname);
|
URL[] urls = Arrays.stream(cp).map(Paths::get).map(IOHelper::toURL).toArray(URL[]::new);
|
||||||
|
ucp = new PublicURLClassLoader(urls);
|
||||||
|
Thread.currentThread().setContextClassLoader(ucp);
|
||||||
|
loader = ucp;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
LogHelper.info("Found %d custom classpath elements");
|
||||||
|
for(String c : cp)
|
||||||
|
ServerAgent.addJVMClassPath(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(config.autoloadLibraries)
|
||||||
|
{
|
||||||
|
if(ServerAgent.inst == null)
|
||||||
|
{
|
||||||
|
throw new UnsupportedOperationException("JavaAgent not found, autoloadLibraries not available");
|
||||||
|
}
|
||||||
|
Path librariesDir = Paths.get(config.librariesDir);
|
||||||
|
LogHelper.info("Load libraries");
|
||||||
|
ServerAgent.loadLibraries(librariesDir);
|
||||||
|
}
|
||||||
|
if(loader != null) mainClass = Class.forName(classname,true, loader);
|
||||||
|
else mainClass = Class.forName(classname);
|
||||||
MethodHandle mainMethod = MethodHandles.publicLookup().findStatic(mainClass, "main", MethodType.methodType(void.class, String[].class));
|
MethodHandle mainMethod = MethodHandles.publicLookup().findStatic(mainClass, "main", MethodType.methodType(void.class, String[].class));
|
||||||
String[] real_args = new String[args.length - 1];
|
String[] real_args = new String[args.length - 1];
|
||||||
System.arraycopy(args, 1, real_args, 0, args.length - 1);
|
System.arraycopy(args, 1, real_args, 0, args.length - 1);
|
||||||
|
@ -143,9 +173,11 @@ public static final class Config extends ConfigObject {
|
||||||
public int port;
|
public int port;
|
||||||
public int reconnectCount;
|
public int reconnectCount;
|
||||||
public int reconnectSleep;
|
public int reconnectSleep;
|
||||||
public boolean customClassLoader;
|
public boolean customClassPath;
|
||||||
|
public boolean autoloadLibraries;
|
||||||
public boolean syncAuth;
|
public boolean syncAuth;
|
||||||
public String classloader;
|
public String classpath;
|
||||||
|
public String librariesDir;
|
||||||
public String mainclass;
|
public String mainclass;
|
||||||
public String login;
|
public String login;
|
||||||
public String password;
|
public String password;
|
||||||
|
@ -158,9 +190,12 @@ protected Config(BlockConfigEntry block) {
|
||||||
login = block.getEntryValue("login", StringConfigEntry.class);
|
login = block.getEntryValue("login", StringConfigEntry.class);
|
||||||
password = block.getEntryValue("password", StringConfigEntry.class);
|
password = block.getEntryValue("password", StringConfigEntry.class);
|
||||||
port = block.getEntryValue("port", IntegerConfigEntry.class);
|
port = block.getEntryValue("port", IntegerConfigEntry.class);
|
||||||
customClassLoader = block.getEntryValue("customClassLoader", BooleanConfigEntry.class);
|
customClassPath = block.getEntryValue("customClassPath", BooleanConfigEntry.class);
|
||||||
if (customClassLoader)
|
autoloadLibraries = block.getEntryValue("autoloadLibraries", BooleanConfigEntry.class);
|
||||||
classloader = block.getEntryValue("classloader", StringConfigEntry.class);
|
if (customClassPath)
|
||||||
|
classpath = block.getEntryValue("classpath", StringConfigEntry.class);
|
||||||
|
if (autoloadLibraries)
|
||||||
|
librariesDir = block.getEntryValue("librariesDir", StringConfigEntry.class);
|
||||||
mainclass = block.getEntryValue("MainClass", StringConfigEntry.class);
|
mainclass = block.getEntryValue("MainClass", StringConfigEntry.class);
|
||||||
reconnectCount = block.hasEntry("reconnectCount") ? block.getEntryValue("reconnectCount", IntegerConfigEntry.class) : 1;
|
reconnectCount = block.hasEntry("reconnectCount") ? block.getEntryValue("reconnectCount", IntegerConfigEntry.class) : 1;
|
||||||
reconnectSleep = block.hasEntry("reconnectSleep") ? block.getEntryValue("reconnectSleep", IntegerConfigEntry.class) : 30000;
|
reconnectSleep = block.hasEntry("reconnectSleep") ? block.getEntryValue("reconnectSleep", IntegerConfigEntry.class) : 30000;
|
||||||
|
|
|
@ -1,6 +1,11 @@
|
||||||
title: "xxxx";
|
title: "xxxx";
|
||||||
customClassLoader: false;
|
|
||||||
classloader: "ru.gravit.utils.PublicURLClassLoader";
|
autoloadLibraries: false;
|
||||||
|
librariesDir: "libraries";
|
||||||
|
|
||||||
|
customClassPath: false;
|
||||||
|
classpath: "server.jar";
|
||||||
|
|
||||||
MainClass: "";
|
MainClass: "";
|
||||||
address: "localhost";
|
address: "localhost";
|
||||||
port: 7240;
|
port: 7240;
|
||||||
|
|
Loading…
Reference in a new issue