2023-06-20 04:33:09 +00:00
|
|
|
#include "StarRootLoader.hpp"
|
|
|
|
#include "StarLexicalCast.hpp"
|
|
|
|
#include "StarJsonExtra.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
Json const BaseAssetsSettings = Json::parseJson(R"JSON(
|
|
|
|
{
|
|
|
|
"assetTimeToLive" : 30,
|
|
|
|
|
|
|
|
// In seconds, audio less than this long will be decompressed in memory.
|
|
|
|
"audioDecompressLimit" : 4.0,
|
|
|
|
|
|
|
|
"workerPoolSize" : 2,
|
|
|
|
|
|
|
|
"pathIgnore" : [
|
|
|
|
"/\\.",
|
|
|
|
"/~",
|
|
|
|
"thumbs\\.db$",
|
|
|
|
"\\.bak$",
|
|
|
|
"\\.tmp$",
|
|
|
|
"\\.zip$",
|
|
|
|
"\\.orig$",
|
|
|
|
"\\.fail$",
|
|
|
|
"\\.psd$",
|
|
|
|
"\\.tmx$"
|
|
|
|
],
|
|
|
|
|
|
|
|
"digestIgnore" : [
|
|
|
|
"\\.ogg$",
|
|
|
|
"\\.wav$",
|
|
|
|
"\\.abc$"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
)JSON");
|
|
|
|
|
|
|
|
Json const BaseDefaultConfiguration = Json::parseJson(R"JSON(
|
|
|
|
{
|
|
|
|
"configurationVersion" : {
|
|
|
|
"basic" : 2
|
|
|
|
},
|
|
|
|
|
|
|
|
"gameServerPort" : 21025,
|
2024-03-08 09:09:27 +00:00
|
|
|
)JSON"
|
|
|
|
#ifdef STAR_SYSTEM_WINDOWS
|
|
|
|
R"JSON(
|
|
|
|
"gameServerBind" : "*",
|
|
|
|
"queryServerBind" : "*",
|
|
|
|
"rconServerBind" : "*",
|
|
|
|
)JSON"
|
|
|
|
#else
|
|
|
|
R"JSON(
|
2023-06-20 04:33:09 +00:00
|
|
|
"gameServerBind" : "::",
|
2024-03-08 09:09:27 +00:00
|
|
|
)JSON"
|
|
|
|
#endif
|
|
|
|
R"JSON(
|
2023-06-20 04:33:09 +00:00
|
|
|
"serverUsers" : {},
|
|
|
|
"allowAnonymousConnections" : true,
|
|
|
|
|
|
|
|
"bannedUuids" : [],
|
|
|
|
"bannedIPs" : [],
|
|
|
|
|
|
|
|
"serverName" : "A Starbound Server",
|
|
|
|
"maxPlayers" : 8,
|
|
|
|
"maxTeamSize" : 4,
|
|
|
|
"serverFidelity" : "automatic",
|
|
|
|
|
|
|
|
"checkAssetsDigest" : false,
|
|
|
|
|
|
|
|
"safeScripts" : true,
|
|
|
|
"scriptRecursionLimit" : 100,
|
|
|
|
"scriptInstructionLimit" : 10000000,
|
|
|
|
"scriptProfilingEnabled" : false,
|
|
|
|
"scriptInstructionMeasureInterval" : 10000,
|
|
|
|
|
|
|
|
"allowAdminCommands" : true,
|
|
|
|
"allowAdminCommandsFromAnyone" : false,
|
|
|
|
"anonymousConnectionsAreAdmin" : false,
|
2024-07-27 04:09:12 +00:00
|
|
|
"connectionSettings" : {
|
|
|
|
"compression" : "Zstd"
|
|
|
|
},
|
2023-06-20 04:33:09 +00:00
|
|
|
|
|
|
|
"clientP2PJoinable" : true,
|
|
|
|
"clientIPJoinable" : false,
|
|
|
|
|
|
|
|
"clearUniverseFiles" : false,
|
|
|
|
"clearPlayerFiles" : false,
|
|
|
|
"playerBackupFileCount" : 3,
|
|
|
|
|
|
|
|
"tutorialMessages" : true,
|
|
|
|
|
|
|
|
"interactiveHighlight" : true,
|
|
|
|
|
|
|
|
"monochromeLighting" : false,
|
|
|
|
|
|
|
|
"crafting" : {
|
|
|
|
"filterHaveMaterials" : false
|
|
|
|
},
|
|
|
|
|
|
|
|
"inventory" : {
|
|
|
|
"pickupToActionBar" : true
|
2024-08-03 08:51:19 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
"discord" : {
|
|
|
|
"activityDetails" : "<playerName> | <worldName>"
|
2023-06-20 04:33:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)JSON");
|
|
|
|
|
|
|
|
RootLoader::RootLoader(Defaults defaults) {
|
|
|
|
String baseConfigFile;
|
|
|
|
Maybe<String> userConfigFile;
|
|
|
|
|
|
|
|
addParameter("bootconfig", "bootconfig", Optional,
|
|
|
|
strf("Boot time configuration file, defaults to sbinit.config"));
|
|
|
|
addParameter("logfile", "logfile", Optional,
|
2023-06-27 10:23:44 +00:00
|
|
|
strf("Log to the given logfile relative to the root directory, defaults to {}",
|
2023-06-20 04:33:09 +00:00
|
|
|
defaults.logFile ? *defaults.logFile : "no log file"));
|
|
|
|
addParameter("loglevel", "level", Optional,
|
2023-06-27 10:23:44 +00:00
|
|
|
strf("Sets the logging level (debug|info|warn|error), defaults to {}",
|
2023-06-20 04:33:09 +00:00
|
|
|
LogLevelNames.getRight(defaults.logLevel)));
|
2023-06-27 10:23:44 +00:00
|
|
|
addSwitch("quiet", strf("Do not log to stdout, defaults to {}", defaults.quiet));
|
|
|
|
addSwitch("verbose", strf("Log to stdout, defaults to {}", !defaults.quiet));
|
2023-06-20 04:33:09 +00:00
|
|
|
addSwitch("runtimeconfig",
|
2023-06-27 10:23:44 +00:00
|
|
|
strf("Sets the path to the runtime configuration storage file relative to root directory, defauts to {}",
|
2023-06-20 04:33:09 +00:00
|
|
|
defaults.runtimeConfigFile ? *defaults.runtimeConfigFile : "no storage file"));
|
2024-02-19 15:55:19 +00:00
|
|
|
m_defaults = std::move(defaults);
|
2023-06-20 04:33:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pair<Root::Settings, RootLoader::Options> RootLoader::parseOrDie(
|
|
|
|
StringList const& cmdLineArguments) const {
|
|
|
|
auto options = VersionOptionParser::parseOrDie(cmdLineArguments);
|
|
|
|
return {rootSettingsForOptions(options), options};
|
|
|
|
}
|
|
|
|
|
|
|
|
pair<RootUPtr, RootLoader::Options> RootLoader::initOrDie(StringList const& cmdLineArguments) const {
|
|
|
|
auto p = parseOrDie(cmdLineArguments);
|
|
|
|
auto root = make_unique<Root>(p.first);
|
2024-02-19 15:55:19 +00:00
|
|
|
return {std::move(root), p.second};
|
2023-06-20 04:33:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pair<Root::Settings, RootLoader::Options> RootLoader::commandParseOrDie(int argc, char** argv) {
|
|
|
|
auto options = VersionOptionParser::commandParseOrDie(argc, argv);
|
|
|
|
return {rootSettingsForOptions(options), options};
|
|
|
|
}
|
|
|
|
|
|
|
|
pair<RootUPtr, RootLoader::Options> RootLoader::commandInitOrDie(int argc, char** argv) {
|
|
|
|
auto p = commandParseOrDie(argc, argv);
|
|
|
|
auto root = make_unique<Root>(p.first);
|
2024-02-19 15:55:19 +00:00
|
|
|
return {std::move(root), p.second};
|
2023-06-20 04:33:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Root::Settings RootLoader::rootSettingsForOptions(Options const& options) const {
|
|
|
|
try {
|
|
|
|
String bootConfigFile = options.parameters.value("bootconfig").maybeFirst().value("sbinit.config");
|
|
|
|
Json bootConfig = Json::parseJson(File::readFileString(bootConfigFile));
|
|
|
|
|
|
|
|
Json assetsSettings = jsonMerge(
|
|
|
|
BaseAssetsSettings,
|
|
|
|
m_defaults.additionalAssetsSettings,
|
|
|
|
bootConfig.get("assetsSettings", {})
|
|
|
|
);
|
|
|
|
|
|
|
|
Root::Settings rootSettings;
|
|
|
|
rootSettings.assetsSettings.assetTimeToLive = assetsSettings.getInt("assetTimeToLive");
|
|
|
|
rootSettings.assetsSettings.audioDecompressLimit = assetsSettings.getFloat("audioDecompressLimit");
|
|
|
|
rootSettings.assetsSettings.workerPoolSize = assetsSettings.getUInt("workerPoolSize");
|
|
|
|
rootSettings.assetsSettings.missingImage = assetsSettings.optString("missingImage");
|
|
|
|
rootSettings.assetsSettings.missingAudio = assetsSettings.optString("missingAudio");
|
|
|
|
rootSettings.assetsSettings.pathIgnore = jsonToStringList(assetsSettings.get("pathIgnore"));
|
|
|
|
rootSettings.assetsSettings.digestIgnore = jsonToStringList(assetsSettings.get("digestIgnore"));
|
|
|
|
|
2024-03-08 09:09:27 +00:00
|
|
|
rootSettings.assetDirectories = jsonToStringList(bootConfig.get("assetDirectories", JsonArray()));
|
|
|
|
rootSettings.assetSources = jsonToStringList(bootConfig.get("assetSources", JsonArray()));
|
2023-11-02 21:07:29 +00:00
|
|
|
|
2023-06-20 04:33:09 +00:00
|
|
|
rootSettings.defaultConfiguration = jsonMerge(
|
2024-03-08 09:09:27 +00:00
|
|
|
BaseDefaultConfiguration,
|
2023-06-20 04:33:09 +00:00
|
|
|
m_defaults.additionalDefaultConfiguration,
|
|
|
|
bootConfig.get("defaultConfiguration", {})
|
|
|
|
);
|
|
|
|
|
|
|
|
rootSettings.storageDirectory = bootConfig.getString("storageDirectory");
|
2024-03-25 04:23:37 +00:00
|
|
|
rootSettings.logDirectory = bootConfig.optString("logDirectory");
|
2023-06-20 04:33:09 +00:00
|
|
|
rootSettings.logFile = options.parameters.value("logfile").maybeFirst().orMaybe(m_defaults.logFile);
|
2024-03-08 09:09:27 +00:00
|
|
|
rootSettings.logFileBackups = bootConfig.getUInt("logFileBackups", 10);
|
2024-06-27 05:49:41 +00:00
|
|
|
rootSettings.includeUGC = bootConfig.getBool("includeUGC", true);
|
2023-06-20 04:33:09 +00:00
|
|
|
|
|
|
|
if (auto ll = options.parameters.value("loglevel").maybeFirst())
|
|
|
|
rootSettings.logLevel = LogLevelNames.getLeft(*ll);
|
|
|
|
else
|
|
|
|
rootSettings.logLevel = m_defaults.logLevel;
|
|
|
|
|
|
|
|
if (options.switches.contains("quiet"))
|
|
|
|
rootSettings.quiet = true;
|
|
|
|
else if (options.switches.contains("verbose"))
|
|
|
|
rootSettings.quiet = false;
|
|
|
|
else
|
|
|
|
rootSettings.quiet = m_defaults.quiet;
|
|
|
|
|
|
|
|
if (auto rc = options.parameters.value("runtimeconfig").maybeFirst())
|
|
|
|
rootSettings.runtimeConfigFile = *rc;
|
|
|
|
else
|
|
|
|
rootSettings.runtimeConfigFile = m_defaults.runtimeConfigFile;
|
|
|
|
|
|
|
|
return rootSettings;
|
|
|
|
|
|
|
|
} catch (std::exception const& e) {
|
|
|
|
throw StarException("Could not perform initial Root load", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|