#include "sqlite3.hpp" #include #include #include #include #include #include #include namespace mapbox { namespace sqlite { class DatabaseImpl { public: DatabaseImpl(const char* filename, int flags) { const int error = sqlite3_open_v2(filename, &db, flags, nullptr); if (error != SQLITE_OK) { const auto message = sqlite3_errmsg(db); db = nullptr; throw Exception { error, message }; } } ~DatabaseImpl() { if (!db) return; const int error = sqlite3_close(db); if (error != SQLITE_OK) { throw Exception { error, sqlite3_errmsg(db) }; } } sqlite3* db = nullptr; }; class StatementImpl { public: StatementImpl(sqlite3* db, const char* sql) { const int error = sqlite3_prepare_v2(db, sql, -1, &stmt, nullptr); if (error != SQLITE_OK) { stmt = nullptr; throw Exception { error, sqlite3_errmsg(db) }; } } ~StatementImpl() { if (!stmt) return; sqlite3_finalize(stmt); } sqlite3_stmt* stmt = nullptr; }; template using optional = std::experimental::optional; static void errorLogCallback(void *, const int err, const char *msg) { if (err == SQLITE_ERROR) { mbgl::Log::Error(mbgl::Event::Database, "%s (Code %i)", msg, err); } else if (err == SQLITE_WARNING) { mbgl::Log::Warning(mbgl::Event::Database, "%s (Code %i)", msg, err); } else { mbgl::Log::Info(mbgl::Event::Database, "%s (Code %i)", msg, err); } } const static bool sqliteVersionCheck __attribute__((unused)) = []() { if (sqlite3_libversion_number() / 1000000 != SQLITE_VERSION_NUMBER / 1000000) { char message[96]; snprintf(message, 96, "sqlite3 libversion mismatch: headers report %d, but library reports %d", SQLITE_VERSION_NUMBER, sqlite3_libversion_number()); throw std::runtime_error(message); } // Enable SQLite logging before initializing the database. sqlite3_config(SQLITE_CONFIG_LOG, errorLogCallback, nullptr); return true; }(); Database::Database(const std::string &filename, int flags) : impl(std::make_unique(filename.c_str(), flags)) { } Database::Database(Database &&other) : impl(std::move(other.impl)) {} Database &Database::operator=(Database &&other) { std::swap(impl, other.impl); return *this; } Database::~Database() { } Database::operator bool() const { return impl.operator bool(); } void Database::setBusyTimeout(std::chrono::milliseconds timeout) { assert(impl); const int err = sqlite3_busy_timeout(impl->db, int(std::min(timeout.count(), std::numeric_limits::max()))); if (err != SQLITE_OK) { throw Exception { err, sqlite3_errmsg(impl->db) }; } } void Database::exec(const std::string &sql) { assert(impl); char *msg = nullptr; const int err = sqlite3_exec(impl->db, sql.c_str(), nullptr, nullptr, &msg); if (msg) { const std::string message = msg; sqlite3_free(msg); throw Exception { err, message }; } else if (err != SQLITE_OK) { throw Exception { err, sqlite3_errmsg(impl->db) }; } } Statement Database::prepare(const char *query) { assert(impl); return Statement(this, query); } int64_t Database::lastInsertRowid() const { assert(impl); return sqlite3_last_insert_rowid(impl->db); } uint64_t Database::changes() const { assert(impl); return sqlite3_changes(impl->db); } Statement::Statement(Database *db, const char *sql) : impl(std::make_unique(db->impl->db, sql)) { } Statement::Statement(Statement &&other) { *this = std::move(other); } Statement &Statement::operator=(Statement &&other) { std::swap(impl, other.impl); return *this; } Statement::~Statement() { } Statement::operator bool() const { return impl.operator bool(); } void Statement::check(int err) { if (err != SQLITE_OK) { throw Exception { err, sqlite3_errmsg(sqlite3_db_handle(impl->stmt)) }; } } template <> void Statement::bind(int offset, std::nullptr_t) { assert(impl); check(sqlite3_bind_null(impl->stmt, offset)); } template <> void Statement::bind(int offset, int8_t value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, int16_t value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, int32_t value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, int64_t value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, uint8_t value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, uint16_t value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, uint32_t value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, float value) { assert(impl); check(sqlite3_bind_double(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, double value) { assert(impl); check(sqlite3_bind_double(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, bool value) { assert(impl); check(sqlite3_bind_int(impl->stmt, offset, value)); } template <> void Statement::bind(int offset, const char *value) { assert(impl); check(sqlite3_bind_text(impl->stmt, offset, value, -1, SQLITE_STATIC)); } // We currently cannot use sqlite3_bind_blob64 / sqlite3_bind_text64 because they // was introduced in SQLite 3.8.7, and we need to support earlier versions: // iOS 7.0: 3.7.13 // iOS 8.2: 3.8.5 // According to http://stackoverflow.com/questions/14288128/what-version-of-sqlite-does-ios-provide, // the first iOS version with 3.8.7+ was 9.0, with 3.8.10.2. void Statement::bind(int offset, const char * value, std::size_t length, bool retain) { assert(impl); if (length > std::numeric_limits::max()) { throw std::range_error("value too long for sqlite3_bind_text"); } check(sqlite3_bind_text(impl->stmt, offset, value, int(length), retain ? SQLITE_TRANSIENT : SQLITE_STATIC)); } void Statement::bind(int offset, const std::string& value, bool retain) { bind(offset, value.data(), value.size(), retain); } void Statement::bindBlob(int offset, const void * value, std::size_t length, bool retain) { assert(impl); if (length > std::numeric_limits::max()) { throw std::range_error("value too long for sqlite3_bind_text"); } check(sqlite3_bind_blob(impl->stmt, offset, value, int(length), retain ? SQLITE_TRANSIENT : SQLITE_STATIC)); } void Statement::bindBlob(int offset, const std::vector& value, bool retain) { bindBlob(offset, value.data(), value.size(), retain); } template <> void Statement::bind( int offset, std::chrono::time_point value) { assert(impl); check(sqlite3_bind_int64(impl->stmt, offset, std::chrono::system_clock::to_time_t(value))); } template <> void Statement::bind(int offset, optional value) { if (!value) { bind(offset, nullptr); } else { bind(offset, *value); } } template <> void Statement::bind( int offset, optional> value) { if (!value) { bind(offset, nullptr); } else { bind(offset, *value); } } bool Statement::run() { assert(impl); const int err = sqlite3_step(impl->stmt); if (err == SQLITE_DONE) { return false; } else if (err == SQLITE_ROW) { return true; } else if (err != SQLITE_OK) { throw Exception { err, sqlite3_errmsg(sqlite3_db_handle(impl->stmt)) }; } else { return false; } } template <> int Statement::get(int offset) { assert(impl); return sqlite3_column_int(impl->stmt, offset); } template <> int64_t Statement::get(int offset) { assert(impl); return sqlite3_column_int64(impl->stmt, offset); } template <> double Statement::get(int offset) { assert(impl); return sqlite3_column_double(impl->stmt, offset); } template <> std::string Statement::get(int offset) { assert(impl); return { reinterpret_cast(sqlite3_column_blob(impl->stmt, offset)), size_t(sqlite3_column_bytes(impl->stmt, offset)) }; } template <> std::vector Statement::get(int offset) { assert(impl); const uint8_t* begin = reinterpret_cast(sqlite3_column_blob(impl->stmt, offset)); const uint8_t* end = begin + sqlite3_column_bytes(impl->stmt, offset); return { begin, end }; } template <> std::chrono::time_point Statement::get(int offset) { assert(impl); return std::chrono::time_point_cast( std::chrono::system_clock::from_time_t(sqlite3_column_int64(impl->stmt, offset))); } template <> optional Statement::get(int offset) { assert(impl); if (sqlite3_column_type(impl->stmt, offset) == SQLITE_NULL) { return optional(); } else { return get(offset); } } template <> optional Statement::get(int offset) { assert(impl); if (sqlite3_column_type(impl->stmt, offset) == SQLITE_NULL) { return optional(); } else { return get(offset); } } template <> optional Statement::get(int offset) { assert(impl); if (sqlite3_column_type(impl->stmt, offset) == SQLITE_NULL) { return optional(); } else { return get(offset); } } template <> optional> Statement::get(int offset) { assert(impl); if (sqlite3_column_type(impl->stmt, offset) == SQLITE_NULL) { return {}; } else { return get>( offset); } } void Statement::reset() { assert(impl); sqlite3_reset(impl->stmt); } void Statement::clearBindings() { assert(impl); sqlite3_clear_bindings(impl->stmt); } Transaction::Transaction(Database& db_, Mode mode) : db(db_) { switch (mode) { case Deferred: db.exec("BEGIN DEFERRED TRANSACTION"); break; case Immediate: db.exec("BEGIN IMMEDIATE TRANSACTION"); break; case Exclusive: db.exec("BEGIN EXCLUSIVE TRANSACTION"); break; } } Transaction::~Transaction() { if (needRollback) { try { rollback(); } catch (...) { // Ignore failed rollbacks in destructor. } } } void Transaction::commit() { needRollback = false; db.exec("COMMIT TRANSACTION"); } void Transaction::rollback() { needRollback = false; db.exec("ROLLBACK TRANSACTION"); } } // namespace sqlite } // namespace mapbox