18 #include "exception.h" 49 explicit Statement(::sqlite3*
db,
const string& sql_statement);
56 template <
class... PARAMS>
57 void bind(PARAMS&&... params) {
58 bindHelper(1, std::forward<PARAMS>(params)...);
63 void bindValue(
int index, std::nullopt_t);
64 void bindValue(
int index,
int value);
65 void bindValue(
int index, int64_t value);
66 void bindValue(
int index,
const string& value);
67 void bindValue(
int index,
const char* value);
68 void bindValue(
int index,
double value);
71 void bindValue(
int index, nullptr_t) =
delete;
74 void bindValue(
int index,
const optional<T>& value) {
75 if (value.has_value())
76 bindValue(index, *value);
78 bindValue(index, nullopt);
85 int columnCount()
const;
89 void columnValue(
int column, optional<int>& value)
const;
90 void columnValue(
int column, optional<int64_t>& value)
const;
91 void columnValue(
int column, optional<string>& value)
const;
92 void columnValue(
int column, optional<double>& value)
const;
95 template <
class T,
class... PARAMS>
96 void bindHelper(
int index, T&& value, PARAMS&&... params) {
97 bindValue(index, std::forward<T>(value));
98 bindHelper(index + 1, std::forward<PARAMS>(params)...);
101 void bindHelper(
int ) {}
104 ::sqlite3_stmt* stmt_ =
nullptr;
109 template <
class... TYPES>
113 using Row = tuple<optional<TYPES>...>;
123 size_t size()
const {
return results_.size(); }
126 bool empty()
const {
return results_.empty(); }
129 auto begin()
const {
return results_.begin(); }
132 auto end()
const {
return results_.end(); }
137 if (tuple_size<Row>::value != 1 || results_.size() != 1)
139 return get<0>(results_[0]);
142 void extractRow(
const Statement& statement) {
143 if (columnCount() > 0 && statement.columnCount() != columnCount()) {
145 "ResultSet column count does not match the statement column count");
149 extractColumns(statement, row, index_sequence_for<TYPES...>{});
150 results_.push_back(std::move(row));
154 template <
size_t... COLUMNS>
155 void extractColumns([[maybe_unused]]
const Statement& statement,
156 [[maybe_unused]] Row& row,
157 index_sequence<COLUMNS...>) {
158 (statement.columnValue(COLUMNS, std::get<COLUMNS>(row)), ...);
162 vector<Row> results_;
181 explicit Connection(
const string& filename,
OpenMode open_mode,
int busy_wait_ms = 500);
195 RowId lastInsertRowId()
const;
208 template <
class... RESULTS,
class... PARAMS>
209 ResultSet<RESULTS...>
exec(
const string& sql_statement, PARAMS&&... params) {
210 Statement prepared_statement(db_, sql_statement);
211 prepared_statement.bind(std::forward<PARAMS>(params)...);
214 while (prepared_statement.step())
215 results.extractRow(prepared_statement);
220 ::sqlite3* db_ =
nullptr;
242 bool completed_ =
false;
int columnCount() const
Count of columns in the results.
Definition: database.h:120
const auto & singleValue() const
Convenience helper for extracting a single value.
Definition: database.h:136
A very simple relational database abstraction on top of Sqlite.
Definition: database.h:178
OpenMode
Open database flag.
Definition: database.h:38
Maps to BEGIN DEFERRED TRANSACTION
The base for exception types in the Darwin framework.
Definition: exception.h:27
int64_t RowId
Represents the ID of a row in the database.
Definition: database.h:44
const Row & operator[](size_t index) const
Indexed access to the rows in the result.
Definition: database.h:117
auto begin() const
ResultSet begin iterator.
Definition: database.h:129
size_t size() const
Count of rows in the results.
Definition: database.h:123
Creates a new database (will fail if file already exists)
A prepared Sqlite statement.
Definition: database.h:47
tuple< optional< TYPES >... > Row
A results row.
Definition: database.h:113
Maps to BEGIN IMMEDIATE TRANSACTION
bool empty() const
Returns true if the ResultSet is empty.
Definition: database.h:126
Represents the results of executing a query.
Definition: database.h:110
A scope-based transaction guard.
Definition: database.h:224
Low level database utilities.
Definition: database.cpp:19
Classes derived from this are not copyable or movable.
Definition: utils.h:69
The database file must exist.
ResultSet< RESULTS... > exec(const string &sql_statement, PARAMS &&... params)
Executes the specified Sqlite statement and returns the results as a ResultSet.
Definition: database.h:209
Maps to BEGIN EXCLUSIVE TRANSACTION
TransactionOption
The type of Sqlite transaction.
Definition: database.h:171
auto end() const
ResultSet end iterator.
Definition: database.h:132