1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-19 22:53:50 +00:00
openmw-tes3mp/apps/openmw-mp/main.cpp

334 lines
11 KiB
C++
Raw Normal View History

#include <iostream>
#include <boost/filesystem/fstream.hpp>
2016-08-17 23:58:09 +00:00
#include <boost/iostreams/concepts.hpp>
#include <boost/iostreams/stream_buffer.hpp>
#include <components/files/configurationmanager.hpp>
#include <components/files/escape.hpp>
#include <components/settings/settings.hpp>
#include <components/version/version.hpp>
#include <components/openmw-mp/ErrorMessages.hpp>
#include <components/openmw-mp/TimedLog.hpp>
#include <components/openmw-mp/NetworkMessages.hpp>
#include <components/openmw-mp/Utils.hpp>
#include <components/openmw-mp/Version.hpp>
#include <BitStream.h>
#include <MessageIdentifiers.h>
#include <RakPeer.h>
#include <RakPeerInterface.h>
#include "Player.hpp"
#include "Networking.hpp"
#include "MasterClient.hpp"
#include "Utils.hpp"
#include <apps/openmw-mp/Script/Script.hpp>
2017-05-28 10:26:38 +00:00
2016-10-23 09:07:46 +00:00
#ifdef ENABLE_BREAKPAD
#include <handler/exception_handler.h>
#endif
using namespace mwmp;
2016-10-23 09:07:46 +00:00
#ifdef ENABLE_BREAKPAD
google_breakpad::ExceptionHandler *pHandler = 0;
#if defined(_WIN32)
bool DumpCallback(const wchar_t* _dump_dir,const wchar_t* _minidump_id,void* context,EXCEPTION_POINTERS* exinfo,MDRawAssertionInfo* assertion,bool success)
2016-10-23 09:07:46 +00:00
#elif defined(__linux)
bool DumpCallback(const google_breakpad::MinidumpDescriptor &md, void *context, bool success)
#endif
{
// NO STACK USE, NO HEAP USE THERE !!!
return success;
}
void breakpad(std::string pathToDump)
{
#ifdef _WIN32
pHandler = new google_breakpad::ExceptionHandler(
L"crashdumps\\",
2016-10-23 09:07:46 +00:00
/*FilterCallback*/ 0,
DumpCallback,
0,
google_breakpad::ExceptionHandler::HANDLER_ALL);
2016-10-23 09:07:46 +00:00
#else
google_breakpad::MinidumpDescriptor md(pathToDump);
pHandler = new google_breakpad::ExceptionHandler(
md,
/*FilterCallback*/ 0,
DumpCallback,
/*context*/ 0,
true,
-1
);
#endif
}
void breakpad_close()
{
delete pHandler;
}
#else
void breakpad(std::string pathToDump){}
void breakpad_close(){}
#endif
std::string loadSettings (Settings::Manager & settings)
{
Files::ConfigurationManager mCfgMgr;
// Create the settings manager and load default settings file
const std::string localdefault = (mCfgMgr.getLocalPath() / "tes3mp-server-default.cfg").string();
const std::string globaldefault = (mCfgMgr.getGlobalPath() / "tes3mp-server-default.cfg").string();
// prefer local
if (boost::filesystem::exists(localdefault))
settings.loadDefault(localdefault);
else if (boost::filesystem::exists(globaldefault))
settings.loadDefault(globaldefault);
else
throw std::runtime_error ("No default settings file found! Make sure the file \"tes3mp-server-default.cfg\" was properly installed.");
// load user settings if they exist
const std::string settingspath = (mCfgMgr.getUserConfigPath() / "tes3mp-server.cfg").string();
if (boost::filesystem::exists(settingspath))
settings.loadUser(settingspath);
return settingspath;
}
2016-08-17 23:58:09 +00:00
class Tee : public boost::iostreams::sink
{
public:
Tee(std::ostream &stream, std::ostream &stream2)
: out(stream), out2(stream2)
{
}
std::streamsize write(const char *str, std::streamsize size)
{
out.write (str, size);
out.flush();
out2.write (str, size);
out2.flush();
return size;
}
private:
std::ostream &out;
std::ostream &out2;
};
2017-05-28 11:10:16 +00:00
boost::program_options::variables_map launchOptions(int argc, char *argv[], Files::ConfigurationManager cfgMgr)
2017-05-28 10:26:38 +00:00
{
namespace bpo = boost::program_options;
bpo::variables_map variables;
bpo::options_description desc;
desc.add_options()
2017-05-28 11:11:40 +00:00
("resources", bpo::value<Files::EscapeHashString>()->default_value("resources"), "set resources directory")
("no-logs", bpo::value<bool>()->implicit_value(true)->default_value(false),
"Do not write logs. Useful for daemonizing.");
2017-05-28 10:26:38 +00:00
cfgMgr.readConfiguration(variables, desc, true);
2017-05-28 11:10:16 +00:00
bpo::parsed_options valid_opts = bpo::command_line_parser(argc, argv).options(desc).allow_unregistered().run();
bpo::store(valid_opts, variables);
bpo::notify(variables);
2017-05-28 10:26:38 +00:00
return variables;
}
int main(int argc, char *argv[])
{
Settings::Manager mgr;
2016-08-17 23:58:09 +00:00
Files::ConfigurationManager cfgMgr;
2016-11-03 16:23:23 +00:00
breakpad(boost::filesystem::path(cfgMgr.getLogPath()).string());
2016-10-23 09:07:46 +00:00
loadSettings(mgr);
2017-05-28 11:10:16 +00:00
auto variables = launchOptions(argc, argv, cfgMgr);
2017-05-28 10:26:38 +00:00
auto version = Version::getOpenmwVersion(variables["resources"].as<Files::EscapeHashString>().toStdString());
int logLevel = mgr.getInt("logLevel", "General");
if (logLevel < TimedLog::LOG_VERBOSE || logLevel > TimedLog::LOG_FATAL)
logLevel = TimedLog::LOG_VERBOSE;
2016-08-17 23:58:09 +00:00
// Some objects used to redirect cout and cerr
// Scope must be here, so this still works inside the catch block for logging exceptions
std::streambuf* cout_rdbuf = std::cout.rdbuf ();
std::streambuf* cerr_rdbuf = std::cerr.rdbuf ();
boost::iostreams::stream_buffer<Tee> coutsb;
boost::iostreams::stream_buffer<Tee> cerrsb;
std::ostream oldcout(cout_rdbuf);
std::ostream oldcerr(cerr_rdbuf);
boost::filesystem::ofstream logfile;
2017-05-28 11:11:40 +00:00
if (!variables["no-logs"].as<bool>())
{
// Redirect cout and cerr to tes3mp server log
logfile.open(boost::filesystem::path(
cfgMgr.getLogPath() / "/tes3mp-server-" += TimedLog::getFilenameTimestamp() += ".log"));
2016-08-17 23:58:09 +00:00
2017-05-28 11:11:40 +00:00
coutsb.open(Tee(logfile, oldcout));
cerrsb.open(Tee(logfile, oldcerr));
2016-08-17 23:58:09 +00:00
2017-05-28 11:11:40 +00:00
std::cout.rdbuf(&coutsb);
std::cerr.rdbuf(&cerrsb);
}
2016-08-17 23:58:09 +00:00
2016-08-14 23:45:42 +00:00
LOG_INIT(logLevel);
int players = mgr.getInt("maximumPlayers", "General");
2021-03-21 13:45:01 +00:00
std::string address = mgr.getString("localAddress", "General");
int port = mgr.getInt("port", "General");
2021-03-21 13:45:01 +00:00
std::string password = mgr.getString("password", "General");
2021-03-21 13:45:01 +00:00
std::string pluginHome = mgr.getString("home", "Plugins");
std::string dataDirectory = Utils::convertPath(pluginHome + "/data");
2021-03-21 13:45:01 +00:00
std::vector<std::string> plugins(Utils::split(mgr.getString("plugins", "Plugins"), ','));
std::string versionInfo = Utils::getVersionInfo("TES3MP dedicated server", TES3MP_VERSION, version.mCommitHash, TES3MP_PROTO_VERSION);
LOG_MESSAGE_SIMPLE(TimedLog::LOG_INFO, "%s", versionInfo.c_str());
Script::SetModDir(dataDirectory);
#ifdef ENABLE_LUA
LangLua::AddPackagePath(Utils::convertPath(pluginHome + "/scripts/?.lua" + ";"
+ pluginHome + "/lib/lua/?.lua" + ";"));
#ifdef _WIN32
LangLua::AddPackageCPath(Utils::convertPath(pluginHome + "/lib/?.dll"));
#else
2019-01-05 20:11:58 +00:00
LangLua::AddPackageCPath(Utils::convertPath(pluginHome + "/lib/?.so"));
#endif
#endif
2017-06-11 12:04:33 +00:00
int code;
RakNet::RakPeerInterface *peer = RakNet::RakPeerInterface::GetInstance();
2021-03-21 13:45:01 +00:00
std::stringstream sstr;
2017-07-02 03:11:33 +00:00
sstr << TES3MP_VERSION;
2016-09-18 03:49:30 +00:00
sstr << TES3MP_PROTO_VERSION;
// Remove carriage returns added to version file on Windows
version.mCommitHash.erase(std::remove(version.mCommitHash.begin(), version.mCommitHash.end(), '\r'), version.mCommitHash.end());
2017-05-28 10:26:38 +00:00
sstr << version.mCommitHash;
2016-09-18 03:49:30 +00:00
2017-06-11 12:04:33 +00:00
peer->SetIncomingPassword(sstr.str().c_str(), (int) sstr.str().size());
if (RakNet::NonNumericHostString(address.c_str()))
2016-08-15 14:49:34 +00:00
{
LOG_MESSAGE_SIMPLE(TimedLog::LOG_ERROR, "You cannot use non-numeric addresses for the server.");
2016-08-15 14:49:34 +00:00
return 1;
}
RakNet::SocketDescriptor sd((unsigned short) port, address.c_str());
2017-06-11 12:04:33 +00:00
try
2016-08-14 23:48:25 +00:00
{
2017-06-11 12:04:33 +00:00
for (auto plugin : plugins)
Script::LoadScript(plugin.c_str(), pluginHome.c_str());
2017-06-11 12:04:33 +00:00
switch (peer->Startup((unsigned) players, &sd, 1))
{
case RakNet::CRABNET_STARTED:
2017-06-11 12:04:33 +00:00
break;
case RakNet::CRABNET_ALREADY_STARTED:
2021-03-21 13:45:01 +00:00
throw std::runtime_error("Already started");
2017-06-11 12:04:33 +00:00
case RakNet::INVALID_SOCKET_DESCRIPTORS:
2021-03-21 13:45:01 +00:00
throw std::runtime_error("Incorrect port or address");
2017-06-11 12:04:33 +00:00
case RakNet::INVALID_MAX_CONNECTIONS:
2021-03-21 13:45:01 +00:00
throw std::runtime_error("Max players cannot be negative or 0");
2017-06-11 12:04:33 +00:00
case RakNet::SOCKET_FAILED_TO_BIND:
case RakNet::SOCKET_PORT_ALREADY_IN_USE:
case RakNet::PORT_CANNOT_BE_ZERO:
2021-03-21 13:45:01 +00:00
throw std::runtime_error("Failed to bind port. Make sure a server isn't already running on that port.");
2017-06-11 12:04:33 +00:00
case RakNet::SOCKET_FAILED_TEST_SEND:
case RakNet::SOCKET_FAMILY_NOT_SUPPORTED:
case RakNet::FAILED_TO_CREATE_NETWORK_THREAD:
case RakNet::COULD_NOT_GENERATE_GUID:
case RakNet::STARTUP_OTHER_FAILURE:
2021-03-21 13:45:01 +00:00
throw std::runtime_error("Cannot start server");
2017-06-11 12:04:33 +00:00
}
peer->SetMaximumIncomingConnections((unsigned short) (players));
Networking networking(peer);
networking.setServerPassword(password);
2017-06-11 12:04:33 +00:00
if (mgr.getBool("enabled", "MasterServer"))
{
LOG_MESSAGE_SIMPLE(TimedLog::LOG_INFO, "Sharing server query info to master enabled.");
2021-03-21 13:45:01 +00:00
std::string masterAddr = mgr.getString("address", "MasterServer");
2017-06-11 12:04:33 +00:00
int masterPort = mgr.getInt("port", "MasterServer");
int updateRate = mgr.getInt("rate", "MasterServer");
// Is this an attempt to connect to the official master server at the old port? If so,
// redirect it to the correct port for the currently used fork of RakNet
if (Misc::StringUtils::ciEqual(masterAddr, "master.tes3mp.com") && masterPort == 25560)
{
masterPort = 25561;
LOG_APPEND(TimedLog::LOG_INFO, "- switching to port %i because the correct official master server for this version is on that port",
masterPort);
}
if (updateRate < 8000)
{
updateRate = 8000;
LOG_APPEND(TimedLog::LOG_INFO, "- switching to updateRate %i because the one in the server config was too low", updateRate);
}
2017-06-11 12:04:33 +00:00
networking.InitQuery(masterAddr, (unsigned short) masterPort);
networking.getMasterClient()->SetMaxPlayers((unsigned) players);
networking.getMasterClient()->SetUpdateRate((unsigned) updateRate);
2021-03-21 13:45:01 +00:00
std::string hostname = mgr.getString("hostname", "General");
2017-06-11 12:04:33 +00:00
networking.getMasterClient()->SetHostname(hostname);
networking.getMasterClient()->SetRuleString("CommitHash", version.mCommitHash.substr(0, 10));
networking.getMasterClient()->Start();
}
networking.postInit();
code = networking.mainLoop();
networking.getMasterClient()->Stop();
}
catch (std::exception &e)
{
LOG_MESSAGE_SIMPLE(TimedLog::LOG_ERROR, e.what());
Script::Call<Script::CallbackIdentity("OnServerScriptCrash")>(e.what());
2017-06-11 12:04:33 +00:00
throw; //fall through
2016-08-14 23:48:25 +00:00
}
RakNet::RakPeerInterface::DestroyInstance(peer);
2016-08-14 23:48:25 +00:00
if (code == 0)
LOG_MESSAGE_SIMPLE(TimedLog::LOG_INFO, "Quitting peacefully.");
LOG_QUIT();
if (!variables["no-logs"].as<bool>())
2017-05-28 11:11:40 +00:00
{
// Restore cout and cerr
std::cout.rdbuf(cout_rdbuf);
std::cerr.rdbuf(cerr_rdbuf);
}
2016-08-17 23:58:09 +00:00
2017-06-11 12:04:33 +00:00
2016-10-23 09:07:46 +00:00
breakpad_close();
return code;
}