pandorafms/extras/anytermd/libpbe/include/Database.hh

679 lines
20 KiB
C++

// src/Database.hh
// This file is part of libpbe; see http://decimail.org
// (C) 2004 - 2007 Philip Endecott
// 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
// 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#ifndef libpbe_Database_hh
#define libpbe_Database_hh
#include "Exception.hh"
#include "FileDescriptor.hh"
#include "endian.hh"
#include <string>
#include <libpq-fe.h>
#include <boost/lexical_cast.hpp>
#include <boost/type_traits.hpp>
#include <boost/noncopyable.hpp>
#include <boost/static_assert.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/preprocessor/repetition/enum.hpp>
#include <boost/iterator/iterator_facade.hpp>
namespace pbe {
// The maximum number of query parameters is set by this macro, which user code can
// define before #including this file.
#ifndef PBE_DB_MAX_QUERY_PARAMS
#define PBE_DB_MAX_QUERY_PARAMS 7
#endif
// Making it too large has the disadvantage that error messages become even more
// incomprehensible, and may also increase compile times.
// private:
// We support queries with variable numbers of parameters by means of
// sentinel default paramter values. The sentinel is of a special empty type
// null_t.
// ? Could we just use void for this ?
struct null_t {};
static const null_t nullval = null_t();
// We need to map between C++ types and PostgreSQL types in query parameters.
// This enum lists all of the PostgreSQL types that we map to, plus two
// sentinels: unknown_type indicates that no PostgreSQL type corresponds to the
// supplied C++ type, and should result in a compile-time error; null_type is used
// in unused arguments slots in the varargs-like query syntax.
enum typecode_t { /*unknown_type=0,*/ null_type,
text_type, numeric_type, timestamptz_type, bytea_type,
float_type, double_type,
typecode_t_max };
// Now we use template specialisation to get the typecode_t corresponding to a C++
// type at compile time. get_typecode<T>() is the typecode_t for type T.
// The generic template is not implemented anywhere, so it will fail at link
// time for these unsupported types.
template <typename T> inline typecode_t get_typecode(void);
template <> inline typecode_t get_typecode<null_t>(void) { return null_type; }
template <> inline typecode_t get_typecode<std::string>(void) { return text_type; }
template <> inline typecode_t get_typecode<int>(void) { return numeric_type; }
template <> inline typecode_t get_typecode<time_t>(void) { return timestamptz_type; }
template <> inline typecode_t get_typecode<float>(void) { return float_type; }
template <> inline typecode_t get_typecode<double>(void) { return double_type; }
// public:
// class Database represents a connection to the database server.
// It's marked non-copyable because it's not clear what should be done with
// the underlying libpq connection object in that case.
class Database: public boost::noncopyable {
public:
// Connect to the database server using the supplied connection info.
Database(std::string conninfo);
// Disconnect on destruction.
~Database();
// Get the file descriptor for the connction to the server.
// This is useful if you want to use select() to wait for asynchronous
// notifications from the server. You obviously mustn't read or write to
// it.
const pbe::FileDescriptor& get_fd(void) const;
// Return any asynchronous notifications received from the server.
std::string get_any_notification(void);
private:
// libpq data structure representing the connection.
PGconn* pgconn;
// Connection file descriptor.
pbe::FileDescriptor conn_fd;
// This is needed so that we can support nested transactions.
// (PostgreSQL now has built-in support for this, so it should be reviewed.)
bool transaction_in_progress;
// Run arbitary queries
void exec_sql(std::string cmd);
friend class QueryCore;
friend class Transaction;
};
// Exceptions thrown by the Database functions:
class DatabaseException: public Exception {
public:
DatabaseException(PGconn* pgconn, std::string doing_what_):
postgres_error(PQerrorMessage(pgconn)),
doing_what(doing_what_) {}
void report(std::ostream& s) const;
private:
std::string postgres_error;
std::string doing_what;
};
class QueryFailed: public DatabaseException {
public:
QueryFailed(PGconn* pgconn, std::string query):
DatabaseException(pgconn,"Executing query " + query) {}
};
class TypeError: public DatabaseException {
public:
TypeError(PGconn* pgconn):
DatabaseException(pgconn,"Type error") {}
};
// Transactions. Typical usage:
// {
// Transaction t(db);
// ... run queries ...
// t.commit();
// }
// If the transaction goes out of scope without commit() having been called, e.g.
// because an exception was thrown and not caught, the transaction will be rolled
// back.
// TODO: PostgreSQL now has some sort of built-in support for nested
// transactions.
class Transaction: public boost::noncopyable {
public:
Transaction(Database& database_);
~Transaction();
void commit(void);
private:
Database& database;
bool nested;
bool committed;
};
// Convert from the representation returned by libpq to a normal C++ type.
template <typename T>
T decode_pq_res(const char* data, int length __attribute__((unused)));
template <>
inline int decode_pq_res<int>(const char* data, int length __attribute__((unused))) {
const int* valp = reinterpret_cast<const int*>(data);
return ntohl(*valp);
}
template <>
inline uint64_t decode_pq_res<uint64_t>(const char* data, int length __attribute__((unused))) {
const uint64_t* valp = reinterpret_cast<const uint64_t*>(data);
return ntoh64(*valp);
}
template <>
inline int64_t decode_pq_res<int64_t>(const char* data, int length __attribute__((unused))) {
const int64_t* valp = reinterpret_cast<const int64_t*>(data);
return ntoh64(*valp);
}
template <>
inline time_t decode_pq_res<time_t>(const char* data, int length __attribute__((unused))) {
// Timestamp values are returned by PostgreSQL as 64-bit microsecond
// values. 946684800000000 is a magic number to convert to the Unix
// 1970 epoch.
int64_t t = decode_pq_res<int64_t>(data, length);
return t/1000000 + 946684800;
}
template <>
inline std::string decode_pq_res<std::string>(const char* data, int length __attribute__((unused))) {
return std::string(data,length);
}
union float_or_int {
float f;
int i;
};
template <>
inline float decode_pq_res<float>(const char* data, int length __attribute__((unused))) {
const float_or_int* uptr = reinterpret_cast<const float_or_int*>(data);
float_or_int ucopy;
ucopy.i = ntohl(uptr->i);
return ucopy.f;
}
union double_or_two_ints {
double d;
int i[2];
};
template <>
inline double decode_pq_res<double>(const char* data, int length __attribute__((unused))) {
const double_or_two_ints* uptr = reinterpret_cast<const double_or_two_ints*>(data);
double_or_two_ints ucopy;
ucopy.i[0] = ntohl(uptr->i[0]);
ucopy.i[1] = ntohl(uptr->i[1]);
return ucopy.d;
}
// Result of a SELECT query:
class Result {
public:
// Construct from the result object from libpq
Result(boost::shared_ptr<PGresult>);
// For insert, update and delete statements the result tells us how many
// rows were inserted, updated or deleted, using the following:
int get_rows_changed(void) const;
// Everything below here is for getting at the table returned for a select
// statement:
// The size of the table
const int rows;
const int cols;
// Map between column names and numbers
int column(std::string name) const ;
std::string column_name(int pos) const;
// Get data
// The pointer returned by rawget is valid for as long as the Result object exists.
char* rawget(int row, int col) const;
int getlength(int row, int col) const;
template <typename T>
T get_nocheck(int row, int col) const {
return decode_pq_res<T>(rawget(row,col),getlength(row,col));
}
template <typename T>
T get(int row, int col) const {
check_column_type<T>(col);
return get_nocheck<T>(row,col);
}
template <typename T>
T get(int row, std::string colname) const {
return get<T>(row, column(colname));
}
#if 0
// ... does this work?
template <typename T>
T operator()(int row, int col) const {
}
template <typename T>
T operator()(int row, std::string colname) const {
return operator()<T>(row, column(colname));
}
#endif
bool is_null(int row, int col) const;
bool is_null(int row, std::string colname) const {
return is_null(row, column(colname));
}
// Column types
typecode_t column_typecode(int col) const;
template <typename T>
void check_column_type(int col) const {
if (column_typecode(col)!=get_typecode<T>()) {
throw StrException("type error for column "
+boost::lexical_cast<std::string>(col)+", expecting typecode "
+boost::lexical_cast<std::string>(get_typecode<T>())+" but got typecode "
+boost::lexical_cast<std::string>(column_typecode(col)));
}
}
private:
boost::shared_ptr<PGresult> res;
};
// Alternative result for a query that generates a single column.
// Values can be accessed using (), e.g. r(i).
template <typename T>
class ColumnResult: public Result {
public:
ColumnResult(Result r): Result(r) {
if (cols!=1) {
throw "Single column expected";
}
if (rows>0) {
check_column_type<T>(0);
}
}
T operator ()(int row) const { return get<T>(row,0); }
bool is_null (int row) const { return Result::is_null(row,0); }
class const_iterator:
public boost::iterator_facade<const_iterator,const T,boost::bidirectional_traversal_tag,T> {
public:
const_iterator():
res(NULL), row(0) {}
const_iterator(const const_iterator& other):
res(other.res), row(other.row) {}
private:
const ColumnResult& res;
int row;
friend class ColumnResult;
const_iterator(const ColumnResult& res_, int row_):
res(res_), row(row_) {}
friend class boost::iterator_core_access;
void increment() { ++row; }
void decrement() { --row; }
void advance(int n) { row+=n; }
int distance_to(const_iterator other) { return other.row-row; }
bool equal(const const_iterator& other) const {
return row==other.row && &res==&(other.res);
}
const T dereference() const {
return res(row);
}
};
const_iterator begin() const {
return const_iterator(*this,0);
}
const_iterator end() const {
return const_iterator(*this,rows);
}
};
// Alternative result for a query that generates a single value
// (e.g. a "count(*)" query).
// This is convertable to the type of that value.
template <typename T>
class SingletonResult: public Result {
public:
SingletonResult(Result r): Result(r) {
if (rows!=1 || cols!=1) {
throw "Singleton expected";
}
check_column_type<T>(0);
}
operator T () const { return get<T>(0,0); }
bool is_null () const { return Result::is_null(0,0); }
};
// Alternative result for a query that generates a zero or one values.
// This is convertable to the type of that value.
template <typename T>
class OptResult: public Result {
public:
OptResult(Result r): Result(r) {
if (rows>1 || cols!=1) {
throw "Zero or one values expected";
}
if (rows==1) {
check_column_type<T>(0);
}
}
operator T () { return get<T>(0,0); }
bool is_null () { return Result::is_null(0,0); }
bool empty() { return rows==0; }
};
// private:
// Generate names for prepared statements
class statement_name_t: public std::string {
public:
statement_name_t(void):
std::string("stmt_"+boost::lexical_cast<std::string>(counter++))
{}
private:
static int counter;
};
// Most of the work of Query is delagated to QueryCore which is not a template
// class and so need not be coded inline.
// This is "semi-private": code can use it if it needs to pass more than
// PBE_DB_MAX_QUERY_PARAMS paramters to a query, but if that's not necessary then it
// should not be used.
class QueryCore {
public:
QueryCore(Database& database_, std::string querystr_, int nparams,
typecode_t* argtypecodes, int* lengths, int* formats);
~QueryCore();
Result operator()(const char* enc_args[]);
Result runonce(const char* enc_args[]);
private:
Database& database;
const std::string querystr;
const bool params_ok;
const statement_name_t statement_name;
int nparams;
int* param_lengths;
int* param_formats;
Oid* argoids;
bool prepared;
void prepare(void);
};
// Convert from C++ types to the representation passed to libpq.
// Types are either binary or character. Binary is used for numeric types and
// character is used for text types. Binary types must be in network byte
// order.
// This uses template specialisation.
// It's OK for these function to modify their paramters in place.
template <typename T>
inline const char* encode_pq_arg(T& t);
template <>
inline const char* encode_pq_arg<null_t>(null_t& i __attribute__((unused)) ) {
return NULL;
}
template <>
inline const char* encode_pq_arg<int>(int& i) {
i = htonl(i);
return reinterpret_cast<const char*>(&i);
}
template <>
inline const char* encode_pq_arg<uint64_t>(uint64_t& i) {
i = hton64(i);
return reinterpret_cast<const char*>(&i);
}
template <>
inline const char* encode_pq_arg<int64_t>(int64_t& i) {
i = hton64(i);
return reinterpret_cast<const char*>(&i);
}
template <>
inline const char* encode_pq_arg<time_t>(time_t& t) {
// No! This leaks!
int64_t* iptr = new int64_t;
*iptr = static_cast<int64_t>(t-946684800)*1000000;
return encode_pq_arg<int64_t>(*iptr);
}
template <>
inline const char* encode_pq_arg<std::string>(std::string& s) {
return s.c_str();
}
static inline float htonfloat(float f) {
uint32_t i;
memcpy(&i,&f,4);
i = htonl(i);
float r;
memcpy(&r,&i,4);
return r;
}
template <>
inline const char* encode_pq_arg<float>(float& f) {
f = htonfloat(f);
return reinterpret_cast<const char*>(&f);
}
template <>
inline const char* encode_pq_arg<double>(double& d) {
uint32_t* ptr = reinterpret_cast<uint32_t*>(&d);
ptr[0] = htonl(ptr[0]);
ptr[1] = htonl(ptr[1]);
return reinterpret_cast<const char*>(&d);
}
// We need to know the size of the encoded data.
// In most cases this is sizeof(T), but there are exceptions when the
// encoding process changes the size.
template <typename T>
inline size_t get_enc_type_size(void) { return sizeof(T); }
template <>
inline size_t get_enc_type_size<time_t>(void) { return 8; }
// The Boost preprocessor library is used to build query parameter lists (etc.) with
// multiple argumnets.
#define PBE_DB_Q_TEMPLATE_PARAM(z,n,data) typename T##n=null_t
#define PBE_DB_Q_TEMPLATE_PARAMS BOOST_PP_ENUM(PBE_DB_MAX_QUERY_PARAMS,PBE_DB_Q_TEMPLATE_PARAM,)
// Expands to e.g. typename T0=null_t,typename T1=null_t
#define PBE_DB_Q_OP_APPLY_PARAM(z,n,data) T##n arg##n=nullval
#define PBE_DB_Q_OP_APPLY_PARAMS BOOST_PP_ENUM(PBE_DB_MAX_QUERY_PARAMS,PBE_DB_Q_OP_APPLY_PARAM,)
// Expands to e.g. T0 arg0=nullval,T1 arg1=nullval
#define PBE_DB_Q_BASE_TEMPL_PARAM(z,n,data) T##n
#define PBE_DB_Q_BASE_TEMPL_PARAMS BOOST_PP_ENUM(PBE_DB_MAX_QUERY_PARAMS,PBE_DB_Q_BASE_TEMPL_PARAM,)
// Expands to e.g. T0,T1
#define PBE_DB_Q_BASE_OP_APPLY_PARAM(z,n,data) arg##n
#define PBE_DB_Q_BASE_OP_APPLY_PARAMS BOOST_PP_ENUM(PBE_DB_MAX_QUERY_PARAMS,PBE_DB_Q_BASE_OP_APPLY_PARAM,)
// Expands to e.g. arg0,arg1
// This base class is used to form the parameters passed to Query, below, into arrays
// that can be passed to QueryCore.
template <PBE_DB_Q_TEMPLATE_PARAMS> struct QueryBase {
typecode_t typecodes[PBE_DB_MAX_QUERY_PARAMS];
int lengths[PBE_DB_MAX_QUERY_PARAMS];
int formats[PBE_DB_MAX_QUERY_PARAMS];
QueryBase() {
#define PBE_DB_QB_INIT_ONE(z,n,data) \
typecodes[n]=get_typecode<T##n>(); \
lengths[n]=get_enc_type_size<T##n>();\
formats[n]=!boost::is_same<T##n,std::string>::value;
BOOST_PP_REPEAT(PBE_DB_MAX_QUERY_PARAMS,PBE_DB_QB_INIT_ONE,)
}
};
// Prepare and run a query. Typical usage:
// Query<std::string,int> insert_item(db,"insert into items(name,qty) values ($1,$2)");
// insert_item("nut",100);
// insert_item("bolt",102);
// The query can have up to PBE_DB_MAX_QUERY_PARAMS parameters. This limit can be
// increased at the top of this file.
template <PBE_DB_Q_TEMPLATE_PARAMS> class Query:
private QueryBase<PBE_DB_Q_BASE_TEMPL_PARAMS> {
public:
// Create a query. SQL is supplied in querystr with parameters indicated by
// $1, $2 etc. C++ types of these parameters are the template paramters.
Query(Database& database, std::string querystr):
core(database, querystr, PBE_DB_MAX_QUERY_PARAMS,
QueryBase<PBE_DB_Q_BASE_TEMPL_PARAMS>::typecodes,
QueryBase<PBE_DB_Q_BASE_TEMPL_PARAMS>::lengths,
QueryBase<PBE_DB_Q_BASE_TEMPL_PARAMS>::formats)
{}
// Run a query. It is prepared the first time that it is run.
Result operator()(PBE_DB_Q_OP_APPLY_PARAMS) {
const char* arg_cs[PBE_DB_MAX_QUERY_PARAMS];
#define PBE_DB_Q_OP_APPLY_ONE(z,n,data) \
T##n arg##n##_c = arg##n; \
arg_cs[n] = encode_pq_arg(arg##n##_c);
BOOST_PP_REPEAT(PBE_DB_MAX_QUERY_PARAMS,PBE_DB_Q_OP_APPLY_ONE,)
return core(arg_cs);
}
// Run a query immediately, without preparation. Use this for queries that will only be
// run once.
Result runonce(PBE_DB_Q_OP_APPLY_PARAMS) {
const char* arg_cs[PBE_DB_MAX_QUERY_PARAMS];
#define PBE_DB_Q_RUNONCE_ONE(z,n,data) \
T##n arg##n##_c = arg##n; \
arg_cs[n] = encode_pq_arg(arg##n##_c);
BOOST_PP_REPEAT(PBE_DB_MAX_QUERY_PARAMS,PBE_DB_Q_RUNONCE_ONE,)
return core.runonce(arg_cs);
}
private:
QueryCore core;
};
// A query that returns a ColumnResult.
template <typename RT, PBE_DB_Q_TEMPLATE_PARAMS>
class ColumnQuery: public Query< PBE_DB_Q_BASE_TEMPL_PARAMS> {
public:
typedef ColumnResult<RT> result_t;
ColumnQuery(Database& database, std::string querystr):
Query<PBE_DB_Q_BASE_TEMPL_PARAMS>(database,querystr)
{}
result_t operator()(PBE_DB_Q_OP_APPLY_PARAMS) {
return Query<PBE_DB_Q_BASE_TEMPL_PARAMS>::operator()(PBE_DB_Q_BASE_OP_APPLY_PARAMS);
}
result_t runonce(PBE_DB_Q_OP_APPLY_PARAMS) {
return Query<PBE_DB_Q_BASE_TEMPL_PARAMS>::runonce(PBE_DB_Q_BASE_OP_APPLY_PARAMS);
}
};
// A query that returns a SingeltonResult. In this case there's no need to
// delcare the result object, thanks to implicit type conversion. For
// example: int a = query(param);
template <typename RT, PBE_DB_Q_TEMPLATE_PARAMS>
class SingletonQuery: public Query< PBE_DB_Q_BASE_TEMPL_PARAMS> {
public:
typedef SingletonResult<RT> result_t;
SingletonQuery(Database& database, std::string querystr):
Query<PBE_DB_Q_BASE_TEMPL_PARAMS>(database,querystr)
{}
result_t operator()(PBE_DB_Q_OP_APPLY_PARAMS) {
return Query<PBE_DB_Q_BASE_TEMPL_PARAMS>::operator()(PBE_DB_Q_BASE_OP_APPLY_PARAMS);
}
result_t runonce(PBE_DB_Q_OP_APPLY_PARAMS) {
return Query<PBE_DB_Q_BASE_TEMPL_PARAMS>::runonce(PBE_DB_Q_BASE_OP_APPLY_PARAMS);
}
};
// A query that returns a OptResult.
template <typename RT, PBE_DB_Q_TEMPLATE_PARAMS>
class OptQuery: public Query< PBE_DB_Q_BASE_TEMPL_PARAMS> {
public:
typedef OptResult<RT> result_t;
OptQuery(Database& database, std::string querystr):
Query<PBE_DB_Q_BASE_TEMPL_PARAMS>(database,querystr)
{}
result_t operator()(PBE_DB_Q_OP_APPLY_PARAMS) {
return Query<PBE_DB_Q_BASE_TEMPL_PARAMS>::operator()(PBE_DB_Q_BASE_OP_APPLY_PARAMS);
}
result_t runonce(PBE_DB_Q_OP_APPLY_PARAMS) {
return Query<PBE_DB_Q_BASE_TEMPL_PARAMS>::runonce(PBE_DB_Q_BASE_OP_APPLY_PARAMS);
}
};
};
#endif