Migrate to LibWeb
This commit is contained in:
parent
0dd79c29f3
commit
46a6578124
13
build.gradle
13
build.gradle
|
@ -3,6 +3,10 @@ plugins {
|
||||||
id 'maven-publish'
|
id 'maven-publish'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
maven { url 'https://jitpack.io' }
|
||||||
|
}
|
||||||
|
|
||||||
sourceCompatibility = JavaVersion.VERSION_1_8
|
sourceCompatibility = JavaVersion.VERSION_1_8
|
||||||
targetCompatibility = JavaVersion.VERSION_1_8
|
targetCompatibility = JavaVersion.VERSION_1_8
|
||||||
|
|
||||||
|
@ -16,12 +20,19 @@ dependencies {
|
||||||
mappings "net.fabricmc:yarn:${project.yarn_mappings}:v2"
|
mappings "net.fabricmc:yarn:${project.yarn_mappings}:v2"
|
||||||
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
|
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
|
||||||
|
|
||||||
modImplementation "net.fabricmc.fabric-api:fabric-api:${project.fabric_version}"
|
modImplementation "net.fabricmc.fabric-api:fabric-command-api-v1:+"
|
||||||
|
modImplementation "net.fabricmc.fabric-api:fabric-lifecycle-events-v1:+"
|
||||||
|
modImplementation "net.fabricmc.fabric-api:fabric-api-base:+"
|
||||||
|
|
||||||
modApi("me.sargunvohra.mcmods:autoconfig1u:3.2.2") {
|
modApi("me.sargunvohra.mcmods:autoconfig1u:3.2.2") {
|
||||||
exclude(group: "net.fabricmc.fabric-api")
|
exclude(group: "net.fabricmc.fabric-api")
|
||||||
}
|
}
|
||||||
include "me.sargunvohra.mcmods:autoconfig1u:3.2.2"
|
include "me.sargunvohra.mcmods:autoconfig1u:3.2.2"
|
||||||
|
|
||||||
|
modImplementation ('com.gitlab.JFronny:LibWeb:-SNAPSHOT') {
|
||||||
|
exclude(group: "net.fabricmc.fabric-api")
|
||||||
|
}
|
||||||
|
include "com.gitlab.JFronny:LibWeb:-SNAPSHOT"
|
||||||
}
|
}
|
||||||
|
|
||||||
processResources {
|
processResources {
|
||||||
|
|
|
@ -2,13 +2,10 @@
|
||||||
org.gradle.jvmargs=-Xmx1G
|
org.gradle.jvmargs=-Xmx1G
|
||||||
# Fabric Properties
|
# Fabric Properties
|
||||||
# check these on https://modmuss50.me/fabric.html
|
# check these on https://modmuss50.me/fabric.html
|
||||||
minecraft_version=1.16.2
|
minecraft_version=1.16.3
|
||||||
yarn_mappings=1.16.2+build.47
|
yarn_mappings=1.16.3+build.11
|
||||||
loader_version=0.9.3+build.207
|
loader_version=0.9.3+build.207
|
||||||
# Mod Properties
|
# Mod Properties
|
||||||
mod_version=1.0
|
mod_version=1.1
|
||||||
maven_group=io.gitlab.jfronny
|
maven_group=io.gitlab.jfronny
|
||||||
archives_base_name=dynres
|
archives_base_name=dynres
|
||||||
# Dependencies
|
|
||||||
# check this on https://modmuss50.me/fabric.html
|
|
||||||
fabric_version=0.20.1+build.401-1.16
|
|
||||||
|
|
|
@ -6,15 +6,6 @@ import me.sargunvohra.mcmods.autoconfig1u.shadowed.blue.endless.jankson.Comment;
|
||||||
|
|
||||||
@Config(name = "dynres")
|
@Config(name = "dynres")
|
||||||
public class Cfg implements ConfigData {
|
public class Cfg implements ConfigData {
|
||||||
@Comment("The base of the link to send to client. This is equal to the address for clients to enter (requires a restart)")
|
|
||||||
public String baseLink = "http://127.0.0.1";
|
|
||||||
|
|
||||||
@Comment("The port to use for hosting the file. Use 0 for a random one (requires a restart)")
|
|
||||||
public int port = 0;
|
|
||||||
|
|
||||||
@Comment("The number of allowed concurrent downloads. Higher values prevent errors on clients but can increase load")
|
|
||||||
public int maxConnections = 20;
|
|
||||||
|
|
||||||
@Comment("The relative path to the resources zip. Expect strange behaviour if changed")
|
@Comment("The relative path to the resources zip. Expect strange behaviour if changed")
|
||||||
public String resourcesFile = "resources.zip";
|
public String resourcesFile = "resources.zip";
|
||||||
|
|
||||||
|
|
|
@ -1,110 +1,43 @@
|
||||||
package io.gitlab.jfronny.dynres;
|
package io.gitlab.jfronny.dynres;
|
||||||
|
|
||||||
import com.mojang.brigadier.Command;
|
import io.gitlab.jfronny.libweb.api.LibWebAPI;
|
||||||
import io.gitlab.jfronny.dynres.web.RequestHandler;
|
import io.gitlab.jfronny.libweb.api.LibWebInit;
|
||||||
import io.gitlab.jfronny.dynres.web.bluemapcore.WebServer;
|
|
||||||
import me.sargunvohra.mcmods.autoconfig1u.AutoConfig;
|
import me.sargunvohra.mcmods.autoconfig1u.AutoConfig;
|
||||||
import me.sargunvohra.mcmods.autoconfig1u.serializer.JanksonConfigSerializer;
|
import me.sargunvohra.mcmods.autoconfig1u.serializer.JanksonConfigSerializer;
|
||||||
import net.fabricmc.api.DedicatedServerModInitializer;
|
import net.fabricmc.api.DedicatedServerModInitializer;
|
||||||
import net.fabricmc.api.EnvType;
|
import net.fabricmc.api.EnvType;
|
||||||
import net.fabricmc.api.Environment;
|
import net.fabricmc.api.Environment;
|
||||||
import net.fabricmc.fabric.api.command.v1.CommandRegistrationCallback;
|
|
||||||
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerLifecycleEvents;
|
|
||||||
import net.fabricmc.loader.api.FabricLoader;
|
import net.fabricmc.loader.api.FabricLoader;
|
||||||
import net.fabricmc.loader.gui.FabricGuiEntry;
|
import net.fabricmc.loader.gui.FabricGuiEntry;
|
||||||
import net.minecraft.server.command.CommandManager;
|
|
||||||
import net.minecraft.text.LiteralText;
|
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileNotFoundException;
|
import java.io.FileNotFoundException;
|
||||||
import java.util.Arrays;
|
import java.io.IOException;
|
||||||
|
|
||||||
@Environment(EnvType.SERVER)
|
public class DynRes implements LibWebInit {
|
||||||
public class DynRes implements DedicatedServerModInitializer {
|
|
||||||
static WebServer server;
|
|
||||||
public static File resFile;
|
public static File resFile;
|
||||||
public static Cfg cfg;
|
public static Cfg cfg;
|
||||||
|
public static String resourceLink = "";
|
||||||
static {
|
private static boolean initialized = false;
|
||||||
AutoConfig.register(Cfg.class, JanksonConfigSerializer::new);
|
|
||||||
cfg = AutoConfig.getConfigHolder(Cfg.class).getConfig();
|
|
||||||
resFile = new File(FabricLoader.getInstance().getGameDir().toFile(), cfg.resourcesFile);
|
|
||||||
if (!resFile.isFile()) {
|
|
||||||
FabricGuiEntry.displayCriticalError(new FileNotFoundException("The file " + resFile + " does not exist in the game directory but is required"), true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onInitializeServer() {
|
public void register(LibWebAPI api) {
|
||||||
ServerLifecycleEvents.SERVER_STOPPED.register(t -> {
|
if (FabricLoader.getInstance().getEnvironmentType() == EnvType.SERVER) {
|
||||||
try {
|
try {
|
||||||
server.close();
|
if (!initialized) {
|
||||||
server.join();
|
initialized = true;
|
||||||
} catch (Throwable e) {
|
AutoConfig.register(Cfg.class, JanksonConfigSerializer::new);
|
||||||
Logger.l.error("Failed to stop web server", e);
|
cfg = AutoConfig.getConfigHolder(Cfg.class).getConfig();
|
||||||
}
|
resFile = new File(FabricLoader.getInstance().getGameDir().toFile(), cfg.resourcesFile);
|
||||||
});
|
if (!resFile.isFile()) {
|
||||||
|
FabricGuiEntry.displayCriticalError(new FileNotFoundException("The file " + resFile + " does not exist in the game directory but is required"), true);
|
||||||
CommandRegistrationCallback.EVENT.register((dispatcher, dedicated) -> {
|
|
||||||
if (dedicated) {
|
|
||||||
dispatcher.register(CommandManager.literal("dynres").requires((serverCommandSource) -> serverCommandSource.hasPermissionLevel(4))
|
|
||||||
.executes(context -> {
|
|
||||||
context.getSource().sendFeedback(new LiteralText("DynRes is active. Use dynres restart to reload"), false);
|
|
||||||
return Command.SINGLE_SUCCESS;
|
|
||||||
}));
|
|
||||||
dispatcher.register(CommandManager.literal("dynres").requires((serverCommandSource) -> serverCommandSource.hasPermissionLevel(4))
|
|
||||||
.then(CommandManager.literal("restart").executes(context -> {
|
|
||||||
try {
|
|
||||||
context.getSource().sendFeedback(new LiteralText("Restarting DynRes"), true);
|
|
||||||
cfg = AutoConfig.getConfigHolder(Cfg.class).getConfig();
|
|
||||||
resFile = new File(FabricLoader.getInstance().getGameDir().toFile(), cfg.resourcesFile);
|
|
||||||
restartServer();
|
|
||||||
}
|
}
|
||||||
catch (Exception e) {
|
Logger.l.info("Initialized DynRes");
|
||||||
Logger.l.error("Failed to run restart command", e);
|
}
|
||||||
context.getSource().sendError(new LiteralText(e.getMessage()));
|
resourceLink = api.registerFile("/resources.zip", resFile.toPath(), !cfg.hashResources);
|
||||||
}
|
} catch (IOException e) {
|
||||||
return Command.SINGLE_SUCCESS;
|
e.printStackTrace();
|
||||||
})));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
Logger.l.error("DYNRES SHOULD NOT BE RUN ON INTERNAL SERVERS!");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
restartServer();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void restartServer() {
|
|
||||||
int port = cfg.port;
|
|
||||||
if (server != null) {
|
|
||||||
port = server.getPort();
|
|
||||||
server.close();
|
|
||||||
try {
|
|
||||||
server.join();
|
|
||||||
} catch (InterruptedException e) {
|
|
||||||
//It is most likely already dead
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
server = new WebServer(port, cfg.maxConnections, null, new RequestHandler("resources.zip"));
|
|
||||||
server.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String getPort() {
|
|
||||||
return Integer.toString(server.getPort());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String simplifyElement(String s) {
|
|
||||||
String path = s.toLowerCase();
|
|
||||||
if (path.startsWith("/")) path = path.substring(1);
|
|
||||||
if (path.endsWith("/")) path = path.substring(0, path.length() - 1);
|
|
||||||
return path;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String removePort(String s) {
|
|
||||||
String[] r = s.split(":");
|
|
||||||
if (r.length > 2)
|
|
||||||
r = Arrays.copyOf(r, 2);
|
|
||||||
return String.join(":", r);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,8 @@ package io.gitlab.jfronny.dynres.mixin;
|
||||||
import io.gitlab.jfronny.dynres.DynRes;
|
import io.gitlab.jfronny.dynres.DynRes;
|
||||||
import io.gitlab.jfronny.dynres.Logger;
|
import io.gitlab.jfronny.dynres.Logger;
|
||||||
import io.gitlab.jfronny.dynres.ServerPropertiesHandlerExt;
|
import io.gitlab.jfronny.dynres.ServerPropertiesHandlerExt;
|
||||||
|
import io.gitlab.jfronny.libweb.extra.LibWeb;
|
||||||
|
import io.gitlab.jfronny.libweb.extra.WebPaths;
|
||||||
import net.minecraft.server.dedicated.ServerPropertiesHandler;
|
import net.minecraft.server.dedicated.ServerPropertiesHandler;
|
||||||
import net.minecraft.util.registry.DynamicRegistryManager;
|
import net.minecraft.util.registry.DynamicRegistryManager;
|
||||||
import org.apache.commons.codec.digest.DigestUtils;
|
import org.apache.commons.codec.digest.DigestUtils;
|
||||||
|
@ -14,7 +16,6 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
||||||
|
|
||||||
import java.io.FileInputStream;
|
import java.io.FileInputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.Objects;
|
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
@Mixin(ServerPropertiesHandler.class)
|
@Mixin(ServerPropertiesHandler.class)
|
||||||
|
@ -32,7 +33,7 @@ public class ServerPropertiesHandlerMixin implements ServerPropertiesHandlerExt
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void applyChanges(boolean print) {
|
public void applyChanges(boolean print) {
|
||||||
resourcePack = DynRes.removePort(DynRes.simplifyElement(DynRes.cfg.baseLink)) + ":" + DynRes.getPort() + "/resources.zip";
|
resourcePack = WebPaths.concat(LibWeb.api.getServerRoot(), "resources.zip");
|
||||||
if (print)
|
if (print)
|
||||||
Logger.l.info("Pack link: " + resourcePack);
|
Logger.l.info("Pack link: " + resourcePack);
|
||||||
resourcePackSha1 = "";
|
resourcePackSha1 = "";
|
||||||
|
|
|
@ -1,53 +0,0 @@
|
||||||
package io.gitlab.jfronny.dynres.web;
|
|
||||||
|
|
||||||
import io.gitlab.jfronny.dynres.DynRes;
|
|
||||||
import io.gitlab.jfronny.dynres.Logger;
|
|
||||||
import io.gitlab.jfronny.dynres.web.bluemapcore.HttpRequest;
|
|
||||||
import io.gitlab.jfronny.dynres.web.bluemapcore.HttpRequestHandler;
|
|
||||||
import io.gitlab.jfronny.dynres.web.bluemapcore.HttpResponse;
|
|
||||||
import io.gitlab.jfronny.dynres.web.bluemapcore.HttpStatusCode;
|
|
||||||
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileInputStream;
|
|
||||||
import java.io.FileNotFoundException;
|
|
||||||
import java.util.Objects;
|
|
||||||
|
|
||||||
public class RequestHandler implements HttpRequestHandler {
|
|
||||||
String r;
|
|
||||||
@Override
|
|
||||||
public HttpResponse handle(HttpRequest request) {
|
|
||||||
try {
|
|
||||||
//TODO remove debug statement
|
|
||||||
Logger.l.info("Got connection");
|
|
||||||
String method = request.getMethod().toUpperCase();
|
|
||||||
if (!method.equals("GET") && !method.equals("HEAD") && !method.equals("POST")) {
|
|
||||||
Logger.l.error("Invalid method: " + method);
|
|
||||||
HttpResponse resp = new HttpResponse(HttpStatusCode.BAD_REQUEST);
|
|
||||||
resp.setData(method + " method not supported");
|
|
||||||
return resp;
|
|
||||||
}
|
|
||||||
|
|
||||||
String path = DynRes.simplifyElement(request.getPath());
|
|
||||||
|
|
||||||
if (Objects.equals(path, r)) {
|
|
||||||
HttpResponse resp = new HttpResponse(HttpStatusCode.OK);
|
|
||||||
resp.addHeader("Server", "DynRes using BlueMap");
|
|
||||||
resp.addHeader("Cache-Control", "no-cache");
|
|
||||||
resp.addHeader("Content-Type", "application/zip");
|
|
||||||
|
|
||||||
FileInputStream fs = new FileInputStream(DynRes.resFile);
|
|
||||||
resp.setData(fs);
|
|
||||||
return resp;
|
|
||||||
}
|
|
||||||
Logger.l.error("An invalid file was requested: " + path);
|
|
||||||
return new HttpResponse(HttpStatusCode.NOT_FOUND);
|
|
||||||
} catch (Throwable e) {
|
|
||||||
Logger.l.error("Cough error while sending", e);
|
|
||||||
return new HttpResponse(HttpStatusCode.INTERNAL_SERVER_ERROR);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public RequestHandler(String relativePath) {
|
|
||||||
r = relativePath;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,127 +0,0 @@
|
||||||
/*
|
|
||||||
* This file is part of BlueMap, licensed under the MIT License (MIT).
|
|
||||||
*
|
|
||||||
* Copyright (c) Blue (Lukas Rieger) <https://bluecolored.de>
|
|
||||||
* Copyright (c) contributors
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
package io.gitlab.jfronny.dynres.web.bluemapcore;
|
|
||||||
|
|
||||||
import io.gitlab.jfronny.dynres.DynRes;
|
|
||||||
import io.gitlab.jfronny.dynres.Logger;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.io.OutputStream;
|
|
||||||
import java.net.ServerSocket;
|
|
||||||
import java.net.Socket;
|
|
||||||
import java.net.SocketException;
|
|
||||||
import java.net.SocketTimeoutException;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
public class HttpConnection implements Runnable {
|
|
||||||
|
|
||||||
private HttpRequestHandler handler;
|
|
||||||
|
|
||||||
private ServerSocket server;
|
|
||||||
private Socket connection;
|
|
||||||
private InputStream in;
|
|
||||||
private OutputStream out;
|
|
||||||
|
|
||||||
public HttpConnection(ServerSocket server, Socket connection, HttpRequestHandler handler, int timeout, TimeUnit timeoutUnit) throws IOException {
|
|
||||||
this.server = server;
|
|
||||||
this.connection = connection;
|
|
||||||
this.handler = handler;
|
|
||||||
|
|
||||||
if (isClosed()){
|
|
||||||
throw new IOException("Socket already closed!");
|
|
||||||
}
|
|
||||||
|
|
||||||
connection.setSoTimeout((int) timeoutUnit.toMillis(timeout));
|
|
||||||
|
|
||||||
in = this.connection.getInputStream();
|
|
||||||
out = this.connection.getOutputStream();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
while (!isClosed() && !server.isClosed()){
|
|
||||||
try {
|
|
||||||
HttpRequest request = acceptRequest();
|
|
||||||
HttpResponse response = handler.handle(request);
|
|
||||||
sendResponse(response);
|
|
||||||
} catch (InvalidRequestException e){
|
|
||||||
try {
|
|
||||||
sendResponse(new HttpResponse(HttpStatusCode.BAD_REQUEST));
|
|
||||||
} catch (IOException e1) {}
|
|
||||||
break;
|
|
||||||
} catch (SocketTimeoutException e) {
|
|
||||||
break;
|
|
||||||
} catch (SocketException e){
|
|
||||||
break;
|
|
||||||
} catch (ConnectionClosedException e){
|
|
||||||
break;
|
|
||||||
} catch (IOException e) {
|
|
||||||
Logger.l.error("Unexpected error while processing a HttpRequest!", e);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
close();
|
|
||||||
} catch (IOException e){
|
|
||||||
Logger.l.error("Error while closing HttpConnection!", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void sendResponse(HttpResponse response) throws IOException {
|
|
||||||
response.write(out);
|
|
||||||
out.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
private HttpRequest acceptRequest() throws ConnectionClosedException, InvalidRequestException, IOException {
|
|
||||||
return HttpRequest.read(in);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isClosed(){
|
|
||||||
return !connection.isBound() || connection.isClosed() || !connection.isConnected() || connection.isOutputShutdown() || connection.isInputShutdown();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void close() throws IOException {
|
|
||||||
try {
|
|
||||||
in.close();
|
|
||||||
} finally {
|
|
||||||
try {
|
|
||||||
out.close();
|
|
||||||
} finally {
|
|
||||||
connection.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class ConnectionClosedException extends IOException {
|
|
||||||
private static final long serialVersionUID = 1L;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class InvalidRequestException extends IOException {
|
|
||||||
private static final long serialVersionUID = 1L;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,245 +0,0 @@
|
||||||
/*
|
|
||||||
* This file is part of BlueMap, licensed under the MIT License (MIT).
|
|
||||||
*
|
|
||||||
* Copyright (c) Blue (Lukas Rieger) <https://bluecolored.de>
|
|
||||||
* Copyright (c) contributors
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
package io.gitlab.jfronny.dynres.web.bluemapcore;
|
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
|
||||||
import java.io.ByteArrayInputStream;
|
|
||||||
import java.io.ByteArrayOutputStream;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.io.InputStreamReader;
|
|
||||||
import java.nio.charset.StandardCharsets;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Map.Entry;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.regex.Matcher;
|
|
||||||
import java.util.regex.Pattern;
|
|
||||||
|
|
||||||
import io.gitlab.jfronny.dynres.web.bluemapcore.HttpConnection.ConnectionClosedException;
|
|
||||||
import io.gitlab.jfronny.dynres.web.bluemapcore.HttpConnection.InvalidRequestException;
|
|
||||||
|
|
||||||
public class HttpRequest {
|
|
||||||
|
|
||||||
private static final Pattern REQUEST_PATTERN = Pattern.compile("^(\\w+) (\\S+) (.+)$");
|
|
||||||
|
|
||||||
private String method;
|
|
||||||
private String adress;
|
|
||||||
private String version;
|
|
||||||
private Map<String, Set<String>> header;
|
|
||||||
private Map<String, Set<String>> headerLC;
|
|
||||||
private byte[] data;
|
|
||||||
|
|
||||||
private String path = null;
|
|
||||||
private Map<String, String> getParams = null;
|
|
||||||
private String getParamString = null;
|
|
||||||
|
|
||||||
public HttpRequest(String method, String adress, String version, Map<String, Set<String>> header) {
|
|
||||||
this.method = method;
|
|
||||||
this.adress = adress;
|
|
||||||
this.version = version;
|
|
||||||
this.header = header;
|
|
||||||
this.headerLC = new HashMap<>();
|
|
||||||
|
|
||||||
for (Entry<String, Set<String>> e : header.entrySet()){
|
|
||||||
Set<String> values = new HashSet<>();
|
|
||||||
for (String v : e.getValue()){
|
|
||||||
values.add(v.toLowerCase());
|
|
||||||
}
|
|
||||||
|
|
||||||
headerLC.put(e.getKey().toLowerCase(), values);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.data = new byte[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getMethod() {
|
|
||||||
return method;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getAdress(){
|
|
||||||
return adress;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getVersion() {
|
|
||||||
return version;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Map<String, Set<String>> getHeader() {
|
|
||||||
return header;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Map<String, Set<String>> getLowercaseHeader() {
|
|
||||||
return headerLC;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Set<String> getHeader(String key){
|
|
||||||
Set<String> headerValues = header.get(key);
|
|
||||||
if (headerValues == null) return Collections.emptySet();
|
|
||||||
return headerValues;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Set<String> getLowercaseHeader(String key){
|
|
||||||
Set<String> headerValues = headerLC.get(key.toLowerCase());
|
|
||||||
if (headerValues == null) return Collections.emptySet();
|
|
||||||
return headerValues;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getPath() {
|
|
||||||
if (path == null) parseAdress();
|
|
||||||
return path;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Map<String, String> getGETParams() {
|
|
||||||
if (getParams == null) parseAdress();
|
|
||||||
return Collections.unmodifiableMap(getParams);
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getGETParamString() {
|
|
||||||
if (getParamString == null) parseAdress();
|
|
||||||
return getParamString;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void parseAdress() {
|
|
||||||
String adress = this.adress;
|
|
||||||
if (adress.isEmpty()) adress = "/";
|
|
||||||
String[] adressParts = adress.split("\\?", 2);
|
|
||||||
String path = adressParts[0];
|
|
||||||
this.getParamString = adressParts.length > 1 ? adressParts[1] : "";
|
|
||||||
|
|
||||||
Map<String, String> getParams = new HashMap<>();
|
|
||||||
for (String getParam : this.getParamString.split("&")){
|
|
||||||
if (getParam.isEmpty()) continue;
|
|
||||||
String[] kv = getParam.split("=", 2);
|
|
||||||
String key = kv[0];
|
|
||||||
String value = kv.length > 1 ? kv[1] : "";
|
|
||||||
getParams.put(key, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.path = path;
|
|
||||||
this.getParams = getParams;
|
|
||||||
}
|
|
||||||
|
|
||||||
public InputStream getData(){
|
|
||||||
return new ByteArrayInputStream(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static HttpRequest read(InputStream in) throws IOException, InvalidRequestException {
|
|
||||||
BufferedReader reader = new BufferedReader(new InputStreamReader(in, StandardCharsets.UTF_8));
|
|
||||||
List<String> header = new ArrayList<>(20);
|
|
||||||
while(header.size() < 1000){
|
|
||||||
String headerLine = readLine(reader);
|
|
||||||
if (headerLine.isEmpty()) break;
|
|
||||||
header.add(headerLine);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (header.isEmpty()) throw new InvalidRequestException();
|
|
||||||
|
|
||||||
Matcher m = REQUEST_PATTERN.matcher(header.remove(0));
|
|
||||||
if (!m.find()) throw new InvalidRequestException();
|
|
||||||
|
|
||||||
String method = m.group(1);
|
|
||||||
if (method == null) throw new InvalidRequestException();
|
|
||||||
|
|
||||||
String adress = m.group(2);
|
|
||||||
if (adress == null) throw new InvalidRequestException();
|
|
||||||
|
|
||||||
String version = m.group(3);
|
|
||||||
if (version == null) throw new InvalidRequestException();
|
|
||||||
|
|
||||||
Map<String, Set<String>> headerMap = new HashMap<String, Set<String>>();
|
|
||||||
for (String line : header){
|
|
||||||
if (line.trim().isEmpty()) continue;
|
|
||||||
|
|
||||||
String[] kv = line.split(":", 2);
|
|
||||||
if (kv.length < 2) continue;
|
|
||||||
|
|
||||||
Set<String> values = new HashSet<>();
|
|
||||||
if (kv[0].trim().equalsIgnoreCase("If-Modified-Since")){
|
|
||||||
values.add(kv[1].trim());
|
|
||||||
} else {
|
|
||||||
for(String v : kv[1].split(",")){
|
|
||||||
values.add(v.trim());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
headerMap.put(kv[0].trim(), values);
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpRequest request = new HttpRequest(method, adress, version, headerMap);
|
|
||||||
|
|
||||||
if (request.getLowercaseHeader("Transfer-Encoding").contains("chunked")){
|
|
||||||
try {
|
|
||||||
ByteArrayOutputStream dataStream = new ByteArrayOutputStream();
|
|
||||||
while (dataStream.size() < 1000000){
|
|
||||||
String hexSize = reader.readLine();
|
|
||||||
int chunkSize = Integer.parseInt(hexSize, 16);
|
|
||||||
if (chunkSize <= 0) break;
|
|
||||||
byte[] data = new byte[chunkSize];
|
|
||||||
in.read(data);
|
|
||||||
dataStream.write(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dataStream.size() >= 1000000) {
|
|
||||||
throw new InvalidRequestException();
|
|
||||||
}
|
|
||||||
|
|
||||||
request.data = dataStream.toByteArray();
|
|
||||||
|
|
||||||
return request;
|
|
||||||
} catch (NumberFormatException ex){
|
|
||||||
return request;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Set<String> clSet = request.getLowercaseHeader("Content-Length");
|
|
||||||
if (clSet.isEmpty()){
|
|
||||||
return request;
|
|
||||||
} else {
|
|
||||||
try {
|
|
||||||
int cl = Integer.parseInt(clSet.iterator().next());
|
|
||||||
byte[] data = new byte[cl];
|
|
||||||
in.read(data);
|
|
||||||
request.data = data;
|
|
||||||
return request;
|
|
||||||
} catch (NumberFormatException ex){
|
|
||||||
return request;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static String readLine(BufferedReader in) throws ConnectionClosedException, IOException {
|
|
||||||
String line = in.readLine();
|
|
||||||
if (line == null){
|
|
||||||
throw new ConnectionClosedException();
|
|
||||||
}
|
|
||||||
return line;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,34 +0,0 @@
|
||||||
/*
|
|
||||||
* This file is part of BlueMap, licensed under the MIT License (MIT).
|
|
||||||
*
|
|
||||||
* Copyright (c) Blue (Lukas Rieger) <https://bluecolored.de>
|
|
||||||
* Copyright (c) contributors
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
package io.gitlab.jfronny.dynres.web.bluemapcore;
|
|
||||||
|
|
||||||
import org.apache.http.MethodNotSupportedException;
|
|
||||||
|
|
||||||
@FunctionalInterface
|
|
||||||
public interface HttpRequestHandler {
|
|
||||||
|
|
||||||
HttpResponse handle(HttpRequest request);
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,155 +0,0 @@
|
||||||
/*
|
|
||||||
* This file is part of BlueMap, licensed under the MIT License (MIT).
|
|
||||||
*
|
|
||||||
* Copyright (c) Blue (Lukas Rieger) <https://bluecolored.de>
|
|
||||||
* Copyright (c) contributors
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
package io.gitlab.jfronny.dynres.web.bluemapcore;
|
|
||||||
|
|
||||||
import java.io.ByteArrayInputStream;
|
|
||||||
import java.io.Closeable;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.io.OutputStream;
|
|
||||||
import java.io.OutputStreamWriter;
|
|
||||||
import java.nio.charset.StandardCharsets;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Map.Entry;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import org.apache.commons.lang3.StringUtils;
|
|
||||||
|
|
||||||
public class HttpResponse implements Closeable {
|
|
||||||
|
|
||||||
private String version;
|
|
||||||
private HttpStatusCode statusCode;
|
|
||||||
private Map<String, Set<String>> header;
|
|
||||||
private InputStream data;
|
|
||||||
|
|
||||||
public HttpResponse(HttpStatusCode statusCode) {
|
|
||||||
this.version = "HTTP/1.1";
|
|
||||||
this.statusCode = statusCode;
|
|
||||||
|
|
||||||
this.header = new HashMap<>();
|
|
||||||
|
|
||||||
addHeader("Connection", "keep-alive");
|
|
||||||
}
|
|
||||||
|
|
||||||
public void addHeader(String key, String value){
|
|
||||||
Set<String> valueSet = header.get(key);
|
|
||||||
if (valueSet == null){
|
|
||||||
valueSet = new HashSet<>();
|
|
||||||
header.put(key, valueSet);
|
|
||||||
}
|
|
||||||
|
|
||||||
valueSet.add(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void removeHeader(String key, String value){
|
|
||||||
Set<String> valueSet = header.get(key);
|
|
||||||
if (valueSet == null){
|
|
||||||
valueSet = new HashSet<>();
|
|
||||||
header.put(key, valueSet);
|
|
||||||
}
|
|
||||||
|
|
||||||
valueSet.remove(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setData(InputStream dataStream){
|
|
||||||
this.data = dataStream;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setData(String data){
|
|
||||||
setData(new ByteArrayInputStream(data.getBytes(StandardCharsets.UTF_8)));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Writes this Response to an Output-Stream.<br>
|
|
||||||
* <br>
|
|
||||||
* This method closes the data-Stream of this response so it can't be used again!
|
|
||||||
*/
|
|
||||||
public void write(OutputStream out) throws IOException {
|
|
||||||
OutputStreamWriter writer = new OutputStreamWriter(out, StandardCharsets.UTF_8);
|
|
||||||
|
|
||||||
if (data != null){
|
|
||||||
addHeader("Transfer-Encoding", "chunked");
|
|
||||||
} else {
|
|
||||||
addHeader("Content-Length", "0");
|
|
||||||
}
|
|
||||||
|
|
||||||
writeLine(writer, version + " " + statusCode.getCode() + " " + statusCode.getMessage());
|
|
||||||
for (Entry<String, Set<String>> e : header.entrySet()){
|
|
||||||
if (e.getValue().isEmpty()) continue;
|
|
||||||
writeLine(writer, e.getKey() + ": " + StringUtils.join(e.getValue(), ", "));
|
|
||||||
}
|
|
||||||
|
|
||||||
writeLine(writer, "");
|
|
||||||
writer.flush();
|
|
||||||
|
|
||||||
if(data != null){
|
|
||||||
chunkedPipe(data, out);
|
|
||||||
out.flush();
|
|
||||||
data.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void close() throws IOException {
|
|
||||||
data.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void writeLine(OutputStreamWriter writer, String line) throws IOException {
|
|
||||||
writer.write(line + "\r\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
private void chunkedPipe(InputStream input, OutputStream output) throws IOException {
|
|
||||||
byte[] buffer = new byte[1024];
|
|
||||||
int byteCount;
|
|
||||||
while ((byteCount = input.read(buffer)) != -1) {
|
|
||||||
output.write((Integer.toHexString(byteCount) + "\r\n").getBytes());
|
|
||||||
output.write(buffer, 0, byteCount);
|
|
||||||
output.write("\r\n".getBytes());
|
|
||||||
}
|
|
||||||
output.write("0\r\n\r\n".getBytes());
|
|
||||||
}
|
|
||||||
|
|
||||||
public HttpStatusCode getStatusCode(){
|
|
||||||
return statusCode;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getVersion(){
|
|
||||||
return version;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Map<String, Set<String>> getHeader() {
|
|
||||||
return header;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Set<String> getHeader(String key){
|
|
||||||
Set<String> headerValues = header.get(key);
|
|
||||||
if (headerValues == null) return Collections.emptySet();
|
|
||||||
return headerValues;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,70 +0,0 @@
|
||||||
/*
|
|
||||||
* This file is part of BlueMap, licensed under the MIT License (MIT).
|
|
||||||
*
|
|
||||||
* Copyright (c) Blue (Lukas Rieger) <https://bluecolored.de>
|
|
||||||
* Copyright (c) contributors
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
package io.gitlab.jfronny.dynres.web.bluemapcore;
|
|
||||||
|
|
||||||
public enum HttpStatusCode {
|
|
||||||
|
|
||||||
CONTINUE (100, "Continue"),
|
|
||||||
PROCESSING (102, "Processing"),
|
|
||||||
|
|
||||||
OK (200, "OK"),
|
|
||||||
|
|
||||||
MOVED_PERMANENTLY (301, "Moved Permanently"),
|
|
||||||
FOUND (302, "Found"),
|
|
||||||
SEE_OTHER (303, "See Other"),
|
|
||||||
NOT_MODIFIED (304, "Not Modified"),
|
|
||||||
|
|
||||||
BAD_REQUEST (400, "Bad Request"),
|
|
||||||
UNAUTHORIZED (401, "Unauthorized"),
|
|
||||||
FORBIDDEN (403, "Forbidden"),
|
|
||||||
NOT_FOUND (404, "Not Found"),
|
|
||||||
|
|
||||||
INTERNAL_SERVER_ERROR (500, "Internal Server Error"),
|
|
||||||
NOT_IMPLEMENTED (501, "Not Implemented"),
|
|
||||||
SERVICE_UNAVAILABLE (503, "Service Unavailable"),
|
|
||||||
HTTP_VERSION_NOT_SUPPORTED (505, "HTTP Version not supported");
|
|
||||||
|
|
||||||
private int code;
|
|
||||||
private String message;
|
|
||||||
|
|
||||||
private HttpStatusCode(int code, String message) {
|
|
||||||
this.code = code;
|
|
||||||
this.message = message;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getCode(){
|
|
||||||
return code;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getMessage(){
|
|
||||||
return message;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return getCode() + " " + getMessage();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,126 +0,0 @@
|
||||||
/*
|
|
||||||
* This file is part of BlueMap, licensed under the MIT License (MIT).
|
|
||||||
*
|
|
||||||
* Copyright (c) Blue (Lukas Rieger) <https://bluecolored.de>
|
|
||||||
* Copyright (c) contributors
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
package io.gitlab.jfronny.dynres.web.bluemapcore;
|
|
||||||
|
|
||||||
import io.gitlab.jfronny.dynres.DynRes;
|
|
||||||
import io.gitlab.jfronny.dynres.Logger;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.net.InetAddress;
|
|
||||||
import java.net.ServerSocket;
|
|
||||||
import java.net.Socket;
|
|
||||||
import java.net.SocketException;
|
|
||||||
import java.util.concurrent.LinkedBlockingQueue;
|
|
||||||
import java.util.concurrent.RejectedExecutionException;
|
|
||||||
import java.util.concurrent.ThreadPoolExecutor;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
public class WebServer extends Thread {
|
|
||||||
|
|
||||||
private int port;
|
|
||||||
private int maxConnections;
|
|
||||||
private final InetAddress bindAdress;
|
|
||||||
|
|
||||||
private HttpRequestHandler handler;
|
|
||||||
|
|
||||||
private ThreadPoolExecutor connectionThreads;
|
|
||||||
|
|
||||||
private ServerSocket server;
|
|
||||||
|
|
||||||
public WebServer(int port, int maxConnections, InetAddress bindAdress, HttpRequestHandler handler) {
|
|
||||||
this.port = port;
|
|
||||||
this.maxConnections = maxConnections;
|
|
||||||
this.bindAdress = bindAdress;
|
|
||||||
|
|
||||||
this.handler = handler;
|
|
||||||
|
|
||||||
connectionThreads = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run(){
|
|
||||||
close();
|
|
||||||
|
|
||||||
connectionThreads = new ThreadPoolExecutor(maxConnections, maxConnections, 10, TimeUnit.SECONDS, new LinkedBlockingQueue<>());
|
|
||||||
connectionThreads.allowCoreThreadTimeOut(true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
server = new ServerSocket(port, maxConnections, bindAdress);
|
|
||||||
server.setSoTimeout(0);
|
|
||||||
} catch (IOException e){
|
|
||||||
Logger.l.error("Error while starting the WebServer!", e);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Logger.l.info("WebServer started.");
|
|
||||||
|
|
||||||
while (!server.isClosed() && server.isBound()){
|
|
||||||
|
|
||||||
try {
|
|
||||||
Socket connection = server.accept();
|
|
||||||
|
|
||||||
try {
|
|
||||||
connectionThreads.execute(new HttpConnection(server, connection, handler, 10, TimeUnit.SECONDS));
|
|
||||||
} catch (RejectedExecutionException e){
|
|
||||||
connection.close();
|
|
||||||
Logger.l.error("Dropped an incoming HttpConnection! (Too many connections?)");
|
|
||||||
}
|
|
||||||
|
|
||||||
} catch (SocketException e){
|
|
||||||
// this mainly occurs if the socket got closed, so we ignore this error
|
|
||||||
} catch (IOException e){
|
|
||||||
Logger.l.error("Error while creating a new HttpConnection!", e);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
Logger.l.info("WebServer closed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getPort() {
|
|
||||||
return server.getLocalPort();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setPort(int port) {
|
|
||||||
this.port = port;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setMaxConnections(int maxConnections) {
|
|
||||||
this.maxConnections = maxConnections;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void close(){
|
|
||||||
if (connectionThreads != null) connectionThreads.shutdown();
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (server != null && !server.isClosed()){
|
|
||||||
server.close();
|
|
||||||
}
|
|
||||||
} catch (IOException e) {
|
|
||||||
Logger.l.error("Error while closing WebServer!", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -15,7 +15,7 @@
|
||||||
"icon": "assets/dynres/icon.png",
|
"icon": "assets/dynres/icon.png",
|
||||||
"environment": "*",
|
"environment": "*",
|
||||||
"entrypoints": {
|
"entrypoints": {
|
||||||
"server": [
|
"libweb": [
|
||||||
"io.gitlab.jfronny.dynres.DynRes"
|
"io.gitlab.jfronny.dynres.DynRes"
|
||||||
],
|
],
|
||||||
"client": [
|
"client": [
|
||||||
|
@ -27,7 +27,6 @@
|
||||||
],
|
],
|
||||||
"depends": {
|
"depends": {
|
||||||
"fabricloader": ">=0.9.3+build.207",
|
"fabricloader": ">=0.9.3+build.207",
|
||||||
"fabric": "*",
|
"minecraft": "1.16.3"
|
||||||
"minecraft": "1.16.2"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue