/*
* The Mana World Server
* Copyright 2004 The Mana World Development Team
*
* This file is part of The Mana World.
*
* The Mana World 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 any later version.
*
* The Mana World 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 The Mana World; if not, write to the Free Software Foundation, Inc.,
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* $Id$
*/
#include <cstdlib>
#include <getopt.h>
#include <signal.h>
#include <iostream>
#include <physfs.h>
#include <enet/enet.h>
#if (defined __USE_UNIX98 || defined __FreeBSD__)
#include "../config.h"
#endif
#include "configuration.h"
#include "resourcemanager.h"
#include "skill.h"
#include "account-server/accounthandler.hpp"
#include "account-server/serverhandler.hpp"
#include "account-server/storage.hpp"
#include "chat-server/chatchannelmanager.hpp"
#include "chat-server/chathandler.hpp"
#include "net/connectionhandler.hpp"
#include "net/messageout.hpp"
#include "utils/logger.h"
#include "utils/stringfilter.h"
// Default options that automake should be able to override.
#define DEFAULT_LOG_FILE "tmwserv-account.log"
#define DEFAULT_CONFIG_FILE "tmwserv.xml"
#define DEFAULT_ITEMSDB_FILE "items.xml"
bool running = true; /**< Determines if server keeps running */
Skill skillTree("base"); /**< Skill tree */
Configuration config; /**< XML config reader */
utils::StringFilter *stringFilter; /**< Slang's Filter */
/** Account message handler */
AccountHandler *accountHandler;
/** Communications (chat) message handler */
ChatHandler *chatHandler;
/** Server message handler */
ServerHandler *serverHandler;
/** Chat Channels Manager */
ChatChannelManager *chatChannelManager;
/** Callback used when SIGQUIT signal is received. */
void closeGracefully(int dummy)
{
running = false;
}
/**
* Initializes the server.
*/
void initialize()
{
// Reset to default segmentation fault handling for debugging purposes
signal(SIGSEGV, SIG_DFL);
// Used to close via process signals
#if (defined __USE_UNIX98 || defined __FreeBSD__)
signal(SIGQUIT, closeGracefully);
#endif
signal(SIGINT, closeGracefully);
// Set enet to quit on exit.
atexit(enet_deinitialize);
/*
* If the path values aren't defined, we set the default
* depending on the platform.
*/
// The config path
#if defined CONFIG_FILE
std::string configPath = CONFIG_FILE;
#else
#if (defined __USE_UNIX98 || defined __FreeBSD__)
std::string configPath = getenv("HOME");
configPath += "/.";
configPath += DEFAULT_CONFIG_FILE;
#else // Win32, ...
std::string configPath = DEFAULT_CONFIG_FILE;
#endif
#endif // defined CONFIG_FILE
// The log path
#if defined LOG_FILE
std::string logPath = LOG_FILE;
#else
#if (defined __USE_UNIX98 || defined __FreeBSD__)
std::string logPath = getenv("HOME");
logPath += "/.";
logPath += DEFAULT_LOG_FILE;
#else // Win32, ...
std::string logPath = DEFAULT_LOG_FILE;
#endif
#endif // defined LOG_FILE
// Initialize PhysicsFS
PHYSFS_init("");
// Initialize the logger.
using namespace utils;
Logger::setLogFile(logPath);
// write the messages to both the screen and the log file.
Logger::setTeeMode(true);
config.init(configPath);
LOG_INFO("Using Config File: " << configPath);
LOG_INFO("Using Log File: " << logPath);
// --- Initialize the managers
// Initialize the slang's and double quotes filter.
stringFilter = new StringFilter(&config);
// Initialize the Chat channels manager
chatChannelManager = new ChatChannelManager();
// --- Initialize the global handlers
// FIXME: Make the global handlers global vars or part of a bigger
// singleton or a local variable in the event-loop
accountHandler = new AccountHandler();
chatHandler = new ChatHandler();
serverHandler = new ServerHandler();
// --- Initialize enet.
if (enet_initialize() != 0) {
LOG_FATAL("An error occurred while initializing ENet");
exit(2);
}
#if defined (MYSQL_SUPPORT)
LOG_INFO("Using MySQL DB Backend.");
#elif defined (POSTGRESQL_SUPPORT)
LOG_INFO("Using PostGreSQL DB Backend.");
#elif defined (SQLITE_SUPPORT)
LOG_INFO("Using SQLite DB Backend.");
#else
LOG_WARN("No Database Backend Support.");
#endif
// Initialize configuration defaults
config.setValue("dbuser", "");
config.setValue("dbpass", "");
config.setValue("dbhost", "");
//Seed the random number generator
std::srand( time(NULL) );
}
/**
* Deinitializes the server.
*/
void deinitialize()
{
delete stringFilter;
// Write configuration file
config.write();
// Quit ENet
enet_deinitialize();
// Destroy message handlers
delete serverHandler;
delete chatHandler;
delete accountHandler;
// Destroy Managers
delete chatChannelManager;
// Get rid of persistent data storage
Storage::destroy();
PHYSFS_deinit();
}
/**
* Show command line arguments
*/
void printHelp()
{
std::cout << "tmwserv" << std::endl << std::endl
<< "Options: " << std::endl
<< " -h --help : Display this help" << std::endl
<< " --verbosity <n> : Set the verbosity level" << std::endl
<< " --port <n> : Set the default port to listen on" << std::endl;
exit(0);
}
/**
* Parse the command line arguments
*/
void parseOptions(int argc, char *argv[])
{
const char *optstring = "h";
const struct option long_options[] = {
{ "help", no_argument, 0, 'h' },
{ "verbosity", required_argument, 0, 'v' },
{ "port", required_argument, 0, 'p' },
{ 0 }
};
while (optind < argc) {
int result = getopt_long(argc, argv, optstring, long_options, NULL);
if (result == -1) {
break;
}
switch (result) {
default: // Unknown option
case 'h':
// Print help
printHelp();
break;
case 'v':
// Set Verbosity to level
unsigned short verbosityLevel;
verbosityLevel = atoi(optarg);
utils::Logger::setVerbosity(utils::Logger::Level(verbosityLevel));
LOG_INFO("Setting Log Verbosity Level to " << verbosityLevel);
break;
case 'p':
// Change the port to listen on.
unsigned short portToListenOn;
portToListenOn = atoi(optarg);
config.setValue("ListenOnPort", portToListenOn);
LOG_INFO("Setting Default Port to " << portToListenOn);
break;
}
}
}
/**
* Main function, initializes and runs server.
*/
int main(int argc, char *argv[])
{
LOG_INFO("The Mana World Account+Chat Server v" << PACKAGE_VERSION);
// Parse Command Line Options
parseOptions(argc, argv);
// General Initialization
initialize();
int port = int(config.getValue("accountServerPort", DEFAULT_SERVER_PORT));
if (!accountHandler->startListen(port) ||
!serverHandler->startListen(port + 1) ||
!chatHandler->startListen(port + 2))
{
LOG_FATAL("Unable to create an ENet server host.");
return 3;
}
// Create storage wrapper
Storage& store = Storage::instance("tmw");
store.setUser(config.getValue("dbuser", ""));
store.setPassword(config.getValue("dbpass", ""));
store.close();
store.open();
while (running) {
accountHandler->process(50);
chatHandler->process(50);
serverHandler->process(50);
}
LOG_INFO("Received: Quit signal, closing down...");
serverHandler->stopListen();
chatHandler->stopListen();
accountHandler->stopListen();
deinitialize();
}