// Cyphesis Online RPG Server and AI Engine
// Copyright (C) 2000,2001 Alistair Riddoch
//
// 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, write to the Free Software Foundation,
// Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
// $Id: Persistance.cpp,v 1.53 2007-12-07 17:42:59 alriddoch Exp $
#include "Persistance.h"
#include "Admin.h"
#include "Player.h"
#include "rulesets/Entity.h"
#include "common/id.h"
#include "common/log.h"
#include "common/const.h"
#include "common/debug.h"
#include "common/Database.h"
#include "common/compose.hpp"
using Atlas::Message::MapType;
using Atlas::Objects::Root;
static const bool debug_flag = false;
Persistance * Persistance::m_instance = NULL;
Persistance::Persistance() : m_connection(*Database::instance())
{
}
Persistance * Persistance::instance()
{
if (m_instance == NULL) {
m_instance = new Persistance();
}
return m_instance;
}
int Persistance::init()
{
assert(this != 0);
if (m_connection.initConnection() != 0) {
if (::instance == CYPHESIS) {
return DATABASE_CONERR;
}
if (m_connection.createInstanceDatabase() != 0) {
log(ERROR, "Database creation failed.");
return DATABASE_CONERR;
}
if (m_connection.initConnection() != 0) {
log(ERROR, "Still couldn't connect.");
return DATABASE_CONERR;
}
log(INFO, String::compose("Auto created database for new instance "
"\"%1\".", ::instance));
}
if (!m_connection.registerEntityIdGenerator()) {
log(ERROR, "Faled to register Id generator in database.");
return DATABASE_TABERR;
}
bool i = m_connection.initRule(true);
MapType tableDesc;
tableDesc["username"] = " ";
tableDesc["password"] = " ";
tableDesc["type"] = " ";
bool j = m_connection.registerSimpleTable("accounts", tableDesc);
bool k = m_connection.registerRelation(m_characterRelation,
"accounts",
"entity_ent");
if (!findAccount("admin")) {
debug(std::cout << "Bootstraping admin account."
<< std::endl << std::flush;);
std::string adminAccountId;
long adminAccountIntId = m_connection.newId(adminAccountId);
if (adminAccountIntId < 0) {
log(CRITICAL, "Unable to get admin account ID from Database");
return -2;
}
Admin dummyAdminAccount(0, "admin", consts::defaultAdminPasswordHash,
adminAccountId, adminAccountIntId);
putAccount(dummyAdminAccount);
}
return (i && j && k) ? 0 : -2;
}
void Persistance::shutdown()
{
m_connection.shutdownConnection();
delete &m_connection;
assert(this == m_instance);
delete m_instance;
m_instance = NULL;
}
bool Persistance::findAccount(const std::string & name)
{
std::string namestr = "'" + name + "'";
DatabaseResult dr = m_connection.selectSimpleRowBy("accounts", "username", namestr);
if (dr.error()) {
log(ERROR, "Failure while find account.");
return false;
}
if (dr.empty()) {
dr.clear();
return false;
}
if (dr.size() > 1) {
log(ERROR, "Duplicate username in accounts database.");
}
dr.clear();
return true;
}
Account * Persistance::getAccount(const std::string & name)
{
std::string namestr = "'" + name + "'";
DatabaseResult dr = m_connection.selectSimpleRowBy("accounts", "username", namestr);
if (dr.error()) {
log(ERROR, "Failure while find account.");
return 0;
}
if (dr.empty()) {
dr.clear();
return 0;
}
if (dr.size() > 1) {
log(ERROR, "Duplicate username in accounts database.");
}
const char * c = dr.field("id");
if (c == 0) {
dr.clear();
log(ERROR, "Unable to find id field in accounts database.");
return 0;
}
std::string id = c;
long intId = integerId(id);
if (intId == -1) {
dr.clear();
log(ERROR, String::compose("Invalid ID \"%1\" for account from database.", id));
return 0;
}
c = dr.field("password");
if (c == 0) {
dr.clear();
log(ERROR, "Unable to find password field in accounts database.");
return 0;
}
std::string passwd = c;
c = dr.field("type");
if (c == 0) {
dr.clear();
log(ERROR, "Unable to find type field in accounts database.");
return 0;
}
std::string type = c;
dr.clear();
if (type == "admin") {
return new Admin(0, name, passwd, id, intId);
} else {
return new Player(0, name, passwd, id, intId);
}
}
void Persistance::putAccount(const Account & ac)
{
std::string columns = "username, type, password";
std::string values = "'";
values += ac.m_username;
values += "', '";
values += ac.getType();
values += "', '";
values += ac.m_password;
values += "'";
m_connection.createSimpleRow("accounts", ac.getId(), columns, values);
}
void Persistance::registerCharacters(Account & ac,
const EntityDict & worldObjects)
{
DatabaseResult dr = m_connection.selectRelation(m_characterRelation,
ac.getId());
if (dr.error()) {
log(ERROR, "Database query failed while looking for characters for account.");
}
DatabaseResult::const_iterator Iend = dr.end();
for (DatabaseResult::const_iterator I = dr.begin(); I != Iend; ++I) {
const char * id = I.column(0);
if (id == 0) {
log(ERROR, "No ID data in relation when examing characters");
continue;
}
long intId = integerId(id);
EntityDict::const_iterator J = worldObjects.find(intId);
if (J == worldObjects.end()) {
log(WARNING, String::compose("Persistance: Got character id \"%1\" "
"from database which does not exist "
"in world.", id));
continue;
}
ac.addCharacter(J->second);
}
dr.clear();
}
void Persistance::addCharacter(const Account & ac, const Entity & e)
{
m_connection.createRelationRow(m_characterRelation, ac.getId(), e.getId());
}
void Persistance::delCharacter(const std::string & id)
{
m_connection.removeRelationRowByOther(m_characterRelation, id);
}
bool Persistance::getRules(std::map<std::string, Root> & t)
{
return m_connection.getTable(m_connection.rule(), t);
}
bool Persistance::clearRules()
{
return m_connection.clearTable(m_connection.rule());
}
syntax highlighted by Code2HTML, v. 0.9.1