aboutsummaryrefslogtreecommitdiff
path: root/frontend/DB.cpp
blob: 28fd164314bf7b8e5e1f247aae28cdd953e467e0 (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
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
#include "DB.h"
#include "Exception.h"

using namespace std;

DB::DB(const string & path) {
	sqlite3 * db = NULL;
	int ret = sqlite3_open_v2(path.c_str(), &db, SQLITE_OPEN_READWRITE, NULL);
	this->db = {
		db,
		[] (sqlite3 * db) {
			sqlite3_close_v2(db);
		},
	};
	if (ret != SQLITE_OK)
		throw Exception("sqlite3_open_v2: %d", ret);
}

DBStatement DB::prepare(const string & query) {
	return DBStatement(*this, query);
}

DBStatement::DBStatement(DB & parent, const string & query) : parent(parent) {
	sqlite3_stmt * stmt = NULL;
	int ret = sqlite3_prepare_v2(this->parent.db.get(), query.c_str(), query.size(), &stmt, NULL);
	this->stmt = {
		stmt,
		[] (sqlite3_stmt * stmt) {
			sqlite3_finalize(stmt);
		},
	};
	if (ret != SQLITE_OK)
		throw Exception("sqlite3_prepare_v2: %d", ret);
}

DBStatement & DBStatement::bind(const string & text) {
	int ret = sqlite3_bind_text(this->stmt.get(), this->param_index, text.data(), text.size(), NULL);
	if (ret != SQLITE_OK)
		throw Exception("sqlite3_bind_text: %d", ret);

	this->param_index++;
	return *this;
}

DBStatement & DBStatement::bind(const int & number) {
	int ret = sqlite3_bind_int(this->stmt.get(), this->param_index, number);
	if (ret != SQLITE_OK)
		throw Exception("sqlite3_bind_int: %d", ret);

	this->param_index++;
	return *this;
}

DBStatement & DBStatement::unbind() {
	this->param_index = 1;
	return *this;
}

void DBStatement::execute() {
	int ret = sqlite3_step(this->stmt.get());
	if (ret != SQLITE_DONE)
		throw Exception("sqlite3_step: %d", ret);
}

DBQueryRow DBStatement::row() {
	int ret = sqlite3_step(this->stmt.get());
	if (ret != SQLITE_ROW)
		throw Exception("sqlite3_step: %d", ret);

	return { *this };
}

DBQueryRow::DBQueryRow(DBStatement & parent) : parent(parent) { }

template <>
const char * DBQueryRow::col<const char*>(int index) {
	return reinterpret_cast<const char *>(sqlite3_column_text(this->parent.stmt.get(), index));
}

template <>
int DBQueryRow::col<int>(int index) {
	return sqlite3_column_int(this->parent.stmt.get(), index);
}