266 lines
9.8 KiB
Java
Raw Normal View History

2018-01-24 12:18:06 +01:00
package net.knarcraft.serverlauncher.server;
2018-01-25 17:34:45 +01:00
import java.io.*;
import java.net.URL;
import java.util.Scanner;
import java.nio.file.*;
import java.util.ArrayList;
2018-01-25 21:17:02 +01:00
/**
* Contains all necessary information to create, run and manage a Minecraft server.
*
2018-01-25 21:21:06 +01:00
* @author Kristian Knarvik <kristian.knarvik@knett.no>
* @version 0.0.0.1
* @since 0.0.0.1
2018-01-25 21:17:02 +01:00
*/
public class Server {
2018-01-26 20:26:16 +01:00
/** Available ram sizes. For GUI dropdown */
private static final String[] ramList = {"512M", "1G", "2G", "3G", "4G", "5G", "6G", "7G", "8G", "9G", "10G", "11G", "12G", "13G", "14G", "15G", "16G"};
2018-01-26 20:34:15 +01:00
private static final ArrayList<Server> servers = new ArrayList<>();
private String name;
private String path;
private boolean enabled;
2018-01-26 20:34:15 +01:00
private final ArrayList<String> playerList;
2018-01-25 19:55:28 +01:00
private ServerType type;
private String serverVersion;
private String maxRam;
private Process process;
public Server(String name) {
this.name = name;
this.path = "";
this.enabled = false;
this.playerList = new ArrayList<>();
this.type = null;
this.serverVersion = null;
this.maxRam = ramList[0];
this.process = null;
2018-01-25 21:17:02 +01:00
servers.add(this);
}
public void addPlayer(String name) {
this.playerList.add(name);
}
public void removePlayer(String name) {
for (int i = 0; i < playerList.size(); i++) {
if (name.equals(playerList.get(i))) {
playerList.remove(i);
return;
}
}
}
2018-01-26 20:26:16 +01:00
/**
* @return A representation of the name of a jarfile.
*/
2018-01-25 21:21:06 +01:00
private String getType() {
return this.type.getName() + this.serverVersion + ".jar";
}
2018-01-25 21:17:02 +01:00
public static ArrayList<Server> getServers() {
return servers;
}
public Process getProcess() {
return this.process;
}
public void toggle() {
this.enabled = !this.enabled;
}
public void setPath(String path) {
this.path = path;
}
2018-01-25 19:55:28 +01:00
public void setType(ServerType type) {
this.type = type;
}
2018-01-25 17:34:45 +01:00
/**
* Sets the server's server version to a valid version, or ignores the request.
*
2018-01-25 21:17:02 +01:00
* @param serverVersion Version number.
2018-01-25 17:34:45 +01:00
*/
2018-01-25 21:17:02 +01:00
public void setServerVersion(String serverVersion) throws IllegalArgumentException {
2018-01-25 11:49:01 +01:00
String[] versions = this.type.getVersions();
for (String version : versions) {
if (version.equals(serverVersion)) {
this.serverVersion = serverVersion;
2018-01-25 21:17:02 +01:00
return;
2018-01-25 11:49:01 +01:00
}
}
2018-01-25 21:17:02 +01:00
throw new IllegalArgumentException("Invalid server version.");
}
public void setMaxRam(String ram) {
this.maxRam = ram;
}
2018-01-25 21:17:02 +01:00
/**
* Runs all enabled servers with their settings.
*/
public static void startServers() {
System.out.println("Starting servers.");
for (Server server : servers) {
server.run();
}
}
2018-01-25 17:34:45 +01:00
/**
* Runs the Minecraft server.
*/
2018-01-25 21:17:02 +01:00
private void run() {
if (this.enabled) {
2018-01-25 17:34:45 +01:00
try {
2018-01-25 21:17:02 +01:00
System.out.println("Downloading jar...");
2018-01-25 17:34:45 +01:00
this.downloadJar();
System.out.println("File downloaded.");
} catch (FileNotFoundException e) {
2018-01-25 21:17:02 +01:00
e.printStackTrace();
2018-01-25 17:34:45 +01:00
System.out.println("File was not found.");
return;
}
try {
ProcessBuilder builder = new ProcessBuilder("java", "-Xmx" + this.maxRam, "-Xms512M", "-Djline.terminal=jline.UnsupportedTerminal", "-Dcom.mojang.eula.agree=true", "-jar", "\"" + this.path + "\\" + this.getType() + "\"");
builder.directory(new File(this.path));
builder.redirectErrorStream(true);
2018-01-26 20:34:15 +01:00
this.process = builder.start();
2018-01-25 17:34:45 +01:00
System.out.println("Success");
} catch (IOException e) {
System.out.println("Error");
}
}
}
/**
* Downloads necessary .jar file for the server.
* This is unfortunately hardcoded since there is no golden standard, and we only host some jars ourselves.
*
* @throws FileNotFoundException if the file was not found and could not be acquired.
*/
2018-01-25 21:21:06 +01:00
private void downloadJar() throws FileNotFoundException {
2018-01-25 19:55:28 +01:00
AdvancedServerType type;
File file = new File(this.path + "\\" + this.getType());
Path filePath = Paths.get(this.path + "\\" + this.getType());
String versionText;
String newestVersion;
boolean success;
switch (this.type.getName()) {
case "Custom":
2018-01-25 21:17:02 +01:00
//TODO: Change Custom to use path as serverdir + input.
if (!file.isFile()) {
throw new FileNotFoundException("Specified custom jar was not found.");
}
break;
case "Spigot":
case "Craftbukkit":
case "MCPCplus":
if (!file.isFile()) {
2018-01-25 19:55:28 +01:00
success = downloadFile(this.type.getDownloadURL() + this.type.getName() + this.serverVersion + ".jar", filePath);
2018-01-25 21:17:02 +01:00
System.out.println(this.type.getDownloadURL() + this.type.getName() + this.serverVersion + ".jar");
if (!success) {
throw new FileNotFoundException("Jar file could not be downloaded.");
}
}
break;
case "Vanilla":
case "Snapshot":
2018-01-25 19:55:28 +01:00
type = (AdvancedServerType) this.type;
if (this.serverVersion.equals("Latest")) {
try {
2018-01-25 19:55:28 +01:00
versionText = readFile(type.getVersionURL());
} catch (IOException e) {
throw new FileNotFoundException("Version file could not be downloaded.");
}
newestVersion = stringBetween(versionText, type.getSrcStart(), type.getSrcEnd());
2018-01-25 19:55:28 +01:00
success = downloadFile(this.type.getDownloadURL() + newestVersion + type.getDownloadURLPart() + newestVersion + ".jar", filePath);
//TODO: Register the new version number, and only download if version mismatch or missing file.
if (!success) {
throw new FileNotFoundException("Jar file could not be downloaded.");
}
} else {
if (!file.isFile()) {
2018-01-25 19:55:28 +01:00
success = downloadFile(this.type.getDownloadURL() + this.serverVersion + type.getDownloadURLPart() + this.serverVersion + ".jar", filePath);
if (!success) {
throw new FileNotFoundException("Jar file could not be downloaded.");
}
}
}
break;
2018-01-25 17:34:45 +01:00
case "SpongeVanilla":
2018-01-25 19:55:28 +01:00
type = (AdvancedServerType) this.type;
try {
2018-01-25 19:55:28 +01:00
versionText = readFile(type.getVersionURL() + this.serverVersion);
} catch (IOException e) {
throw new FileNotFoundException("Version file could not be downloaded.");
}
newestVersion = stringBetween(versionText, type.getSrcStart(), type.getSrcEnd());
2018-01-25 19:55:28 +01:00
success = downloadFile(this.type.getDownloadURL() + newestVersion + type.getDownloadURLPart() + newestVersion + ".jar", filePath);
//TODO: Register the new version number, and only download if version mismatch or missing file.
if (!success) {
throw new FileNotFoundException("Jar file could not be downloaded.");
}
break;
case "Bungee":
2018-01-25 19:55:28 +01:00
/*type = (AdvancedServerType) this.type;
try {
versionText = readFile(this.type.getVersionURL());
} catch (IOException e) {
throw new FileNotFoundException("Version file could not be downloaded.");
}
newestVersion = stringBetween(versionText, type.getSrcStart(), type.getSrcEnd());*/
2018-01-25 19:55:28 +01:00
success = downloadFile(this.type.getDownloadURL(), filePath);
//TODO: Register the new version number, and only download if version mismatch or missing file.
if (!success) {
throw new FileNotFoundException("Jar file could not be downloaded.");
}
}
}
/**
* Reads a file from a website.
* This is used to find the newest version of the software.
*
* @param path The full url of the file to read.
2018-01-26 20:34:15 +01:00
* @return True if successful. False otherwise.
*/
private static String readFile(String path) throws IOException {
URL url = new URL(path);
return new Scanner(url.openStream()).useDelimiter("\\Z").next();
}
/**
* Downloads a file from a website.
*
* @param path The full url of the file to download.
* @param outfile The file to save to.
* @return True if successful. False otherwise.
*/
private static boolean downloadFile(String path, Path outfile) {
try {
URL url = new URL(path);
InputStream in = url.openStream();
Files.copy(in, outfile, StandardCopyOption.REPLACE_EXISTING);
return true;
} catch (IOException e) {
return false;
}
}
/**
* Finds a substring between two substrings in a string.
*
* @param string The string containing the substrings.
* @param start The substring before the wanted substring.
* @param end The substring after the wanted substring.
* @return The wanted substring.
*/
private String stringBetween(String string, String start, String end) {
int startPos = string.indexOf(start) + start.length();
return string.substring(startPos, string.indexOf(end, startPos));
}
}