2022-04-17 20:43:07 +08:00
|
|
|
package emu.grasscutter;
|
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.FileReader;
|
|
|
|
import java.io.FileWriter;
|
2022-05-03 20:44:52 +08:00
|
|
|
import java.io.IOError;
|
2022-04-17 20:43:07 +08:00
|
|
|
import java.net.InetSocketAddress;
|
2022-04-30 05:29:34 +08:00
|
|
|
import java.util.Calendar;
|
2022-04-17 20:43:07 +08:00
|
|
|
|
2022-04-20 20:21:38 +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;
|
2022-04-29 13:19:14 +08:00
|
|
|
import emu.grasscutter.scripts.ScriptLoader;
|
2022-04-18 13:11:27 +08:00
|
|
|
import emu.grasscutter.utils.Utils;
|
2022-05-03 20:44:52 +08:00
|
|
|
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;
|
|
|
|
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;
|
2022-05-03 20:44:52 +08:00
|
|
|
private static LineReader consoleLineReader = null;
|
2022-05-03 14:23:25 +08:00
|
|
|
private static Language language;
|
2022-05-03 22:51:12 +08:00
|
|
|
private static CNLanguage cn_language;
|
2022-05-03 20:44:52 +08:00
|
|
|
|
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-04-17 20:43:07 +08:00
|
|
|
|
2022-04-30 05:29:34 +08:00
|
|
|
private static int day; // Current day of week
|
|
|
|
|
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-04-17 20:43:07 +08:00
|
|
|
|
2022-04-26 12:39:05 +08:00
|
|
|
public static final Reflections reflector = new Reflections("emu.grasscutter");
|
2022-04-19 12:35:01 +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");
|
|
|
|
|
|
|
|
// Load server configuration.
|
2022-04-18 13:11:27 +08:00
|
|
|
Grasscutter.loadConfig();
|
2022-05-03 14:23:25 +08:00
|
|
|
|
|
|
|
// Load Language
|
|
|
|
Grasscutter.loadLanguage();
|
2022-04-19 17:22:21 +08:00
|
|
|
|
2022-04-18 13:11:27 +08:00
|
|
|
// Check server structure.
|
|
|
|
Utils.startupCheck();
|
|
|
|
}
|
|
|
|
|
2022-04-17 20:43:07 +08:00
|
|
|
public static void main(String[] args) throws Exception {
|
|
|
|
Crypto.loadKeys();
|
|
|
|
|
|
|
|
for (String arg : args) {
|
|
|
|
switch (arg.toLowerCase()) {
|
2022-04-23 13:17:35 +08:00
|
|
|
case "-handbook" -> {
|
|
|
|
Tools.createGmHandbook(); return;
|
|
|
|
}
|
2022-05-02 14:17:18 +08:00
|
|
|
case "-gachamap" -> {
|
|
|
|
Tools.createGachaMapping(); return;
|
|
|
|
}
|
2022-04-17 20:43:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-18 13:11:27 +08:00
|
|
|
// Initialize server.
|
2022-05-03 14:23:25 +08:00
|
|
|
Grasscutter.getLogger().info(language.Starting_Grasscutter);
|
2022-04-17 20:43:07 +08:00
|
|
|
|
2022-04-18 13:11:27 +08:00
|
|
|
// Load all resources.
|
2022-04-30 05:29:34 +08:00
|
|
|
Grasscutter.updateDayOfWeek();
|
2022-04-17 20:43:07 +08:00
|
|
|
ResourceLoader.loadAll();
|
2022-04-29 13:19:14 +08:00
|
|
|
ScriptLoader.init();
|
2022-04-30 05:29:34 +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-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-04-23 13:17:35 +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) {
|
2022-04-21 18:04:00 +08:00
|
|
|
dispatchServer.start();
|
|
|
|
gameServer.start();
|
2022-05-01 13:52:09 +08:00
|
|
|
} else if (getConfig().RunMode == ServerRunMode.DISPATCH_ONLY) {
|
2022-04-21 18:04:00 +08:00
|
|
|
dispatchServer.start();
|
2022-05-01 13:52:09 +08:00
|
|
|
} else if (getConfig().RunMode == ServerRunMode.GAME_ONLY) {
|
2022-04-21 18:04:00 +08:00
|
|
|
gameServer.start();
|
|
|
|
} else {
|
2022-05-03 14:23:25 +08:00
|
|
|
getLogger().error(language.Invalid_server_run_mode + " " + getConfig().RunMode);
|
|
|
|
getLogger().error(language.Server_run_mode);
|
|
|
|
getLogger().error(language.Shutting_down);
|
2022-04-21 18:04:00 +08:00
|
|
|
System.exit(1);
|
|
|
|
}
|
2022-04-23 13:17:35 +08:00
|
|
|
|
|
|
|
// Enable all plugins.
|
|
|
|
pluginManager.enablePlugins();
|
2022-05-03 20:44:52 +08:00
|
|
|
|
2022-04-23 13:17:35 +08:00
|
|
|
// Hook into shutdown event.
|
|
|
|
Runtime.getRuntime().addShutdownHook(new Thread(Grasscutter::onShutdown));
|
2022-05-03 20:44:52 +08:00
|
|
|
|
|
|
|
// 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-05-03 20:44:52 +08:00
|
|
|
|
2022-04-17 20:43:07 +08:00
|
|
|
public static void loadConfig() {
|
|
|
|
try (FileReader file = new FileReader(configFile)) {
|
|
|
|
config = gson.fromJson(file, Config.class);
|
2022-04-22 05:44:55 +08:00
|
|
|
saveConfig();
|
2022-04-17 20:43:07 +08:00
|
|
|
} catch (Exception e) {
|
2022-04-22 05:44:55 +08:00
|
|
|
Grasscutter.config = new Config();
|
|
|
|
saveConfig();
|
2022-04-17 20:43:07 +08:00
|
|
|
}
|
|
|
|
}
|
2022-05-03 14:23:25 +08:00
|
|
|
|
|
|
|
public static void loadLanguage() {
|
2022-05-03 18:19:43 +08:00
|
|
|
try (FileReader file = new FileReader(String.format(getConfig().LANGUAGE_FOLDER + "%s.json", Grasscutter.config.Language))) {
|
2022-05-03 14:23:25 +08:00
|
|
|
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-03 14:23:25 +08:00
|
|
|
Grasscutter.config.Language = "en_us";
|
|
|
|
saveConfig();
|
|
|
|
|
|
|
|
try {
|
|
|
|
File folder = new File("./language");
|
|
|
|
if (!folder.exists() && !folder.isDirectory()) {
|
|
|
|
//noinspection ResultOfMethodCallIgnored
|
|
|
|
folder.mkdirs();
|
|
|
|
}
|
|
|
|
} catch (Exception ee) {
|
|
|
|
Grasscutter.getLogger().error("Unable to create language folder.");
|
|
|
|
}
|
|
|
|
try (FileWriter file = new FileWriter("./language/en_us.json")) {
|
|
|
|
file.write(gson.toJson(language));
|
|
|
|
} catch (Exception ee) {
|
|
|
|
Grasscutter.getLogger().error("Unable to create language file.");
|
|
|
|
}
|
2022-05-03 22:51:12 +08:00
|
|
|
try (FileWriter file = new FileWriter("./languages/zh_cn.json")) {
|
|
|
|
file.write(gson.toJson(cn_language));
|
|
|
|
} catch (Exception ee) {
|
|
|
|
Grasscutter.getLogger().error("无法创建中文语言文件。");
|
|
|
|
}
|
2022-05-03 14:23:25 +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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void startConsole() {
|
2022-05-03 20:44:52 +08:00
|
|
|
// Console should not start in dispatch only mode.
|
|
|
|
if (getConfig().RunMode == ServerRunMode.DISPATCH_ONLY) {
|
|
|
|
getLogger().info(language.Dispatch_mode_not_support_command);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-05-03 14:23:25 +08:00
|
|
|
getLogger().info(language.Start_done);
|
2022-05-03 20:44:52 +08:00
|
|
|
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
|
|
|
}
|
2022-05-03 20:44:52 +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 {
|
|
|
|
CommandMap.getInstance().invoke(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;
|
|
|
|
}
|
|
|
|
|
2022-05-03 14:23:25 +08:00
|
|
|
public static Language getLanguage() {
|
|
|
|
return language;
|
|
|
|
}
|
|
|
|
|
2022-04-18 13:11:27 +08:00
|
|
|
public static Logger getLogger() {
|
|
|
|
return log;
|
|
|
|
}
|
|
|
|
|
2022-05-03 20:44:52 +08:00
|
|
|
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-04-23 13:17:35 +08:00
|
|
|
|
|
|
|
public static PluginManager getPluginManager() {
|
|
|
|
return pluginManager;
|
|
|
|
}
|
2022-04-30 05:29:34 +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-01 13:52:09 +08:00
|
|
|
|
|
|
|
public enum ServerRunMode {
|
|
|
|
HYBRID, DISPATCH_ONLY, GAME_ONLY
|
|
|
|
}
|
|
|
|
|
|
|
|
public enum ServerDebugMode {
|
|
|
|
ALL, MISSING, NONE
|
|
|
|
}
|
2022-04-17 20:43:07 +08:00
|
|
|
}
|