1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
|
// Cyphesis Online RPG Server and AI Engine
// Copyright (C) 2001-2004 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
#ifndef COMMON_DATABASE_H
#define COMMON_DATABASE_H
#include <Atlas/Message/DecoderBase.h>
#include <Atlas/Message/Element.h>
#include <Atlas/Objects/Decoder.h>
#include <Atlas/Objects/Root.h>
#include <Atlas/Objects/SmartPtr.h>
#include <libpq-fe.h>
#include <set>
#include <memory>
/// \brief Class to handle decoding Atlas encoded database records
class Decoder : public Atlas::Message::DecoderBase {
private:
virtual void messageArrived(const Atlas::Message::MapType & msg) {
m_check = true;
m_msg = msg;
}
bool m_check;
Atlas::Message::MapType m_msg;
public:
Decoder() : m_check (false) {
}
bool check() const {
return m_check;
}
const Atlas::Message::MapType & get() {
m_check = false;
return m_msg;
}
};
/// \brief Class to handle decoding Atlas encoded database records
class ObjectDecoder : public Atlas::Objects::ObjectsDecoder {
private:
virtual void objectArrived(const Atlas::Objects::Root & obj) {
m_check = true;
m_obj = obj;
}
bool m_check;
Atlas::Objects::Root m_obj;
public:
ObjectDecoder() : m_check (false) {
}
bool check() const {
return m_check;
}
const Atlas::Objects::Root & get() {
m_check = false;
return m_obj;
}
};
class DatabaseResult;
typedef std::vector<std::string> StringVector;
typedef std::set<std::string> TableSet;
typedef std::pair<std::string, ExecStatusType> DatabaseQuery;
typedef std::deque<DatabaseQuery> QueryQue;
/// \brief Class to provide interface to Database connection
///
/// Most SQL is generated from here, including queries for handling all
/// table creation, queries to simple non-inherited tables and more
class Database {
protected:
static Database * m_instance;
std::string m_rule_db;
TableSet allTables;
QueryQue pendingQueries;
bool m_queryInProgress;
Decoder m_d;
ObjectDecoder m_od;
PGconn * m_connection;
Database();
~Database();
// bool command(const std::string & cmd);
bool tuplesOk();
int commandOk();
public:
static const int MAINTAIN_VACUUM = 0x0100;
static const int MAINTAIN_VACUUM_FULL = 0x0001;
static const int MAINTAIN_VACUUM_ANALYZE = 0x0002;
static const int MAINTAIN_REINDEX = 0x0200;
typedef enum { OneToMany, ManyToMany, ManyToOne, OneToOne } RelationType;
typedef std::map<std::string, std::string> KeyValues;
PGconn * getConnection() const { return m_connection; }
const std::string & rule() const { return m_rule_db; }
bool queryInProgress() const { return m_queryInProgress; }
size_t queryQueueSize() const {
return pendingQueries.size();
}
int decodeObject(const std::string & data,
Atlas::Objects::Root &);
int decodeMessage(const std::string & data,
Atlas::Message::MapType &);
int encodeObject(const Atlas::Message::MapType &,
std::string &);
int putObject(const std::string & table,
const std::string &,
const Atlas::Message::MapType &,
const StringVector & = StringVector());
int getObject(const std::string & table,
const std::string & key,
Atlas::Message::MapType &);
int updateObject(const std::string & table,
const std::string & key,
const Atlas::Message::MapType&);
int delObject(const std::string &, const std::string & key);
bool hasKey(const std::string &, const std::string & key);
int getTable(const std::string & table,
std::map<std::string, Atlas::Objects::Root> &);
int clearTable(const std::string & table);
void reportError();
int connect(const std::string & context, std::string & error_msg);
static Database * instance();
static void cleanup();
int initConnection();
int createInstanceDatabase();
int initRule(bool createTables = false);
void shutdownConnection();
const DatabaseResult runSimpleSelectQuery(const std::string & query);
int runCommandQuery(const std::string & query);
// Interface for relations between tables.
int registerRelation(std::string & tablename,
const std::string & sourcetable,
const std::string & targettable,
RelationType kind = OneToMany);
const DatabaseResult selectRelation(const std::string & name,
const std::string & id);
int createRelationRow(const std::string & name,
const std::string & id,
const std::string & other);
int removeRelationRow(const std::string & name,
const std::string & id);
int removeRelationRowByOther(const std::string & name,
const std::string & other);
// Interface for simple tables that mainly just store Atlasish data.
int registerSimpleTable(const std::string & name,
const Atlas::Message::MapType & row);
const DatabaseResult selectSimpleRow(const std::string & name,
const std::string & id);
const DatabaseResult selectSimpleRowBy(const std::string & name,
const std::string & column,
const std::string & value);
int createSimpleRow(const std::string & name,
const std::string & id,
const std::string & columns,
const std::string & values);
int updateSimpleRow(const std::string & name,
const std::string & key,
const std::string & value,
const std::string & columns);
// Interface for the ID generation sequence.
int registerEntityIdGenerator();
/// Creates a new unique id for the database.
/// Note that this method will access the database, so it's a fairly expensive method.
long newId(std::string & id);
// Interface for Entity and Property tables.
int registerEntityTable(const std::map<std::string, int> & chunks);
int insertEntity(const std::string & id,
const std::string & loc,
const std::string & type,
int seq,
const std::string & value);
int updateEntityWithoutLoc(const std::string & id,
int seq,
const std::string & location_data);
int updateEntity(const std::string & id,
int seq,
const std::string & location_data,
const std::string & location_entity_id);
const DatabaseResult selectEntities(const std::string & loc);
int dropEntity(long id);
int registerPropertyTable();
int insertProperties(const std::string & id,
const KeyValues & tuples);
const DatabaseResult selectProperties(const std::string & loc);
int updateProperties(const std::string & id,
const KeyValues & tuples);
int registerThoughtsTable();
const DatabaseResult selectThoughts(const std::string & loc);
int replaceThoughts(const std::string & id,
const std::vector<std::string>& thoughts);
// Interface for CommPSQLSocket, so it can give us feedback
void queryResult(ExecStatusType);
void queryComplete();
int launchNewQuery();
int scheduleCommand(const std::string & query);
int clearPendingQuery();
int runMaintainance(int command = MAINTAIN_VACUUM);
};
/// \brief Class to encapsulate a result from the database.
///
/// This allows the result to be used in the upper layers in a database
/// independant way.
class DatabaseResult {
private:
struct PGresultDeleter {
void operator()(PGresult* p) const {
PQclear(p);
}
};
static PGresultDeleter deleter;
std::shared_ptr<PGresult> m_res;
public:
explicit DatabaseResult(PGresult * r) : m_res(r, deleter) { }
DatabaseResult(const DatabaseResult & dr) : m_res(dr.m_res) { }
DatabaseResult & operator=(const DatabaseResult & other) {
m_res = other.m_res;
return *this;
}
/// \brief Iterator for DatabaseResult
///
/// Minics STL iterator API
class const_iterator {
private:
const DatabaseResult & m_dr;
int m_row;
const_iterator(const DatabaseResult & dr, int r = 0) : m_dr(dr),
m_row(r) {
if (m_row != -1) {
if (m_row >= m_dr.size()) {
m_row = -1;
}
}
}
public:
const_iterator(const const_iterator & ci) : m_dr(ci.m_dr),
m_row(ci.m_row) { }
bool operator==(const const_iterator & other) {
return (m_row == other.m_row);
}
bool operator!=(const const_iterator & other) {
return (m_row != other.m_row);
}
const_iterator operator++() {
if (m_row != -1) {
if (++m_row >= m_dr.size()) {
m_row = -1;
}
}
return *this;
}
const char * column(int column) const {
if (m_row == -1) {
return 0;
}
return PQgetvalue(m_dr.m_res.get(), m_row, column);
}
const char * column(const char *) const;
void readColumn(const char *, int &) const;
void readColumn(const char *, float &) const;
void readColumn(const char *, double &) const;
void readColumn(const char *, std::string &) const;
void readColumn(const char *, Atlas::Message::MapType &) const;
friend class DatabaseResult;
};
int size() const { return PQntuples(m_res.get()); }
int empty() const { return (size() == 0); }
int columns() const { return PQnfields(m_res.get()); }
bool error() const { return (m_res.get() == NULL); }
const_iterator begin() const {
return const_iterator(*this);
}
const_iterator end() const {
return const_iterator(*this, -1);
}
// const_iterator find() perhaps
const char * field(int column, int row = 0) const {
return PQgetvalue(m_res.get(), row, column);
}
const char * field(const char * column, int row = 0) const;
};
#endif // COMMON_DATABASE_H
|