diff options
Diffstat (limited to 'kexi/kexidb/connection.cpp')
-rw-r--r-- | kexi/kexidb/connection.cpp | 3552 |
1 files changed, 3552 insertions, 0 deletions
diff --git a/kexi/kexidb/connection.cpp b/kexi/kexidb/connection.cpp new file mode 100644 index 00000000..1a401a8a --- /dev/null +++ b/kexi/kexidb/connection.cpp @@ -0,0 +1,3552 @@ +/* This file is part of the KDE project + Copyright (C) 2003-2007 Jaroslaw Staniek <js@iidea.pl> + + This program is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This program 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 + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public License + along with this program; see the file COPYING. If not, write to + the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. +*/ + +#include <kexidb/connection.h> + +#include "error.h" +#include "connection_p.h" +#include "connectiondata.h" +#include "driver.h" +#include "driver_p.h" +#include "schemadata.h" +#include "tableschema.h" +#include "relationship.h" +#include "transaction.h" +#include "cursor.h" +#include "global.h" +#include "roweditbuffer.h" +#include "utils.h" +#include "dbproperties.h" +#include "lookupfieldschema.h" +#include "parser/parser.h" + +#include <kexiutils/utils.h> +#include <kexiutils/identifier.h> + +#include <qdir.h> +#include <qfileinfo.h> +#include <qguardedptr.h> +#include <qdom.h> + +#include <klocale.h> +#include <kdebug.h> + +#define KEXIDB_EXTENDED_TABLE_SCHEMA_VERSION 1 + +//#define KEXIDB_LOOKUP_FIELD_TEST + +namespace KexiDB { + +Connection::SelectStatementOptions::SelectStatementOptions() + : identifierEscaping(Driver::EscapeDriver|Driver::EscapeAsNecessary) + , alsoRetrieveROWID(false) + , addVisibleLookupColumns(true) +{ +} + +Connection::SelectStatementOptions::~SelectStatementOptions() +{ +} + +//================================================ + +ConnectionInternal::ConnectionInternal(Connection *conn) + : connection(conn) +{ +} + +ConnectionInternal::~ConnectionInternal() +{ +} + +//================================================ +//! @internal +class ConnectionPrivate +{ + public: + ConnectionPrivate(Connection* const conn, ConnectionData &conn_data) + : conn(conn) + , conn_data(&conn_data) + , tableSchemaChangeListeners(101) + , m_parser(0) + , tables_byname(101, false) + , queries_byname(101, false) + , kexiDBSystemTables(101) + , dont_remove_transactions(false) + , skip_databaseExists_check_in_useDatabase(false) + , default_trans_started_inside(false) + , isConnected(false) + , autoCommit(true) + { + tableSchemaChangeListeners.setAutoDelete(true); + obsoleteQueries.setAutoDelete(true); + + tables.setAutoDelete(true); + tables_byname.setAutoDelete(false);//tables is owner, not me + kexiDBSystemTables.setAutoDelete(true);//only system tables + queries.setAutoDelete(true); + queries_byname.setAutoDelete(false);//queries is owner, not me + + //reasonable sizes: TODO + tables.resize(101); + queries.resize(101); + } + ~ConnectionPrivate() + { + delete m_parser; + } + + void errorInvalidDBContents(const QString& details) { + conn->setError( ERR_INVALID_DATABASE_CONTENTS, i18n("Invalid database contents. ")+details); + } + + QString strItIsASystemObject() const { + return i18n("It is a system object."); + } + + inline Parser *parser() { return m_parser ? m_parser : (m_parser = new Parser(conn)); } + + Connection* const conn; //!< The \a Connection instance this \a ConnectionPrivate belongs to. + QGuardedPtr<ConnectionData> conn_data; //!< the \a ConnectionData used within that connection. + + /*! Default transaction handle. + If transactions are supported: Any operation on database (e.g. inserts) + that is started without specifying transaction context, will be performed + in the context of this transaction. */ + Transaction default_trans; + QValueList<Transaction> transactions; + + QPtrDict< QPtrList<Connection::TableSchemaChangeListenerInterface> > tableSchemaChangeListeners; + + //! Used in Connection::setQuerySchemaObsolete( const QString& queryName ) + //! to collect obsolete queries. THese are deleted on connection deleting. + QPtrList<QuerySchema> obsoleteQueries; + + + //! server version information for this connection. + KexiDB::ServerVersionInfo serverVersion; + + //! Daabase version information for this connection. + KexiDB::DatabaseVersionInfo databaseVersion; + + Parser *m_parser; + + //! Table schemas retrieved on demand with tableSchema() + QIntDict<TableSchema> tables; + QDict<TableSchema> tables_byname; + QIntDict<QuerySchema> queries; + QDict<QuerySchema> queries_byname; + + //! used just for removing system TableSchema objects on db close. + QPtrDict<TableSchema> kexiDBSystemTables; + + //! Database properties + DatabaseProperties* dbProperties; + + QString availableDatabaseName; //!< used by anyAvailableDatabaseName() + QString usedDatabase; //!< database name that is opened now (the currentDatabase() name) + + //! true if rollbackTransaction() and commitTransaction() shouldn't remove + //! the transaction object from 'transactions' list; used by closeDatabase() + bool dont_remove_transactions : 1; + + //! used to avoid endless recursion between useDatabase() and databaseExists() + //! when useTemporaryDatabaseIfNeeded() works + bool skip_databaseExists_check_in_useDatabase : 1; + + /*! Used when single transactions are only supported (Driver::SingleTransactions). + True value means default transaction has been started inside connection object + (by beginAutoCommitTransaction()), otherwise default transaction has been started outside + of the object (e.g. before createTable()), so we shouldn't autocommit the transaction + in commitAutoCommitTransaction(). Also, beginAutoCommitTransaction() doesn't restarts + transaction if default_trans_started_inside is false. Such behaviour allows user to + execute a sequence of actions like CREATE TABLE...; INSERT DATA...; within a single transaction + and commit it or rollback by hand. */ + bool default_trans_started_inside : 1; + + bool isConnected : 1; + + bool autoCommit : 1; + + /*! True for read only connection. Used especially for file-based drivers. */ + bool readOnly : 1; +}; + +}//namespace KexiDB + +//================================================ +using namespace KexiDB; + +//! static: list of internal KexiDB system table names +QStringList KexiDB_kexiDBSystemTableNames; + +Connection::Connection( Driver *driver, ConnectionData &conn_data ) + : QObject() + ,KexiDB::Object() + ,d(new ConnectionPrivate(this, conn_data)) + ,m_driver(driver) + ,m_destructor_started(false) +{ + d->dbProperties = new DatabaseProperties(this); + m_cursors.setAutoDelete(true); +// d->transactions.setAutoDelete(true); + //reasonable sizes: TODO + m_cursors.resize(101); +// d->transactions.resize(101);//woohoo! so many transactions? + m_sql.reserve(0x4000); +} + +void Connection::destroy() +{ + disconnect(); + //do not allow the driver to touch me: I will kill myself. + m_driver->d->connections.take( this ); +} + +Connection::~Connection() +{ + m_destructor_started = true; +// KexiDBDbg << "Connection::~Connection()" << endl; + delete d->dbProperties; + delete d; + d = 0; +/* if (m_driver) { + if (m_is_connected) { + //delete own table schemas + d->tables.clear(); + //delete own cursors: + m_cursors.clear(); + } + //do not allow the driver to touch me: I will kill myself. + m_driver->m_connections.take( this ); + }*/ +} + +ConnectionData* Connection::data() const +{ + return d->conn_data; +} + +bool Connection::connect() +{ + clearError(); + if (d->isConnected) { + setError(ERR_ALREADY_CONNECTED, i18n("Connection already established.") ); + return false; + } + + d->serverVersion.clear(); + if (!(d->isConnected = drv_connect(d->serverVersion))) { + setError(m_driver->isFileDriver() ? + i18n("Could not open \"%1\" project file.").arg(QDir::convertSeparators(d->conn_data->fileName())) + : i18n("Could not connect to \"%1\" database server.").arg(d->conn_data->serverInfoString()) ); + } + return d->isConnected; +} + +bool Connection::isDatabaseUsed() const +{ + return !d->usedDatabase.isEmpty() && d->isConnected && drv_isDatabaseUsed(); +} + +void Connection::clearError() +{ + Object::clearError(); + m_sql = QString::null; +} + +bool Connection::disconnect() +{ + clearError(); + if (!d->isConnected) + return true; + + if (!closeDatabase()) + return false; + + bool ok = drv_disconnect(); + if (ok) + d->isConnected = false; + return ok; +} + +bool Connection::isConnected() const +{ + return d->isConnected; +} + +bool Connection::checkConnected() +{ + if (d->isConnected) { + clearError(); + return true; + } + setError(ERR_NO_CONNECTION, i18n("Not connected to the database server.") ); + return false; +} + +bool Connection::checkIsDatabaseUsed() +{ + if (isDatabaseUsed()) { + clearError(); + return true; + } + setError(ERR_NO_DB_USED, i18n("Currently no database is used.") ); + return false; +} + +QStringList Connection::databaseNames(bool also_system_db) +{ + KexiDBDbg << "Connection::databaseNames("<<also_system_db<<")"<< endl; + if (!checkConnected()) + return QStringList(); + + QString tmpdbName; + //some engines need to have opened any database before executing "create database" + if (!useTemporaryDatabaseIfNeeded(tmpdbName)) + return QStringList(); + + QStringList list, non_system_list; + + bool ret = drv_getDatabasesList( list ); + + if (!tmpdbName.isEmpty()) { + //whatever result is - now we have to close temporary opened database: + if (!closeDatabase()) + return QStringList(); + } + + if (!ret) + return QStringList(); + + if (also_system_db) + return list; + //filter system databases: + for (QStringList::ConstIterator it = list.constBegin(); it!=list.constEnd(); ++it) { + KexiDBDbg << "Connection::databaseNames(): " << *it << endl; + if (!m_driver->isSystemDatabaseName(*it)) { + KexiDBDbg << "add " << *it << endl; + non_system_list << (*it); + } + } + return non_system_list; +} + +bool Connection::drv_getDatabasesList( QStringList &list ) +{ + list.clear(); + return true; +} + +bool Connection::drv_databaseExists( const QString &dbName, bool ignoreErrors ) +{ + QStringList list = databaseNames(true);//also system + if (error()) { + return false; + } + + if (list.find( dbName )==list.end()) { + if (!ignoreErrors) + setError(ERR_OBJECT_NOT_FOUND, i18n("The database \"%1\" does not exist.").arg(dbName)); + return false; + } + + return true; +} + +bool Connection::databaseExists( const QString &dbName, bool ignoreErrors ) +{ +// KexiDBDbg << "Connection::databaseExists(" << dbName << "," << ignoreErrors << ")" << endl; + if (!checkConnected()) + return false; + clearError(); + + if (m_driver->isFileDriver()) { + //for file-based db: file must exists and be accessible +//js: moved from useDatabase(): + QFileInfo file(d->conn_data->fileName()); + if (!file.exists() || ( !file.isFile() && !file.isSymLink()) ) { + if (!ignoreErrors) + setError(ERR_OBJECT_NOT_FOUND, i18n("Database file \"%1\" does not exist.") + .arg(QDir::convertSeparators(d->conn_data->fileName())) ); + return false; + } + if (!file.isReadable()) { + if (!ignoreErrors) + setError(ERR_ACCESS_RIGHTS, i18n("Database file \"%1\" is not readable.") + .arg(QDir::convertSeparators(d->conn_data->fileName())) ); + return false; + } + if (!file.isWritable()) { + if (!ignoreErrors) + setError(ERR_ACCESS_RIGHTS, i18n("Database file \"%1\" is not writable.") + .arg(QDir::convertSeparators(d->conn_data->fileName())) ); + return false; + } + return true; + } + + QString tmpdbName; + //some engines need to have opened any database before executing "create database" + const bool orig_skip_databaseExists_check_in_useDatabase = d->skip_databaseExists_check_in_useDatabase; + d->skip_databaseExists_check_in_useDatabase = true; + bool ret = useTemporaryDatabaseIfNeeded(tmpdbName); + d->skip_databaseExists_check_in_useDatabase = orig_skip_databaseExists_check_in_useDatabase; + if (!ret) + return false; + + ret = drv_databaseExists(dbName, ignoreErrors); + + if (!tmpdbName.isEmpty()) { + //whatever result is - now we have to close temporary opened database: + if (!closeDatabase()) + return false; + } + + return ret; +} + +#define createDatabase_CLOSE \ + { if (!closeDatabase()) { \ + setError(i18n("Database \"%1\" created but could not be closed after creation.").arg(dbName) ); \ + return false; \ + } } + +#define createDatabase_ERROR \ + { createDatabase_CLOSE; return false; } + + +bool Connection::createDatabase( const QString &dbName ) +{ + if (!checkConnected()) + return false; + + if (databaseExists( dbName )) { + setError(ERR_OBJECT_EXISTS, i18n("Database \"%1\" already exists.").arg(dbName) ); + return false; + } + if (m_driver->isSystemDatabaseName( dbName )) { + setError(ERR_SYSTEM_NAME_RESERVED, + i18n("Cannot create database \"%1\". This name is reserved for system database.").arg(dbName) ); + return false; + } + if (m_driver->isFileDriver()) { + //update connection data if filename differs + d->conn_data->setFileName( dbName ); + } + + QString tmpdbName; + //some engines need to have opened any database before executing "create database" + if (!useTemporaryDatabaseIfNeeded(tmpdbName)) + return false; + + //low-level create + if (!drv_createDatabase( dbName )) { + setError(i18n("Error creating database \"%1\" on the server.").arg(dbName) ); + closeDatabase();//sanity + return false; + } + + if (!tmpdbName.isEmpty()) { + //whatever result is - now we have to close temporary opened database: + if (!closeDatabase()) + return false; + } + + if (!tmpdbName.isEmpty() || !m_driver->d->isDBOpenedAfterCreate) { + //db need to be opened + if (!useDatabase( dbName, false/*not yet kexi compatible!*/ )) { + setError(i18n("Database \"%1\" created but could not be opened.").arg(dbName) ); + return false; + } + } + else { + //just for the rule + d->usedDatabase = dbName; + } + + Transaction trans; + if (m_driver->transactionsSupported()) { + trans = beginTransaction(); + if (!trans.active()) + return false; + } +//not needed since closeDatabase() rollbacks transaction: TransactionGuard trans_g(this); +// if (error()) +// return false; + + //-create system tables schema objects + if (!setupKexiDBSystemSchema()) + return false; + + //-physically create system tables + for (QPtrDictIterator<TableSchema> it(d->kexiDBSystemTables); it.current(); ++it) { + if (!drv_createTable( it.current()->name() )) + createDatabase_ERROR; + } + +/* moved to KexiProject... + + //-create default part info + TableSchema *ts; + if (!(ts = tableSchema("kexi__parts"))) + createDatabase_ERROR; + FieldList *fl = ts->subList("p_id", "p_name", "p_mime", "p_url"); + if (!fl) + createDatabase_ERROR; + if (!insertRecord(*fl, QVariant(1), QVariant("Tables"), QVariant("kexi/table"), QVariant("http://koffice.org/kexi/"))) + createDatabase_ERROR; + if (!insertRecord(*fl, QVariant(2), QVariant("Queries"), QVariant("kexi/query"), QVariant("http://koffice.org/kexi/"))) + createDatabase_ERROR; +*/ + + //-insert KexiDB version info: + TableSchema *t_db = tableSchema("kexi__db"); + if (!t_db) + createDatabase_ERROR; + if ( !insertRecord(*t_db, "kexidb_major_ver", KexiDB::version().major) + || !insertRecord(*t_db, "kexidb_minor_ver", KexiDB::version().minor)) + createDatabase_ERROR; + + if (trans.active() && !commitTransaction(trans)) + createDatabase_ERROR; + + createDatabase_CLOSE; + return true; +} + +#undef createDatabase_CLOSE +#undef createDatabase_ERROR + +bool Connection::useDatabase( const QString &dbName, bool kexiCompatible, bool *cancelled, MessageHandler* msgHandler ) +{ + if (cancelled) + *cancelled = false; + KexiDBDbg << "Connection::useDatabase(" << dbName << "," << kexiCompatible <<")" << endl; + if (!checkConnected()) + return false; + + if (dbName.isEmpty()) + return false; + QString my_dbName = dbName; +// if (my_dbName.isEmpty()) { +// const QStringList& db_lst = databaseNames(); +// if (!db_lst.isEmpty()) +// my_dbName = db_lst.first(); +// } + if (d->usedDatabase == my_dbName) + return true; //already used + + if (!d->skip_databaseExists_check_in_useDatabase) { + if (!databaseExists(my_dbName, false /*don't ignore errors*/)) + return false; //database must exist + } + + if (!d->usedDatabase.isEmpty() && !closeDatabase()) //close db if already used + return false; + + d->usedDatabase = ""; + + if (!drv_useDatabase( my_dbName, cancelled, msgHandler )) { + if (cancelled && *cancelled) + return false; + QString msg(i18n("Opening database \"%1\" failed.").arg( my_dbName )); + if (error()) + setError( this, msg ); + else + setError( msg ); + return false; + } + + //-create system tables schema objects + if (!setupKexiDBSystemSchema()) + return false; + + if (kexiCompatible && my_dbName.lower()!=anyAvailableDatabaseName().lower()) { + //-get global database information + int num; + bool ok; +// static QString notfound_str = i18n("\"%1\" database property not found"); + num = d->dbProperties->value("kexidb_major_ver").toInt(&ok); + if (!ok) + return false; + d->databaseVersion.major = num; +/* if (true!=querySingleNumber( + "select db_value from kexi__db where db_property=" + m_driver->escapeString(QString("kexidb_major_ver")), num)) { + d->errorInvalidDBContents(notfound_str.arg("kexidb_major_ver")); + return false; + }*/ + num = d->dbProperties->value("kexidb_minor_ver").toInt(&ok); + if (!ok) + return false; + d->databaseVersion.minor = num; +/* if (true!=querySingleNumber( + "select db_value from kexi__db where db_property=" + m_driver->escapeString(QString("kexidb_minor_ver")), num)) { + d->errorInvalidDBContents(notfound_str.arg("kexidb_minor_ver")); + return false; + }*/ + +#if 0 //this is already checked in DriverManagerInternal::lookupDrivers() + //** error if major version does not match + if (m_driver->versionMajor()!=KexiDB::versionMajor()) { + setError(ERR_INCOMPAT_DATABASE_VERSION, + i18n("Database version (%1) does not match Kexi application's version (%2)") + .arg( QString("%1.%2").arg(versionMajor()).arg(versionMinor()) ) + .arg( QString("%1.%2").arg(KexiDB::versionMajor()).arg(KexiDB::versionMinor()) ) ); + return false; + } + if (m_driver->versionMinor()!=KexiDB::versionMinor()) { + //js TODO: COMPATIBILITY CODE HERE! + //js TODO: CONVERSION CODE HERE (or signal that conversion is needed) + } +#endif + } + d->usedDatabase = my_dbName; + return true; +} + +bool Connection::closeDatabase() +{ + if (d->usedDatabase.isEmpty()) + return true; //no db used + if (!checkConnected()) + return true; + + bool ret = true; + +/*! \todo (js) add CLEVER algorithm here for nested transactions */ + if (m_driver->transactionsSupported()) { + //rollback all transactions + QValueList<Transaction>::ConstIterator it; + d->dont_remove_transactions=true; //lock! + for (it=d->transactions.constBegin(); it!= d->transactions.constEnd(); ++it) { + if (!rollbackTransaction(*it)) {//rollback as much as you can, don't stop on prev. errors + ret = false; + } + else { + KexiDBDbg << "Connection::closeDatabase(): transaction rolled back!" << endl; + KexiDBDbg << "Connection::closeDatabase(): trans.refcount==" << + ((*it).m_data ? QString::number((*it).m_data->refcount) : "(null)") << endl; + } + } + d->dont_remove_transactions=false; //unlock! + d->transactions.clear(); //free trans. data + } + + //delete own cursors: + m_cursors.clear(); + //delete own schemas + d->tables.clear(); + d->kexiDBSystemTables.clear(); + d->queries.clear(); + + if (!drv_closeDatabase()) + return false; + + d->usedDatabase = ""; +// KexiDBDbg << "Connection::closeDatabase(): " << ret << endl; + return ret; +} + +QString Connection::currentDatabase() const +{ + return d->usedDatabase; +} + +bool Connection::useTemporaryDatabaseIfNeeded(QString &tmpdbName) +{ + if (!m_driver->isFileDriver() && m_driver->beh->USING_DATABASE_REQUIRED_TO_CONNECT + && !isDatabaseUsed()) { + //we have no db used, but it is required by engine to have used any! + tmpdbName = anyAvailableDatabaseName(); + if (tmpdbName.isEmpty()) { + setError(ERR_NO_DB_USED, i18n("Cannot find any database for temporary connection.") ); + return false; + } + const bool orig_skip_databaseExists_check_in_useDatabase = d->skip_databaseExists_check_in_useDatabase; + d->skip_databaseExists_check_in_useDatabase = true; + bool ret = useDatabase(tmpdbName, false); + d->skip_databaseExists_check_in_useDatabase = orig_skip_databaseExists_check_in_useDatabase; + if (!ret) { + setError(errorNum(), + i18n("Error during starting temporary connection using \"%1\" database name.") + .arg(tmpdbName) ); + return false; + } + } + return true; +} + +bool Connection::dropDatabase( const QString &dbName ) +{ + if (!checkConnected()) + return false; + + QString dbToDrop; + if (dbName.isEmpty() && d->usedDatabase.isEmpty()) { + if (!m_driver->isFileDriver() + || (m_driver->isFileDriver() && d->conn_data->fileName().isEmpty()) ) { + setError(ERR_NO_NAME_SPECIFIED, i18n("Cannot drop database - name not specified.") ); + return false; + } + //this is a file driver so reuse previously passed filename + dbToDrop = d->conn_data->fileName(); + } + else { + if (dbName.isEmpty()) { + dbToDrop = d->usedDatabase; + } else { + if (m_driver->isFileDriver()) //lets get full path + dbToDrop = QFileInfo(dbName).absFilePath(); + else + dbToDrop = dbName; + } + } + + if (dbToDrop.isEmpty()) { + setError(ERR_NO_NAME_SPECIFIED, i18n("Cannot delete database - name not specified.") ); + return false; + } + + if (m_driver->isSystemDatabaseName( dbToDrop )) { + setError(ERR_SYSTEM_NAME_RESERVED, i18n("Cannot delete system database \"%1\".").arg(dbToDrop) ); + return false; + } + + if (isDatabaseUsed() && d->usedDatabase == dbToDrop) { + //we need to close database because cannot drop used this database + if (!closeDatabase()) + return false; + } + + QString tmpdbName; + //some engines need to have opened any database before executing "drop database" + if (!useTemporaryDatabaseIfNeeded(tmpdbName)) + return false; + + //ok, now we have access to dropping + bool ret = drv_dropDatabase( dbToDrop ); + + if (!tmpdbName.isEmpty()) { + //whatever result is - now we have to close temporary opened database: + if (!closeDatabase()) + return false; + } + return ret; +} + +QStringList Connection::objectNames(int objType, bool* ok) +{ + QStringList list; + + if (!checkIsDatabaseUsed()) { + if(ok) + *ok = false; + return list; + } + + QString sql; + if (objType==KexiDB::AnyObjectType) + sql = "SELECT o_name FROM kexi__objects"; + else + sql = QString::fromLatin1("SELECT o_name FROM kexi__objects WHERE o_type=%1").arg(objType); + + Cursor *c = executeQuery(sql); + if (!c) { + if(ok) + *ok = false; + return list; + } + + for (c->moveFirst(); !c->eof(); c->moveNext()) { + QString name = c->value(0).toString(); + if (KexiUtils::isIdentifier( name )) { + list.append(name); + } + } + + if (!deleteCursor(c)) { + if(ok) + *ok = false; + return list; + } + + if(ok) + *ok = true; + return list; +} + +QStringList Connection::tableNames(bool also_system_tables) +{ + bool ok = true; + QStringList list = objectNames(TableObjectType, &ok); + if (also_system_tables && ok) { + list += Connection::kexiDBSystemTableNames(); + } + return list; +} + +//! \todo (js): this will depend on KexiDB lib version +const QStringList& Connection::kexiDBSystemTableNames() +{ + if (KexiDB_kexiDBSystemTableNames.isEmpty()) { + KexiDB_kexiDBSystemTableNames + << "kexi__objects" + << "kexi__objectdata" + << "kexi__fields" +// << "kexi__querydata" +// << "kexi__queryfields" +// << "kexi__querytables" + << "kexi__db" + ; + } + return KexiDB_kexiDBSystemTableNames; +} + +KexiDB::ServerVersionInfo* Connection::serverVersion() const +{ + return isConnected() ? &d->serverVersion : 0; +} + +KexiDB::DatabaseVersionInfo* Connection::databaseVersion() const +{ + return isDatabaseUsed() ? &d->databaseVersion : 0; +} + +DatabaseProperties& Connection::databaseProperties() +{ + return *d->dbProperties; +} + +QValueList<int> Connection::tableIds() +{ + return objectIds(KexiDB::TableObjectType); +} + +QValueList<int> Connection::queryIds() +{ + return objectIds(KexiDB::QueryObjectType); +} + +QValueList<int> Connection::objectIds(int objType) +{ + QValueList<int> list; + + if (!checkIsDatabaseUsed()) + return list; + + Cursor *c = executeQuery( + QString::fromLatin1("SELECT o_id, o_name FROM kexi__objects WHERE o_type=%1").arg(objType)); + if (!c) + return list; + for (c->moveFirst(); !c->eof(); c->moveNext()) + { + QString tname = c->value(1).toString(); //kexi__objects.o_name + if (KexiUtils::isIdentifier( tname )) { + list.append(c->value(0).toInt()); //kexi__objects.o_id + } + } + + deleteCursor(c); + + return list; +} + +QString Connection::createTableStatement( const KexiDB::TableSchema& tableSchema ) const +{ +// Each SQL identifier needs to be escaped in the generated query. + QString sql; + sql.reserve(4096); + sql = "CREATE TABLE " + escapeIdentifier(tableSchema.name()) + " ("; + bool first=true; + Field::ListIterator it( tableSchema.m_fields ); + Field *field; + for (;(field = it.current())!=0; ++it) { + if (first) + first = false; + else + sql += ", "; + QString v = escapeIdentifier(field->name()) + " "; + const bool autoinc = field->isAutoIncrement(); + const bool pk = field->isPrimaryKey() || (autoinc && m_driver->beh->AUTO_INCREMENT_REQUIRES_PK); +//TODO: warning: ^^^^^ this allows only one autonumber per table when AUTO_INCREMENT_REQUIRES_PK==true! + if (autoinc && m_driver->beh->SPECIAL_AUTO_INCREMENT_DEF) { + if (pk) + v += m_driver->beh->AUTO_INCREMENT_TYPE + " " + m_driver->beh->AUTO_INCREMENT_PK_FIELD_OPTION; + else + v += m_driver->beh->AUTO_INCREMENT_TYPE + " " + m_driver->beh->AUTO_INCREMENT_FIELD_OPTION; + } + else { + if (autoinc && !m_driver->beh->AUTO_INCREMENT_TYPE.isEmpty()) + v += m_driver->beh->AUTO_INCREMENT_TYPE; + else + v += m_driver->sqlTypeName(field->type(), field->precision()); + + if (field->isUnsigned()) + v += (" " + m_driver->beh->UNSIGNED_TYPE_KEYWORD); + + if (field->isFPNumericType() && field->precision()>0) { + if (field->scale()>0) + v += QString::fromLatin1("(%1,%2)").arg(field->precision()).arg(field->scale()); + else + v += QString::fromLatin1("(%1)").arg(field->precision()); + } + else if (field->type()==Field::Text && field->length()>0) + v += QString::fromLatin1("(%1)").arg(field->length()); + + if (autoinc) + v += (" " + + (pk ? m_driver->beh->AUTO_INCREMENT_PK_FIELD_OPTION : m_driver->beh->AUTO_INCREMENT_FIELD_OPTION)); + else + //TODO: here is automatically a single-field key created + if (pk) + v += " PRIMARY KEY"; + if (!pk && field->isUniqueKey()) + v += " UNIQUE"; +///@todo IS this ok for all engines?: if (!autoinc && !field->isPrimaryKey() && field->isNotNull()) + if (!autoinc && !pk && field->isNotNull()) + v += " NOT NULL"; //only add not null option if no autocommit is set + if (field->defaultValue().isValid()) { + QString valToSQL( m_driver->valueToSQL( field, field->defaultValue() ) ); + if (!valToSQL.isEmpty()) //for sanity + v += QString::fromLatin1(" DEFAULT ") + valToSQL; + } + } + sql += v; + } + sql += ")"; + return sql; +} + +//yeah, it is very efficient: +#define C_A(a) , const QVariant& c ## a + +#define V_A0 m_driver->valueToSQL( tableSchema.field(0), c0 ) +#define V_A(a) +","+m_driver->valueToSQL( \ + tableSchema.field(a) ? tableSchema.field(a)->type() : Field::Text, c ## a ) + +// KexiDBDbg << "******** " << QString("INSERT INTO ") + +// escapeIdentifier(tableSchema.name()) + +// " VALUES (" + vals + ")" <<endl; + +#define C_INS_REC(args, vals) \ + bool Connection::insertRecord(KexiDB::TableSchema &tableSchema args) {\ + return executeSQL( \ + QString("INSERT INTO ") + escapeIdentifier(tableSchema.name()) + " VALUES (" + vals + ")" \ + ); \ + } + +#define C_INS_REC_ALL \ +C_INS_REC( C_A(0), V_A0 ) \ +C_INS_REC( C_A(0) C_A(1), V_A0 V_A(1) ) \ +C_INS_REC( C_A(0) C_A(1) C_A(2), V_A0 V_A(1) V_A(2) ) \ +C_INS_REC( C_A(0) C_A(1) C_A(2) C_A(3), V_A0 V_A(1) V_A(2) V_A(3) ) \ +C_INS_REC( C_A(0) C_A(1) C_A(2) C_A(3) C_A(4), V_A0 V_A(1) V_A(2) V_A(3) V_A(4) ) \ +C_INS_REC( C_A(0) C_A(1) C_A(2) C_A(3) C_A(4) C_A(5), V_A0 V_A(1) V_A(2) V_A(3) V_A(4) V_A(5) ) \ +C_INS_REC( C_A(0) C_A(1) C_A(2) C_A(3) C_A(4) C_A(5) C_A(6), V_A0 V_A(1) V_A(2) V_A(3) V_A(4) V_A(5) V_A(6) ) \ +C_INS_REC( C_A(0) C_A(1) C_A(2) C_A(3) C_A(4) C_A(5) C_A(6) C_A(7), V_A0 V_A(1) V_A(2) V_A(3) V_A(4) V_A(5) V_A(6) V_A(7) ) + +C_INS_REC_ALL + +#undef V_A0 +#undef V_A +#undef C_INS_REC + +#define V_A0 value += m_driver->valueToSQL( flist->first(), c0 ); +#define V_A( a ) value += ("," + m_driver->valueToSQL( flist->next(), c ## a )); +//#define V_ALAST( a ) valueToSQL( flist->last(), c ## a ) + + +#define C_INS_REC(args, vals) \ + bool Connection::insertRecord(FieldList& fields args) \ + { \ + QString value; \ + Field::List *flist = fields.fields(); \ + vals \ + return executeSQL( \ + QString("INSERT INTO ") + \ + ((fields.fields()->first() && fields.fields()->first()->table()) ? \ + escapeIdentifier(fields.fields()->first()->table()->name()) : \ + "??") \ + + "(" + fields.sqlFieldsList(m_driver) + ") VALUES (" + value + ")" \ + ); \ + } + +C_INS_REC_ALL + +#undef C_A +#undef V_A +#undef V_ALAST +#undef C_INS_REC +#undef C_INS_REC_ALL + +bool Connection::insertRecord(TableSchema &tableSchema, QValueList<QVariant>& values) +{ +// Each SQL identifier needs to be escaped in the generated query. + Field::List *fields = tableSchema.fields(); + Field *f = fields->first(); +// QString s_val; +// s_val.reserve(4096); + m_sql = QString::null; + QValueList<QVariant>::ConstIterator it = values.constBegin(); +// int i=0; + while (f && (it!=values.end())) { + if (m_sql.isEmpty()) + m_sql = QString("INSERT INTO ") + + escapeIdentifier(tableSchema.name()) + + " VALUES ("; + else + m_sql += ","; + m_sql += m_driver->valueToSQL( f, *it ); +// KexiDBDbg << "val" << i++ << ": " << m_driver->valueToSQL( f, *it ) << endl; + ++it; + f=fields->next(); + } + m_sql += ")"; + +// KexiDBDbg<<"******** "<< m_sql << endl; + return executeSQL(m_sql); +} + +bool Connection::insertRecord(FieldList& fields, QValueList<QVariant>& values) +{ +// Each SQL identifier needs to be escaped in the generated query. + Field::List *flist = fields.fields(); + Field *f = flist->first(); + if (!f) + return false; +// QString s_val; +// s_val.reserve(4096); + m_sql = QString::null; + QValueList<QVariant>::ConstIterator it = values.constBegin(); +// int i=0; + while (f && (it!=values.constEnd())) { + if (m_sql.isEmpty()) + m_sql = QString("INSERT INTO ") + + escapeIdentifier(flist->first()->table()->name()) + "(" + + fields.sqlFieldsList(m_driver) + ") VALUES ("; + else + m_sql += ","; + m_sql += m_driver->valueToSQL( f, *it ); +// KexiDBDbg << "val" << i++ << ": " << m_driver->valueToSQL( f, *it ) << endl; + ++it; + f=flist->next(); + } + m_sql += ")"; + + return executeSQL(m_sql); +} + +bool Connection::executeSQL( const QString& statement ) +{ + m_sql = statement; //remember for error handling + if (!drv_executeSQL( m_sql )) { + m_errMsg = QString::null; //clear as this could be most probably jsut "Unknown error" string. + m_errorSql = statement; + setError(this, ERR_SQL_EXECUTION_ERROR, i18n("Error while executing SQL statement.")); + return false; + } + return true; +} + +QString Connection::selectStatement( KexiDB::QuerySchema& querySchema, + const QValueList<QVariant>& params, + const SelectStatementOptions& options) const +{ +//"SELECT FROM ..." is theoretically allowed " +//if (querySchema.fieldCount()<1) +// return QString::null; +// Each SQL identifier needs to be escaped in the generated query. + + if (!querySchema.statement().isEmpty()) + return querySchema.statement(); + +//! @todo looking at singleTable is visually nice but a field name can conflict +//! with function or variable name... + Field *f; + uint number = 0; + bool singleTable = querySchema.tables()->count() <= 1; + if (singleTable) { + //make sure we will have single table: + for (Field::ListIterator it = querySchema.fieldsIterator(); (f = it.current()); ++it, number++) { + if (querySchema.isColumnVisible(number) && f->table() && f->table()->lookupFieldSchema( *f )) { + //uups, no, there's at least one left join + singleTable = false; + break; + } + } + } + + QString sql; //final sql string + sql.reserve(4096); +//unused QString s_from_additional; //additional tables list needed for lookup fields + QString s_additional_joins; //additional joins needed for lookup fields + QString s_additional_fields; //additional fields to append to the fields list + uint internalUniqueTableAliasNumber = 0; //used to build internalUniqueTableAliases + uint internalUniqueQueryAliasNumber = 0; //used to build internalUniqueQueryAliases + number = 0; + QPtrList<QuerySchema> subqueries_for_lookup_data; // subqueries will be added to FROM section + QString kexidb_subquery_prefix("__kexidb_subquery_"); + for (Field::ListIterator it = querySchema.fieldsIterator(); (f = it.current()); ++it, number++) { + if (querySchema.isColumnVisible(number)) { + if (!sql.isEmpty()) + sql += QString::fromLatin1(", "); + + if (f->isQueryAsterisk()) { + if (!singleTable && static_cast<QueryAsterisk*>(f)->isSingleTableAsterisk()) //single-table * + sql += escapeIdentifier(f->table()->name(), options.identifierEscaping) + + QString::fromLatin1(".*"); + else //all-tables * (or simplified table.* when there's only one table) + sql += QString::fromLatin1("*"); + } + else { + if (f->isExpression()) { + sql += f->expression()->toString(); + } + else { + if (!f->table()) //sanity check + return QString::null; + + QString tableName; + int tablePosition = querySchema.tableBoundToColumn(number); + if (tablePosition>=0) + tableName = querySchema.tableAlias(tablePosition); + if (tableName.isEmpty()) + tableName = f->table()->name(); + + if (!singleTable) { + sql += (escapeIdentifier(tableName, options.identifierEscaping) + "."); + } + sql += escapeIdentifier(f->name(), options.identifierEscaping); + } + QString aliasString = QString(querySchema.columnAlias(number)); + if (!aliasString.isEmpty()) + sql += (QString::fromLatin1(" AS ") + aliasString); +//! @todo add option that allows to omit "AS" keyword + } + LookupFieldSchema *lookupFieldSchema = (options.addVisibleLookupColumns && f->table()) + ? f->table()->lookupFieldSchema( *f ) : 0; + if (lookupFieldSchema && lookupFieldSchema->boundColumn()>=0) { + // Lookup field schema found + // Now we also need to fetch "visible" value from the lookup table, not only the value of binding. + // -> build LEFT OUTER JOIN clause for this purpose (LEFT, not INNER because the binding can be broken) + // "LEFT OUTER JOIN lookupTable ON thisTable.thisField=lookupTable.boundField" + LookupFieldSchema::RowSource& rowSource = lookupFieldSchema->rowSource(); + if (rowSource.type()==LookupFieldSchema::RowSource::Table) { + TableSchema *lookupTable = querySchema.connection()->tableSchema( rowSource.name() ); + FieldList* visibleColumns = 0; + Field *boundField = 0; + if (lookupTable + && (uint)lookupFieldSchema->boundColumn() < lookupTable->fieldCount() + && (visibleColumns = lookupTable->subList( lookupFieldSchema->visibleColumns() )) + && (boundField = lookupTable->field( lookupFieldSchema->boundColumn() ))) + { + //add LEFT OUTER JOIN + if (!s_additional_joins.isEmpty()) + s_additional_joins += QString::fromLatin1(" "); + QString internalUniqueTableAlias( QString("__kexidb_") + lookupTable->name() + "_" + + QString::number(internalUniqueTableAliasNumber++) ); + s_additional_joins += QString("LEFT OUTER JOIN %1 AS %2 ON %3.%4=%5.%6") + .arg(escapeIdentifier(lookupTable->name(), options.identifierEscaping)) + .arg(internalUniqueTableAlias) + .arg(escapeIdentifier(f->table()->name(), options.identifierEscaping)) + .arg(escapeIdentifier(f->name(), options.identifierEscaping)) + .arg(internalUniqueTableAlias) + .arg(escapeIdentifier(boundField->name(), options.identifierEscaping)); + + //add visibleField to the list of SELECTed fields //if it is not yet present there +//not needed if (!querySchema.findTableField( visibleField->table()->name()+"."+visibleField->name() )) { +#if 0 + if (!querySchema.table( visibleField->table()->name() )) { +/* not true + //table should be added after FROM + if (!s_from_additional.isEmpty()) + s_from_additional += QString::fromLatin1(", "); + s_from_additional += escapeIdentifier(visibleField->table()->name(), options.identifierEscaping); + */ + } +#endif + if (!s_additional_fields.isEmpty()) + s_additional_fields += QString::fromLatin1(", "); +// s_additional_fields += (internalUniqueTableAlias + "." //escapeIdentifier(visibleField->table()->name(), options.identifierEscaping) + "." +// escapeIdentifier(visibleField->name(), options.identifierEscaping)); +//! @todo Add lookup schema option for separator other than ' ' or even option for placeholders like "Name ? ?" +//! @todo Add possibility for joining the values at client side. + s_additional_fields += visibleColumns->sqlFieldsList( + driver(), " || ' ' || ", internalUniqueTableAlias, options.identifierEscaping); + } + delete visibleColumns; + } + else if (rowSource.type()==LookupFieldSchema::RowSource::Query) { + QuerySchema *lookupQuery = querySchema.connection()->querySchema( rowSource.name() ); + if (!lookupQuery) { + KexiDBWarn << "Connection::selectStatement(): !lookupQuery" << endl; + return QString::null; + } + const QueryColumnInfo::Vector fieldsExpanded( lookupQuery->fieldsExpanded() ); + if ((uint)lookupFieldSchema->boundColumn() >= fieldsExpanded.count()) { + KexiDBWarn << "Connection::selectStatement(): (uint)lookupFieldSchema->boundColumn() >= fieldsExpanded.count()" << endl; + return QString::null; + } + QueryColumnInfo *boundColumnInfo = fieldsExpanded.at( lookupFieldSchema->boundColumn() ); + if (!boundColumnInfo) { + KexiDBWarn << "Connection::selectStatement(): !boundColumnInfo" << endl; + return QString::null; + } + Field *boundField = boundColumnInfo->field; + if (!boundField) { + KexiDBWarn << "Connection::selectStatement(): !boundField" << endl; + return QString::null; + } + //add LEFT OUTER JOIN + if (!s_additional_joins.isEmpty()) + s_additional_joins += QString::fromLatin1(" "); + QString internalUniqueQueryAlias( + kexidb_subquery_prefix + lookupQuery->name() + "_" + + QString::number(internalUniqueQueryAliasNumber++) ); + s_additional_joins += QString("LEFT OUTER JOIN (%1) AS %2 ON %3.%4=%5.%6") + .arg(selectStatement( *lookupQuery, params, options )) + .arg(internalUniqueQueryAlias) + .arg(escapeIdentifier(f->table()->name(), options.identifierEscaping)) + .arg(escapeIdentifier(f->name(), options.identifierEscaping)) + .arg(internalUniqueQueryAlias) + .arg(escapeIdentifier(boundColumnInfo->aliasOrName(), options.identifierEscaping)); + + if (!s_additional_fields.isEmpty()) + s_additional_fields += QString::fromLatin1(", "); + const QValueList<uint> visibleColumns( lookupFieldSchema->visibleColumns() ); + QString expression; + foreach (QValueList<uint>::ConstIterator, visibleColumnsIt, visibleColumns) { +//! @todo Add lookup schema option for separator other than ' ' or even option for placeholders like "Name ? ?" +//! @todo Add possibility for joining the values at client side. + if (fieldsExpanded.count() <= (*visibleColumnsIt)) { + KexiDBWarn << "Connection::selectStatement(): fieldsExpanded.count() <= (*visibleColumnsIt) : " + << fieldsExpanded.count() << " <= " << *visibleColumnsIt << endl; + return QString::null; + } + if (!expression.isEmpty()) + expression += " || ' ' || "; + expression += (internalUniqueQueryAlias + "." + + escapeIdentifier(fieldsExpanded[*visibleColumnsIt]->aliasOrName(), + options.identifierEscaping)); + } + s_additional_fields += expression; +//subqueries_for_lookup_data.append(lookupQuery); + } + else { + KexiDBWarn << "Connection::selectStatement(): unsupported row source type " + << rowSource.typeName() << endl; + return QString(); + } + } + } + } + + //add lookup fields + if (!s_additional_fields.isEmpty()) + sql += (QString::fromLatin1(", ") + s_additional_fields); + + if (options.alsoRetrieveROWID) { //append rowid column + QString s; + if (!sql.isEmpty()) + s = QString::fromLatin1(", "); + if (querySchema.masterTable()) + s += (escapeIdentifier(querySchema.masterTable()->name())+"."); + s += m_driver->beh->ROW_ID_FIELD_NAME; + sql += s; + } + + sql.prepend("SELECT "); + TableSchema::List* tables = querySchema.tables(); + if ((tables && !tables->isEmpty()) || !subqueries_for_lookup_data.isEmpty()) { + sql += QString::fromLatin1(" FROM "); + QString s_from; + if (tables) { + TableSchema *table; + number = 0; + for (TableSchema::ListIterator it(*tables); (table = it.current()); + ++it, number++) + { + if (!s_from.isEmpty()) + s_from += QString::fromLatin1(", "); + s_from += escapeIdentifier(table->name(), options.identifierEscaping); + QString aliasString = QString(querySchema.tableAlias(number)); + if (!aliasString.isEmpty()) + s_from += (QString::fromLatin1(" AS ") + aliasString); + } + /*unused if (!s_from_additional.isEmpty()) {//additional tables list needed for lookup fields + if (!s_from.isEmpty()) + s_from += QString::fromLatin1(", "); + s_from += s_from_additional; + }*/ + } + // add subqueries for lookup data + uint subqueries_for_lookup_data_counter = 0; + for (QPtrListIterator<QuerySchema> it(subqueries_for_lookup_data); + subqueries_for_lookup_data.current(); ++it, subqueries_for_lookup_data_counter++) + { + if (!s_from.isEmpty()) + s_from += QString::fromLatin1(", "); + s_from += QString::fromLatin1("("); + s_from += selectStatement( *it.current(), params, options ); + s_from += QString::fromLatin1(") AS %1%2") + .arg(kexidb_subquery_prefix).arg(subqueries_for_lookup_data_counter); + } + sql += s_from; + } + QString s_where; + s_where.reserve(4096); + + //JOINS + if (!s_additional_joins.isEmpty()) { + sql += QString::fromLatin1(" ") + s_additional_joins + QString::fromLatin1(" "); + } + +//@todo: we're using WHERE for joins now; use INNER/LEFT/RIGHT JOIN later + + //WHERE + Relationship *rel; + bool wasWhere = false; //for later use + for (Relationship::ListIterator it(*querySchema.relationships()); (rel = it.current()); ++it) { + if (s_where.isEmpty()) { + wasWhere = true; + } + else + s_where += QString::fromLatin1(" AND "); + Field::Pair *pair; + QString s_where_sub; + for (QPtrListIterator<Field::Pair> p_it(*rel->fieldPairs()); (pair = p_it.current()); ++p_it) { + if (!s_where_sub.isEmpty()) + s_where_sub += QString::fromLatin1(" AND "); + s_where_sub += ( + escapeIdentifier(pair->first->table()->name(), options.identifierEscaping) + + QString::fromLatin1(".") + + escapeIdentifier(pair->first->name(), options.identifierEscaping) + + QString::fromLatin1(" = ") + + escapeIdentifier(pair->second->table()->name(), options.identifierEscaping) + + QString::fromLatin1(".") + + escapeIdentifier(pair->second->name(), options.identifierEscaping)); + } + if (rel->fieldPairs()->count()>1) { + s_where_sub.prepend("("); + s_where_sub += QString::fromLatin1(")"); + } + s_where += s_where_sub; + } + //EXPLICITLY SPECIFIED WHERE EXPRESSION + if (querySchema.whereExpression()) { + QuerySchemaParameterValueListIterator paramValuesIt(*m_driver, params); + QuerySchemaParameterValueListIterator *paramValuesItPtr = params.isEmpty() ? 0 : ¶mValuesIt; + if (wasWhere) { +//TODO: () are not always needed + s_where = "(" + s_where + ") AND (" + querySchema.whereExpression()->toString(paramValuesItPtr) + ")"; + } + else { + s_where = querySchema.whereExpression()->toString(paramValuesItPtr); + } + } + if (!s_where.isEmpty()) + sql += QString::fromLatin1(" WHERE ") + s_where; +//! \todo (js) add other sql parts + //(use wasWhere here) + + // ORDER BY + QString orderByString( + querySchema.orderByColumnList().toSQLString(!singleTable/*includeTableName*/, + driver(), options.identifierEscaping) ); + const QValueVector<int> pkeyFieldsOrder( querySchema.pkeyFieldsOrder() ); + if (orderByString.isEmpty() && !pkeyFieldsOrder.isEmpty()) { + //add automatic ORDER BY if there is no explicity defined (especially helps when there are complex JOINs) + OrderByColumnList automaticPKOrderBy; + const QueryColumnInfo::Vector fieldsExpanded( querySchema.fieldsExpanded() ); + foreach (QValueVector<int>::ConstIterator, it, pkeyFieldsOrder) { + if ((*it) < 0) // no field mentioned in this query + continue; + if ((*it) >= (int)fieldsExpanded.count()) { + KexiDBWarn << "Connection::selectStatement(): ORDER BY: (*it) >= fieldsExpanded.count() - " + << (*it) << " >= " << fieldsExpanded.count() << endl; + continue; + } + QueryColumnInfo *ci = fieldsExpanded[ *it ]; + automaticPKOrderBy.appendColumn( *ci ); + } + orderByString = automaticPKOrderBy.toSQLString(!singleTable/*includeTableName*/, + driver(), options.identifierEscaping); + } + if (!orderByString.isEmpty()) + sql += (" ORDER BY " + orderByString); + + //KexiDBDbg << sql << endl; + return sql; +} + +QString Connection::selectStatement( KexiDB::TableSchema& tableSchema, + const SelectStatementOptions& options) const +{ + return selectStatement( *tableSchema.query(), options ); +} + +Field* Connection::findSystemFieldName(KexiDB::FieldList* fieldlist) +{ + Field *f = fieldlist->fields()->first(); + while (f) { + if (m_driver->isSystemFieldName( f->name() )) + return f; + f = fieldlist->fields()->next(); + } + return 0; +} + +Q_ULLONG Connection::lastInsertedAutoIncValue(const QString& aiFieldName, const QString& tableName, + Q_ULLONG* ROWID) +{ + Q_ULLONG row_id = drv_lastInsertRowID(); + if (ROWID) + *ROWID = row_id; + if (m_driver->beh->ROW_ID_FIELD_RETURNS_LAST_AUTOINCREMENTED_VALUE) { + return row_id; + } + RowData rdata; + if (row_id<=0 || true!=querySingleRecord( + QString::fromLatin1("SELECT ") + tableName + QString::fromLatin1(".") + aiFieldName + QString::fromLatin1(" FROM ") + tableName + + QString::fromLatin1(" WHERE ") + m_driver->beh->ROW_ID_FIELD_NAME + QString::fromLatin1("=") + QString::number(row_id), rdata)) + { +// KexiDBDbg << "Connection::lastInsertedAutoIncValue(): row_id<=0 || true!=querySingleRecord()" << endl; + return (Q_ULLONG)-1; //ULL; + } + return rdata[0].toULongLong(); +} + +Q_ULLONG Connection::lastInsertedAutoIncValue(const QString& aiFieldName, + const KexiDB::TableSchema& table, Q_ULLONG* ROWID) +{ + return lastInsertedAutoIncValue(aiFieldName,table.name(), ROWID); +} + +//! Creates a Field list for kexi__fields, for sanity. Used by createTable() +static FieldList* createFieldListForKexi__Fields(TableSchema *kexi__fieldsSchema) +{ + if (!kexi__fieldsSchema) + return 0; + return kexi__fieldsSchema->subList( + "t_id", + "f_type", + "f_name", + "f_length", + "f_precision", + "f_constraints", + "f_options", + "f_default", + "f_order", + "f_caption", + "f_help" + ); +} + +//! builds a list of values for field's \a f properties. Used by createTable(). +void buildValuesForKexi__Fields(QValueList<QVariant>& vals, Field* f) +{ + vals.clear(); + vals + << QVariant(f->table()->id()) + << QVariant(f->type()) + << QVariant(f->name()) + << QVariant(f->isFPNumericType() ? f->scale() : f->length()) + << QVariant(f->isFPNumericType() ? f->precision() : 0) + << QVariant(f->constraints()) + << QVariant(f->options()) + // KexiDB::variantToString() is needed here because the value can be of any QVariant type, + // depending on f->type() + << (f->defaultValue().isNull() + ? QVariant() : QVariant(KexiDB::variantToString( f->defaultValue() ))) + << QVariant(f->order()) + << QVariant(f->caption()) + << QVariant(f->description()); +} + +bool Connection::storeMainFieldSchema(Field *field) +{ + if (!field || !field->table()) + return false; + FieldList *fl = createFieldListForKexi__Fields(d->tables_byname["kexi__fields"]); + if (!fl) + return false; + + QValueList<QVariant> vals; + buildValuesForKexi__Fields(vals, field); + QValueList<QVariant>::ConstIterator valsIt = vals.constBegin(); + Field *f; + bool first = true; + QString sql = "UPDATE kexi__fields SET "; + for (Field::ListIterator it( fl->fieldsIterator() ); (f = it.current()); ++it, ++valsIt) { + sql.append( (first ? QString::null : QString(", ")) + + f->name() + "=" + m_driver->valueToSQL( f, *valsIt ) ); + if (first) + first = false; + } + delete fl; + + sql.append(QString(" WHERE t_id=") + QString::number( field->table()->id() ) + + " AND f_name=" + m_driver->valueToSQL( Field::Text, field->name() ) ); + return executeSQL( sql ); +} + +#define createTable_ERR \ + { KexiDBDbg << "Connection::createTable(): ERROR!" <<endl; \ + setError(this, i18n("Creating table failed.")); \ + rollbackAutoCommitTransaction(tg.transaction()); \ + return false; } + //setError( errorNum(), i18n("Creating table failed.") + " " + errorMsg()); + +//! Creates a table according to the given schema +/*! Creates a table according to the given TableSchema, adding the table and + column definitions to kexi__* tables. Checks that a database is in use, + that the table name is not that of a system table, and that the schema + defines at least one column. + If the table exists, and replaceExisting is true, the table is replaced. + Otherwise, the table is not replaced. +*/ +bool Connection::createTable( KexiDB::TableSchema* tableSchema, bool replaceExisting ) +{ + if (!tableSchema || !checkIsDatabaseUsed()) + return false; + + //check if there are any fields + if (tableSchema->fieldCount()<1) { + clearError(); + setError(ERR_CANNOT_CREATE_EMPTY_OBJECT, i18n("Cannot create table without fields.")); + return false; + } + const bool internalTable = dynamic_cast<InternalTableSchema*>(tableSchema); + + const QString &tableName = tableSchema->name().lower(); + + if (!internalTable) { + if (m_driver->isSystemObjectName( tableName )) { + clearError(); + setError(ERR_SYSTEM_NAME_RESERVED, i18n("System name \"%1\" cannot be used as table name.") + .arg(tableSchema->name())); + return false; + } + + Field *sys_field = findSystemFieldName(tableSchema); + if (sys_field) { + clearError(); + setError(ERR_SYSTEM_NAME_RESERVED, + i18n("System name \"%1\" cannot be used as one of fields in \"%2\" table.") + .arg(sys_field->name()).arg(tableName)); + return false; + } + } + + bool previousSchemaStillKept = false; + + KexiDB::TableSchema *existingTable = 0; + if (replaceExisting) { + //get previous table (do not retrieve, though) + existingTable = d->tables_byname[tableName]; + if (existingTable) { + if (existingTable == tableSchema) { + clearError(); + setError(ERR_OBJECT_EXISTS, + i18n("Could not create the same table \"%1\" twice.").arg(tableSchema->name()) ); + return false; + } +//TODO(js): update any structure (e.g. queries) that depend on this table! + if (existingTable->id()>0) + tableSchema->m_id = existingTable->id(); //copy id from existing table + previousSchemaStillKept = true; + if (!dropTable( existingTable, false /*alsoRemoveSchema*/ )) + return false; + } + } + else { + if (this->tableSchema( tableSchema->name() ) != 0) { + clearError(); + setError(ERR_OBJECT_EXISTS, i18n("Table \"%1\" already exists.").arg(tableSchema->name()) ); + return false; + } + } + +/* if (replaceExisting) { + //get previous table (do not retrieve, though) + KexiDB::TableSchema *existingTable = d->tables_byname.take(name); + if (oldTable) { + }*/ + + TransactionGuard tg; + if (!beginAutoCommitTransaction(tg)) + return false; + + if (!drv_createTable(*tableSchema)) + createTable_ERR; + + //add schema data to kexi__* tables + if (!internalTable) { + //update kexi__objects + if (!storeObjectSchemaData( *tableSchema, true )) + createTable_ERR; + + TableSchema *ts = d->tables_byname["kexi__fields"]; + if (!ts) + return false; + //for sanity: remove field info (if any) for this table id + if (!KexiDB::deleteRow(*this, ts, "t_id", tableSchema->id())) + return false; + + FieldList *fl = createFieldListForKexi__Fields(d->tables_byname["kexi__fields"]); + if (!fl) + return false; + +// int order = 0; + Field *f; + for (Field::ListIterator it( *tableSchema->fields() ); (f = it.current()); ++it/*, order++*/) { + QValueList<QVariant> vals; + buildValuesForKexi__Fields(vals, f); + if (!insertRecord(*fl, vals )) + createTable_ERR; + } + delete fl; + + if (!storeExtendedTableSchemaData(*tableSchema)) + createTable_ERR; + } + + //finally: +/* if (replaceExisting) { + if (existingTable) { + d->tables.take(existingTable->id()); + delete existingTable; + } + }*/ + + bool res = commitAutoCommitTransaction(tg.transaction()); + + if (res) { + if (internalTable) { + //insert the internal table into structures + insertInternalTableSchema(tableSchema); + } + else { + if (previousSchemaStillKept) { + //remove previous table schema + removeTableSchemaInternal(tableSchema); + } + //store one schema object locally: + d->tables.insert(tableSchema->id(), tableSchema); + d->tables_byname.insert(tableSchema->name().lower(), tableSchema); + } + //ok, this table is not created by the connection + tableSchema->m_conn = this; + } + return res; +} + +void Connection::removeTableSchemaInternal(TableSchema *tableSchema) +{ + d->tables_byname.remove(tableSchema->name()); + d->tables.remove(tableSchema->id()); +} + +bool Connection::removeObject( uint objId ) +{ + clearError(); + //remove table schema from kexi__* tables + if (!KexiDB::deleteRow(*this, d->tables_byname["kexi__objects"], "o_id", objId) //schema entry + || !KexiDB::deleteRow(*this, d->tables_byname["kexi__objectdata"], "o_id", objId)) {//data blocks + setError(ERR_DELETE_SERVER_ERROR, i18n("Could not remove object's data.")); + return false; + } + return true; +} + +bool Connection::drv_dropTable( const QString& name ) +{ + m_sql = "DROP TABLE " + escapeIdentifier(name); + return executeSQL(m_sql); +} + +//! Drops a table corresponding to the name in the given schema +/*! Drops a table according to the name given by the TableSchema, removing the + table and column definitions to kexi__* tables. Checks first that the + table is not a system table. + + TODO: Should check that a database is currently in use? (c.f. createTable) +*/ +tristate Connection::dropTable( KexiDB::TableSchema* tableSchema ) +{ + return dropTable( tableSchema, true ); +} + +tristate Connection::dropTable( KexiDB::TableSchema* tableSchema, bool alsoRemoveSchema) +{ +// Each SQL identifier needs to be escaped in the generated query. + clearError(); + if (!tableSchema) + return false; + + QString errmsg(i18n("Table \"%1\" cannot be removed.\n")); + //be sure that we handle the correct TableSchema object: + if (tableSchema->id() < 0 + || this->tableSchema(tableSchema->name())!=tableSchema + || this->tableSchema(tableSchema->id())!=tableSchema) + { + setError(ERR_OBJECT_NOT_FOUND, errmsg.arg(tableSchema->name()) + +i18n("Unexpected name or identifier.")); + return false; + } + + tristate res = closeAllTableSchemaChangeListeners(*tableSchema); + if (true!=res) + return res; + + //sanity checks: + if (m_driver->isSystemObjectName( tableSchema->name() )) { + setError(ERR_SYSTEM_NAME_RESERVED, errmsg.arg(tableSchema->name()) + d->strItIsASystemObject()); + return false; + } + + TransactionGuard tg; + if (!beginAutoCommitTransaction(tg)) + return false; + + //for sanity we're checking if this table exists physically + if (drv_containsTable(tableSchema->name())) { + if (!drv_dropTable(tableSchema->name())) + return false; + } + + TableSchema *ts = d->tables_byname["kexi__fields"]; + if (!KexiDB::deleteRow(*this, ts, "t_id", tableSchema->id())) //field entries + return false; + + //remove table schema from kexi__objects table + if (!removeObject( tableSchema->id() )) { + return false; + } + + if (alsoRemoveSchema) { +//! \todo js: update any structure (e.g. queries) that depend on this table! + tristate res = removeDataBlock( tableSchema->id(), "extended_schema"); + if (!res) + return false; + removeTableSchemaInternal(tableSchema); + } + return commitAutoCommitTransaction(tg.transaction()); +} + +tristate Connection::dropTable( const QString& table ) +{ + clearError(); + TableSchema* ts = tableSchema( table ); + if (!ts) { + setError(ERR_OBJECT_NOT_FOUND, i18n("Table \"%1\" does not exist.") + .arg(table)); + return false; + } + return dropTable(ts); +} + +tristate Connection::alterTable( TableSchema& tableSchema, TableSchema& newTableSchema ) +{ + clearError(); + tristate res = closeAllTableSchemaChangeListeners(tableSchema); + if (true!=res) + return res; + + if (&tableSchema == &newTableSchema) { + setError(ERR_OBJECT_THE_SAME, i18n("Could not alter table \"%1\" using the same table.") + .arg(tableSchema.name())); + return false; + } +//TODO(js): implement real altering +//TODO(js): update any structure (e.g. query) that depend on this table! + bool ok, empty; +#if 0//TODO ucomment: + empty = isEmpty( tableSchema, ok ) && ok; +#else + empty = true; +#endif + if (empty) { + ok = createTable(&newTableSchema, true/*replace*/); + } + return ok; +} + +bool Connection::alterTableName(TableSchema& tableSchema, const QString& newName, bool replace) +{ + clearError(); + if (&tableSchema!=d->tables[tableSchema.id()]) { + setError(ERR_OBJECT_NOT_FOUND, i18n("Unknown table \"%1\"").arg(tableSchema.name())); + return false; + } + if (newName.isEmpty() || !KexiUtils::isIdentifier(newName)) { + setError(ERR_INVALID_IDENTIFIER, i18n("Invalid table name \"%1\"").arg(newName)); + return false; + } + const QString oldTableName = tableSchema.name(); + const QString newTableName = newName.lower().stripWhiteSpace(); + if (oldTableName.lower().stripWhiteSpace() == newTableName) { + setError(ERR_OBJECT_THE_SAME, i18n("Could rename table \"%1\" using the same name.") + .arg(newTableName)); + return false; + } +//TODO: alter table name for server DB backends! +//TODO: what about objects (queries/forms) that use old name? +//TODO + TableSchema *tableToReplace = this->tableSchema( newName ); + const bool destTableExists = tableToReplace != 0; + const int origID = destTableExists ? tableToReplace->id() : -1; //will be reused in the new table + if (!replace && destTableExists) { + setError(ERR_OBJECT_EXISTS, + i18n("Could not rename table \"%1\" to \"%2\". Table \"%3\" already exists.") + .arg(tableSchema.name()).arg(newName).arg(newName)); + return false; + } + +//helper: +#define alterTableName_ERR \ + tableSchema.setName(oldTableName) //restore old name + + TransactionGuard tg; + if (!beginAutoCommitTransaction(tg)) + return false; + + // drop the table replaced (with schema) + if (destTableExists) { + if (!replace) { + return false; + } + if (!dropTable( newName )) { + return false; + } + + // the new table owns the previous table's id: + if (!executeSQL(QString::fromLatin1("UPDATE kexi__objects SET o_id=%1 WHERE o_id=%2 AND o_type=%3") + .arg(origID).arg(tableSchema.id()).arg((int)TableObjectType))) + { + return false; + } + if (!executeSQL(QString::fromLatin1("UPDATE kexi__fields SET t_id=%1 WHERE t_id=%2") + .arg(origID).arg(tableSchema.id()))) + { + return false; + } + d->tables.take(tableSchema.id()); + d->tables.insert(origID, &tableSchema); + //maintain table ID + tableSchema.m_id = origID; + } + + if (!drv_alterTableName(tableSchema, newTableName)) { + alterTableName_ERR; + return false; + } + + // Update kexi__objects + //TODO + if (!executeSQL(QString::fromLatin1("UPDATE kexi__objects SET o_name=%1 WHERE o_id=%2") + .arg(m_driver->escapeString(tableSchema.name())).arg(tableSchema.id()))) + { + alterTableName_ERR; + return false; + } +//TODO what about caption? + + //restore old name: it will be changed soon! + tableSchema.setName(oldTableName); + + if (!commitAutoCommitTransaction(tg.transaction())) { + alterTableName_ERR; + return false; + } + + //update tableSchema: + d->tables_byname.take(tableSchema.name()); + tableSchema.setName(newTableName); + d->tables_byname.insert(tableSchema.name(), &tableSchema); + return true; +} + +bool Connection::drv_alterTableName(TableSchema& tableSchema, const QString& newName) +{ + const QString oldTableName = tableSchema.name(); + tableSchema.setName(newName); + + if (!executeSQL(QString::fromLatin1("ALTER TABLE %1 RENAME TO %2") + .arg(escapeIdentifier(oldTableName)).arg(escapeIdentifier(newName)))) + { + tableSchema.setName(oldTableName); //restore old name + return false; + } + return true; +} + +bool Connection::dropQuery( KexiDB::QuerySchema* querySchema ) +{ + clearError(); + if (!querySchema) + return false; + + TransactionGuard tg; + if (!beginAutoCommitTransaction(tg)) + return false; + +/* TableSchema *ts = d->tables_byname["kexi__querydata"]; + if (!KexiDB::deleteRow(*this, ts, "q_id", querySchema->id())) + return false; + + ts = d->tables_byname["kexi__queryfields"]; + if (!KexiDB::deleteRow(*this, ts, "q_id", querySchema->id())) + return false; + + ts = d->tables_byname["kexi__querytables"]; + if (!KexiDB::deleteRow(*this, ts, "q_id", querySchema->id())) + return false;*/ + + //remove query schema from kexi__objects table + if (!removeObject( querySchema->id() )) { + return false; + } + +//TODO(js): update any structure that depend on this table! + d->queries_byname.remove(querySchema->name()); + d->queries.remove(querySchema->id()); + + return commitAutoCommitTransaction(tg.transaction()); +} + +bool Connection::dropQuery( const QString& query ) +{ + clearError(); + QuerySchema* qs = querySchema( query ); + if (!qs) { + setError(ERR_OBJECT_NOT_FOUND, i18n("Query \"%1\" does not exist.") + .arg(query)); + return false; + } + return dropQuery(qs); +} + +bool Connection::drv_createTable( const KexiDB::TableSchema& tableSchema ) +{ + m_sql = createTableStatement(tableSchema); + KexiDBDbg<<"******** "<<m_sql<<endl; + return executeSQL(m_sql); +} + +bool Connection::drv_createTable( const QString& tableSchemaName ) +{ + TableSchema *ts = d->tables_byname[tableSchemaName]; + if (!ts) + return false; + return drv_createTable(*ts); +} + +bool Connection::beginAutoCommitTransaction(TransactionGuard &tg) +{ + if ((m_driver->d->features & Driver::IgnoreTransactions) + || !d->autoCommit) + { + tg.setTransaction( Transaction() ); + return true; + } + + // commit current transaction (if present) for drivers + // that allow single transaction per connection + if (m_driver->d->features & Driver::SingleTransactions) { + if (d->default_trans_started_inside) //only commit internally started transaction + if (!commitTransaction(d->default_trans, true)) { + tg.setTransaction( Transaction() ); + return false; //we have a real error + } + + d->default_trans_started_inside = d->default_trans.isNull(); + if (!d->default_trans_started_inside) { + tg.setTransaction( d->default_trans ); + tg.doNothing(); + return true; //reuse externally started transaction + } + } + else if (!(m_driver->d->features & Driver::MultipleTransactions)) { + tg.setTransaction( Transaction() ); + return true; //no trans. supported at all - just return + } + tg.setTransaction( beginTransaction() ); + return !error(); +} + +bool Connection::commitAutoCommitTransaction(const Transaction& trans) +{ + if (m_driver->d->features & Driver::IgnoreTransactions) + return true; + if (trans.isNull() || !m_driver->transactionsSupported()) + return true; + if (m_driver->d->features & Driver::SingleTransactions) { + if (!d->default_trans_started_inside) //only commit internally started transaction + return true; //give up + } + return commitTransaction(trans, true); +} + +bool Connection::rollbackAutoCommitTransaction(const Transaction& trans) +{ + if (trans.isNull() || !m_driver->transactionsSupported()) + return true; + return rollbackTransaction(trans); +} + +#define SET_ERR_TRANS_NOT_SUPP \ + { setError(ERR_UNSUPPORTED_DRV_FEATURE, \ + i18n("Transactions are not supported for \"%1\" driver.").arg(m_driver->name() )); } + +#define SET_BEGIN_TR_ERROR \ + { if (!error()) \ + setError(ERR_ROLLBACK_OR_COMMIT_TRANSACTION, i18n("Begin transaction failed")); } + +Transaction Connection::beginTransaction() +{ + if (!checkIsDatabaseUsed()) + return Transaction::null; + Transaction trans; + if (m_driver->d->features & Driver::IgnoreTransactions) { + //we're creating dummy transaction data here, + //so it will look like active + trans.m_data = new TransactionData(this); + d->transactions.append(trans); + return trans; + } + if (m_driver->d->features & Driver::SingleTransactions) { + if (d->default_trans.active()) { + setError(ERR_TRANSACTION_ACTIVE, i18n("Transaction already started.") ); + return Transaction::null; + } + if (!(trans.m_data = drv_beginTransaction())) { + SET_BEGIN_TR_ERROR; + return Transaction::null; + } + d->default_trans = trans; + d->transactions.append(trans); + return d->default_trans; + } + if (m_driver->d->features & Driver::MultipleTransactions) { + if (!(trans.m_data = drv_beginTransaction())) { + SET_BEGIN_TR_ERROR; + return Transaction::null; + } + d->transactions.append(trans); + return trans; + } + + SET_ERR_TRANS_NOT_SUPP; + return Transaction::null; +} + +bool Connection::commitTransaction(const Transaction trans, bool ignore_inactive) +{ + if (!isDatabaseUsed()) + return false; +// if (!checkIsDatabaseUsed()) + //return false; + if ( !m_driver->transactionsSupported() + && !(m_driver->d->features & Driver::IgnoreTransactions)) + { + SET_ERR_TRANS_NOT_SUPP; + return false; + } + Transaction t = trans; + if (!t.active()) { //try default tr. + if (!d->default_trans.active()) { + if (ignore_inactive) + return true; + clearError(); + setError(ERR_NO_TRANSACTION_ACTIVE, i18n("Transaction not started.") ); + return false; + } + t = d->default_trans; + d->default_trans = Transaction::null; //now: no default tr. + } + bool ret = true; + if (! (m_driver->d->features & Driver::IgnoreTransactions) ) + ret = drv_commitTransaction(t.m_data); + if (t.m_data) + t.m_data->m_active = false; //now this transaction if inactive + if (!d->dont_remove_transactions) //true=transaction obj will be later removed from list + d->transactions.remove(t); + if (!ret && !error()) + setError(ERR_ROLLBACK_OR_COMMIT_TRANSACTION, i18n("Error on commit transaction")); + return ret; +} + +bool Connection::rollbackTransaction(const Transaction trans, bool ignore_inactive) +{ + if (!isDatabaseUsed()) + return false; +// if (!checkIsDatabaseUsed()) +// return false; + if ( !m_driver->transactionsSupported() + && !(m_driver->d->features & Driver::IgnoreTransactions)) + { + SET_ERR_TRANS_NOT_SUPP; + return false; + } + Transaction t = trans; + if (!t.active()) { //try default tr. + if (!d->default_trans.active()) { + if (ignore_inactive) + return true; + clearError(); + setError(ERR_NO_TRANSACTION_ACTIVE, i18n("Transaction not started.") ); + return false; + } + t = d->default_trans; + d->default_trans = Transaction::null; //now: no default tr. + } + bool ret = true; + if (! (m_driver->d->features & Driver::IgnoreTransactions) ) + ret = drv_rollbackTransaction(t.m_data); + if (t.m_data) + t.m_data->m_active = false; //now this transaction if inactive + if (!d->dont_remove_transactions) //true=transaction obj will be later removed from list + d->transactions.remove(t); + if (!ret && !error()) + setError(ERR_ROLLBACK_OR_COMMIT_TRANSACTION, i18n("Error on rollback transaction")); + return ret; +} + +#undef SET_ERR_TRANS_NOT_SUPP +#undef SET_BEGIN_TR_ERROR + +/*bool Connection::duringTransaction() +{ + return drv_duringTransaction(); +}*/ + +Transaction& Connection::defaultTransaction() const +{ + return d->default_trans; +} + +void Connection::setDefaultTransaction(const Transaction& trans) +{ + if (!isDatabaseUsed()) + return; +// if (!checkIsDatabaseUsed()) + // return; + if ( !(m_driver->d->features & Driver::IgnoreTransactions) + && (!trans.active() || !m_driver->transactionsSupported()) ) + { + return; + } + d->default_trans = trans; +} + +const QValueList<Transaction>& Connection::transactions() +{ + return d->transactions; +} + +bool Connection::autoCommit() const +{ + return d->autoCommit; +} + +bool Connection::setAutoCommit(bool on) +{ + if (d->autoCommit == on || m_driver->d->features & Driver::IgnoreTransactions) + return true; + if (!drv_setAutoCommit(on)) + return false; + d->autoCommit = on; + return true; +} + +TransactionData* Connection::drv_beginTransaction() +{ + QString old_sql = m_sql; //don't + if (!executeSQL( "BEGIN" )) + return 0; + return new TransactionData(this); +} + +bool Connection::drv_commitTransaction(TransactionData *) +{ + return executeSQL( "COMMIT" ); +} + +bool Connection::drv_rollbackTransaction(TransactionData *) +{ + return executeSQL( "ROLLBACK" ); +} + +bool Connection::drv_setAutoCommit(bool /*on*/) +{ + return true; +} + +Cursor* Connection::executeQuery( const QString& statement, uint cursor_options ) +{ + if (statement.isEmpty()) + return 0; + Cursor *c = prepareQuery( statement, cursor_options ); + if (!c) + return 0; + if (!c->open()) {//err - kill that + setError(c); + delete c; + return 0; + } + return c; +} + +Cursor* Connection::executeQuery( QuerySchema& query, const QValueList<QVariant>& params, + uint cursor_options ) +{ + Cursor *c = prepareQuery( query, params, cursor_options ); + if (!c) + return 0; + if (!c->open()) {//err - kill that + setError(c); + delete c; + return 0; + } + return c; +} + +Cursor* Connection::executeQuery( QuerySchema& query, uint cursor_options ) +{ + return executeQuery(query, QValueList<QVariant>(), cursor_options); +} + +Cursor* Connection::executeQuery( TableSchema& table, uint cursor_options ) +{ + return executeQuery( *table.query(), cursor_options ); +} + +Cursor* Connection::prepareQuery( TableSchema& table, uint cursor_options ) +{ + return prepareQuery( *table.query(), cursor_options ); +} + +Cursor* Connection::prepareQuery( QuerySchema& query, const QValueList<QVariant>& params, + uint cursor_options ) +{ + Cursor* cursor = prepareQuery(query, cursor_options); + if (cursor) + cursor->setQueryParameters(params); + return cursor; +} + +bool Connection::deleteCursor(Cursor *cursor) +{ + if (!cursor) + return false; + if (cursor->connection()!=this) {//illegal call + KexiDBWarn << "Connection::deleteCursor(): Cannot delete the cursor not owned by the same connection!" << endl; + return false; + } + const bool ret = cursor->close(); + delete cursor; + return ret; +} + +bool Connection::setupObjectSchemaData( const RowData &data, SchemaData &sdata ) +{ + //not found: retrieve schema +/* KexiDB::Cursor *cursor; + if (!(cursor = executeQuery( QString("select * from kexi__objects where o_id='%1'").arg(objId) ))) + return false; + if (!cursor->moveFirst()) { + deleteCursor(cursor); + return false; + }*/ + //if (!ok) { + //deleteCursor(cursor); + //return 0; +// } + bool ok; + sdata.m_id = data[0].toInt(&ok); + if (!ok) { + return false; + } + sdata.m_name = data[2].toString(); + if (!KexiUtils::isIdentifier( sdata.m_name )) { + setError(ERR_INVALID_IDENTIFIER, i18n("Invalid object name \"%1\"").arg(sdata.m_name)); + return false; + } + sdata.m_caption = data[3].toString(); + sdata.m_desc = data[4].toString(); + +// KexiDBDbg<<"@@@ Connection::setupObjectSchemaData() == " << sdata.schemaDataDebugString() << endl; + return true; +} + +tristate Connection::loadObjectSchemaData( int objectID, SchemaData &sdata ) +{ + RowData data; + if (true!=querySingleRecord(QString::fromLatin1( + "SELECT o_id, o_type, o_name, o_caption, o_desc FROM kexi__objects WHERE o_id=%1") + .arg(objectID), data)) + return cancelled; + return setupObjectSchemaData( data, sdata ); +} + +tristate Connection::loadObjectSchemaData( int objectType, const QString& objectName, SchemaData &sdata ) +{ + RowData data; + if (true!=querySingleRecord(QString::fromLatin1("SELECT o_id, o_type, o_name, o_caption, o_desc " + "FROM kexi__objects WHERE o_type=%1 AND lower(o_name)=%2") + .arg(objectType).arg(m_driver->valueToSQL(Field::Text, objectName.lower())), data)) + return cancelled; + return setupObjectSchemaData( data, sdata ); +} + +bool Connection::storeObjectSchemaData( SchemaData &sdata, bool newObject ) +{ + TableSchema *ts = d->tables_byname["kexi__objects"]; + if (!ts) + return false; + if (newObject) { + int existingID; + if (true == querySingleNumber(QString::fromLatin1( + "SELECT o_id FROM kexi__objects WHERE o_type=%1 AND lower(o_name)=%2") + .arg(sdata.type()).arg(m_driver->valueToSQL(Field::Text, sdata.name().lower())), existingID)) + { + //we already have stored a schema data with the same name and type: + //just update it's properties as it would be existing object + sdata.m_id = existingID; + newObject = false; + } + } + if (newObject) { + FieldList *fl; + bool ok; + if (sdata.id()<=0) {//get new ID + fl = ts->subList("o_type", "o_name", "o_caption", "o_desc"); + ok = fl!=0; + if (ok && !insertRecord(*fl, QVariant(sdata.type()), QVariant(sdata.name()), + QVariant(sdata.caption()), QVariant(sdata.description()) )) + ok = false; + delete fl; + if (!ok) + return false; + //fetch newly assigned ID +//! @todo safe to cast it? + int obj_id = (int)lastInsertedAutoIncValue("o_id",*ts); + KexiDBDbg << "######## NEW obj_id == " << obj_id << endl; + if (obj_id<=0) + return false; + sdata.m_id = obj_id; + return true; + } else { + fl = ts->subList("o_id", "o_type", "o_name", "o_caption", "o_desc"); + ok = fl!=0; + if (ok && !insertRecord(*fl, QVariant(sdata.id()), QVariant(sdata.type()), QVariant(sdata.name()), + QVariant(sdata.caption()), QVariant(sdata.description()) )) + ok = false; + delete fl; + return ok; + } + } + //existing object: + return executeSQL(QString("UPDATE kexi__objects SET o_type=%2, o_caption=%3, o_desc=%4 WHERE o_id=%1") + .arg(sdata.id()).arg(sdata.type()) + .arg(m_driver->valueToSQL(KexiDB::Field::Text, sdata.caption())) + .arg(m_driver->valueToSQL(KexiDB::Field::Text, sdata.description())) ); +} + +tristate Connection::querySingleRecordInternal(RowData &data, const QString* sql, QuerySchema* query, + bool addLimitTo1) +{ + Q_ASSERT(sql || query); +//! @todo does not work with non-SQL data sources + if (sql) + m_sql = addLimitTo1 ? (*sql + " LIMIT 1") : *sql; // is this safe? + KexiDB::Cursor *cursor; + if (!(cursor = sql ? executeQuery( m_sql ) : executeQuery( *query ))) { + KexiDBWarn << "Connection::querySingleRecord(): !executeQuery() " << m_sql << endl; + return false; + } + if (!cursor->moveFirst() || cursor->eof()) { + const tristate result = cursor->error() ? false : cancelled; + KexiDBWarn << "Connection::querySingleRecord(): !cursor->moveFirst() || cursor->eof() m_sql=" << m_sql << endl; + setError(cursor); + deleteCursor(cursor); + return result; + } + cursor->storeCurrentRow(data); + return deleteCursor(cursor); +} + +tristate Connection::querySingleRecord(const QString& sql, RowData &data, bool addLimitTo1) +{ + return querySingleRecordInternal(data, &sql, 0, addLimitTo1); +} + +tristate Connection::querySingleRecord(QuerySchema& query, RowData &data, bool addLimitTo1) +{ + return querySingleRecordInternal(data, 0, &query, addLimitTo1); +} + +bool Connection::checkIfColumnExists(Cursor *cursor, uint column) +{ + if (column >= cursor->fieldCount()) { + setError(ERR_CURSOR_RECORD_FETCHING, i18n("Column %1 does not exist for the query.").arg(column)); + return false; + } + return true; +} + +tristate Connection::querySingleString(const QString& sql, QString &value, uint column, bool addLimitTo1) +{ + KexiDB::Cursor *cursor; + m_sql = addLimitTo1 ? (sql + " LIMIT 1") : sql; // is this safe?; + if (!(cursor = executeQuery( m_sql ))) { + KexiDBWarn << "Connection::querySingleRecord(): !executeQuery() " << m_sql << endl; + return false; + } + if (!cursor->moveFirst() || cursor->eof()) { + const tristate result = cursor->error() ? false : cancelled; + KexiDBWarn << "Connection::querySingleRecord(): !cursor->moveFirst() || cursor->eof() " << m_sql << endl; + deleteCursor(cursor); + return result; + } + if (!checkIfColumnExists(cursor, column)) { + deleteCursor(cursor); + return false; + } + value = cursor->value(column).toString(); + return deleteCursor(cursor); +} + +tristate Connection::querySingleNumber(const QString& sql, int &number, uint column, bool addLimitTo1) +{ + static QString str; + static bool ok; + const tristate result = querySingleString(sql, str, column, addLimitTo1); + if (result!=true) + return result; + number = str.toInt(&ok); + return ok; +} + +bool Connection::queryStringList(const QString& sql, QStringList& list, uint column) +{ + KexiDB::Cursor *cursor; + clearError(); + m_sql = sql; + if (!(cursor = executeQuery( m_sql ))) { + KexiDBWarn << "Connection::queryStringList(): !executeQuery() " << m_sql << endl; + return false; + } + cursor->moveFirst(); + if (cursor->error()) { + setError(cursor); + deleteCursor(cursor); + return false; + } + if (!cursor->eof() && !checkIfColumnExists(cursor, column)) { + deleteCursor(cursor); + return false; + } + list.clear(); + while (!cursor->eof()) { + list.append( cursor->value(column).toString() ); + if (!cursor->moveNext() && cursor->error()) { + setError(cursor); + deleteCursor(cursor); + return false; + } + } + return deleteCursor(cursor); +} + +bool Connection::resultExists(const QString& sql, bool &success, bool addLimitTo1) +{ + KexiDB::Cursor *cursor; + //optimization + if (m_driver->beh->SELECT_1_SUBQUERY_SUPPORTED) { + //this is at least for sqlite + if (addLimitTo1 && sql.left(6).upper() == "SELECT") + m_sql = QString("SELECT 1 FROM (") + sql + ") LIMIT 1"; // is this safe?; + else + m_sql = sql; + } + else { + if (addLimitTo1 && sql.left(6).upper() == "SELECT") + m_sql = sql + " LIMIT 1"; //not always safe! + else + m_sql = sql; + } + if (!(cursor = executeQuery( m_sql ))) { + KexiDBWarn << "Connection::querySingleRecord(): !executeQuery() " << m_sql << endl; + success = false; + return false; + } + if (!cursor->moveFirst() || cursor->eof()) { + success = !cursor->error(); + KexiDBWarn << "Connection::querySingleRecord(): !cursor->moveFirst() || cursor->eof() " << m_sql << endl; + setError(cursor); + deleteCursor(cursor); + return false; + } + success = deleteCursor(cursor); + return true; +} + +bool Connection::isEmpty( TableSchema& table, bool &success ) +{ + return !resultExists( selectStatement( *table.query() ), success ); +} + +//! Used by addFieldPropertyToExtendedTableSchemaData() +static void createExtendedTableSchemaMainElementIfNeeded( + QDomDocument& doc, QDomElement& extendedTableSchemaMainEl, + bool& extendedTableSchemaStringIsEmpty) +{ + if (!extendedTableSchemaStringIsEmpty) + return; + //init document + extendedTableSchemaMainEl = doc.createElement("EXTENDED_TABLE_SCHEMA"); + doc.appendChild( extendedTableSchemaMainEl ); + extendedTableSchemaMainEl.setAttribute("version", QString::number(KEXIDB_EXTENDED_TABLE_SCHEMA_VERSION)); + extendedTableSchemaStringIsEmpty = false; +} + +//! Used by addFieldPropertyToExtendedTableSchemaData() +static void createExtendedTableSchemaFieldElementIfNeeded(QDomDocument& doc, + QDomElement& extendedTableSchemaMainEl, const QString& fieldName, QDomElement& extendedTableSchemaFieldEl, + bool append = true) +{ + if (!extendedTableSchemaFieldEl.isNull()) + return; + extendedTableSchemaFieldEl = doc.createElement("field"); + if (append) + extendedTableSchemaMainEl.appendChild( extendedTableSchemaFieldEl ); + extendedTableSchemaFieldEl.setAttribute("name", fieldName); +} + +/*! @internal used by storeExtendedTableSchemaData() + Creates DOM node for \a propertyName and \a propertyValue. + Creates enclosing EXTENDED_TABLE_SCHEMA element if EXTENDED_TABLE_SCHEMA is true. + Updates extendedTableSchemaStringIsEmpty and extendedTableSchemaMainEl afterwards. + If extendedTableSchemaFieldEl is null, creates <field> element (with optional + "custom" attribute is \a custom is false). */ +static void addFieldPropertyToExtendedTableSchemaData( + Field *f, const char* propertyName, const QVariant& propertyValue, + QDomDocument& doc, QDomElement& extendedTableSchemaMainEl, + QDomElement& extendedTableSchemaFieldEl, + bool& extendedTableSchemaStringIsEmpty, + bool custom = false ) +{ + createExtendedTableSchemaMainElementIfNeeded(doc, + extendedTableSchemaMainEl, extendedTableSchemaStringIsEmpty); + createExtendedTableSchemaFieldElementIfNeeded( + doc, extendedTableSchemaMainEl, f->name(), extendedTableSchemaFieldEl); + + //create <property> + QDomElement extendedTableSchemaFieldPropertyEl = doc.createElement("property"); + extendedTableSchemaFieldEl.appendChild( extendedTableSchemaFieldPropertyEl ); + if (custom) + extendedTableSchemaFieldPropertyEl.setAttribute("custom", "true"); + extendedTableSchemaFieldPropertyEl.setAttribute("name", propertyName); + QDomElement extendedTableSchemaFieldPropertyValueEl; + switch (propertyValue.type()) { + case QVariant::String: + extendedTableSchemaFieldPropertyValueEl = doc.createElement("string"); + break; + case QVariant::CString: + extendedTableSchemaFieldPropertyValueEl = doc.createElement("cstring"); + break; + case QVariant::Int: + case QVariant::Double: + case QVariant::UInt: + case QVariant::LongLong: + case QVariant::ULongLong: + extendedTableSchemaFieldPropertyValueEl = doc.createElement("number"); + break; + case QVariant::Bool: + extendedTableSchemaFieldPropertyValueEl = doc.createElement("bool"); + break; + default: +//! @todo add more QVariant types + KexiDBFatal << "addFieldPropertyToExtendedTableSchemaData(): impl. error" << endl; + } + extendedTableSchemaFieldPropertyEl.appendChild( extendedTableSchemaFieldPropertyValueEl ); + extendedTableSchemaFieldPropertyValueEl.appendChild( + doc.createTextNode( propertyValue.toString() ) ); +} + +bool Connection::storeExtendedTableSchemaData(TableSchema& tableSchema) +{ +//! @todo future: save in older versions if neeed + QDomDocument doc("EXTENDED_TABLE_SCHEMA"); + QDomElement extendedTableSchemaMainEl; + bool extendedTableSchemaStringIsEmpty = true; + + //for each field: + Field *f; + for (Field::ListIterator it( *tableSchema.fields() ); (f = it.current()); ++it) { + QDomElement extendedTableSchemaFieldEl; + if (f->visibleDecimalPlaces()>=0/*nondefault*/ && KexiDB::supportsVisibleDecimalPlacesProperty(f->type())) { + addFieldPropertyToExtendedTableSchemaData( + f, "visibleDecimalPlaces", f->visibleDecimalPlaces(), doc, + extendedTableSchemaMainEl, extendedTableSchemaFieldEl, + extendedTableSchemaStringIsEmpty ); + } + // boolean field with "not null" + + // add custom properties + const Field::CustomPropertiesMap customProperties(f->customProperties()); + foreach( Field::CustomPropertiesMap::ConstIterator, itCustom, customProperties ) { + addFieldPropertyToExtendedTableSchemaData( + f, itCustom.key(), itCustom.data(), doc, + extendedTableSchemaMainEl, extendedTableSchemaFieldEl, extendedTableSchemaStringIsEmpty, + /*custom*/true ); + } + // save lookup table specification, if present + LookupFieldSchema *lookupFieldSchema = tableSchema.lookupFieldSchema( *f ); + if (lookupFieldSchema) { + createExtendedTableSchemaFieldElementIfNeeded( + doc, extendedTableSchemaMainEl, f->name(), extendedTableSchemaFieldEl, false/* !append */); + LookupFieldSchema::saveToDom(*lookupFieldSchema, doc, extendedTableSchemaFieldEl); + + if (extendedTableSchemaFieldEl.hasChildNodes()) { + // this element provides the definition, so let's append it now + createExtendedTableSchemaMainElementIfNeeded(doc, extendedTableSchemaMainEl, + extendedTableSchemaStringIsEmpty); + extendedTableSchemaMainEl.appendChild( extendedTableSchemaFieldEl ); + } + } + } + + // Store extended schema information (see ExtendedTableSchemaInformation in Kexi Wiki) + if (extendedTableSchemaStringIsEmpty) { +#ifdef KEXI_DEBUG_GUI + KexiUtils::addAlterTableActionDebug(QString("** Extended table schema REMOVED.")); +#endif + if (!removeDataBlock( tableSchema.id(), "extended_schema")) + return false; + } + else { +#ifdef KEXI_DEBUG_GUI + KexiUtils::addAlterTableActionDebug(QString("** Extended table schema set to:\n")+doc.toString(4)); +#endif + if (!storeDataBlock( tableSchema.id(), doc.toString(1), "extended_schema" )) + return false; + } + return true; +} + +bool Connection::loadExtendedTableSchemaData(TableSchema& tableSchema) +{ +#define loadExtendedTableSchemaData_ERR \ + { setError(i18n("Error while loading extended table schema information.")); \ + return false; } +#define loadExtendedTableSchemaData_ERR2(details) \ + { setError(i18n("Error while loading extended table schema information."), details); \ + return false; } +#define loadExtendedTableSchemaData_ERR3(data) \ + { setError(i18n("Error while loading extended table schema information."), \ + i18n("Invalid XML data: ") + data.left(1024) ); \ + return false; } + + // Load extended schema information, if present (see ExtendedTableSchemaInformation in Kexi Wiki) + QString extendedTableSchemaString; + tristate res = loadDataBlock( tableSchema.id(), extendedTableSchemaString, "extended_schema" ); + if (!res) + loadExtendedTableSchemaData_ERR; + // extendedTableSchemaString will be just empty if there is no such data block + +#ifdef KEXIDB_LOOKUP_FIELD_TEST +//<temp. for LookupFieldSchema tests> + if (tableSchema.name()=="cars") { + LookupFieldSchema *lookupFieldSchema = new LookupFieldSchema(); + lookupFieldSchema->rowSource().setType(LookupFieldSchema::RowSource::Table); + lookupFieldSchema->rowSource().setName("persons"); + lookupFieldSchema->setBoundColumn(0); //id + lookupFieldSchema->setVisibleColumn(3); //surname + tableSchema.setLookupFieldSchema( "owner", lookupFieldSchema ); + } +//</temp. for LookupFieldSchema tests> +#endif + + if (extendedTableSchemaString.isEmpty()) + return true; + + QDomDocument doc; + QString errorMsg; + int errorLine, errorColumn; + if (!doc.setContent( extendedTableSchemaString, &errorMsg, &errorLine, &errorColumn )) + loadExtendedTableSchemaData_ERR2( i18n("Error in XML data: \"%1\" in line %2, column %3.\nXML data: ") + .arg(errorMsg).arg(errorLine).arg(errorColumn) + extendedTableSchemaString.left(1024)); + +//! @todo look at the current format version (KEXIDB_EXTENDED_TABLE_SCHEMA_VERSION) + + if (doc.doctype().name()!="EXTENDED_TABLE_SCHEMA") + loadExtendedTableSchemaData_ERR3( extendedTableSchemaString ); + + QDomElement docEl = doc.documentElement(); + if (docEl.tagName()!="EXTENDED_TABLE_SCHEMA") + loadExtendedTableSchemaData_ERR3( extendedTableSchemaString ); + + for (QDomNode n = docEl.firstChild(); !n.isNull(); n = n.nextSibling()) { + QDomElement fieldEl = n.toElement(); + if (fieldEl.tagName()=="field") { + Field *f = tableSchema.field( fieldEl.attribute("name") ); + if (f) { + //set properties of the field: +//! @todo more properties + for (QDomNode propNode = fieldEl.firstChild(); + !propNode.isNull(); propNode = propNode.nextSibling()) + { + QDomElement propEl = propNode.toElement(); + bool ok; + int intValue; + if (propEl.tagName()=="property") { + QCString propertyName = propEl.attribute("name").latin1(); + if (propEl.attribute("custom")=="true") { + //custom property + f->setCustomProperty(propertyName, + KexiDB::loadPropertyValueFromDom( propEl.firstChild() )); + } + else if (propertyName == "visibleDecimalPlaces" + && KexiDB::supportsVisibleDecimalPlacesProperty(f->type())) + { + intValue = KexiDB::loadIntPropertyValueFromDom( propEl.firstChild(), &ok ); + if (ok) + f->setVisibleDecimalPlaces(intValue); + } +//! @todo more properties... + } + else if (propEl.tagName()=="lookup-column") { + LookupFieldSchema *lookupFieldSchema = LookupFieldSchema::loadFromDom(propEl); + if (lookupFieldSchema) + lookupFieldSchema->debug(); + tableSchema.setLookupFieldSchema( f->name(), lookupFieldSchema ); + } + } + } + else { + KexiDBWarn << "Connection::loadExtendedTableSchemaData(): no such field \"" + << fieldEl.attribute("name") << "\" in table \"" << tableSchema.name() << "\"" << endl; + } + } + } + + return true; +} + +KexiDB::Field* Connection::setupField( const RowData &data ) +{ + bool ok = true; + int f_int_type = data.at(1).toInt(&ok); + if (f_int_type<=Field::InvalidType || f_int_type>Field::LastType) + ok = false; + if (!ok) + return 0; + Field::Type f_type = (Field::Type)f_int_type; + int f_len = QMAX( 0, data.at(3).toInt(&ok) ); + if (!ok) + return 0; + int f_prec = data.at(4).toInt(&ok); + if (!ok) + return 0; + int f_constr = data.at(5).toInt(&ok); + if (!ok) + return 0; + int f_opts = data.at(6).toInt(&ok); + if (!ok) + return 0; + + if (!KexiUtils::isIdentifier( data.at(2).toString() )) { + setError(ERR_INVALID_IDENTIFIER, i18n("Invalid object name \"%1\"") + .arg( data.at(2).toString() )); + ok = false; + return 0; + } + + Field *f = new Field( + data.at(2).toString(), f_type, f_constr, f_opts, f_len, f_prec ); + + f->setDefaultValue( KexiDB::stringToVariant(data.at(7).toString(), Field::variantType( f_type ), ok) ); + if (!ok) { + KexiDBWarn << "Connection::setupTableSchema() problem with KexiDB::stringToVariant(" + << data.at(7).toString() << ")" << endl; + } + ok = true; //problem with defaultValue is not critical + + f->m_caption = data.at(9).toString(); + f->m_desc = data.at(10).toString(); + return f; +} + +KexiDB::TableSchema* Connection::setupTableSchema( const RowData &data ) +{ + TableSchema *t = new TableSchema( this ); + if (!setupObjectSchemaData( data, *t )) { + delete t; + return 0; + } + + KexiDB::Cursor *cursor; + if (!(cursor = executeQuery( + QString::fromLatin1("SELECT t_id, f_type, f_name, f_length, f_precision, f_constraints, " + "f_options, f_default, f_order, f_caption, f_help" + " FROM kexi__fields WHERE t_id=%1 ORDER BY f_order").arg(t->m_id) ))) + { + delete t; + return 0; + } + if (!cursor->moveFirst()) { + if (!cursor->error() && cursor->eof()) { + setError(i18n("Table has no fields defined.")); + } + deleteCursor(cursor); + delete t; + return 0; + } + + // For each field: load its schema + RowData fieldData; + bool ok = true; + while (!cursor->eof()) { +// KexiDBDbg<<"@@@ f_name=="<<cursor->value(2).asCString()<<endl; + cursor->storeCurrentRow(fieldData); + Field *f = setupField(fieldData); + if (!f) { + ok = false; + break; + } + t->addField(f); + cursor->moveNext(); + } + + if (!ok) {//error: + deleteCursor(cursor); + delete t; + return 0; + } + + if (!deleteCursor(cursor)) { + delete t; + return 0; + } + + if (!loadExtendedTableSchemaData(*t)) { + delete t; + return 0; + } + //store locally: + d->tables.insert(t->m_id, t); + d->tables_byname.insert(t->m_name.lower(), t); + return t; +} + +TableSchema* Connection::tableSchema( const QString& tableName ) +{ + QString m_tableName = tableName.lower(); + TableSchema *t = d->tables_byname[m_tableName]; + if (t) + return t; + //not found: retrieve schema + RowData data; + if (true!=querySingleRecord(QString::fromLatin1( + "SELECT o_id, o_type, o_name, o_caption, o_desc FROM kexi__objects WHERE lower(o_name)='%1' AND o_type=%2") + .arg(m_tableName).arg(KexiDB::TableObjectType), data)) + return 0; + + return setupTableSchema(data); +} + +TableSchema* Connection::tableSchema( int tableId ) +{ + TableSchema *t = d->tables[tableId]; + if (t) + return t; + //not found: retrieve schema + RowData data; + if (true!=querySingleRecord(QString::fromLatin1( + "SELECT o_id, o_type, o_name, o_caption, o_desc FROM kexi__objects WHERE o_id=%1") + .arg(tableId), data)) + return 0; + + return setupTableSchema(data); +} + +tristate Connection::loadDataBlock( int objectID, QString &dataString, const QString& dataID ) +{ + if (objectID<=0) + return false; + return querySingleString( + QString("SELECT o_data FROM kexi__objectdata WHERE o_id=") + QString::number(objectID) + + " AND " + KexiDB::sqlWhere(m_driver, KexiDB::Field::Text, "o_sub_id", dataID), + dataString ); +} + +bool Connection::storeDataBlock( int objectID, const QString &dataString, const QString& dataID ) +{ + if (objectID<=0) + return false; + QString sql(QString::fromLatin1("SELECT kexi__objectdata.o_id FROM kexi__objectdata WHERE o_id=%1").arg(objectID)); + QString sql_sub( KexiDB::sqlWhere(m_driver, KexiDB::Field::Text, "o_sub_id", dataID) ); + + bool ok, exists; + exists = resultExists(sql + " and " + sql_sub, ok); + if (!ok) + return false; + if (exists) { + return executeSQL( "UPDATE kexi__objectdata SET o_data=" + + m_driver->valueToSQL( KexiDB::Field::LongText, dataString ) + + " WHERE o_id=" + QString::number(objectID) + " AND " + sql_sub ); + } + return executeSQL( + QString::fromLatin1("INSERT INTO kexi__objectdata (o_id, o_data, o_sub_id) VALUES (") + + QString::number(objectID) +"," + m_driver->valueToSQL( KexiDB::Field::LongText, dataString ) + + "," + m_driver->valueToSQL( KexiDB::Field::Text, dataID ) + ")" ); +} + +bool Connection::removeDataBlock( int objectID, const QString& dataID) +{ + if (objectID<=0) + return false; + if (dataID.isEmpty()) + return KexiDB::deleteRow(*this, "kexi__objectdata", "o_id", QString::number(objectID)); + else + return KexiDB::deleteRow(*this, "kexi__objectdata", + "o_id", KexiDB::Field::Integer, objectID, "o_sub_id", KexiDB::Field::Text, dataID); +} + +KexiDB::QuerySchema* Connection::setupQuerySchema( const RowData &data ) +{ + bool ok = true; + const int objID = data[0].toInt(&ok); + if (!ok) + return false; + QString sqlText; + if (!loadDataBlock( objID, sqlText, "sql" )) { + setError(ERR_OBJECT_NOT_FOUND, + i18n("Could not find definition for query \"%1\". Removing this query is recommended.") + .arg(data[2].toString())); + return 0; + } + d->parser()->parse( sqlText ); + KexiDB::QuerySchema *query = d->parser()->query(); + //error? + if (!query) { + setError(ERR_SQL_PARSE_ERROR, + i18n("<p>Could not load definition for query \"%1\". " + "SQL statement for this query is invalid:<br><tt>%2</tt></p>\n" + "<p>You can open this query in Text View and correct it.</p>").arg(data[2].toString()) + .arg(d->parser()->statement())); + return 0; + } + if (!setupObjectSchemaData( data, *query )) { + delete query; + return 0; + } + d->queries.insert(query->m_id, query); + d->queries_byname.insert(query->m_name, query); + return query; +} + +QuerySchema* Connection::querySchema( const QString& queryName ) +{ + QString m_queryName = queryName.lower(); + QuerySchema *q = d->queries_byname[m_queryName]; + if (q) + return q; + //not found: retrieve schema + RowData data; + if (true!=querySingleRecord(QString::fromLatin1( + "SELECT o_id, o_type, o_name, o_caption, o_desc FROM kexi__objects WHERE lower(o_name)='%1' AND o_type=%2") + .arg(m_queryName).arg(KexiDB::QueryObjectType), data)) + return 0; + + return setupQuerySchema(data); +} + +QuerySchema* Connection::querySchema( int queryId ) +{ + QuerySchema *q = d->queries[queryId]; + if (q) + return q; + //not found: retrieve schema + clearError(); + RowData data; + if (true!=querySingleRecord(QString::fromLatin1( + "SELECT o_id, o_type, o_name, o_caption, o_desc FROM kexi__objects WHERE o_id=%1").arg(queryId), data)) + return 0; + + return setupQuerySchema(data); +} + +bool Connection::setQuerySchemaObsolete( const QString& queryName ) +{ + QuerySchema* oldQuery = querySchema( queryName ); + if (!oldQuery) + return false; + d->obsoleteQueries.append(oldQuery); + d->queries_byname.take(queryName); + d->queries.take(oldQuery->id()); + return true; +} + +TableSchema* Connection::newKexiDBSystemTableSchema(const QString& tsname) +{ + TableSchema *ts = new TableSchema(tsname.lower()); + insertInternalTableSchema( ts ); + return ts; +} + +bool Connection::isInternalTableSchema(const QString& tableName) +{ + return (d->kexiDBSystemTables[ d->tables_byname[tableName] ]) + // these are here for compatiblility because we're no longer instantiate + // them but can exist in projects created with previous Kexi versions: + || tableName=="kexi__final" || tableName=="kexi__useractions"; +} + +void Connection::insertInternalTableSchema(TableSchema *tableSchema) +{ + tableSchema->setKexiDBSystem(true); + d->kexiDBSystemTables.insert(tableSchema, tableSchema); + d->tables_byname.insert(tableSchema->name(), tableSchema); +} + +//! Creates kexi__* tables. +bool Connection::setupKexiDBSystemSchema() +{ + if (!d->kexiDBSystemTables.isEmpty()) + return true; //already set up + + TableSchema *t_objects = newKexiDBSystemTableSchema("kexi__objects"); + t_objects->addField( new Field("o_id", Field::Integer, Field::PrimaryKey | Field::AutoInc, Field::Unsigned) ) + .addField( new Field("o_type", Field::Byte, 0, Field::Unsigned) ) + .addField( new Field("o_name", Field::Text) ) + .addField( new Field("o_caption", Field::Text ) ) + .addField( new Field("o_desc", Field::LongText ) ); + + t_objects->debug(); + + TableSchema *t_objectdata = newKexiDBSystemTableSchema("kexi__objectdata"); + t_objectdata->addField( new Field("o_id", Field::Integer, Field::NotNull, Field::Unsigned) ) + .addField( new Field("o_data", Field::LongText) ) + .addField( new Field("o_sub_id", Field::Text) ); + + TableSchema *t_fields = newKexiDBSystemTableSchema("kexi__fields"); + t_fields->addField( new Field("t_id", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("f_type", Field::Byte, 0, Field::Unsigned) ) + .addField( new Field("f_name", Field::Text ) ) + .addField( new Field("f_length", Field::Integer ) ) + .addField( new Field("f_precision", Field::Integer ) ) + .addField( new Field("f_constraints", Field::Integer ) ) + .addField( new Field("f_options", Field::Integer ) ) + .addField( new Field("f_default", Field::Text ) ) + //these are additional properties: + .addField( new Field("f_order", Field::Integer ) ) + .addField( new Field("f_caption", Field::Text ) ) + .addField( new Field("f_help", Field::LongText ) ); + +/* TableSchema *t_querydata = newKexiDBSystemTableSchema("kexi__querydata"); + t_querydata->addField( new Field("q_id", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("q_sql", Field::LongText ) ) + .addField( new Field("q_valid", Field::Boolean ) ); + + TableSchema *t_queryfields = newKexiDBSystemTableSchema("kexi__queryfields"); + t_queryfields->addField( new Field("q_id", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("f_order", Field::Integer ) ) + .addField( new Field("f_id", Field::Integer ) ) + .addField( new Field("f_tab_asterisk", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("f_alltab_asterisk", Field::Boolean) ); + + TableSchema *t_querytables = newKexiDBSystemTableSchema("kexi__querytables"); + t_querytables->addField( new Field("q_id", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("t_id", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("t_order", Field::Integer, 0, Field::Unsigned) );*/ + + TableSchema *t_db = newKexiDBSystemTableSchema("kexi__db"); + t_db->addField( new Field("db_property", Field::Text, Field::NoConstraints, Field::NoOptions, 32 ) ) + .addField( new Field("db_value", Field::LongText ) ); + +/* moved to KexiProject... + TableSchema *t_parts = newKexiDBSystemTableSchema("kexi__parts"); + t_parts->addField( new Field("p_id", Field::Integer, Field::PrimaryKey | Field::AutoInc, Field::Unsigned) ) + .addField( new Field("p_name", Field::Text) ) + .addField( new Field("p_mime", Field::Text ) ) + .addField( new Field("p_url", Field::Text ) ); +*/ + +/*UNUSED + TableSchema *t_final = newKexiDBSystemTableSchema("kexi__final"); + t_final->addField( new Field("p_id", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("property", Field::LongText ) ) + .addField( new Field("value", Field::BLOB) ); + + TableSchema *t_useractions = newKexiDBSystemTableSchema("kexi__useractions"); + t_useractions->addField( new Field("p_id", Field::Integer, 0, Field::Unsigned) ) + .addField( new Field("scope", Field::Integer ) ) + .addField( new Field("name", Field::LongText ) ) + .addField( new Field("text", Field::LongText ) ) + .addField( new Field("icon", Field::LongText ) ) + .addField( new Field("method", Field::Integer ) ) + .addField( new Field("arguments", Field::LongText) ); +*/ + return true; +} + +void Connection::removeMe(TableSchema *ts) +{ + if (ts && !m_destructor_started) { + d->tables.take(ts->id()); + d->tables_byname.take(ts->name()); + } +} + +QString Connection::anyAvailableDatabaseName() +{ + if (!d->availableDatabaseName.isEmpty()) { + return d->availableDatabaseName; + } + return m_driver->beh->ALWAYS_AVAILABLE_DATABASE_NAME; +} + +void Connection::setAvailableDatabaseName(const QString& dbName) +{ + d->availableDatabaseName = dbName; +} + +//! @internal used in updateRow(), insertRow(), +inline void updateRowDataWithNewValues(QuerySchema &query, RowData& data, KexiDB::RowEditBuffer::DBMap& b, + QMap<QueryColumnInfo*,int>& columnsOrderExpanded) +{ + columnsOrderExpanded = query.columnsOrder(QuerySchema::ExpandedList); + QMap<QueryColumnInfo*,int>::ConstIterator columnsOrderExpandedIt; + for (KexiDB::RowEditBuffer::DBMap::ConstIterator it=b.constBegin();it!=b.constEnd();++it) { + columnsOrderExpandedIt = columnsOrderExpanded.find( it.key() ); + if (columnsOrderExpandedIt == columnsOrderExpanded.constEnd()) { + KexiDBWarn << "(Connection) updateRowDataWithNewValues(): \"now also assign new value in memory\" step " + "- could not find item '" << it.key()->aliasOrName() << "'" << endl; + continue; + } + data[ columnsOrderExpandedIt.data() ] = it.data(); + } +} + +bool Connection::updateRow(QuerySchema &query, RowData& data, RowEditBuffer& buf, bool useROWID) +{ +// Each SQL identifier needs to be escaped in the generated query. +// query.debug(); + + KexiDBDbg << "Connection::updateRow.." << endl; + clearError(); + //--get PKEY + if (buf.dbBuffer().isEmpty()) { + KexiDBDbg << " -- NO CHANGES DATA!" << endl; + return true; + } + TableSchema *mt = query.masterTable(); + if (!mt) { + KexiDBWarn << " -- NO MASTER TABLE!" << endl; + setError(ERR_UPDATE_NO_MASTER_TABLE, + i18n("Could not update row because there is no master table defined.")); + return false; + } + IndexSchema *pkey = (mt->primaryKey() && !mt->primaryKey()->fields()->isEmpty()) ? mt->primaryKey() : 0; + if (!useROWID && !pkey) { + KexiDBWarn << " -- NO MASTER TABLE's PKEY!" << endl; + setError(ERR_UPDATE_NO_MASTER_TABLES_PKEY, + i18n("Could not update row because master table has no primary key defined.")); +//! @todo perhaps we can try to update without using PKEY? + return false; + } + //update the record: + m_sql = "UPDATE " + escapeIdentifier(mt->name()) + " SET "; + QString sqlset, sqlwhere; + sqlset.reserve(1024); + sqlwhere.reserve(1024); + KexiDB::RowEditBuffer::DBMap b = buf.dbBuffer(); + for (KexiDB::RowEditBuffer::DBMap::ConstIterator it=b.constBegin();it!=b.constEnd();++it) { + if (it.key()->field->table()!=mt) + continue; // skip values for fields outside of the master table (e.g. a "visible value" of the lookup field) + if (!sqlset.isEmpty()) + sqlset+=","; + sqlset += (escapeIdentifier(it.key()->field->name()) + "=" + + m_driver->valueToSQL(it.key()->field,it.data())); + } + if (pkey) { + const QValueVector<int> pkeyFieldsOrder( query.pkeyFieldsOrder() ); + KexiDBDbg << pkey->fieldCount() << " ? " << query.pkeyFieldsCount() << endl; + if (pkey->fieldCount() != query.pkeyFieldsCount()) { //sanity check + KexiDBWarn << " -- NO ENTIRE MASTER TABLE's PKEY SPECIFIED!" << endl; + setError(ERR_UPDATE_NO_ENTIRE_MASTER_TABLES_PKEY, + i18n("Could not update row because it does not contain entire master table's primary key.")); + return false; + } + if (!pkey->fields()->isEmpty()) { + uint i=0; + for (Field::ListIterator it = pkey->fieldsIterator(); it.current(); i++, ++it) { + if (!sqlwhere.isEmpty()) + sqlwhere+=" AND "; + QVariant val = data[ pkeyFieldsOrder[i] ]; + if (val.isNull() || !val.isValid()) { + setError(ERR_UPDATE_NULL_PKEY_FIELD, + i18n("Primary key's field \"%1\" cannot be empty.").arg(it.current()->name())); + //js todo: pass the field's name somewhere! + return false; + } + sqlwhere += ( escapeIdentifier(it.current()->name()) + "=" + + m_driver->valueToSQL( it.current(), val ) ); + } + } + } + else {//use ROWID + sqlwhere = ( escapeIdentifier(m_driver->beh->ROW_ID_FIELD_NAME) + "=" + + m_driver->valueToSQL(Field::BigInteger, data[data.size()-1])); + } + m_sql += (sqlset + " WHERE " + sqlwhere); + KexiDBDbg << " -- SQL == " << ((m_sql.length() > 400) ? (m_sql.left(400)+"[.....]") : m_sql) << endl; + + if (!executeSQL(m_sql)) { + setError(ERR_UPDATE_SERVER_ERROR, i18n("Row updating on the server failed.")); + return false; + } + //success: now also assign new values in memory: + QMap<QueryColumnInfo*,int> columnsOrderExpanded; + updateRowDataWithNewValues(query, data, b, columnsOrderExpanded); + return true; +} + +bool Connection::insertRow(QuerySchema &query, RowData& data, RowEditBuffer& buf, bool getROWID) +{ +// Each SQL identifier needs to be escaped in the generated query. + KexiDBDbg << "Connection::updateRow.." << endl; + clearError(); + //--get PKEY + /*disabled: there may be empty rows (with autoinc) + if (buf.dbBuffer().isEmpty()) { + KexiDBDbg << " -- NO CHANGES DATA!" << endl; + return true; }*/ + TableSchema *mt = query.masterTable(); + if (!mt) { + KexiDBWarn << " -- NO MASTER TABLE!" << endl; + setError(ERR_INSERT_NO_MASTER_TABLE, + i18n("Could not insert row because there is no master table defined.")); + return false; + } + IndexSchema *pkey = (mt->primaryKey() && !mt->primaryKey()->fields()->isEmpty()) ? mt->primaryKey() : 0; + if (!getROWID && !pkey) + KexiDBWarn << " -- WARNING: NO MASTER TABLE's PKEY" << endl; + + QString sqlcols, sqlvals; + sqlcols.reserve(1024); + sqlvals.reserve(1024); + + //insert the record: + m_sql = "INSERT INTO " + escapeIdentifier(mt->name()) + " ("; + KexiDB::RowEditBuffer::DBMap b = buf.dbBuffer(); + + // add default values, if available (for any column without value explicitly set) + const QueryColumnInfo::Vector fieldsExpanded( query.fieldsExpanded( QuerySchema::Unique ) ); + for (uint i=0; i<fieldsExpanded.count(); i++) { + QueryColumnInfo *ci = fieldsExpanded.at(i); + if (ci->field && KexiDB::isDefaultValueAllowed(ci->field) + && !ci->field->defaultValue().isNull() + && !b.contains( ci )) + { + KexiDBDbg << "Connection::insertRow(): adding default value '" << ci->field->defaultValue().toString() + << "' for column '" << ci->field->name() << "'" << endl; + b.insert( ci, ci->field->defaultValue() ); + } + } + + if (b.isEmpty()) { + // empty row inserting requested: + if (!getROWID && !pkey) { + KexiDBWarn << "MASTER TABLE's PKEY REQUIRED FOR INSERTING EMPTY ROWS: INSERT CANCELLED" << endl; + setError(ERR_INSERT_NO_MASTER_TABLES_PKEY, + i18n("Could not insert row because master table has no primary key defined.")); + return false; + } + if (pkey) { + const QValueVector<int> pkeyFieldsOrder( query.pkeyFieldsOrder() ); +// KexiDBDbg << pkey->fieldCount() << " ? " << query.pkeyFieldsCount() << endl; + if (pkey->fieldCount() != query.pkeyFieldsCount()) { //sanity check + KexiDBWarn << "NO ENTIRE MASTER TABLE's PKEY SPECIFIED!" << endl; + setError(ERR_INSERT_NO_ENTIRE_MASTER_TABLES_PKEY, + i18n("Could not insert row because it does not contain entire master table's primary key.") + .arg(query.name())); + return false; + } + } + //at least one value is needed for VALUES section: find it and set to NULL: + Field *anyField = mt->anyNonPKField(); + if (!anyField) { + if (!pkey) { + KexiDBWarn << "WARNING: NO FIELD AVAILABLE TO SET IT TO NULL" << endl; + return false; + } + else { + //try to set NULL in pkey field (could not work for every SQL engine!) + anyField = pkey->fields()->first(); + } + } + sqlcols += escapeIdentifier(anyField->name()); + sqlvals += m_driver->valueToSQL(anyField,QVariant()/*NULL*/); + } + else { + // non-empty row inserting requested: + for (KexiDB::RowEditBuffer::DBMap::ConstIterator it=b.constBegin();it!=b.constEnd();++it) { + if (it.key()->field->table()!=mt) + continue; // skip values for fields outside of the master table (e.g. a "visible value" of the lookup field) + if (!sqlcols.isEmpty()) { + sqlcols+=","; + sqlvals+=","; + } + sqlcols += escapeIdentifier(it.key()->field->name()); + sqlvals += m_driver->valueToSQL(it.key()->field,it.data()); + } + } + m_sql += (sqlcols + ") VALUES (" + sqlvals + ")"); +// KexiDBDbg << " -- SQL == " << m_sql << endl; + + bool res = executeSQL(m_sql); + + if (!res) { + setError(ERR_INSERT_SERVER_ERROR, i18n("Row inserting on the server failed.")); + return false; + } + //success: now also assign a new value in memory: + QMap<QueryColumnInfo*,int> columnsOrderExpanded; + updateRowDataWithNewValues(query, data, b, columnsOrderExpanded); + + //fetch autoincremented values + QueryColumnInfo::List *aif_list = query.autoIncrementFields(); + Q_ULLONG ROWID = 0; + if (pkey && !aif_list->isEmpty()) { + //! @todo now only if PKEY is present, this should also work when there's no PKEY + QueryColumnInfo *id_columnInfo = aif_list->first(); +//! @todo safe to cast it? + Q_ULLONG last_id = lastInsertedAutoIncValue( + id_columnInfo->field->name(), id_columnInfo->field->table()->name(), &ROWID); + if (last_id==(Q_ULLONG)-1 || last_id<=0) { + //! @todo show error +//! @todo remove just inserted row. How? Using ROLLBACK? + return false; + } + RowData aif_data; + QString getAutoIncForInsertedValue = QString::fromLatin1("SELECT ") + + query.autoIncrementSQLFieldsList(m_driver) + + QString::fromLatin1(" FROM ") + + escapeIdentifier(id_columnInfo->field->table()->name()) + + QString::fromLatin1(" WHERE ") + + escapeIdentifier(id_columnInfo->field->name()) + "=" + + QString::number(last_id); + if (true!=querySingleRecord(getAutoIncForInsertedValue, aif_data)) { + //! @todo show error + return false; + } + QueryColumnInfo::ListIterator ci_it(*aif_list); + QueryColumnInfo *ci; + for (uint i=0; (ci = ci_it.current()); ++ci_it, i++) { +// KexiDBDbg << "Connection::insertRow(): AUTOINCREMENTED FIELD " << fi->field->name() << " == " +// << aif_data[i].toInt() << endl; + ( data[ columnsOrderExpanded[ ci ] ] = aif_data[i] ).cast( ci->field->variantType() ); //cast to get proper type + } + } + else { + ROWID = drv_lastInsertRowID(); +// KexiDBDbg << "Connection::insertRow(): new ROWID == " << (uint)ROWID << endl; + if (m_driver->beh->ROW_ID_FIELD_RETURNS_LAST_AUTOINCREMENTED_VALUE) { + KexiDBWarn << "Connection::insertRow(): m_driver->beh->ROW_ID_FIELD_RETURNS_LAST_AUTOINCREMENTED_VALUE" << endl; + return false; + } + } + if (getROWID && /*sanity check*/data.size() > fieldsExpanded.size()) { +// KexiDBDbg << "Connection::insertRow(): new ROWID == " << (uint)ROWID << endl; + data[data.size()-1] = ROWID; + } + return true; +} + +bool Connection::deleteRow(QuerySchema &query, RowData& data, bool useROWID) +{ +// Each SQL identifier needs to be escaped in the generated query. + KexiDBWarn << "Connection::deleteRow.." << endl; + clearError(); + TableSchema *mt = query.masterTable(); + if (!mt) { + KexiDBWarn << " -- NO MASTER TABLE!" << endl; + setError(ERR_DELETE_NO_MASTER_TABLE, + i18n("Could not delete row because there is no master table defined.") + .arg(query.name())); + return false; + } + IndexSchema *pkey = (mt->primaryKey() && !mt->primaryKey()->fields()->isEmpty()) ? mt->primaryKey() : 0; + +//! @todo allow to delete from a table without pkey + if (!useROWID && !pkey) { + KexiDBWarn << " -- WARNING: NO MASTER TABLE's PKEY" << endl; + setError(ERR_DELETE_NO_MASTER_TABLES_PKEY, + i18n("Could not delete row because there is no primary key for master table defined.")); + return false; + } + + //update the record: + m_sql = "DELETE FROM " + escapeIdentifier(mt->name()) + " WHERE "; + QString sqlwhere; + sqlwhere.reserve(1024); + + if (pkey) { + const QValueVector<int> pkeyFieldsOrder( query.pkeyFieldsOrder() ); + KexiDBDbg << pkey->fieldCount() << " ? " << query.pkeyFieldsCount() << endl; + if (pkey->fieldCount() != query.pkeyFieldsCount()) { //sanity check + KexiDBWarn << " -- NO ENTIRE MASTER TABLE's PKEY SPECIFIED!" << endl; + setError(ERR_DELETE_NO_ENTIRE_MASTER_TABLES_PKEY, + i18n("Could not delete row because it does not contain entire master table's primary key.")); + return false; + } + uint i=0; + for (Field::ListIterator it = pkey->fieldsIterator(); it.current(); i++, ++it) { + if (!sqlwhere.isEmpty()) + sqlwhere+=" AND "; + QVariant val = data[ pkeyFieldsOrder[i] ]; + if (val.isNull() || !val.isValid()) { + setError(ERR_DELETE_NULL_PKEY_FIELD, i18n("Primary key's field \"%1\" cannot be empty.") + .arg(it.current()->name())); +//js todo: pass the field's name somewhere! + return false; + } + sqlwhere += ( escapeIdentifier(it.current()->name()) + "=" + + m_driver->valueToSQL( it.current(), val ) ); + } + } + else {//use ROWID + sqlwhere = ( escapeIdentifier(m_driver->beh->ROW_ID_FIELD_NAME) + "=" + + m_driver->valueToSQL(Field::BigInteger, data[data.size()-1])); + } + m_sql += sqlwhere; + KexiDBDbg << " -- SQL == " << m_sql << endl; + + if (!executeSQL(m_sql)) { + setError(ERR_DELETE_SERVER_ERROR, i18n("Row deletion on the server failed.")); + return false; + } + return true; +} + +bool Connection::deleteAllRows(QuerySchema &query) +{ + clearError(); + TableSchema *mt = query.masterTable(); + if (!mt) { + KexiDBWarn << " -- NO MASTER TABLE!" << endl; + return false; + } + IndexSchema *pkey = mt->primaryKey(); + if (!pkey || pkey->fields()->isEmpty()) + KexiDBWarn << "Connection::deleteAllRows -- WARNING: NO MASTER TABLE's PKEY" << endl; + + m_sql = "DELETE FROM " + escapeIdentifier(mt->name()); + + KexiDBDbg << " -- SQL == " << m_sql << endl; + + if (!executeSQL(m_sql)) { + setError(ERR_DELETE_SERVER_ERROR, i18n("Row deletion on the server failed.")); + return false; + } + return true; +} + +void Connection::registerForTableSchemaChanges(TableSchemaChangeListenerInterface& listener, + TableSchema &schema) +{ + QPtrList<TableSchemaChangeListenerInterface>* listeners = d->tableSchemaChangeListeners[&schema]; + if (!listeners) { + listeners = new QPtrList<TableSchemaChangeListenerInterface>(); + d->tableSchemaChangeListeners.insert(&schema, listeners); + } +//TODO: inefficient + if (listeners->findRef( &listener )==-1) + listeners->append( &listener ); +} + +void Connection::unregisterForTableSchemaChanges(TableSchemaChangeListenerInterface& listener, + TableSchema &schema) +{ + QPtrList<TableSchemaChangeListenerInterface>* listeners = d->tableSchemaChangeListeners[&schema]; + if (!listeners) + return; +//TODO: inefficient + listeners->remove( &listener ); +} + +void Connection::unregisterForTablesSchemaChanges(TableSchemaChangeListenerInterface& listener) +{ + for (QPtrDictIterator< QPtrList<TableSchemaChangeListenerInterface> > it(d->tableSchemaChangeListeners); + it.current(); ++it) + { + if (-1!=it.current()->find(&listener)) + it.current()->take(); + } +} + +QPtrList<Connection::TableSchemaChangeListenerInterface>* +Connection::tableSchemaChangeListeners(TableSchema& tableSchema) const +{ + KexiDBDbg << d->tableSchemaChangeListeners.count() << endl; + return d->tableSchemaChangeListeners[&tableSchema]; +} + +tristate Connection::closeAllTableSchemaChangeListeners(TableSchema& tableSchema) +{ + QPtrList<Connection::TableSchemaChangeListenerInterface> *listeners = d->tableSchemaChangeListeners[&tableSchema]; + if (!listeners) + return true; + QPtrListIterator<KexiDB::Connection::TableSchemaChangeListenerInterface> tmpListeners(*listeners); //safer copy + tristate res = true; + //try to close every window + for (QPtrListIterator<KexiDB::Connection::TableSchemaChangeListenerInterface> it(tmpListeners); + it.current() && res==true; ++it) + { + res = it.current()->closeListener(); + } + return res; +} + +/*PreparedStatement::Ptr Connection::prepareStatement(PreparedStatement::StatementType, + TableSchema&) +{ + //safe? + return 0; +}*/ + +void Connection::setReadOnly(bool set) +{ + if (d->isConnected) + return; //sanity + d->readOnly = set; +} + +bool Connection::isReadOnly() const +{ + return d->readOnly; +} + +#include "connection.moc" |