mirror of
https://github.com/TrinityCore/TrinityCore.git
synced 2026-01-16 23:50:44 +01:00
Make acessible all the info about current realm (e.g name) anywhere, not only realm id Reduce the number of differences between the two branches Original changes by Shauren Partial port ofbacc90b6baand63def8aa32
322 lines
10 KiB
C++
322 lines
10 KiB
C++
/*
|
|
* Copyright (C) 2008-2016 TrinityCore <http://www.trinitycore.org/>
|
|
* Copyright (C) 2005-2009 MaNGOS <http://getmangos.com/>
|
|
*
|
|
* 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 "AuthSocketMgr.h"
|
|
#include "Common.h"
|
|
#include "Config.h"
|
|
#include "DatabaseEnv.h"
|
|
#include "DatabaseLoader.h"
|
|
#include "Log.h"
|
|
#include "AppenderDB.h"
|
|
#include "ProcessPriority.h"
|
|
#include "RealmList.h"
|
|
#include "GitRevision.h"
|
|
#include "Util.h"
|
|
#include <iostream>
|
|
#include <boost/program_options.hpp>
|
|
#include <openssl/opensslv.h>
|
|
#include <openssl/crypto.h>
|
|
|
|
using boost::asio::ip::tcp;
|
|
using namespace boost::program_options;
|
|
|
|
#ifndef _TRINITY_REALM_CONFIG
|
|
# define _TRINITY_REALM_CONFIG "authserver.conf"
|
|
#endif
|
|
|
|
#if PLATFORM == 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;
|
|
|
|
boost::asio::deadline_timer* _serviceStatusWatchTimer;
|
|
void ServiceStatusWatcher(boost::system::error_code const& error);
|
|
#endif
|
|
|
|
bool StartDB();
|
|
void StopDB();
|
|
void SignalHandler(const boost::system::error_code& error, int signalNumber);
|
|
void KeepDatabaseAliveHandler(const boost::system::error_code& error);
|
|
void BanExpiryHandler(boost::system::error_code const& error);
|
|
variables_map GetConsoleArguments(int argc, char** argv, std::string& configFile, std::string& configService);
|
|
|
|
boost::asio::io_service* _ioService;
|
|
boost::asio::deadline_timer* _dbPingTimer;
|
|
uint32 _dbPingInterval;
|
|
boost::asio::deadline_timer* _banExpiryCheckTimer;
|
|
uint32 _banExpiryCheckInterval;
|
|
LoginDatabaseWorkerPool LoginDatabase;
|
|
|
|
int main(int argc, char** argv)
|
|
{
|
|
std::string configFile = _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 PLATFORM == 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, configError))
|
|
{
|
|
printf("Error in config file: %s\n", configError.c_str());
|
|
return 1;
|
|
}
|
|
|
|
sLog->RegisterAppender<AppenderDB>();
|
|
sLog->Initialize(nullptr);
|
|
|
|
TC_LOG_INFO("server.authserver", "%s (authserver)", GitRevision::GetFullVersion());
|
|
TC_LOG_INFO("server.authserver", "<Ctrl-C> to stop.\n");
|
|
TC_LOG_INFO("server.authserver", "Using configuration file %s.", configFile.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);
|
|
|
|
// 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;
|
|
|
|
_ioService = new boost::asio::io_service();
|
|
|
|
// Get the list of realms for the server
|
|
sRealmList->Initialize(*_ioService, sConfigMgr->GetIntDefault("RealmsStateUpdateDelay", 20));
|
|
|
|
if (sRealmList->GetRealms().empty())
|
|
{
|
|
TC_LOG_ERROR("server.authserver", "No valid realms specified.");
|
|
StopDB();
|
|
delete _ioService;
|
|
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)");
|
|
StopDB();
|
|
delete _ioService;
|
|
return 1;
|
|
}
|
|
|
|
std::string bindIp = sConfigMgr->GetStringDefault("BindIP", "0.0.0.0");
|
|
|
|
sAuthSocketMgr.StartNetwork(*_ioService, bindIp, port);
|
|
|
|
// Set signal handlers
|
|
boost::asio::signal_set signals(*_ioService, SIGINT, SIGTERM);
|
|
#if PLATFORM == PLATFORM_WINDOWS
|
|
signals.add(SIGBREAK);
|
|
#endif
|
|
signals.async_wait(SignalHandler);
|
|
|
|
// Set process priority according to configuration settings
|
|
SetProcessPriority("server.authserver");
|
|
|
|
// Enabled a timed callback for handling the database keep alive ping
|
|
_dbPingInterval = sConfigMgr->GetIntDefault("MaxPingTime", 30);
|
|
_dbPingTimer = new boost::asio::deadline_timer(*_ioService);
|
|
_dbPingTimer->expires_from_now(boost::posix_time::minutes(_dbPingInterval));
|
|
_dbPingTimer->async_wait(KeepDatabaseAliveHandler);
|
|
|
|
_banExpiryCheckInterval = sConfigMgr->GetIntDefault("BanExpiryCheckInterval", 60);
|
|
_banExpiryCheckTimer = new boost::asio::deadline_timer(*_ioService);
|
|
_banExpiryCheckTimer->expires_from_now(boost::posix_time::seconds(_banExpiryCheckInterval));
|
|
_banExpiryCheckTimer->async_wait(BanExpiryHandler);
|
|
|
|
#if PLATFORM == PLATFORM_WINDOWS
|
|
if (m_ServiceStatus != -1)
|
|
{
|
|
_serviceStatusWatchTimer = new boost::asio::deadline_timer(*_ioService);
|
|
_serviceStatusWatchTimer->expires_from_now(boost::posix_time::seconds(1));
|
|
_serviceStatusWatchTimer->async_wait(ServiceStatusWatcher);
|
|
}
|
|
#endif
|
|
|
|
// Start the io service worker loop
|
|
_ioService->run();
|
|
|
|
_banExpiryCheckTimer->cancel();
|
|
_dbPingTimer->cancel();
|
|
|
|
sAuthSocketMgr.StopNetwork();
|
|
|
|
// Close the Database Pool and library
|
|
StopDB();
|
|
|
|
TC_LOG_INFO("server.authserver", "Halting process...");
|
|
|
|
signals.cancel();
|
|
|
|
delete _banExpiryCheckTimer;
|
|
delete _dbPingTimer;
|
|
delete _ioService;
|
|
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(const boost::system::error_code& error, int /*signalNumber*/)
|
|
{
|
|
if (!error)
|
|
_ioService->stop();
|
|
}
|
|
|
|
void KeepDatabaseAliveHandler(const boost::system::error_code& error)
|
|
{
|
|
if (!error)
|
|
{
|
|
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(KeepDatabaseAliveHandler);
|
|
}
|
|
}
|
|
|
|
void BanExpiryHandler(boost::system::error_code const& error)
|
|
{
|
|
if (!error)
|
|
{
|
|
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(BanExpiryHandler);
|
|
}
|
|
}
|
|
|
|
#if PLATFORM == PLATFORM_WINDOWS
|
|
void ServiceStatusWatcher(boost::system::error_code const& error)
|
|
{
|
|
if (!error)
|
|
{
|
|
if (m_ServiceStatus == 0)
|
|
{
|
|
_ioService->stop();
|
|
delete _serviceStatusWatchTimer;
|
|
}
|
|
else
|
|
{
|
|
_serviceStatusWatchTimer->expires_from_now(boost::posix_time::seconds(1));
|
|
_serviceStatusWatchTimer->async_wait(ServiceStatusWatcher);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
variables_map GetConsoleArguments(int argc, char** argv, std::string& 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<std::string>(&configFile)->default_value(_TRINITY_REALM_CONFIG), "use <arg> as configuration file")
|
|
;
|
|
#if PLATFORM == 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;
|
|
}
|