mirror of
https://github.com/GravitLauncher/Launcher
synced 2024-12-31 20:59:54 +03:00
Введение MirrorManager, теперь можно создавать свои "зеркала", откуда можно скачивать клиенты и ассеты. Список зеркал указывается в LaunchServer.cfg
Модули могут добавлять свои зеркала
This commit is contained in:
parent
81a867e631
commit
615ed25e00
7 changed files with 105 additions and 12 deletions
|
@ -4,6 +4,7 @@
|
||||||
import java.io.BufferedWriter;
|
import java.io.BufferedWriter;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
import java.nio.file.DirectoryStream;
|
import java.nio.file.DirectoryStream;
|
||||||
import java.nio.file.FileVisitResult;
|
import java.nio.file.FileVisitResult;
|
||||||
|
@ -32,6 +33,8 @@
|
||||||
import ru.gravit.launcher.Launcher;
|
import ru.gravit.launcher.Launcher;
|
||||||
import ru.gravit.launcher.LauncherAPI;
|
import ru.gravit.launcher.LauncherAPI;
|
||||||
import ru.gravit.launcher.hasher.HashedDir;
|
import ru.gravit.launcher.hasher.HashedDir;
|
||||||
|
import ru.gravit.launcher.serialize.config.entry.*;
|
||||||
|
import ru.gravit.launchserver.manangers.MirrorManager;
|
||||||
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.JVMHelper;
|
import ru.gravit.utils.helper.JVMHelper;
|
||||||
|
@ -43,10 +46,6 @@
|
||||||
import ru.gravit.launcher.serialize.config.ConfigObject;
|
import ru.gravit.launcher.serialize.config.ConfigObject;
|
||||||
import ru.gravit.launcher.serialize.config.TextConfigReader;
|
import ru.gravit.launcher.serialize.config.TextConfigReader;
|
||||||
import ru.gravit.launcher.serialize.config.TextConfigWriter;
|
import ru.gravit.launcher.serialize.config.TextConfigWriter;
|
||||||
import ru.gravit.launcher.serialize.config.entry.BlockConfigEntry;
|
|
||||||
import ru.gravit.launcher.serialize.config.entry.BooleanConfigEntry;
|
|
||||||
import ru.gravit.launcher.serialize.config.entry.IntegerConfigEntry;
|
|
||||||
import ru.gravit.launcher.serialize.config.entry.StringConfigEntry;
|
|
||||||
import ru.gravit.launcher.serialize.signed.SignedObjectHolder;
|
import ru.gravit.launcher.serialize.signed.SignedObjectHolder;
|
||||||
import ru.gravit.launchserver.auth.AuthLimiter;
|
import ru.gravit.launchserver.auth.AuthLimiter;
|
||||||
import ru.gravit.launchserver.auth.handler.AuthHandler;
|
import ru.gravit.launchserver.auth.handler.AuthHandler;
|
||||||
|
@ -101,6 +100,7 @@ public static final class Config extends ConfigObject {
|
||||||
|
|
||||||
public final boolean genMappings;
|
public final boolean genMappings;
|
||||||
|
|
||||||
|
public ListConfigEntry mirrors;
|
||||||
public final String binaryName;
|
public final String binaryName;
|
||||||
private final StringConfigEntry address;
|
private final StringConfigEntry address;
|
||||||
private final String bindAddress;
|
private final String bindAddress;
|
||||||
|
@ -136,6 +136,7 @@ private Config(BlockConfigEntry block, Path coredir,LaunchServer server) {
|
||||||
|
|
||||||
// Set misc config
|
// Set misc config
|
||||||
genMappings = block.getEntryValue("proguardPrintMappings", BooleanConfigEntry.class);
|
genMappings = block.getEntryValue("proguardPrintMappings", BooleanConfigEntry.class);
|
||||||
|
mirrors = block.getEntry("mirrors",ListConfigEntry.class);
|
||||||
launch4j = new ExeConf(block.getEntry("launch4J", BlockConfigEntry.class));
|
launch4j = new ExeConf(block.getEntry("launch4J", BlockConfigEntry.class));
|
||||||
sign = new SignConf(block.getEntry("signing", BlockConfigEntry.class), coredir);
|
sign = new SignConf(block.getEntry("signing", BlockConfigEntry.class), coredir);
|
||||||
binaryName = block.getEntryValue("binaryName", StringConfigEntry.class);
|
binaryName = block.getEntryValue("binaryName", StringConfigEntry.class);
|
||||||
|
@ -314,6 +315,8 @@ public static void main(String... args) throws Throwable {
|
||||||
|
|
||||||
public final ModulesManager modulesManager;
|
public final ModulesManager modulesManager;
|
||||||
|
|
||||||
|
public final MirrorManager mirrorManager;
|
||||||
|
|
||||||
|
|
||||||
public final BuildHookManager buildHookManager;
|
public final BuildHookManager buildHookManager;
|
||||||
|
|
||||||
|
@ -390,7 +393,7 @@ public LaunchServer(Path dir, boolean portable) throws IOException, InvalidKeySp
|
||||||
|
|
||||||
// Print keypair fingerprints
|
// Print keypair fingerprints
|
||||||
CRC32 crc = new CRC32();
|
CRC32 crc = new CRC32();
|
||||||
crc.update(publicKey.getModulus().toByteArray());
|
crc.update(publicKey.getModulus().toByteArray()); // IDEA говорит, что это Java 9 API. WTF?
|
||||||
LogHelper.subInfo("Modulus CRC32: 0x%08x", crc.getValue());
|
LogHelper.subInfo("Modulus CRC32: 0x%08x", crc.getValue());
|
||||||
|
|
||||||
// pre init modules
|
// pre init modules
|
||||||
|
@ -411,8 +414,16 @@ public LaunchServer(Path dir, boolean portable) throws IOException, InvalidKeySp
|
||||||
limiter = new AuthLimiter(this);
|
limiter = new AuthLimiter(this);
|
||||||
proguardConf = new ProguardConf(this);
|
proguardConf = new ProguardConf(this);
|
||||||
sessionManager = new SessionManager();
|
sessionManager = new SessionManager();
|
||||||
|
mirrorManager = new MirrorManager();
|
||||||
GarbageManager.registerNeedGC(sessionManager);
|
GarbageManager.registerNeedGC(sessionManager);
|
||||||
GarbageManager.registerNeedGC(limiter);
|
GarbageManager.registerNeedGC(limiter);
|
||||||
|
config.mirrors.stream(StringConfigEntry.class).forEach(s -> {
|
||||||
|
try {
|
||||||
|
mirrorManager.addMirror(s);
|
||||||
|
} catch (MalformedURLException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// init modules
|
// init modules
|
||||||
modulesManager.initModules();
|
modulesManager.initModules();
|
||||||
|
|
|
@ -172,7 +172,7 @@ private void stdBuild() throws IOException {
|
||||||
jaConfigurator.setAddress(server.config.getAddress());
|
jaConfigurator.setAddress(server.config.getAddress());
|
||||||
jaConfigurator.setPort(server.config.port);
|
jaConfigurator.setPort(server.config.port);
|
||||||
jaConfigurator.setProjectName(server.config.projectName);
|
jaConfigurator.setProjectName(server.config.projectName);
|
||||||
jaConfigurator.setSecretKey(SecurityHelper.randomStringToken());
|
jaConfigurator.setSecretKey(SecurityHelper.randomStringAESKey());
|
||||||
jaConfigurator.setClientPort(32148 + SecurityHelper.newRandom().nextInt(512));
|
jaConfigurator.setClientPort(32148 + SecurityHelper.newRandom().nextInt(512));
|
||||||
server.buildHookManager.registerAllClientModuleClass(jaConfigurator);
|
server.buildHookManager.registerAllClientModuleClass(jaConfigurator);
|
||||||
try (ZipInputStream input = new ZipInputStream(
|
try (ZipInputStream input = new ZipInputStream(
|
||||||
|
|
|
@ -13,7 +13,6 @@
|
||||||
import ru.gravit.launchserver.command.Command;
|
import ru.gravit.launchserver.command.Command;
|
||||||
|
|
||||||
public final class DownloadAssetCommand extends Command {
|
public final class DownloadAssetCommand extends Command {
|
||||||
private static final String ASSET_URL_MASK = "http://launcher.sashok724.net/download/assets/%s.zip";
|
|
||||||
|
|
||||||
public DownloadAssetCommand(LaunchServer server) {
|
public DownloadAssetCommand(LaunchServer server) {
|
||||||
super(server);
|
super(server);
|
||||||
|
@ -42,7 +41,7 @@ public void invoke(String... args) throws Exception {
|
||||||
|
|
||||||
// Download required asset
|
// Download required asset
|
||||||
LogHelper.subInfo("Downloading asset, it may take some time");
|
LogHelper.subInfo("Downloading asset, it may take some time");
|
||||||
HttpDownloader.downloadZip(new URL(String.format(ASSET_URL_MASK, IOHelper.urlEncode(version.name))), assetDir);
|
HttpDownloader.downloadZip(server.mirrorManager.getDefaultMirror().getAssetsURL(version.name), assetDir);
|
||||||
|
|
||||||
// Finished
|
// Finished
|
||||||
server.syncUpdatesDir(Collections.singleton(dirName));
|
server.syncUpdatesDir(Collections.singleton(dirName));
|
||||||
|
|
|
@ -21,7 +21,6 @@
|
||||||
import ru.gravit.launchserver.command.CommandException;
|
import ru.gravit.launchserver.command.CommandException;
|
||||||
|
|
||||||
public final class DownloadClientCommand extends Command {
|
public final class DownloadClientCommand extends Command {
|
||||||
private static final String CLIENT_URL_MASK = "http://launcher.sashok724.net/download/clients/%s.zip";
|
|
||||||
|
|
||||||
public DownloadClientCommand(LaunchServer server) {
|
public DownloadClientCommand(LaunchServer server) {
|
||||||
super(server);
|
super(server);
|
||||||
|
@ -50,8 +49,7 @@ public void invoke(String... args) throws IOException, CommandException {
|
||||||
|
|
||||||
// Download required client
|
// Download required client
|
||||||
LogHelper.subInfo("Downloading client, it may take some time");
|
LogHelper.subInfo("Downloading client, it may take some time");
|
||||||
HttpDownloader.downloadZip(new URL(String.format(CLIENT_URL_MASK,
|
HttpDownloader.downloadZip(server.mirrorManager.getDefaultMirror().getClientsURL(version.name), clientDir);
|
||||||
IOHelper.urlEncode(version.name))), clientDir);
|
|
||||||
|
|
||||||
// Create profile file
|
// Create profile file
|
||||||
LogHelper.subInfo("Creaing profile file: '%s'", dirName);
|
LogHelper.subInfo("Creaing profile file: '%s'", dirName);
|
||||||
|
|
|
@ -0,0 +1,64 @@
|
||||||
|
package ru.gravit.launchserver.manangers;
|
||||||
|
|
||||||
|
import ru.gravit.utils.helper.IOHelper;
|
||||||
|
|
||||||
|
import java.net.MalformedURLException;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
|
public class MirrorManager {
|
||||||
|
public class Mirror
|
||||||
|
{
|
||||||
|
URL url;
|
||||||
|
String assetsURLMask;
|
||||||
|
String clientsURLMask;
|
||||||
|
boolean enabled;
|
||||||
|
Mirror(String url)
|
||||||
|
{
|
||||||
|
assetsURLMask = url.concat("assets/%s.zip");
|
||||||
|
clientsURLMask = url.concat("clients/%s.zip");
|
||||||
|
}
|
||||||
|
private URL formatArg(String mask,String arg) throws MalformedURLException {
|
||||||
|
return new URL(String.format(mask, IOHelper.urlEncode(arg)));
|
||||||
|
}
|
||||||
|
public URL getAssetsURL(String assets) throws MalformedURLException {
|
||||||
|
return formatArg(assetsURLMask,assets);
|
||||||
|
}
|
||||||
|
public URL getClientsURL(String client) throws MalformedURLException {
|
||||||
|
return formatArg(clientsURLMask,client);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
protected ArrayList<Mirror> list = new ArrayList<>();
|
||||||
|
private Mirror defaultMirror;
|
||||||
|
public void addMirror(String mirror) throws MalformedURLException {
|
||||||
|
Mirror m = new Mirror(mirror);
|
||||||
|
m.enabled = true;
|
||||||
|
if(defaultMirror == null) defaultMirror = m;
|
||||||
|
}
|
||||||
|
public void addMirror(String mirror,boolean enabled) throws MalformedURLException {
|
||||||
|
Mirror m = new Mirror(mirror);
|
||||||
|
m.url = new URL(mirror);
|
||||||
|
m.enabled = enabled;
|
||||||
|
if(defaultMirror == null && enabled) defaultMirror = m;
|
||||||
|
}
|
||||||
|
public Mirror getDefaultMirror()
|
||||||
|
{
|
||||||
|
return defaultMirror;
|
||||||
|
}
|
||||||
|
public void setDefaultMirror(Mirror m)
|
||||||
|
{
|
||||||
|
defaultMirror = m;
|
||||||
|
}
|
||||||
|
public void disableMirror(int index)
|
||||||
|
{
|
||||||
|
list.get(index).enabled = false;
|
||||||
|
}
|
||||||
|
public void enableMirror(int index)
|
||||||
|
{
|
||||||
|
list.get(index).enabled = true;
|
||||||
|
}
|
||||||
|
public int size()
|
||||||
|
{
|
||||||
|
return list.size();
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,7 +1,7 @@
|
||||||
address: "x";
|
address: "x";
|
||||||
bindAddress: "0.0.0.0";
|
bindAddress: "0.0.0.0";
|
||||||
port: 7240;
|
port: 7240;
|
||||||
|
mirrors: ["http://launcher.sashok724.net/download/"];
|
||||||
# Auth rate limit
|
# Auth rate limit
|
||||||
authRateLimit: 2;
|
authRateLimit: 2;
|
||||||
authRateLimitMilis: 5000;
|
authRateLimitMilis: 5000;
|
||||||
|
|
|
@ -90,6 +90,8 @@ public byte[] verify(byte[] digest) {
|
||||||
@LauncherAPI
|
@LauncherAPI
|
||||||
public static final int TOKEN_LENGTH = 16;
|
public static final int TOKEN_LENGTH = 16;
|
||||||
@LauncherAPI
|
@LauncherAPI
|
||||||
|
public static final int AES_KEY_LENGTH = 24;
|
||||||
|
@LauncherAPI
|
||||||
public static final int TOKEN_STRING_LENGTH = TOKEN_LENGTH << 1;
|
public static final int TOKEN_STRING_LENGTH = TOKEN_LENGTH << 1;
|
||||||
@LauncherAPI
|
@LauncherAPI
|
||||||
public static final int RSA_KEY_LENGTH_BITS = 2048;
|
public static final int RSA_KEY_LENGTH_BITS = 2048;
|
||||||
|
@ -310,6 +312,25 @@ public static byte[] randomToken(Random random) {
|
||||||
return randomBytes(random, TOKEN_LENGTH);
|
return randomBytes(random, TOKEN_LENGTH);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@LauncherAPI
|
||||||
|
public static String randomStringAESKey() {
|
||||||
|
return toHex(randomAESKey(newRandom()));
|
||||||
|
}
|
||||||
|
@LauncherAPI
|
||||||
|
public static String randomStringAESKey(Random random) {
|
||||||
|
return toHex(randomAESKey(random));
|
||||||
|
}
|
||||||
|
|
||||||
|
@LauncherAPI
|
||||||
|
public static byte[] randomAESKey() {
|
||||||
|
return randomAESKey(newRandom());
|
||||||
|
}
|
||||||
|
|
||||||
|
@LauncherAPI
|
||||||
|
public static byte[] randomAESKey(Random random) {
|
||||||
|
return randomBytes(random, AES_KEY_LENGTH);
|
||||||
|
}
|
||||||
|
|
||||||
@LauncherAPI
|
@LauncherAPI
|
||||||
public static String randomUsername() {
|
public static String randomUsername() {
|
||||||
return randomUsername(newRandom());
|
return randomUsername(newRandom());
|
||||||
|
|
Loading…
Reference in a new issue