Grasscutter/src/main/java/emu/grasscutter/Grasscutter.java

296 lines
8.2 KiB
Java
Raw Normal View History

2022-04-17 20:43:07 +08:00
package emu.grasscutter;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOError;
2022-04-17 20:43:07 +08:00
import java.net.InetSocketAddress;
import java.util.Calendar;
2022-05-05 13:24:49 +08:00
import java.util.Locale;
2022-04-17 20:43:07 +08:00
import emu.grasscutter.command.CommandMap;
2022-04-23 13:17:35 +08:00
import emu.grasscutter.plugin.PluginManager;
2022-05-03 09:20:24 +08:00
import emu.grasscutter.plugin.api.ServerHook;
import emu.grasscutter.scripts.ScriptLoader;
2022-04-18 13:11:27 +08:00
import emu.grasscutter.utils.Utils;
import org.jline.reader.EndOfFileException;
import org.jline.reader.LineReader;
import org.jline.reader.LineReaderBuilder;
import org.jline.reader.UserInterruptException;
import org.jline.terminal.Terminal;
import org.jline.terminal.TerminalBuilder;
2022-04-19 12:35:01 +08:00
import org.reflections.Reflections;
2022-04-17 20:43:07 +08:00
import org.slf4j.LoggerFactory;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import ch.qos.logback.classic.Logger;
import emu.grasscutter.data.ResourceLoader;
import emu.grasscutter.database.DatabaseManager;
2022-05-04 15:44:33 +08:00
import emu.grasscutter.languages.CNLanguage;
import emu.grasscutter.languages.Language;
2022-04-17 20:43:07 +08:00
import emu.grasscutter.server.dispatch.DispatchServer;
import emu.grasscutter.server.game.GameServer;
import emu.grasscutter.tools.Tools;
import emu.grasscutter.utils.Crypto;
2022-04-18 13:11:27 +08:00
public final class Grasscutter {
private static final Logger log = (Logger) LoggerFactory.getLogger(Grasscutter.class);
2022-04-17 20:43:07 +08:00
private static Config config;
private static LineReader consoleLineReader = null;
private static Language language;
2022-05-03 22:51:12 +08:00
private static CNLanguage cn_language;
2022-04-18 13:11:27 +08:00
private static final Gson gson = new GsonBuilder().setPrettyPrinting().create();
private static final File configFile = new File("./config.json");
2022-05-05 13:24:49 +08:00
private static int day; // Current day of week
2022-05-05 13:24:49 +08:00
2022-04-17 20:43:07 +08:00
private static DispatchServer dispatchServer;
private static GameServer gameServer;
2022-04-23 13:17:35 +08:00
private static PluginManager pluginManager;
2022-05-05 13:24:49 +08:00
2022-04-26 12:39:05 +08:00
public static final Reflections reflector = new Reflections("emu.grasscutter");
2022-05-05 13:24:49 +08:00
2022-04-18 13:11:27 +08:00
static {
2022-04-19 12:35:01 +08:00
// Declare logback configuration.
System.setProperty("logback.configurationFile", "src/main/resources/logback.xml");
2022-05-05 13:24:49 +08:00
2022-04-19 12:35:01 +08:00
// Load server configuration.
2022-04-18 13:11:27 +08:00
Grasscutter.loadConfig();
// Load Language
Grasscutter.loadLanguage();
2022-05-05 13:24:49 +08:00
2022-04-18 13:11:27 +08:00
// Check server structure.
Utils.startupCheck();
}
2022-05-05 13:24:49 +08:00
2022-04-17 20:43:07 +08:00
public static void main(String[] args) throws Exception {
Crypto.loadKeys();
2022-05-05 13:24:49 +08:00
2022-04-17 20:43:07 +08:00
for (String arg : args) {
switch (arg.toLowerCase()) {
2022-04-23 13:17:35 +08:00
case "-handbook" -> {
Tools.createGmHandbook(); return;
}
case "-gachamap" -> {
Tools.createGachaMapping("./gacha_mappings.js"); return;
}
2022-04-17 20:43:07 +08:00
}
}
2022-05-05 13:24:49 +08:00
2022-04-18 13:11:27 +08:00
// Initialize server.
Grasscutter.getLogger().info(language.Starting_Grasscutter);
2022-05-05 13:24:49 +08:00
2022-04-18 13:11:27 +08:00
// Load all resources.
Grasscutter.updateDayOfWeek();
2022-04-17 20:43:07 +08:00
ResourceLoader.loadAll();
ScriptLoader.init();
2022-05-05 13:24:49 +08:00
2022-04-17 20:43:07 +08:00
// Database
DatabaseManager.initialize();
2022-04-23 13:17:35 +08:00
2022-04-26 14:07:00 +08:00
// Create plugin manager instance.
pluginManager = new PluginManager();
2022-05-05 13:24:49 +08:00
2022-04-23 13:17:35 +08:00
// Create server instances.
dispatchServer = new DispatchServer();
gameServer = new GameServer(new InetSocketAddress(getConfig().getGameServerOptions().Ip, getConfig().getGameServerOptions().Port));
2022-05-03 09:20:24 +08:00
// Create a server hook instance with both servers.
new ServerHook(gameServer, dispatchServer);
2022-05-05 13:24:49 +08:00
2022-04-18 13:11:27 +08:00
// Start servers.
2022-05-01 13:52:09 +08:00
if (getConfig().RunMode == ServerRunMode.HYBRID) {
dispatchServer.start();
gameServer.start();
2022-05-01 13:52:09 +08:00
} else if (getConfig().RunMode == ServerRunMode.DISPATCH_ONLY) {
dispatchServer.start();
2022-05-01 13:52:09 +08:00
} else if (getConfig().RunMode == ServerRunMode.GAME_ONLY) {
gameServer.start();
} else {
getLogger().error(language.Invalid_server_run_mode + " " + getConfig().RunMode);
getLogger().error(language.Server_run_mode);
getLogger().error(language.Shutting_down);
System.exit(1);
}
2022-05-05 13:24:49 +08:00
2022-04-23 13:17:35 +08:00
// Enable all plugins.
pluginManager.enablePlugins();
2022-04-23 13:17:35 +08:00
// Hook into shutdown event.
Runtime.getRuntime().addShutdownHook(new Thread(Grasscutter::onShutdown));
// Open console.
startConsole();
2022-04-17 20:43:07 +08:00
}
2022-04-23 13:17:35 +08:00
/**
* Server shutdown event.
*/
private static void onShutdown() {
// Disable all plugins.
pluginManager.disablePlugins();
}
2022-04-17 20:43:07 +08:00
public static void loadConfig() {
try (FileReader file = new FileReader(configFile)) {
config = gson.fromJson(file, Config.class);
saveConfig();
2022-04-17 20:43:07 +08:00
} catch (Exception e) {
Grasscutter.config = new Config();
saveConfig();
2022-04-17 20:43:07 +08:00
}
}
public static void loadLanguage() {
2022-05-05 13:24:49 +08:00
try (FileReader file = new FileReader(String.format("%s%s.json", getConfig().LANGUAGE_FOLDER, Grasscutter.config.LocaleLanguage))) {
language = gson.fromJson(file, Language.class);
} catch (Exception e) {
Grasscutter.language = new Language();
2022-05-03 22:51:12 +08:00
Grasscutter.cn_language = new CNLanguage();
2022-05-05 13:24:49 +08:00
Grasscutter.config.LocaleLanguage = Locale.getDefault();
saveConfig();
try {
2022-05-03 23:11:28 +08:00
File folder = new File("./languages");
if (!folder.exists() && !folder.isDirectory()) {
//noinspection ResultOfMethodCallIgnored
folder.mkdirs();
}
} catch (Exception ee) {
Grasscutter.getLogger().error("Unable to create language folder.");
}
2022-05-05 13:24:49 +08:00
try (FileWriter file = new FileWriter("./languages/" + Locale.US + ".json")) {
file.write(gson.toJson(language));
} catch (Exception ee) {
Grasscutter.getLogger().error("Unable to create language file.");
}
2022-05-05 13:24:49 +08:00
try (FileWriter file = new FileWriter("./languages/" + Locale.SIMPLIFIED_CHINESE + ".json")) {
2022-05-03 22:51:12 +08:00
file.write(gson.toJson(cn_language));
} catch (Exception ee) {
2022-05-05 13:24:49 +08:00
Grasscutter.getLogger().error("无法创建简体中文语言文件。");
}
// try again
try (FileReader file = new FileReader(String.format("%s%s.json", getConfig().LANGUAGE_FOLDER, Grasscutter.config.LocaleLanguage))) {
language = gson.fromJson(file, Language.class);
} catch (Exception ee) {
Grasscutter.getLogger().error("Unable to load " + Grasscutter.config.LocaleLanguage + ".json");
2022-05-03 22:51:12 +08:00
}
}
}
2022-05-05 13:24:49 +08:00
2022-04-17 20:43:07 +08:00
public static void saveConfig() {
try (FileWriter file = new FileWriter(configFile)) {
file.write(gson.toJson(config));
} catch (Exception e) {
2022-04-23 13:17:35 +08:00
Grasscutter.getLogger().error("Unable to save config file.");
2022-04-17 20:43:07 +08:00
}
}
2022-05-05 13:24:49 +08:00
2022-04-17 20:43:07 +08:00
public static void startConsole() {
// Console should not start in dispatch only mode.
if (getConfig().RunMode == ServerRunMode.DISPATCH_ONLY) {
getLogger().info(language.Dispatch_mode_not_support_command);
return;
}
getLogger().info(language.Start_done);
String input = null;
boolean isLastInterrupted = false;
while (true) {
try {
input = consoleLineReader.readLine("> ");
} catch (UserInterruptException e) {
if (!isLastInterrupted) {
isLastInterrupted = true;
Grasscutter.getLogger().info("Press Ctrl-C again to shutdown.");
continue;
} else {
Runtime.getRuntime().exit(0);
2022-04-19 12:35:01 +08:00
}
} catch (EndOfFileException e) {
Grasscutter.getLogger().info("EOF detected.");
continue;
} catch (IOError e) {
Grasscutter.getLogger().error("An IO error occurred.", e);
continue;
}
isLastInterrupted = false;
try {
2022-05-04 14:32:09 +08:00
CommandMap.getInstance().invoke(null, null, input);
} catch (Exception e) {
Grasscutter.getLogger().error(language.Command_error, e);
2022-04-17 20:43:07 +08:00
}
}
}
2022-04-18 13:11:27 +08:00
public static Config getConfig() {
return config;
}
public static Language getLanguage() {
return language;
}
2022-04-18 13:11:27 +08:00
public static Logger getLogger() {
return log;
}
public static LineReader getConsole() {
if (consoleLineReader == null) {
Terminal terminal = null;
try {
terminal = TerminalBuilder.builder().jna(true).build();
} catch (Exception e) {
try {
// Fallback to a dumb jline terminal.
terminal = TerminalBuilder.builder().dumb(true).build();
} catch (Exception ignored) {
// When dumb is true, build() never throws.
}
}
consoleLineReader = LineReaderBuilder.builder()
.terminal(terminal)
.build();
}
return consoleLineReader;
}
2022-04-18 13:11:27 +08:00
public static Gson getGsonFactory() {
return gson;
}
public static DispatchServer getDispatchServer() {
return dispatchServer;
}
public static GameServer getGameServer() {
return gameServer;
}
2022-05-05 13:24:49 +08:00
2022-04-23 13:17:35 +08:00
public static PluginManager getPluginManager() {
return pluginManager;
}
2022-05-05 13:24:49 +08:00
public static void updateDayOfWeek() {
Calendar calendar = Calendar.getInstance();
day = calendar.get(Calendar.DAY_OF_WEEK);
}
public static int getCurrentDayOfWeek() {
return day;
}
2022-05-05 13:24:49 +08:00
2022-05-01 13:52:09 +08:00
public enum ServerRunMode {
HYBRID, DISPATCH_ONLY, GAME_ONLY
}
2022-05-05 13:24:49 +08:00
2022-05-01 13:52:09 +08:00
public enum ServerDebugMode {
ALL, MISSING, NONE
}
2022-04-17 20:43:07 +08:00
}