Files
sqlitebrowser/src/sqlitedb.h
Martin Kleusberg 74e0df376a When updating a field try to respect the affinity data type if possible
When modifying the value of a field in the Browse Data tab, we used to
always hand over the new value to SQLite as text. This works most of the
time but can cause problems e.g. when a CHECK constraint checks the data
type of the value and expects something other than text. It is also a
pretty lazy approach in general.

This commit checks if the new value matches the affinity data type of
the column, and if it does hands over the new value as the correct data
type.

See issue #1952.
2019-08-01 13:30:01 +02:00

294 lines
11 KiB
C++

#ifndef SQLITEDB_H
#define SQLITEDB_H
#include "sql/ObjectIdentifier.h"
#include "sql/sqlitetypes.h"
#include <condition_variable>
#include <memory>
#include <mutex>
#include <functional>
#include <vector>
#include <map>
#include <QObject>
#include <QByteArray>
#include <QStringList>
struct sqlite3;
class CipherSettings;
enum LogMessageType
{
kLogMsg_User,
kLogMsg_App,
kLogMsg_ErrorLog
};
using objectMap = std::multimap<std::string, sqlb::ObjectPtr>; // Maps from object type (table, index, view, trigger) to a pointer to the object representation
using schemaMap = std::map<std::string, objectMap>; // Maps from the schema name (main, temp, attached schemas) to the object map for that schema
int collCompare(void* pArg, int sizeA, const void* sA, int sizeB, const void* sB);
namespace sqlb
{
QString escapeIdentifier(const QString& id);
}
/// represents a single SQLite database. except when noted otherwise,
/// all member functions are to be called from the main UI thread
/// only.
class DBBrowserDB : public QObject
{
Q_OBJECT
private:
/// custom unique_ptr deleter releases database for further use by others
struct DatabaseReleaser
{
explicit DatabaseReleaser(DBBrowserDB * pParent_ = nullptr) : pParent(pParent_) {}
DBBrowserDB * pParent;
void operator() (sqlite3 * db) const
{
if(!db || !pParent)
return;
std::unique_lock<std::mutex> lk(pParent->m);
pParent->db_used = false;
lk.unlock();
emit pParent->databaseInUseChanged(false, QString());
pParent->cv.notify_one();
}
};
public:
explicit DBBrowserDB();
~DBBrowserDB () override {}
bool open(const QString& db, bool readOnly = false);
bool attach(const QString& filename, QString attach_as = "");
bool create ( const QString & db);
bool close();
// This returns the SQLite version as well as the SQLCipher if DB4S is compiled with encryption support
static void getSqliteVersion(QString& sqlite, QString& sqlcipher);
using db_pointer_type = std::unique_ptr<sqlite3, DatabaseReleaser>;
/**
borrow exclusive address to the currently open database, until
releasing the returned unique_ptr.
the intended use case is that the main UI thread can call this
any time, and then optionally pass the obtained pointer to a
background worker, or release it after doing work immediately.
if database is currently used by somebody else, opens a dialog
box and gives user the opportunity to sqlite3_interrupt() the
operation of the current owner, then tries again.
\param user a string that identifies the new user, and which
can be displayed in the dialog box.
\param force_wait if set to true we won't ask the user to cancel
the running query but just wait until it is done.
\returns a unique_ptr containing the SQLite database handle, or
nullptr in case no database is open.
**/
db_pointer_type get (const QString& user, bool force_wait = false);
bool setSavepoint(const QString& pointname = "RESTOREPOINT");
bool releaseSavepoint(const QString& pointname = "RESTOREPOINT");
bool revertToSavepoint(const QString& pointname = "RESTOREPOINT");
bool releaseAllSavepoints();
bool revertAll();
bool dump(const QString& filename, const QStringList& tablesToDump, bool insertColNames, bool insertNew, bool exportSchema, bool exportData, bool keepOldSchema);
enum ChoiceOnUse
{
Ask,
Wait,
CancelOther
};
// Callback to get results from executeSQL(). It is invoked for
// each result row coming out of the evaluated SQL statements. If
// a callback returns true (abort), the executeSQL() method
// returns false (error) without invoking the callback again and
// without running any subsequent SQL statements. The 1st argument
// is the number of columns in the result. The 2nd argument to the
// callback is the text representation of the values, one for each
// column. The 3rd argument is a list of strings where each entry
// represents the name of corresponding result column.
using execCallback = std::function<bool(int, QStringList, QStringList)>;
bool executeSQL(QString statement, bool dirtyDB = true, bool logsql = true, execCallback callback = nullptr);
bool executeMultiSQL(QByteArray query, bool dirty = true, bool log = false);
QByteArray querySingleValueFromDb(const QString& sql, bool log = true, ChoiceOnUse choice = Ask);
const QString& lastError() const { return lastErrorMessage; }
/**
* @brief getRow Executes a sqlite statement to get the rowdata(columns)
* for the given rowid.
* @param schemaName Name of the database schema.
* @param sTableName Table to query.
* @param rowid The rowid to fetch.
* @param rowdata A list of QByteArray containing the row data.
* @return true if statement execution was ok, else false.
*/
bool getRow(const sqlb::ObjectIdentifier& table, const QString& rowid, std::vector<QByteArray>& rowdata);
/**
* @brief Interrupts the currenty running statement as soon as possible.
*/
void interruptQuery();
private:
/**
* @brief max Queries the table t for the max value of field.
* @param tableName Table to query
* @param field Field to get the max value
* @return the max value of the field or 0 on error
*/
QString max(const sqlb::ObjectIdentifier& tableName, const sqlb::Field& field) const;
static int callbackWrapper (void* callback, int numberColumns, char** values, char** columnNames);
public:
void updateSchema();
private:
/**
* @brief Creates an empty insert statement.
* @param schemaName The name of the database schema in which to find the table
* @param pk_value This optional parameter can be used to manually set a specific value for the primary key column
* @return An sqlite conform INSERT INTO statement with empty values. (NULL,'',0)
*/
QString emptyInsertStmt(const std::string& schemaName, const sqlb::Table& t, const QString& pk_value = QString()) const;
public:
QString addRecord(const sqlb::ObjectIdentifier& tablename);
bool deleteRecords(const sqlb::ObjectIdentifier& table, const QStringList& rowids, const sqlb::StringVector& pseudo_pk = {});
bool updateRecord(const sqlb::ObjectIdentifier& table, const std::string& column, const QString& rowid, const QByteArray& value, int force_type = 0, const sqlb::StringVector& pseudo_pk = {});
bool createTable(const sqlb::ObjectIdentifier& name, const sqlb::FieldVector& structure);
bool renameTable(const std::string& schema, const std::string& from_table, const std::string& to_table);
bool addColumn(const sqlb::ObjectIdentifier& tablename, const sqlb::Field& field);
/**
* @brief This type maps from old column names to new column names. Given the old and the new table definition, this suffices to
* track fields between the two.
* USE CASES:
* 1) Don't specify a column at all or specify equal column names: Keep its name as-is.
* 2) Specify different column names: Rename the field.
* 3) Map from an existing column name to a Null string: Delete the column.
* 4) Map from a Null column name to a new column name: Add the column.
*/
using AlterTableTrackColumns = std::map<QString, QString>;
/**
* @brief alterTable Can be used to rename, modify or drop existing columns of a given table
* @param tablename Specifies the schema and name of the table to edit
* @param new_table Specifies the new table schema. This is exactly how the new table is going to look like.
* @param track_columns Maps old column names to new column names. This is used to copy the data from the old table to the new one.
* @param newSchema Set this to a non-empty string to move the table to a new schema
* @return true if renaming was successful, false if not. In the latter case also lastErrorMessage is set
*/
bool alterTable(const sqlb::ObjectIdentifier& tablename, const sqlb::Table& new_table, AlterTableTrackColumns track_columns, std::string newSchemaName = "");
objectMap getBrowsableObjects(const std::string& schema) const;
template<typename T = sqlb::Object>
const std::shared_ptr<T> getObjectByName(const sqlb::ObjectIdentifier& name) const
{
for(auto& it : schemata.at(name.schema()))
{
if(it.second->name() == name.name())
return std::dynamic_pointer_cast<T>(it.second);
}
return std::shared_ptr<T>();
}
bool isOpen() const;
bool encrypted() const { return isEncrypted; }
bool readOnly() const { return isReadOnly; }
bool getDirty() const;
QString currentFile() const { return curDBFilename; }
/// log an SQL statement [thread-safe]
void logSQL(QString statement, LogMessageType msgtype);
QString getPragma(const QString& pragma);
bool setPragma(const QString& pragma, const QString& value);
bool setPragma(const QString& pragma, const QString& value, QString& originalvalue);
bool setPragma(const QString& pragma, int value, int& originalvalue);
bool loadExtension(const QString& filename);
void loadExtensionsFromSettings();
static QStringList Datatypes;
private:
std::vector<std::pair<std::string, std::string> > queryColumnInformation(const std::string& schema_name, const std::string& object_name);
public:
QString generateSavepointName(const QString& identifier = QString()) const;
// This function generates the name for a temporary table. It guarantees that there is no table with this name yet
std::string generateTemporaryTableName(const std::string& schema) const;
schemaMap schemata;
signals:
void sqlExecuted(QString sql, int msgtype);
void dbChanged(bool dirty);
void structureUpdated();
void requestCollation(QString name, int eTextRep);
void databaseInUseChanged(bool busy, QString user);
private:
/// external code needs to go through get() to obtain access to the database
sqlite3 * _db;
std::mutex m;
std::condition_variable cv;
bool db_used;
QString db_user;
/// wait for release of the DB locked through a previous get(),
/// giving users the option to discard running task through a
/// message box.
void waitForDbRelease(ChoiceOnUse choice = Ask);
QString curDBFilename;
QString lastErrorMessage;
QStringList savepointList;
bool isEncrypted;
bool isReadOnly;
sqlb::StringVector primaryKeyForEditing(const sqlb::ObjectIdentifier& table, const sqlb::StringVector& pseudo_pk) const;
// SQLite Callbacks
void collationNeeded(void* pData, sqlite3* db, int eTextRep, const char* sCollationName);
void errorLogCallback(void* user_data, int error_code, const char* message);
bool tryEncryptionSettings(const QString& filename, bool* encrypted, CipherSettings*& cipherSettings);
bool dontCheckForStructureUpdates;
class NoStructureUpdateChecks
{
public:
explicit NoStructureUpdateChecks(DBBrowserDB& db) : m_db(db) { m_db.dontCheckForStructureUpdates = true; }
~NoStructureUpdateChecks() { m_db.dontCheckForStructureUpdates = false; }
private:
DBBrowserDB& m_db;
};
};
#endif