2016-12-26 16:06:55 +01:00
|
|
|
|
#include "PgsqlConn.h"
|
2017-02-13 19:51:19 +01:00
|
|
|
|
#include "Pgsql_declare.h"
|
2016-12-26 16:06:55 +01:00
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
using namespace Pgsql;
|
|
|
|
|
|
|
2016-12-29 13:48:35 +01:00
|
|
|
|
|
|
|
|
|
|
|
2016-12-26 16:06:55 +01:00
|
|
|
|
|
2016-12-29 13:48:35 +01:00
|
|
|
|
Canceller::Canceller(PGcancel *c)
|
|
|
|
|
|
: m_cancel(c)
|
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
|
|
Canceller::Canceller(Canceller&& rhs)
|
|
|
|
|
|
: m_cancel(rhs.m_cancel)
|
|
|
|
|
|
{
|
|
|
|
|
|
rhs.m_cancel = nullptr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Canceller& Canceller::operator=(Canceller&& rhs)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (m_cancel) {
|
|
|
|
|
|
PQfreeCancel(m_cancel);
|
|
|
|
|
|
}
|
|
|
|
|
|
m_cancel = rhs.m_cancel;
|
|
|
|
|
|
rhs.m_cancel = nullptr;
|
|
|
|
|
|
return *this;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Canceller::~Canceller()
|
|
|
|
|
|
{
|
|
|
|
|
|
if (m_cancel) {
|
|
|
|
|
|
PQfreeCancel(m_cancel);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-01-08 09:58:34 +01:00
|
|
|
|
bool Canceller::cancel(std::string *error)
|
2016-12-29 13:48:35 +01:00
|
|
|
|
{
|
|
|
|
|
|
const int errbuf_size = 256;
|
|
|
|
|
|
char errbuf[errbuf_size];
|
2017-01-08 09:58:34 +01:00
|
|
|
|
bool res = PQcancel(m_cancel, errbuf, errbuf_size);
|
|
|
|
|
|
if (!res && error) {
|
|
|
|
|
|
*error = errbuf;
|
|
|
|
|
|
}
|
|
|
|
|
|
return res;
|
2016-12-29 13:48:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-12-26 16:06:55 +01:00
|
|
|
|
Connection::Connection() = default;
|
|
|
|
|
|
|
|
|
|
|
|
Connection::~Connection()
|
|
|
|
|
|
{
|
|
|
|
|
|
close();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Connection::Connection(Connection &&rhs)
|
|
|
|
|
|
: conn(rhs.conn)
|
|
|
|
|
|
{
|
|
|
|
|
|
rhs.conn = nullptr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Connection& Connection::operator=(Connection &&rhs)
|
|
|
|
|
|
{
|
|
|
|
|
|
close();
|
|
|
|
|
|
conn = rhs.conn;
|
|
|
|
|
|
rhs.conn = nullptr;
|
|
|
|
|
|
return *this;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Connection::close()
|
|
|
|
|
|
{
|
|
|
|
|
|
if (conn) {
|
|
|
|
|
|
PQfinish(conn);
|
|
|
|
|
|
conn = nullptr;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2016-12-29 13:48:35 +01:00
|
|
|
|
Canceller Connection::getCancel()
|
|
|
|
|
|
{
|
|
|
|
|
|
Canceller c(PQgetCancel(conn));
|
|
|
|
|
|
return c;
|
|
|
|
|
|
}
|
2016-12-26 16:06:55 +01:00
|
|
|
|
|
|
|
|
|
|
bool Connection::connect(const char *params)
|
|
|
|
|
|
{
|
|
|
|
|
|
bool result = false;
|
|
|
|
|
|
conn = PQconnectdb(params);
|
|
|
|
|
|
if (conn) {
|
|
|
|
|
|
ConnStatusType status = PQstatus(conn);
|
|
|
|
|
|
result = (status == CONNECTION_OK);
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-02-04 11:52:42 +01:00
|
|
|
|
bool Connection::connect(const char *const * keywords, const char* const * values, int expand_dbname)
|
|
|
|
|
|
{
|
|
|
|
|
|
bool result = false;
|
|
|
|
|
|
conn = PQconnectdbParams(keywords, values, expand_dbname);
|
|
|
|
|
|
if (conn) {
|
|
|
|
|
|
ConnStatusType status = PQstatus(conn);
|
|
|
|
|
|
result = (status == CONNECTION_OK);
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2016-12-27 15:41:11 +01:00
|
|
|
|
bool Connection::connectStart(const char* params)
|
|
|
|
|
|
{
|
|
|
|
|
|
conn = PQconnectStart(params);
|
|
|
|
|
|
return conn != nullptr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-01-15 21:01:40 +01:00
|
|
|
|
bool Connection::connectStart(const char * const *keywords,
|
|
|
|
|
|
const char * const *values)
|
|
|
|
|
|
{
|
|
|
|
|
|
conn = PQconnectStartParams(keywords, values, 0);
|
|
|
|
|
|
return conn != nullptr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2016-12-27 15:41:11 +01:00
|
|
|
|
PostgresPollingStatusType Connection::connectPoll()
|
|
|
|
|
|
{
|
|
|
|
|
|
return PQconnectPoll(conn);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ConnStatusType Connection::status()
|
|
|
|
|
|
{
|
|
|
|
|
|
return PQstatus(conn);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Connection::socket()
|
|
|
|
|
|
{
|
|
|
|
|
|
return PQsocket(conn);
|
|
|
|
|
|
}
|
2016-12-26 16:06:55 +01:00
|
|
|
|
|
|
|
|
|
|
std::string Connection::getErrorMessage() const
|
|
|
|
|
|
{
|
|
|
|
|
|
std::string result;
|
|
|
|
|
|
if (conn) {
|
|
|
|
|
|
result = PQerrorMessage(conn);
|
|
|
|
|
|
}
|
|
|
|
|
|
else {
|
|
|
|
|
|
result = "no connection";
|
|
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2016-12-27 15:41:11 +01:00
|
|
|
|
Result Connection::query(const char * command)
|
2016-12-26 16:06:55 +01:00
|
|
|
|
{
|
2016-12-27 15:41:11 +01:00
|
|
|
|
PGresult *result = PQexec(conn, command);
|
2016-12-29 13:48:35 +01:00
|
|
|
|
return Result(result);
|
2016-12-26 16:06:55 +01:00
|
|
|
|
}
|
2016-12-27 15:41:11 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool Connection::sendQuery(const char *query)
|
|
|
|
|
|
{
|
|
|
|
|
|
int res = PQsendQuery(conn, query);
|
|
|
|
|
|
return res == 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-01-08 09:58:34 +01:00
|
|
|
|
std::shared_ptr<Result> Connection::getResult()
|
2016-12-27 15:41:11 +01:00
|
|
|
|
{
|
|
|
|
|
|
PGresult *r = PQgetResult(conn);
|
|
|
|
|
|
if (r) {
|
2017-01-08 09:58:34 +01:00
|
|
|
|
return std::make_shared<Result>(r);
|
2016-12-27 15:41:11 +01:00
|
|
|
|
}
|
|
|
|
|
|
else {
|
|
|
|
|
|
return nullptr;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2016-12-27 21:22:49 +01:00
|
|
|
|
|
|
|
|
|
|
bool Connection::consumeInput()
|
|
|
|
|
|
{
|
|
|
|
|
|
int res = PQconsumeInput(conn);
|
|
|
|
|
|
return res == 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Connection::isBusy()
|
|
|
|
|
|
{
|
|
|
|
|
|
int res = PQisBusy(conn);
|
|
|
|
|
|
return res == 1;
|
|
|
|
|
|
}
|
2016-12-29 13:48:35 +01:00
|
|
|
|
|
|
|
|
|
|
void Connection::setNoticeReceiver(std::function<void(const PGresult *)> callback)
|
|
|
|
|
|
{
|
|
|
|
|
|
notifyReceiver = callback;
|
|
|
|
|
|
PQsetNoticeReceiver(conn,
|
|
|
|
|
|
Connection::notifyReceiveFunc
|
|
|
|
|
|
, reinterpret_cast<void*>(this));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Connection::notifyReceiveFunc(void *arg, const PGresult *result)
|
|
|
|
|
|
{
|
|
|
|
|
|
Connection *c = reinterpret_cast<Connection *>(arg);
|
|
|
|
|
|
c->notifyReceiver(result);
|
|
|
|
|
|
}
|