/* * Copyright (C) by Klaas Freitag * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include #include #include #include #include #include "ownsql.h" #include "common/utility.h" #include "common/asserts.h" #include #define SQLITE_SLEEP_TIME_USEC 100000 #define SQLITE_REPEAT_COUNT 20 #define SQLITE_DO(A) \ if (1) { \ _errId = (A); \ if (_errId != SQLITE_OK && _errId != SQLITE_DONE && _errId != SQLITE_ROW) { \ _error = QString::fromUtf8(sqlite3_errmsg(_db)); \ } \ } namespace OCC { Q_LOGGING_CATEGORY(lcSql, "nextcloud.sync.database.sql", QtInfoMsg) SqlDatabase::SqlDatabase() = default; SqlDatabase::~SqlDatabase() { close(); } bool SqlDatabase::isOpen() { return _db != nullptr; } bool SqlDatabase::openHelper(const QString &filename, int sqliteFlags) { if (isOpen()) { return true; } sqliteFlags |= SQLITE_OPEN_NOMUTEX; SQLITE_DO(sqlite3_open_v2(filename.toUtf8().constData(), &_db, sqliteFlags, nullptr)); if (_errId != SQLITE_OK) { qCWarning(lcSql) << "Error:" << _error << "for" << filename; if (_errId == SQLITE_CANTOPEN) { qCWarning(lcSql) << "CANTOPEN extended errcode: " << sqlite3_extended_errcode(_db); #if SQLITE_VERSION_NUMBER >= 3012000 qCWarning(lcSql) << "CANTOPEN system errno: " << sqlite3_system_errno(_db); #endif } close(); return false; } if (!_db) { qCWarning(lcSql) << "Error: no database for" << filename; return false; } sqlite3_busy_timeout(_db, 5000); return true; } SqlDatabase::CheckDbResult SqlDatabase::checkDb() { // quick_check can fail with a disk IO error when diskspace is low SqlQuery quick_check(*this); if (quick_check.prepare("PRAGMA quick_check;", /*allow_failure=*/true) != SQLITE_OK) { qCWarning(lcSql) << "Error preparing quick_check on database"; _errId = quick_check.errorId(); _error = quick_check.error(); return CheckDbResult::CantPrepare; } if (!quick_check.exec()) { qCWarning(lcSql) << "Error running quick_check on database"; _errId = quick_check.errorId(); _error = quick_check.error(); return CheckDbResult::CantExec; } quick_check.next(); QString result = quick_check.stringValue(0); if (result != QLatin1String("ok")) { qCWarning(lcSql) << "quick_check returned failure:" << result; return CheckDbResult::NotOk; } return CheckDbResult::Ok; } bool SqlDatabase::openOrCreateReadWrite(const QString &filename) { if (isOpen()) { return true; } if (!openHelper(filename, SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE)) { return false; } auto checkResult = checkDb(); if (checkResult != CheckDbResult::Ok) { if (checkResult == CheckDbResult::CantPrepare) { // When disk space is low, preparing may fail even though the db is fine. // Typically CANTOPEN or IOERR. qint64 freeSpace = Utility::freeDiskSpace(QFileInfo(filename).dir().absolutePath()); if (freeSpace != -1 && freeSpace < 1000000) { qCWarning(lcSql) << "Can't prepare consistency check and disk space is low:" << freeSpace; close(); return false; } // Even when there's enough disk space, it might very well be that the // file is on a read-only filesystem and can't be opened because of that. if (_errId == SQLITE_CANTOPEN) { qCWarning(lcSql) << "Can't open db to prepare consistency check, aborting"; close(); return false; } } qCCritical(lcSql) << "Consistency check failed, removing broken db" << filename; close(); QFile::remove(filename); return openHelper(filename, SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE); } return true; } bool SqlDatabase::openReadOnly(const QString &filename) { if (isOpen()) { return true; } if (!openHelper(filename, SQLITE_OPEN_READONLY)) { return false; } if (checkDb() != CheckDbResult::Ok) { qCWarning(lcSql) << "Consistency check failed in readonly mode, giving up" << filename; close(); return false; } return true; } QString SqlDatabase::error() const { const QString err(_error); // _error.clear(); return err; } void SqlDatabase::close() { if (_db) { foreach (auto q, _queries) { q->finish(); } SQLITE_DO(sqlite3_close(_db)); if (_errId != SQLITE_OK) qCWarning(lcSql) << "Closing database failed" << _error; _db = nullptr; } } bool SqlDatabase::transaction() { if (!_db) { return false; } SQLITE_DO(sqlite3_exec(_db, "BEGIN", nullptr, nullptr, nullptr)); return _errId == SQLITE_OK; } bool SqlDatabase::commit() { if (!_db) { return false; } SQLITE_DO(sqlite3_exec(_db, "COMMIT", nullptr, nullptr, nullptr)); return _errId == SQLITE_OK; } sqlite3 *SqlDatabase::sqliteDb() { return _db; } /* =========================================================================================== */ SqlQuery::SqlQuery(SqlDatabase &db) : _sqldb(&db) , _db(db.sqliteDb()) { } SqlQuery::~SqlQuery() { if (_stmt) { finish(); } } SqlQuery::SqlQuery(const QByteArray &sql, SqlDatabase &db) : _sqldb(&db) , _db(db.sqliteDb()) { prepare(sql); } int SqlQuery::prepare(const QByteArray &sql, bool allow_failure) { _sql = sql.trimmed(); if (_stmt) { finish(); } if (!_sql.isEmpty()) { int n = 0; int rc = 0; do { rc = sqlite3_prepare_v2(_db, _sql.constData(), -1, &_stmt, nullptr); if ((rc == SQLITE_BUSY) || (rc == SQLITE_LOCKED)) { n++; OCC::Utility::usleep(SQLITE_SLEEP_TIME_USEC); } } while ((n < SQLITE_REPEAT_COUNT) && ((rc == SQLITE_BUSY) || (rc == SQLITE_LOCKED))); _errId = rc; if (_errId != SQLITE_OK) { _error = QString::fromUtf8(sqlite3_errmsg(_db)); qCWarning(lcSql) << "Sqlite prepare statement error:" << _error << "in" << _sql; ENFORCE(allow_failure, "SQLITE Prepare error"); } else { ASSERT(_stmt); _sqldb->_queries.insert(this); } } return _errId; } /** * There is no overloads to QByteArray::startWith that takes Qt::CaseInsensitive. * Returns true if 'a' starts with 'b' in a case insensitive way */ static bool startsWithInsensitive(const QByteArray &a, const char *b) { size_t len = strlen(b); return a.size() >= len && qstrnicmp(a.constData(), b, Utility::convertSizeToUint(len)) == 0; } bool SqlQuery::isSelect() { return startsWithInsensitive(_sql, "SELECT"); } bool SqlQuery::isPragma() { return startsWithInsensitive(_sql, "PRAGMA"); } bool SqlQuery::exec() { qCDebug(lcSql) << "SQL exec" << _sql; if (!_stmt) { qCWarning(lcSql) << "Can't exec query, statement unprepared."; return false; } // Don't do anything for selects, that is how we use the lib :-| if (!isSelect() && !isPragma()) { int rc = 0, n = 0; do { rc = sqlite3_step(_stmt); if (rc == SQLITE_LOCKED) { rc = sqlite3_reset(_stmt); /* This will also return SQLITE_LOCKED */ n++; OCC::Utility::usleep(SQLITE_SLEEP_TIME_USEC); } else if (rc == SQLITE_BUSY) { OCC::Utility::usleep(SQLITE_SLEEP_TIME_USEC); n++; } } while ((n < SQLITE_REPEAT_COUNT) && ((rc == SQLITE_BUSY) || (rc == SQLITE_LOCKED))); _errId = rc; if (_errId != SQLITE_DONE && _errId != SQLITE_ROW) { _error = QString::fromUtf8(sqlite3_errmsg(_db)); qCWarning(lcSql) << "Sqlite exec statement error:" << _errId << _error << "in" << _sql; if (_errId == SQLITE_IOERR) { qCWarning(lcSql) << "IOERR extended errcode: " << sqlite3_extended_errcode(_db); #if SQLITE_VERSION_NUMBER >= 3012000 qCWarning(lcSql) << "IOERR system errno: " << sqlite3_system_errno(_db); #endif } } else { qCDebug(lcSql) << "Last exec affected" << numRowsAffected() << "rows."; } return (_errId == SQLITE_DONE); // either SQLITE_ROW or SQLITE_DONE } return true; } auto SqlQuery::next() -> NextResult { const bool firstStep = !sqlite3_stmt_busy(_stmt); int n = 0; forever { _errId = sqlite3_step(_stmt); if (n < SQLITE_REPEAT_COUNT && firstStep && (_errId == SQLITE_LOCKED || _errId == SQLITE_BUSY)) { sqlite3_reset(_stmt); // not necessary after sqlite version 3.6.23.1 n++; OCC::Utility::usleep(SQLITE_SLEEP_TIME_USEC); } else { break; } } NextResult result; result.ok = _errId == SQLITE_ROW || _errId == SQLITE_DONE; result.hasData = _errId == SQLITE_ROW; if (!result.ok) { _error = QString::fromUtf8(sqlite3_errmsg(_db)); qCWarning(lcSql) << "Sqlite step statement error:" << _errId << _error << "in" << _sql; } return result; } void SqlQuery::bindValueInternal(int pos, const QVariant &value) { int res = -1; if (!_stmt) { ASSERT(false); return; } switch (value.type()) { case QVariant::Int: case QVariant::Bool: res = sqlite3_bind_int(_stmt, pos, value.toInt()); break; case QVariant::Double: res = sqlite3_bind_double(_stmt, pos, value.toDouble()); break; case QVariant::UInt: case QVariant::LongLong: case QVariant::ULongLong: res = sqlite3_bind_int64(_stmt, pos, value.toLongLong()); break; case QVariant::DateTime: { const QDateTime dateTime = value.toDateTime(); const QString str = dateTime.toString(QStringLiteral("yyyy-MM-ddThh:mm:ss.zzz")); res = sqlite3_bind_text16(_stmt, pos, str.utf16(), str.size() * static_cast(sizeof(ushort)), SQLITE_TRANSIENT); break; } case QVariant::Time: { const QTime time = value.toTime(); const QString str = time.toString(QStringLiteral("hh:mm:ss.zzz")); res = sqlite3_bind_text16(_stmt, pos, str.utf16(), str.size() * static_cast(sizeof(ushort)), SQLITE_TRANSIENT); break; } case QVariant::String: { if (!value.toString().isNull()) { // lifetime of string == lifetime of its qvariant const auto *str = static_cast(value.constData()); res = sqlite3_bind_text16(_stmt, pos, str->utf16(), (str->size()) * static_cast(sizeof(QChar)), SQLITE_TRANSIENT); } else { res = sqlite3_bind_null(_stmt, pos); } break; } case QVariant::ByteArray: { auto ba = value.toByteArray(); res = sqlite3_bind_text(_stmt, pos, ba.constData(), ba.size(), SQLITE_TRANSIENT); break; } default: { QString str = value.toString(); // SQLITE_TRANSIENT makes sure that sqlite buffers the data res = sqlite3_bind_text16(_stmt, pos, str.utf16(), (str.size()) * static_cast(sizeof(QChar)), SQLITE_TRANSIENT); break; } } if (res != SQLITE_OK) { qCWarning(lcSql) << "ERROR binding SQL value:" << value << "error:" << res; } ASSERT(res == SQLITE_OK); } bool SqlQuery::nullValue(int index) { return sqlite3_column_type(_stmt, index) == SQLITE_NULL; } QString SqlQuery::stringValue(int index) { return QString::fromUtf16(static_cast(sqlite3_column_text16(_stmt, index))); } int SqlQuery::intValue(int index) { return sqlite3_column_int(_stmt, index); } quint64 SqlQuery::int64Value(int index) { return sqlite3_column_int64(_stmt, index); } QByteArray SqlQuery::baValue(int index) { return QByteArray(static_cast(sqlite3_column_blob(_stmt, index)), sqlite3_column_bytes(_stmt, index)); } QString SqlQuery::error() const { return _error; } int SqlQuery::errorId() const { return _errId; } const QByteArray &SqlQuery::lastQuery() const { return _sql; } int SqlQuery::numRowsAffected() { return sqlite3_changes(_db); } void SqlQuery::finish() { if (!_stmt) return; SQLITE_DO(sqlite3_finalize(_stmt)); _stmt = nullptr; if (_sqldb) { _sqldb->_queries.remove(this); } } void SqlQuery::reset_and_clear_bindings() { if (_stmt) { SQLITE_DO(sqlite3_reset(_stmt)); SQLITE_DO(sqlite3_clear_bindings(_stmt)); } } bool SqlQuery::initOrReset(const QByteArray &sql, OCC::SqlDatabase &db) { ENFORCE(!_sqldb || &db == _sqldb); _sqldb = &db; _db = db.sqliteDb(); if (_stmt) { reset_and_clear_bindings(); return true; } else { return prepare(sql) == 0; } } } // namespace OCC