summaryrefslogtreecommitdiff
path: root/libpqpp/pq-modifycommand.cpp
blob: e140f0a0b609921ac7bef9a740241fafcbc2b1b3 (plain)
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
#include "pq-modifycommand.h"
#include "pq-error.h"
#include <stdlib.h>
#include "pq-connection.h"

PQ::ModifyCommand::ModifyCommand(Connection * conn, const std::string & sql, unsigned int no) :
	DB::Command(sql),
	DB::ModifyCommand(sql),
	PQ::Command(conn, sql, no),
	pstmt(prepare()->second)
{
}

PQ::ModifyCommand::~ModifyCommand()
{
}

PQ::Connection::PreparedStatements::const_iterator
PQ::ModifyCommand::prepare() const
{
	auto hash(std::hash<std::string>()(sql));
	auto i = c->preparedStatements.find(hash);
	if (i != c->preparedStatements.end()) {
		return i;
	}
	std::string psql;
	psql.reserve(sql.length() + 20);
	prepareSql(psql, sql);
	c->checkResultFree(PQprepare(
				c->conn, stmntName.c_str(), psql.c_str(), values.size(), NULL), PGRES_COMMAND_OK);
	return c->preparedStatements.insert({hash, stmntName}).first;
}

unsigned int
PQ::ModifyCommand::execute(bool anc)
{
	PGresult * res = PQexecPrepared(c->conn, pstmt.c_str(), values.size(), &values.front(), &lengths.front(), NULL, 0);
	c->checkResult(res, PGRES_COMMAND_OK, PGRES_TUPLES_OK);
	unsigned int rows = atoi(PQcmdTuples(res));
	PQclear(res);
	if (rows == 0 && !anc) {
		throw DB::NoRowsAffected();
	}
	return rows;
}