forked from qt-creator/qt-creator
Carray is not changing the pointer so it is fine to use a const pointer. Change-Id: Iad89666b6f543496e39cfeff17e069feb2002ff5 Reviewed-by: Thomas Hartmann <thomas.hartmann@qt.io> Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
816 lines
25 KiB
C++
816 lines
25 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "sqlitebasestatement.h"
|
|
|
|
#include "sqlitedatabase.h"
|
|
#include "sqlitedatabasebackend.h"
|
|
#include "sqliteexception.h"
|
|
|
|
#include "sqlite.h"
|
|
|
|
#include <condition_variable>
|
|
#include <mutex>
|
|
|
|
#if defined(__GNUC__)
|
|
# pragma GCC diagnostic ignored "-Wignored-qualifiers"
|
|
#endif
|
|
|
|
#define CARRAY_INT32 0 /* Data is 32-bit signed integers */
|
|
#define CARRAY_INT64 1 /* Data is 64-bit signed integers */
|
|
#define CARRAY_DOUBLE 2 /* Data is doubles */
|
|
#define CARRAY_TEXT 3 /* Data is char* */
|
|
|
|
extern "C" int sqlite3_carray_bind(
|
|
sqlite3_stmt *pStmt, int idx, void *aData, int nData, int mFlags, void (*xDestroy)(void *));
|
|
|
|
namespace Sqlite {
|
|
|
|
BaseStatement::BaseStatement(Utils::SmallStringView sqlStatement, Database &database)
|
|
: m_compiledStatement(nullptr, deleteCompiledStatement)
|
|
, m_database(database)
|
|
, m_bindingParameterCount(0)
|
|
, m_columnCount(0)
|
|
{
|
|
prepare(sqlStatement);
|
|
setBindingParameterCount();
|
|
setColumnCount();
|
|
}
|
|
|
|
void BaseStatement::deleteCompiledStatement(sqlite3_stmt *compiledStatement)
|
|
{
|
|
sqlite3_finalize(compiledStatement);
|
|
}
|
|
|
|
class UnlockNotification
|
|
{
|
|
public:
|
|
static void unlockNotifyCallBack(void **arguments, int argumentCount)
|
|
{
|
|
for (int index = 0; index < argumentCount; index++) {
|
|
UnlockNotification *unlockNotification = static_cast<UnlockNotification *>(arguments[index]);
|
|
unlockNotification->wakeupWaitCondition();
|
|
}
|
|
}
|
|
|
|
void wakeupWaitCondition()
|
|
{
|
|
{
|
|
std::lock_guard<std::mutex> lock(m_mutex);
|
|
m_fired = 1;
|
|
}
|
|
m_waitCondition.notify_all();
|
|
}
|
|
|
|
void wait()
|
|
{
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
|
|
|
m_waitCondition.wait(lock, [&] () { return m_fired; });
|
|
}
|
|
|
|
private:
|
|
bool m_fired = false;
|
|
std::condition_variable m_waitCondition;
|
|
std::mutex m_mutex;
|
|
};
|
|
|
|
void BaseStatement::waitForUnlockNotify() const
|
|
{
|
|
UnlockNotification unlockNotification;
|
|
int resultCode = sqlite3_unlock_notify(sqliteDatabaseHandle(),
|
|
UnlockNotification::unlockNotifyCallBack,
|
|
&unlockNotification);
|
|
|
|
if (resultCode == SQLITE_LOCKED)
|
|
throw DeadLock("SqliteStatement::waitForUnlockNotify: database is in a dead lock!");
|
|
|
|
unlockNotification.wait();
|
|
}
|
|
|
|
void BaseStatement::reset() const noexcept
|
|
{
|
|
sqlite3_reset(m_compiledStatement.get());
|
|
}
|
|
|
|
bool BaseStatement::next() const
|
|
{
|
|
int resultCode;
|
|
|
|
do {
|
|
resultCode = sqlite3_step(m_compiledStatement.get());
|
|
if (resultCode == SQLITE_LOCKED) {
|
|
waitForUnlockNotify();
|
|
sqlite3_reset(m_compiledStatement.get());
|
|
}
|
|
|
|
} while (resultCode == SQLITE_LOCKED);
|
|
|
|
if (resultCode == SQLITE_ROW)
|
|
return true;
|
|
else if (resultCode == SQLITE_DONE)
|
|
return false;
|
|
|
|
checkForStepError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::step() const
|
|
{
|
|
next();
|
|
}
|
|
|
|
int BaseStatement::columnCount() const
|
|
{
|
|
return m_columnCount;
|
|
}
|
|
|
|
void BaseStatement::bind(int index, NullValue)
|
|
{
|
|
int resultCode = sqlite3_bind_null(m_compiledStatement.get(), index);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, int value)
|
|
{
|
|
int resultCode = sqlite3_bind_int(m_compiledStatement.get(), index, value);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, long long value)
|
|
{
|
|
int resultCode = sqlite3_bind_int64(m_compiledStatement.get(), index, value);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, double value)
|
|
{
|
|
int resultCode = sqlite3_bind_double(m_compiledStatement.get(), index, value);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, void *pointer)
|
|
{
|
|
int resultCode = sqlite3_bind_pointer(m_compiledStatement.get(),
|
|
index,
|
|
pointer,
|
|
"carray",
|
|
nullptr);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, Utils::span<const int> values)
|
|
{
|
|
int resultCode = sqlite3_carray_bind(m_compiledStatement.get(),
|
|
index,
|
|
const_cast<int *>(values.data()),
|
|
static_cast<int>(values.size()),
|
|
CARRAY_INT32,
|
|
SQLITE_STATIC);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, Utils::span<const long long> values)
|
|
{
|
|
int resultCode = sqlite3_carray_bind(m_compiledStatement.get(),
|
|
index,
|
|
const_cast<long long *>(values.data()),
|
|
static_cast<int>(values.size()),
|
|
CARRAY_INT64,
|
|
SQLITE_STATIC);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, Utils::span<const double> values)
|
|
{
|
|
int resultCode = sqlite3_carray_bind(m_compiledStatement.get(),
|
|
index,
|
|
const_cast<double *>(values.data()),
|
|
static_cast<int>(values.size()),
|
|
CARRAY_DOUBLE,
|
|
SQLITE_STATIC);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, Utils::span<const char *> values)
|
|
{
|
|
int resultCode = sqlite3_carray_bind(m_compiledStatement.get(),
|
|
index,
|
|
values.data(),
|
|
static_cast<int>(values.size()),
|
|
CARRAY_TEXT,
|
|
SQLITE_STATIC);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, Utils::SmallStringView text)
|
|
{
|
|
int resultCode = sqlite3_bind_text(m_compiledStatement.get(),
|
|
index,
|
|
text.data(),
|
|
int(text.size()),
|
|
SQLITE_STATIC);
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, BlobView blobView)
|
|
{
|
|
int resultCode = SQLITE_OK;
|
|
|
|
if (blobView.empty()) {
|
|
sqlite3_bind_null(m_compiledStatement.get(), index);
|
|
} else {
|
|
resultCode = sqlite3_bind_blob64(m_compiledStatement.get(),
|
|
index,
|
|
blobView.data(),
|
|
blobView.size(),
|
|
SQLITE_STATIC);
|
|
}
|
|
|
|
if (resultCode != SQLITE_OK)
|
|
checkForBindingError(resultCode);
|
|
}
|
|
|
|
void BaseStatement::bind(int index, const Value &value)
|
|
{
|
|
switch (value.type()) {
|
|
case ValueType::Integer:
|
|
bind(index, value.toInteger());
|
|
break;
|
|
case ValueType::Float:
|
|
bind(index, value.toFloat());
|
|
break;
|
|
case ValueType::String:
|
|
bind(index, value.toStringView());
|
|
break;
|
|
case ValueType::Blob:
|
|
bind(index, value.toBlobView());
|
|
break;
|
|
case ValueType::Null:
|
|
bind(index, NullValue{});
|
|
break;
|
|
}
|
|
}
|
|
|
|
void BaseStatement::bind(int index, ValueView value)
|
|
{
|
|
switch (value.type()) {
|
|
case ValueType::Integer:
|
|
bind(index, value.toInteger());
|
|
break;
|
|
case ValueType::Float:
|
|
bind(index, value.toFloat());
|
|
break;
|
|
case ValueType::String:
|
|
bind(index, value.toStringView());
|
|
break;
|
|
case ValueType::Blob:
|
|
bind(index, value.toBlobView());
|
|
break;
|
|
case ValueType::Null:
|
|
bind(index, NullValue{});
|
|
break;
|
|
}
|
|
}
|
|
|
|
void BaseStatement::prepare(Utils::SmallStringView sqlStatement)
|
|
{
|
|
int resultCode;
|
|
|
|
do {
|
|
sqlite3_stmt *sqliteStatement = nullptr;
|
|
resultCode = sqlite3_prepare_v2(sqliteDatabaseHandle(),
|
|
sqlStatement.data(),
|
|
int(sqlStatement.size()),
|
|
&sqliteStatement,
|
|
nullptr);
|
|
m_compiledStatement.reset(sqliteStatement);
|
|
|
|
if (resultCode == SQLITE_LOCKED)
|
|
waitForUnlockNotify();
|
|
|
|
} while (resultCode == SQLITE_LOCKED);
|
|
|
|
|
|
if (resultCode != SQLITE_OK)
|
|
checkForPrepareError(resultCode);
|
|
}
|
|
|
|
sqlite3 *BaseStatement::sqliteDatabaseHandle() const
|
|
{
|
|
return m_database.backend().sqliteDatabaseHandle();
|
|
}
|
|
|
|
void BaseStatement::checkForStepError(int resultCode) const
|
|
{
|
|
switch (resultCode) {
|
|
case SQLITE_BUSY_RECOVERY:
|
|
case SQLITE_BUSY_SNAPSHOT:
|
|
case SQLITE_BUSY_TIMEOUT:
|
|
case SQLITE_BUSY:
|
|
throwStatementIsBusy("SqliteStatement::stepStatement: database engine was unable to "
|
|
"acquire the database locks!");
|
|
case SQLITE_ERROR_MISSING_COLLSEQ:
|
|
case SQLITE_ERROR_RETRY:
|
|
case SQLITE_ERROR_SNAPSHOT:
|
|
case SQLITE_ERROR:
|
|
throwStatementHasError("SqliteStatement::stepStatement: run-time error (such as a "
|
|
"constraint violation) has occurred!");
|
|
case SQLITE_MISUSE:
|
|
throwStatementIsMisused("SqliteStatement::stepStatement: was called inappropriately!");
|
|
case SQLITE_CONSTRAINT_CHECK:
|
|
case SQLITE_CONSTRAINT_COMMITHOOK:
|
|
case SQLITE_CONSTRAINT_FOREIGNKEY:
|
|
case SQLITE_CONSTRAINT_FUNCTION:
|
|
case SQLITE_CONSTRAINT_NOTNULL:
|
|
case SQLITE_CONSTRAINT_PINNED:
|
|
case SQLITE_CONSTRAINT_PRIMARYKEY:
|
|
case SQLITE_CONSTRAINT_ROWID:
|
|
case SQLITE_CONSTRAINT_TRIGGER:
|
|
case SQLITE_CONSTRAINT_UNIQUE:
|
|
case SQLITE_CONSTRAINT_VTAB:
|
|
case SQLITE_CONSTRAINT:
|
|
throwConstraintPreventsModification(
|
|
"SqliteStatement::stepStatement: contraint prevent insert or update!");
|
|
case SQLITE_TOOBIG:
|
|
throwTooBig("SqliteStatement::stepStatement: Some is to bigger than SQLITE_MAX_LENGTH.");
|
|
case SQLITE_SCHEMA:
|
|
throwSchemaChangeError("SqliteStatement::stepStatement: Schema changed but the statement "
|
|
"cannot be recompiled.");
|
|
case SQLITE_READONLY_CANTINIT:
|
|
case SQLITE_READONLY_CANTLOCK:
|
|
case SQLITE_READONLY_DBMOVED:
|
|
case SQLITE_READONLY_DIRECTORY:
|
|
case SQLITE_READONLY_RECOVERY:
|
|
case SQLITE_READONLY_ROLLBACK:
|
|
case SQLITE_READONLY:
|
|
throwCannotWriteToReadOnlyConnection(
|
|
"SqliteStatement::stepStatement: Cannot write to read only connection");
|
|
case SQLITE_PROTOCOL:
|
|
throwProtocolError(
|
|
"SqliteStatement::stepStatement: Something strang with the file locking happened.");
|
|
case SQLITE_NOMEM:
|
|
throw std::bad_alloc();
|
|
case SQLITE_NOLFS:
|
|
throwDatabaseExceedsMaximumFileSize(
|
|
"SqliteStatement::stepStatement: Database exceeds maximum file size.");
|
|
case SQLITE_MISMATCH:
|
|
throwDataTypeMismatch(
|
|
"SqliteStatement::stepStatement: Most probably you used not an integer for a rowid.");
|
|
case SQLITE_LOCKED_SHAREDCACHE:
|
|
case SQLITE_LOCKED_VTAB:
|
|
case SQLITE_LOCKED:
|
|
throwConnectionIsLocked("SqliteStatement::stepStatement: Database connection is locked.");
|
|
case SQLITE_IOERR_AUTH:
|
|
case SQLITE_IOERR_BEGIN_ATOMIC:
|
|
case SQLITE_IOERR_BLOCKED:
|
|
case SQLITE_IOERR_CHECKRESERVEDLOCK:
|
|
case SQLITE_IOERR_CLOSE:
|
|
case SQLITE_IOERR_COMMIT_ATOMIC:
|
|
case SQLITE_IOERR_CONVPATH:
|
|
case SQLITE_IOERR_DATA:
|
|
case SQLITE_IOERR_DELETE:
|
|
case SQLITE_IOERR_DELETE_NOENT:
|
|
case SQLITE_IOERR_DIR_CLOSE:
|
|
case SQLITE_IOERR_DIR_FSYNC:
|
|
case SQLITE_IOERR_FSTAT:
|
|
case SQLITE_IOERR_FSYNC:
|
|
case SQLITE_IOERR_GETTEMPPATH:
|
|
case SQLITE_IOERR_LOCK:
|
|
case SQLITE_IOERR_MMAP:
|
|
case SQLITE_IOERR_NOMEM:
|
|
case SQLITE_IOERR_RDLOCK:
|
|
case SQLITE_IOERR_READ:
|
|
case SQLITE_IOERR_ROLLBACK_ATOMIC:
|
|
case SQLITE_IOERR_SEEK:
|
|
case SQLITE_IOERR_SHMLOCK:
|
|
case SQLITE_IOERR_SHMMAP:
|
|
case SQLITE_IOERR_SHMOPEN:
|
|
case SQLITE_IOERR_SHMSIZE:
|
|
case SQLITE_IOERR_SHORT_READ:
|
|
case SQLITE_IOERR_TRUNCATE:
|
|
case SQLITE_IOERR_UNLOCK:
|
|
case SQLITE_IOERR_VNODE:
|
|
case SQLITE_IOERR_WRITE:
|
|
case SQLITE_IOERR:
|
|
throwInputOutputError("SqliteStatement::stepStatement: An IO error happened.");
|
|
case SQLITE_INTERRUPT:
|
|
throwExecutionInterrupted("SqliteStatement::stepStatement: Execution was interrupted.");
|
|
case SQLITE_CORRUPT_INDEX:
|
|
case SQLITE_CORRUPT_SEQUENCE:
|
|
case SQLITE_CORRUPT_VTAB:
|
|
case SQLITE_CORRUPT:
|
|
throwDatabaseIsCorrupt("SqliteStatement::stepStatement: Database is corrupt.");
|
|
case SQLITE_CANTOPEN_CONVPATH:
|
|
case SQLITE_CANTOPEN_DIRTYWAL:
|
|
case SQLITE_CANTOPEN_FULLPATH:
|
|
case SQLITE_CANTOPEN_ISDIR:
|
|
case SQLITE_CANTOPEN_NOTEMPDIR:
|
|
case SQLITE_CANTOPEN_SYMLINK:
|
|
case SQLITE_CANTOPEN:
|
|
throwCannotOpen("SqliteStatement::stepStatement: Cannot open database or temporary file.");
|
|
}
|
|
|
|
throwUnknowError("SqliteStatement::stepStatement: unknown error has happened");
|
|
}
|
|
|
|
void BaseStatement::checkForPrepareError(int resultCode) const
|
|
{
|
|
switch (resultCode) {
|
|
case SQLITE_BUSY_RECOVERY:
|
|
case SQLITE_BUSY_SNAPSHOT:
|
|
case SQLITE_BUSY_TIMEOUT:
|
|
case SQLITE_BUSY:
|
|
throwStatementIsBusy("SqliteStatement::prepareStatement: database engine was unable to "
|
|
"acquire the database locks!");
|
|
case SQLITE_ERROR_MISSING_COLLSEQ:
|
|
case SQLITE_ERROR_RETRY:
|
|
case SQLITE_ERROR_SNAPSHOT:
|
|
case SQLITE_ERROR:
|
|
throwStatementHasError("SqliteStatement::prepareStatement: run-time error (such as a "
|
|
"constraint violation) has occurred!");
|
|
case SQLITE_MISUSE:
|
|
throwStatementIsMisused("SqliteStatement::prepareStatement: was called inappropriately!");
|
|
case SQLITE_IOERR_AUTH:
|
|
case SQLITE_IOERR_BEGIN_ATOMIC:
|
|
case SQLITE_IOERR_BLOCKED:
|
|
case SQLITE_IOERR_CHECKRESERVEDLOCK:
|
|
case SQLITE_IOERR_CLOSE:
|
|
case SQLITE_IOERR_COMMIT_ATOMIC:
|
|
case SQLITE_IOERR_CONVPATH:
|
|
case SQLITE_IOERR_DATA:
|
|
case SQLITE_IOERR_DELETE:
|
|
case SQLITE_IOERR_DELETE_NOENT:
|
|
case SQLITE_IOERR_DIR_CLOSE:
|
|
case SQLITE_IOERR_DIR_FSYNC:
|
|
case SQLITE_IOERR_FSTAT:
|
|
case SQLITE_IOERR_FSYNC:
|
|
case SQLITE_IOERR_GETTEMPPATH:
|
|
case SQLITE_IOERR_LOCK:
|
|
case SQLITE_IOERR_MMAP:
|
|
case SQLITE_IOERR_NOMEM:
|
|
case SQLITE_IOERR_RDLOCK:
|
|
case SQLITE_IOERR_READ:
|
|
case SQLITE_IOERR_ROLLBACK_ATOMIC:
|
|
case SQLITE_IOERR_SEEK:
|
|
case SQLITE_IOERR_SHMLOCK:
|
|
case SQLITE_IOERR_SHMMAP:
|
|
case SQLITE_IOERR_SHMOPEN:
|
|
case SQLITE_IOERR_SHMSIZE:
|
|
case SQLITE_IOERR_SHORT_READ:
|
|
case SQLITE_IOERR_TRUNCATE:
|
|
case SQLITE_IOERR_UNLOCK:
|
|
case SQLITE_IOERR_VNODE:
|
|
case SQLITE_IOERR_WRITE:
|
|
case SQLITE_IOERR:
|
|
throwInputOutputError("SqliteStatement::prepareStatement: IO error happened!");
|
|
}
|
|
|
|
throwUnknowError("SqliteStatement::prepareStatement: unknown error has happened");
|
|
}
|
|
|
|
void BaseStatement::checkForBindingError(int resultCode) const
|
|
{
|
|
switch (resultCode) {
|
|
case SQLITE_TOOBIG: throwBingingTooBig("SqliteStatement::bind: string or blob are over size limits(SQLITE_LIMIT_LENGTH)!");
|
|
case SQLITE_RANGE : throwBindingIndexIsOutOfRange("SqliteStatement::bind: binding index is out of range!");
|
|
case SQLITE_NOMEM: throw std::bad_alloc();
|
|
case SQLITE_MISUSE: throwStatementIsMisused("SqliteStatement::bind: was called inappropriately!");
|
|
}
|
|
|
|
throwUnknowError("SqliteStatement::bind: unknown error has happened");
|
|
}
|
|
|
|
void BaseStatement::checkColumnCount(int columnCount) const
|
|
{
|
|
if (columnCount != m_columnCount)
|
|
throw ColumnCountDoesNotMatch("SqliteStatement::values: column count does not match!");
|
|
}
|
|
|
|
void BaseStatement::checkBindingName(int index) const
|
|
{
|
|
if (index <= 0 || index > m_bindingParameterCount)
|
|
throwWrongBingingName("SqliteStatement::bind: binding name are not exists in this statement!");
|
|
}
|
|
|
|
void BaseStatement::setBindingParameterCount()
|
|
{
|
|
m_bindingParameterCount = sqlite3_bind_parameter_count(m_compiledStatement.get());
|
|
}
|
|
|
|
Utils::SmallStringView chopFirstLetter(const char *rawBindingName)
|
|
{
|
|
if (rawBindingName != nullptr)
|
|
return Utils::SmallStringView(++rawBindingName);
|
|
|
|
return Utils::SmallStringView("");
|
|
}
|
|
|
|
void BaseStatement::setColumnCount()
|
|
{
|
|
m_columnCount = sqlite3_column_count(m_compiledStatement.get());
|
|
}
|
|
|
|
bool BaseStatement::isReadOnlyStatement() const
|
|
{
|
|
return sqlite3_stmt_readonly(m_compiledStatement.get());
|
|
}
|
|
|
|
void BaseStatement::throwStatementIsBusy(const char *whatHasHappened) const
|
|
{
|
|
throw StatementIsBusy(whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle()));
|
|
}
|
|
|
|
void BaseStatement::throwStatementHasError(const char *whatHasHappened) const
|
|
{
|
|
throw StatementHasError(whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle()));
|
|
}
|
|
|
|
void BaseStatement::throwStatementIsMisused(const char *whatHasHappened) const
|
|
{
|
|
throw StatementIsMisused(whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle()));
|
|
}
|
|
|
|
void BaseStatement::throwInputOutputError(const char *whatHasHappened) const
|
|
{
|
|
throw InputOutputError(whatHasHappened);
|
|
}
|
|
|
|
void BaseStatement::throwConstraintPreventsModification(const char *whatHasHappened) const
|
|
{
|
|
throw ConstraintPreventsModification(whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle()));
|
|
}
|
|
|
|
void BaseStatement::throwNoValuesToFetch(const char *whatHasHappened) const
|
|
{
|
|
throw NoValuesToFetch(whatHasHappened);
|
|
}
|
|
|
|
void BaseStatement::throwBindingIndexIsOutOfRange(const char *whatHasHappened) const
|
|
{
|
|
throw BindingIndexIsOutOfRange(whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle()));
|
|
}
|
|
|
|
void BaseStatement::throwWrongBingingName(const char *whatHasHappened) const
|
|
{
|
|
throw WrongBindingName(whatHasHappened);
|
|
}
|
|
|
|
void BaseStatement::throwUnknowError(const char *whatHasHappened) const
|
|
{
|
|
if (sqliteDatabaseHandle())
|
|
throw UnknowError(whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle()));
|
|
else
|
|
throw UnknowError(whatHasHappened);
|
|
}
|
|
|
|
void BaseStatement::throwBingingTooBig(const char *whatHasHappened) const
|
|
{
|
|
throw BindingTooBig(whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle()));
|
|
}
|
|
|
|
void BaseStatement::throwTooBig(const char *whatHasHappened) const
|
|
{
|
|
throw TooBig{whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle())};
|
|
}
|
|
|
|
void BaseStatement::throwSchemaChangeError(const char *whatHasHappened) const
|
|
{
|
|
throw SchemeChangeError{whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle())};
|
|
}
|
|
|
|
void BaseStatement::throwCannotWriteToReadOnlyConnection(const char *whatHasHappened) const
|
|
{
|
|
throw CannotWriteToReadOnlyConnection{whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle())};
|
|
}
|
|
|
|
void BaseStatement::throwProtocolError(const char *whatHasHappened) const
|
|
{
|
|
throw ProtocolError{whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle())};
|
|
}
|
|
|
|
void BaseStatement::throwDatabaseExceedsMaximumFileSize(const char *whatHasHappened) const
|
|
{
|
|
throw DatabaseExceedsMaximumFileSize{whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle())};
|
|
}
|
|
|
|
void BaseStatement::throwDataTypeMismatch(const char *whatHasHappened) const
|
|
{
|
|
throw DataTypeMismatch{whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle())};
|
|
}
|
|
|
|
void BaseStatement::throwConnectionIsLocked(const char *whatHasHappened) const
|
|
{
|
|
throw ConnectionIsLocked{whatHasHappened, sqlite3_errmsg(sqliteDatabaseHandle())};
|
|
}
|
|
|
|
void BaseStatement::throwExecutionInterrupted(const char *whatHasHappened) const
|
|
{
|
|
throw ExecutionInterrupted{whatHasHappened};
|
|
}
|
|
|
|
void BaseStatement::throwDatabaseIsCorrupt(const char *whatHasHappened) const
|
|
{
|
|
throw DatabaseIsCorrupt{whatHasHappened};
|
|
}
|
|
|
|
void BaseStatement::throwCannotOpen(const char *whatHasHappened) const
|
|
{
|
|
throw CannotOpen{whatHasHappened};
|
|
}
|
|
|
|
QString BaseStatement::columnName(int column) const
|
|
{
|
|
return QString::fromUtf8(sqlite3_column_name(m_compiledStatement.get(), column));
|
|
}
|
|
|
|
Database &BaseStatement::database() const
|
|
{
|
|
return m_database;
|
|
}
|
|
|
|
namespace {
|
|
template<typename StringType>
|
|
StringType textForColumn(sqlite3_stmt *sqlStatment, int column)
|
|
{
|
|
const char *text = reinterpret_cast<const char*>(sqlite3_column_text(sqlStatment, column));
|
|
std::size_t size = std::size_t(sqlite3_column_bytes(sqlStatment, column));
|
|
|
|
return StringType(text, size);
|
|
}
|
|
|
|
BlobView blobForColumn(sqlite3_stmt *sqlStatment, int column)
|
|
{
|
|
const std::byte *blob = reinterpret_cast<const std::byte *>(
|
|
sqlite3_column_blob(sqlStatment, column));
|
|
std::size_t size = std::size_t(sqlite3_column_bytes(sqlStatment, column));
|
|
|
|
return {blob, size};
|
|
}
|
|
|
|
BlobView convertToBlobForColumn(sqlite3_stmt *sqlStatment, int column)
|
|
{
|
|
int dataType = sqlite3_column_type(sqlStatment, column);
|
|
if (dataType == SQLITE_BLOB)
|
|
return blobForColumn(sqlStatment, column);
|
|
|
|
return {};
|
|
}
|
|
|
|
template<typename StringType>
|
|
StringType convertToTextForColumn(sqlite3_stmt *sqlStatment, int column)
|
|
{
|
|
int dataType = sqlite3_column_type(sqlStatment, column);
|
|
switch (dataType) {
|
|
case SQLITE_INTEGER:
|
|
case SQLITE_FLOAT:
|
|
case SQLITE3_TEXT:
|
|
return textForColumn<StringType>(sqlStatment, column);
|
|
case SQLITE_BLOB:
|
|
case SQLITE_NULL:
|
|
break;
|
|
}
|
|
|
|
return {};
|
|
}
|
|
} // namespace
|
|
|
|
Type BaseStatement::fetchType(int column) const
|
|
{
|
|
auto dataType = sqlite3_column_type(m_compiledStatement.get(), column);
|
|
|
|
switch (dataType) {
|
|
case SQLITE_INTEGER:
|
|
return Type::Integer;
|
|
case SQLITE_FLOAT:
|
|
return Type::Float;
|
|
case SQLITE3_TEXT:
|
|
return Type::Text;
|
|
case SQLITE_BLOB:
|
|
return Type::Blob;
|
|
case SQLITE_NULL:
|
|
return Type::Null;
|
|
}
|
|
|
|
return Type::Invalid;
|
|
}
|
|
|
|
int BaseStatement::fetchIntValue(int column) const
|
|
{
|
|
return sqlite3_column_int(m_compiledStatement.get(), column);
|
|
}
|
|
|
|
template<>
|
|
int BaseStatement::fetchValue<int>(int column) const
|
|
{
|
|
return fetchIntValue(column);
|
|
}
|
|
|
|
long BaseStatement::fetchLongValue(int column) const
|
|
{
|
|
return long(fetchValue<long long>(column));
|
|
}
|
|
|
|
template<>
|
|
long BaseStatement::fetchValue<long>(int column) const
|
|
{
|
|
return fetchLongValue(column);
|
|
}
|
|
|
|
long long BaseStatement::fetchLongLongValue(int column) const
|
|
{
|
|
return sqlite3_column_int64(m_compiledStatement.get(), column);
|
|
}
|
|
|
|
template<>
|
|
long long BaseStatement::fetchValue<long long>(int column) const
|
|
{
|
|
return fetchLongLongValue(column);
|
|
}
|
|
|
|
double BaseStatement::fetchDoubleValue(int column) const
|
|
{
|
|
return sqlite3_column_double(m_compiledStatement.get(), column);
|
|
}
|
|
|
|
BlobView BaseStatement::fetchBlobValue(int column) const
|
|
{
|
|
return convertToBlobForColumn(m_compiledStatement.get(), column);
|
|
}
|
|
|
|
template<>
|
|
double BaseStatement::fetchValue<double>(int column) const
|
|
{
|
|
return fetchDoubleValue(column);
|
|
}
|
|
|
|
template<typename StringType>
|
|
StringType BaseStatement::fetchValue(int column) const
|
|
{
|
|
return convertToTextForColumn<StringType>(m_compiledStatement.get(), column);
|
|
}
|
|
|
|
template SQLITE_EXPORT Utils::SmallStringView BaseStatement::fetchValue<Utils::SmallStringView>(
|
|
int column) const;
|
|
template SQLITE_EXPORT Utils::SmallString BaseStatement::fetchValue<Utils::SmallString>(
|
|
int column) const;
|
|
template SQLITE_EXPORT Utils::PathString BaseStatement::fetchValue<Utils::PathString>(
|
|
int column) const;
|
|
|
|
Utils::SmallStringView BaseStatement::fetchSmallStringViewValue(int column) const
|
|
{
|
|
return fetchValue<Utils::SmallStringView>(column);
|
|
}
|
|
|
|
ValueView BaseStatement::fetchValueView(int column) const
|
|
{
|
|
int dataType = sqlite3_column_type(m_compiledStatement.get(), column);
|
|
switch (dataType) {
|
|
case SQLITE_NULL:
|
|
return ValueView::create(NullValue{});
|
|
case SQLITE_INTEGER:
|
|
return ValueView::create(fetchLongLongValue(column));
|
|
case SQLITE_FLOAT:
|
|
return ValueView::create(fetchDoubleValue(column));
|
|
case SQLITE3_TEXT:
|
|
return ValueView::create(fetchValue<Utils::SmallStringView>(column));
|
|
case SQLITE_BLOB:
|
|
break;
|
|
}
|
|
|
|
return ValueView::create(NullValue{});
|
|
}
|
|
|
|
} // namespace Sqlite
|