postgresql_engine.cpp 2.61 KB
Newer Older
1 2 3
#include <biboumi.h>
#ifdef PQ_FOUND

4 5
#include <utils/scopeguard.hpp>

6 7
#include <database/query.hpp>

louiz’'s avatar
louiz’ committed
8 9 10 11 12 13
#include <database/postgresql_engine.hpp>

#include <database/postgresql_statement.hpp>

#include <logger/logger.hpp>

14 15
#include <cstring>

louiz’'s avatar
louiz’ committed
16 17 18 19 20 21 22 23 24
PostgresqlEngine::PostgresqlEngine(PGconn*const conn):
    conn(conn)
{}

PostgresqlEngine::~PostgresqlEngine()
{
  PQfinish(this->conn);
}

25 26 27 28 29 30 31 32 33
static void logging_notice_processor(void*, const char* original)
{
  if (original && std::strlen(original) > 0)
    {
      std::string message{original, std::strlen(original) - 1};
      log_warning("PostgreSQL: ", message);
    }
}

louiz’'s avatar
louiz’ committed
34 35 36 37 38 39 40 41 42 43 44 45 46 47
std::unique_ptr<DatabaseEngine> PostgresqlEngine::open(const std::string& conninfo)
{
  PGconn* con = PQconnectdb(conninfo.data());

  if (!con)
    {
      log_error("Failed to allocate a Postgresql connection");
      throw std::runtime_error("");
    }
  const auto status = PQstatus(con);
  if (status != CONNECTION_OK)
    {
      const char* errmsg = PQerrorMessage(con);
      log_error("Postgresql connection failed: ", errmsg);
48
      PQfinish(con);
louiz’'s avatar
louiz’ committed
49 50
      throw std::runtime_error("failed to open connection.");
    }
51
  PQsetNoticeProcessor(con, &logging_notice_processor, nullptr);
louiz’'s avatar
louiz’ committed
52 53 54
  return std::make_unique<PostgresqlEngine>(con);
}

55
std::set<std::string> PostgresqlEngine::get_all_columns_from_table(const std::string& table_name)
louiz’'s avatar
louiz’ committed
56
{
57
  const auto query = "SELECT column_name from information_schema.columns where table_name='" + table_name + "'";
louiz’'s avatar
louiz’ committed
58
  auto statement = this->prepare(query);
59
  std::set<std::string> columns;
louiz’'s avatar
louiz’ committed
60 61

  while (statement->step() == StepResult::Row)
62
    columns.insert(statement->get_column_text(0));
louiz’'s avatar
louiz’ committed
63 64 65 66 67 68

  return columns;
}

std::tuple<bool, std::string> PostgresqlEngine::raw_exec(const std::string& query)
{
69 70 71 72
#ifdef DEBUG_SQL_QUERIES
  log_debug("SQL QUERY: ", query);
  const auto timer = make_sql_timer();
#endif
louiz’'s avatar
louiz’ committed
73
  PGresult* res = PQexec(this->conn, query.data());
74 75 76 77
  auto sg = utils::make_scope_guard([res](){
      PQclear(res);
  });

louiz’'s avatar
louiz’ committed
78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
  auto res_status = PQresultStatus(res);
  if (res_status != PGRES_COMMAND_OK)
    return std::make_tuple(false, PQresultErrorMessage(res));
  return std::make_tuple(true, std::string{});
}

std::unique_ptr<Statement> PostgresqlEngine::prepare(const std::string& query)
{
  return std::make_unique<PostgresqlStatement>(query, this->conn);
}

void PostgresqlEngine::extract_last_insert_rowid(Statement& statement)
{
  this->last_inserted_rowid = statement.get_column_int64(0);
}

std::string PostgresqlEngine::get_returning_id_sql_string(const std::string& col_name)
{
  return " RETURNING " + col_name;
}

99
std::string PostgresqlEngine::id_column_type()
louiz’'s avatar
louiz’ committed
100 101 102
{
  return "SERIAL";
}
103 104

#endif