mirror of
https://github.com/TrinityCore/TrinityCore.git
synced 2026-01-20 09:17:36 +01:00
* Core/Config: Implement config override with env vars Implement overriding of configuration from the .conf file with environment variables. Environment variables keys are autogenerated based on the keys defined in .conf file. Usage example: $ export TC_DATA_DIR=/usr $ TC_WORLD_SERVER_PORT=8080 ./worldserver * Core/Config Fix typo in logs Co-authored-by: Giacomo Pozzoni <giacomopoz@gmail.com> * Core/Config Fix code style in EnvVarForIniKey Co-authored-by: Shauren <shauren.trinity@gmail.com> * Update tests/common/Config.cpp * Apply suggestions from code review Co-authored-by: Peter Keresztes Schmidt <carbenium@outlook.com> * Apply suggestions from code review Co-authored-by: Peter Keresztes Schmidt <carbenium@outlook.com> Co-authored-by: Anton Popovichenko <anton.popovichenko@mendix.com> Co-authored-by: Giacomo Pozzoni <giacomopoz@gmail.com> Co-authored-by: Shauren <shauren.trinity@gmail.com> Co-authored-by: Peter Keresztes Schmidt <carbenium@outlook.com>
354 lines
13 KiB
C++
354 lines
13 KiB
C++
/*
|
|
* This file is part of the TrinityCore Project. See AUTHORS file for Copyright information
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation; either version 2 of the License, or (at your
|
|
* option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/**
|
|
* @file main.cpp
|
|
* @brief Authentication Server main program
|
|
*
|
|
* This file contains the main program for the
|
|
* authentication server
|
|
*/
|
|
|
|
#include "AppenderDB.h"
|
|
#include "AuthSocketMgr.h"
|
|
#include "Banner.h"
|
|
#include "Config.h"
|
|
#include "DatabaseEnv.h"
|
|
#include "DatabaseLoader.h"
|
|
#include "DeadlineTimer.h"
|
|
#include "IoContext.h"
|
|
#include "IPLocation.h"
|
|
#include "GitRevision.h"
|
|
#include "MySQLThreading.h"
|
|
#include "ProcessPriority.h"
|
|
#include "RealmList.h"
|
|
#include "SecretMgr.h"
|
|
#include "SharedDefines.h"
|
|
#include "Util.h"
|
|
#include <boost/asio/signal_set.hpp>
|
|
#include <boost/program_options.hpp>
|
|
#include <boost/filesystem/operations.hpp>
|
|
#include <openssl/crypto.h>
|
|
#include <openssl/opensslv.h>
|
|
#include <iostream>
|
|
#include <csignal>
|
|
|
|
using boost::asio::ip::tcp;
|
|
using namespace boost::program_options;
|
|
namespace fs = boost::filesystem;
|
|
|
|
#ifndef _TRINITY_REALM_CONFIG
|
|
# define _TRINITY_REALM_CONFIG "authserver.conf"
|
|
#endif
|
|
|
|
#if TRINITY_PLATFORM == TRINITY_PLATFORM_WINDOWS
|
|
#include "ServiceWin32.h"
|
|
char serviceName[] = "authserver";
|
|
char serviceLongName[] = "TrinityCore auth service";
|
|
char serviceDescription[] = "TrinityCore World of Warcraft emulator auth service";
|
|
/*
|
|
* -1 - not in service mode
|
|
* 0 - stopped
|
|
* 1 - running
|
|
* 2 - paused
|
|
*/
|
|
int m_ServiceStatus = -1;
|
|
|
|
void ServiceStatusWatcher(std::weak_ptr<Trinity::Asio::DeadlineTimer> serviceStatusWatchTimerRef, std::weak_ptr<Trinity::Asio::IoContext> ioContextRef, boost::system::error_code const& error);
|
|
#endif
|
|
|
|
bool StartDB();
|
|
void StopDB();
|
|
void SignalHandler(std::weak_ptr<Trinity::Asio::IoContext> ioContextRef, boost::system::error_code const& error, int signalNumber);
|
|
void KeepDatabaseAliveHandler(std::weak_ptr<Trinity::Asio::DeadlineTimer> dbPingTimerRef, int32 dbPingInterval, boost::system::error_code const& error);
|
|
void BanExpiryHandler(std::weak_ptr<Trinity::Asio::DeadlineTimer> banExpiryCheckTimerRef, int32 banExpiryCheckInterval, boost::system::error_code const& error);
|
|
variables_map GetConsoleArguments(int argc, char** argv, fs::path& configFile, std::string& configService);
|
|
|
|
int main(int argc, char** argv)
|
|
{
|
|
Trinity::Impl::CurrentServerProcessHolder::_type = SERVER_PROCESS_AUTHSERVER;
|
|
signal(SIGABRT, &Trinity::AbortHandler);
|
|
|
|
auto configFile = fs::absolute(_TRINITY_REALM_CONFIG);
|
|
std::string configService;
|
|
auto vm = GetConsoleArguments(argc, argv, configFile, configService);
|
|
// exit if help or version is enabled
|
|
if (vm.count("help") || vm.count("version"))
|
|
return 0;
|
|
|
|
#if TRINITY_PLATFORM == TRINITY_PLATFORM_WINDOWS
|
|
if (configService.compare("install") == 0)
|
|
return WinServiceInstall() == true ? 0 : 1;
|
|
else if (configService.compare("uninstall") == 0)
|
|
return WinServiceUninstall() == true ? 0 : 1;
|
|
else if (configService.compare("run") == 0)
|
|
return WinServiceRun() ? 0 : 1;
|
|
#endif
|
|
|
|
std::string configError;
|
|
if (!sConfigMgr->LoadInitial(configFile.generic_string(),
|
|
std::vector<std::string>(argv, argv + argc),
|
|
configError))
|
|
{
|
|
printf("Error in config file: %s\n", configError.c_str());
|
|
return 1;
|
|
}
|
|
|
|
std::vector<std::string> overriddenKeys = sConfigMgr->OverrideWithEnvVariablesIfAny();
|
|
|
|
sLog->RegisterAppender<AppenderDB>();
|
|
sLog->Initialize(nullptr);
|
|
|
|
Trinity::Banner::Show("authserver",
|
|
[](char const* text)
|
|
{
|
|
TC_LOG_INFO("server.authserver", "%s", text);
|
|
},
|
|
[]()
|
|
{
|
|
TC_LOG_INFO("server.authserver", "Using configuration file %s.", sConfigMgr->GetFilename().c_str());
|
|
TC_LOG_INFO("server.authserver", "Using SSL version: %s (library: %s)", OPENSSL_VERSION_TEXT, SSLeay_version(SSLEAY_VERSION));
|
|
TC_LOG_INFO("server.authserver", "Using Boost version: %i.%i.%i", BOOST_VERSION / 100000, BOOST_VERSION / 100 % 1000, BOOST_VERSION % 100);
|
|
}
|
|
);
|
|
|
|
for (std::string const& key : overriddenKeys)
|
|
TC_LOG_INFO("server.authserver", "Configuration field '%s' was overridden with environment variable.", key.c_str());
|
|
|
|
// authserver PID file creation
|
|
std::string pidFile = sConfigMgr->GetStringDefault("PidFile", "");
|
|
if (!pidFile.empty())
|
|
{
|
|
if (uint32 pid = CreatePIDFile(pidFile))
|
|
TC_LOG_INFO("server.authserver", "Daemon PID: %u\n", pid);
|
|
else
|
|
{
|
|
TC_LOG_ERROR("server.authserver", "Cannot create PID file %s.\n", pidFile.c_str());
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
// Initialize the database connection
|
|
if (!StartDB())
|
|
return 1;
|
|
|
|
sSecretMgr->Initialize();
|
|
|
|
// Load IP Location Database
|
|
sIPLocation->Load();
|
|
|
|
std::shared_ptr<void> dbHandle(nullptr, [](void*) { StopDB(); });
|
|
|
|
std::shared_ptr<Trinity::Asio::IoContext> ioContext = std::make_shared<Trinity::Asio::IoContext>();
|
|
|
|
// Get the list of realms for the server
|
|
sRealmList->Initialize(*ioContext, sConfigMgr->GetIntDefault("RealmsStateUpdateDelay", 20));
|
|
|
|
std::shared_ptr<void> sRealmListHandle(nullptr, [](void*) { sRealmList->Close(); });
|
|
|
|
if (sRealmList->GetRealms().empty())
|
|
{
|
|
TC_LOG_ERROR("server.authserver", "No valid realms specified.");
|
|
return 1;
|
|
}
|
|
|
|
// Start the listening port (acceptor) for auth connections
|
|
int32 port = sConfigMgr->GetIntDefault("RealmServerPort", 3724);
|
|
if (port < 0 || port > 0xFFFF)
|
|
{
|
|
TC_LOG_ERROR("server.authserver", "Specified port out of allowed range (1-65535)");
|
|
return 1;
|
|
}
|
|
|
|
std::string bindIp = sConfigMgr->GetStringDefault("BindIP", "0.0.0.0");
|
|
|
|
if (!sAuthSocketMgr.StartNetwork(*ioContext, bindIp, port))
|
|
{
|
|
TC_LOG_ERROR("server.authserver", "Failed to initialize network");
|
|
return 1;
|
|
}
|
|
|
|
std::shared_ptr<void> sAuthSocketMgrHandle(nullptr, [](void*) { sAuthSocketMgr.StopNetwork(); });
|
|
|
|
// Set signal handlers
|
|
boost::asio::signal_set signals(*ioContext, SIGINT, SIGTERM);
|
|
#if TRINITY_PLATFORM == TRINITY_PLATFORM_WINDOWS
|
|
signals.add(SIGBREAK);
|
|
#endif
|
|
signals.async_wait(std::bind(&SignalHandler, std::weak_ptr<Trinity::Asio::IoContext>(ioContext), std::placeholders::_1, std::placeholders::_2));
|
|
|
|
// Set process priority according to configuration settings
|
|
SetProcessPriority("server.authserver", sConfigMgr->GetIntDefault(CONFIG_PROCESSOR_AFFINITY, 0), sConfigMgr->GetBoolDefault(CONFIG_HIGH_PRIORITY, false));
|
|
|
|
// Enabled a timed callback for handling the database keep alive ping
|
|
int32 dbPingInterval = sConfigMgr->GetIntDefault("MaxPingTime", 30);
|
|
std::shared_ptr<Trinity::Asio::DeadlineTimer> dbPingTimer = std::make_shared<Trinity::Asio::DeadlineTimer>(*ioContext);
|
|
dbPingTimer->expires_from_now(boost::posix_time::minutes(dbPingInterval));
|
|
dbPingTimer->async_wait(std::bind(&KeepDatabaseAliveHandler, std::weak_ptr<Trinity::Asio::DeadlineTimer>(dbPingTimer), dbPingInterval, std::placeholders::_1));
|
|
|
|
int32 banExpiryCheckInterval = sConfigMgr->GetIntDefault("BanExpiryCheckInterval", 60);
|
|
std::shared_ptr<Trinity::Asio::DeadlineTimer> banExpiryCheckTimer = std::make_shared<Trinity::Asio::DeadlineTimer>(*ioContext);
|
|
banExpiryCheckTimer->expires_from_now(boost::posix_time::seconds(banExpiryCheckInterval));
|
|
banExpiryCheckTimer->async_wait(std::bind(&BanExpiryHandler, std::weak_ptr<Trinity::Asio::DeadlineTimer>(banExpiryCheckTimer), banExpiryCheckInterval, std::placeholders::_1));
|
|
|
|
#if TRINITY_PLATFORM == TRINITY_PLATFORM_WINDOWS
|
|
std::shared_ptr<Trinity::Asio::DeadlineTimer> serviceStatusWatchTimer;
|
|
if (m_ServiceStatus != -1)
|
|
{
|
|
serviceStatusWatchTimer = std::make_shared<Trinity::Asio::DeadlineTimer>(*ioContext);
|
|
serviceStatusWatchTimer->expires_from_now(boost::posix_time::seconds(1));
|
|
serviceStatusWatchTimer->async_wait(std::bind(&ServiceStatusWatcher,
|
|
std::weak_ptr<Trinity::Asio::DeadlineTimer>(serviceStatusWatchTimer),
|
|
std::weak_ptr<Trinity::Asio::IoContext>(ioContext),
|
|
std::placeholders::_1));
|
|
}
|
|
#endif
|
|
|
|
// Start the io service worker loop
|
|
ioContext->run();
|
|
|
|
banExpiryCheckTimer->cancel();
|
|
dbPingTimer->cancel();
|
|
|
|
TC_LOG_INFO("server.authserver", "Halting process...");
|
|
|
|
signals.cancel();
|
|
|
|
return 0;
|
|
}
|
|
|
|
/// Initialize connection to the database
|
|
bool StartDB()
|
|
{
|
|
MySQL::Library_Init();
|
|
|
|
// Load databases
|
|
// NOTE: While authserver is singlethreaded you should keep synch_threads == 1.
|
|
// Increasing it is just silly since only 1 will be used ever.
|
|
DatabaseLoader loader("server.authserver", DatabaseLoader::DATABASE_NONE);
|
|
loader
|
|
.AddDatabase(LoginDatabase, "Login");
|
|
|
|
if (!loader.Load())
|
|
return false;
|
|
|
|
TC_LOG_INFO("server.authserver", "Started auth database connection pool.");
|
|
sLog->SetRealmId(0); // Enables DB appenders when realm is set.
|
|
return true;
|
|
}
|
|
|
|
/// Close the connection to the database
|
|
void StopDB()
|
|
{
|
|
LoginDatabase.Close();
|
|
MySQL::Library_End();
|
|
}
|
|
|
|
void SignalHandler(std::weak_ptr<Trinity::Asio::IoContext> ioContextRef, boost::system::error_code const& error, int /*signalNumber*/)
|
|
{
|
|
if (!error)
|
|
if (std::shared_ptr<Trinity::Asio::IoContext> ioContext = ioContextRef.lock())
|
|
ioContext->stop();
|
|
}
|
|
|
|
void KeepDatabaseAliveHandler(std::weak_ptr<Trinity::Asio::DeadlineTimer> dbPingTimerRef, int32 dbPingInterval, boost::system::error_code const& error)
|
|
{
|
|
if (!error)
|
|
{
|
|
if (std::shared_ptr<Trinity::Asio::DeadlineTimer> dbPingTimer = dbPingTimerRef.lock())
|
|
{
|
|
TC_LOG_INFO("server.authserver", "Ping MySQL to keep connection alive");
|
|
LoginDatabase.KeepAlive();
|
|
|
|
dbPingTimer->expires_from_now(boost::posix_time::minutes(dbPingInterval));
|
|
dbPingTimer->async_wait(std::bind(&KeepDatabaseAliveHandler, dbPingTimerRef, dbPingInterval, std::placeholders::_1));
|
|
}
|
|
}
|
|
}
|
|
|
|
void BanExpiryHandler(std::weak_ptr<Trinity::Asio::DeadlineTimer> banExpiryCheckTimerRef, int32 banExpiryCheckInterval, boost::system::error_code const& error)
|
|
{
|
|
if (!error)
|
|
{
|
|
if (std::shared_ptr<Trinity::Asio::DeadlineTimer> banExpiryCheckTimer = banExpiryCheckTimerRef.lock())
|
|
{
|
|
LoginDatabase.Execute(LoginDatabase.GetPreparedStatement(LOGIN_DEL_EXPIRED_IP_BANS));
|
|
LoginDatabase.Execute(LoginDatabase.GetPreparedStatement(LOGIN_UPD_EXPIRED_ACCOUNT_BANS));
|
|
|
|
banExpiryCheckTimer->expires_from_now(boost::posix_time::seconds(banExpiryCheckInterval));
|
|
banExpiryCheckTimer->async_wait(std::bind(&BanExpiryHandler, banExpiryCheckTimerRef, banExpiryCheckInterval, std::placeholders::_1));
|
|
}
|
|
}
|
|
}
|
|
|
|
#if TRINITY_PLATFORM == TRINITY_PLATFORM_WINDOWS
|
|
void ServiceStatusWatcher(std::weak_ptr<Trinity::Asio::DeadlineTimer> serviceStatusWatchTimerRef, std::weak_ptr<Trinity::Asio::IoContext> ioContextRef, boost::system::error_code const& error)
|
|
{
|
|
if (!error)
|
|
{
|
|
if (std::shared_ptr<Trinity::Asio::IoContext> ioContext = ioContextRef.lock())
|
|
{
|
|
if (m_ServiceStatus == 0)
|
|
ioContext->stop();
|
|
else if (std::shared_ptr<Trinity::Asio::DeadlineTimer> serviceStatusWatchTimer = serviceStatusWatchTimerRef.lock())
|
|
{
|
|
serviceStatusWatchTimer->expires_from_now(boost::posix_time::seconds(1));
|
|
serviceStatusWatchTimer->async_wait(std::bind(&ServiceStatusWatcher, serviceStatusWatchTimerRef, ioContextRef, std::placeholders::_1));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
variables_map GetConsoleArguments(int argc, char** argv, fs::path& configFile, std::string& configService)
|
|
{
|
|
options_description all("Allowed options");
|
|
all.add_options()
|
|
("help,h", "print usage message")
|
|
("version,v", "print version build info")
|
|
("config,c", value<fs::path>(&configFile)->default_value(fs::absolute(_TRINITY_REALM_CONFIG)),
|
|
"use <arg> as configuration file")
|
|
;
|
|
#if TRINITY_PLATFORM == TRINITY_PLATFORM_WINDOWS
|
|
options_description win("Windows platform specific options");
|
|
win.add_options()
|
|
("service,s", value<std::string>(&configService)->default_value(""), "Windows service options: [install | uninstall]")
|
|
;
|
|
|
|
all.add(win);
|
|
#else
|
|
(void)configService;
|
|
#endif
|
|
variables_map variablesMap;
|
|
try
|
|
{
|
|
store(command_line_parser(argc, argv).options(all).allow_unregistered().run(), variablesMap);
|
|
notify(variablesMap);
|
|
}
|
|
catch (std::exception& e)
|
|
{
|
|
std::cerr << e.what() << "\n";
|
|
}
|
|
|
|
if (variablesMap.count("help"))
|
|
std::cout << all << "\n";
|
|
else if (variablesMap.count("version"))
|
|
std::cout << GitRevision::GetFullVersion() << "\n";
|
|
|
|
return variablesMap;
|
|
}
|