summaryrefslogtreecommitdiffstats
path: root/tests/auto/qsqldatabase/tst_databases.h
diff options
context:
space:
mode:
Diffstat (limited to 'tests/auto/qsqldatabase/tst_databases.h')
-rw-r--r--tests/auto/qsqldatabase/tst_databases.h454
1 files changed, 454 insertions, 0 deletions
diff --git a/tests/auto/qsqldatabase/tst_databases.h b/tests/auto/qsqldatabase/tst_databases.h
new file mode 100644
index 0000000..b529b7e
--- /dev/null
+++ b/tests/auto/qsqldatabase/tst_databases.h
@@ -0,0 +1,454 @@
+/****************************************************************************
+**
+** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
+** Contact: Qt Software Information (qt-info@nokia.com)
+**
+** This file is part of the test suite of the Qt Toolkit.
+**
+** $QT_BEGIN_LICENSE:LGPL$
+** No Commercial Usage
+** This file contains pre-release code and may not be distributed.
+** You may use this file in accordance with the terms and conditions
+** contained in the either Technology Preview License Agreement or the
+** Beta Release License Agreement.
+**
+** GNU Lesser General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU Lesser
+** General Public License version 2.1 as published by the Free Software
+** Foundation and appearing in the file LICENSE.LGPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU Lesser General Public License version 2.1 requirements
+** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** In addition, as a special exception, Nokia gives you certain
+** additional rights. These rights are described in the Nokia Qt LGPL
+** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this
+** package.
+**
+** GNU General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU
+** General Public License version 3.0 as published by the Free Software
+** Foundation and appearing in the file LICENSE.GPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU General Public License version 3.0 requirements will be
+** met: http://www.gnu.org/copyleft/gpl.html.
+**
+** If you are unsure which license is appropriate for your use, please
+** contact the sales department at qt-sales@nokia.com.
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+/* possible connection parameters */
+
+#ifndef TST_DATABASES_H
+#define TST_DATABASES_H
+
+#include <QSqlDatabase>
+#include <QSqlDriver>
+#include <QSqlError>
+#include <QSqlQuery>
+#include <QRegExp>
+#include <QDir>
+#include <QVariant>
+#include <QDebug>
+
+#include <QtTest/QtTest>
+
+#if defined (Q_OS_WIN) || defined (Q_OS_WIN32)
+# include <qt_windows.h>
+# if defined (Q_OS_WINCE)
+# include <winsock2.h>
+# endif
+#else
+#include <unistd.h>
+#endif
+
+#define CHECK_DATABASE( db ) \
+ if ( !db.isValid() ) { qFatal( "db is Invalid" ); }
+
+#define QVERIFY_SQL(q, stmt) QVERIFY2((q).stmt, tst_Databases::printError((q).lastError(), db))
+#define QFAIL_SQL(q, stmt) QVERIFY2(!(q).stmt, tst_Databases::printError((q).lastError(), db))
+
+#define DBMS_SPECIFIC(db, driver) \
+ if (!db.driverName().startsWith(driver)) { QSKIP(driver " specific test", SkipSingle); return; }
+
+// ### use QSystem::hostName if it is integrated in qtest/main
+static QString qGetHostName()
+{
+ static QString hostname;
+
+ if ( !hostname.isEmpty() )
+ return hostname;
+
+ char hn[257];
+
+ if ( gethostname( hn, 255 ) == 0 ) {
+ hn[256] = '\0';
+ hostname = QString::fromLatin1( hn );
+ hostname.replace( QLatin1Char( '.' ), QLatin1Char( '_' ) );
+ hostname.replace( QLatin1Char( '-' ), QLatin1Char( '_' ) );
+ }
+
+ return hostname;
+}
+
+// to prevent nameclashes on our database server, each machine
+// will use its own set of table names. Call this function to get
+// "tablename_hostname"
+inline static QString qTableName( const QString& prefix, QSqlDriver* driver = 0 )
+{
+ if ( !driver )
+ return prefix + "_" + qGetHostName().replace( "-", "_" );
+ else
+ return driver->escapeIdentifier( prefix + "_" + qGetHostName(), QSqlDriver::TableName );
+}
+
+inline static bool testWhiteSpaceNames( const QString &name )
+{
+ return name.startsWith( "QPSQL" )
+ || name.startsWith( "QODBC" )
+ || name.startsWith( "QSQLITE" )
+ || name.startsWith( "QMYSQL" );
+}
+
+inline static QString toHex( const QString& binary )
+{
+ QString str;
+ static char const hexchars[] = "0123456789ABCDEF";
+
+ for ( int i = 0; i < binary.size(); i++ ) {
+ ushort code = binary.at(i).unicode();
+ str += (QChar)(hexchars[ (code >> 12) & 0x0F ]);
+ str += (QChar)(hexchars[ (code >> 8) & 0x0F ]);
+ str += (QChar)(hexchars[ (code >> 4) & 0x0F ]);
+ str += (QChar)(hexchars[ code & 0x0F ]);
+ }
+
+ return str;
+}
+
+
+class tst_Databases
+{
+
+public:
+ tst_Databases(): counter( 0 )
+ {
+ }
+
+ ~tst_Databases()
+ {
+ close();
+ }
+
+ // returns a testtable consisting of the names of all database connections if
+ // driverPrefix is empty, otherwise only those that start with driverPrefix.
+ int fillTestTable( const QString& driverPrefix = QString() ) const
+ {
+ QTest::addColumn<QString>( "dbName" );
+ int count = 0;
+
+ for ( int i = 0; i < dbNames.count(); ++i ) {
+ QSqlDatabase db = QSqlDatabase::database( dbNames.at( i ) );
+
+ if ( !db.isValid() )
+ continue;
+
+ if ( driverPrefix.isEmpty() || db.driverName().startsWith( driverPrefix ) ) {
+ QTest::newRow( dbNames.at( i ).toLatin1() ) << dbNames.at( i );
+ ++count;
+ }
+ }
+
+ return count;
+ }
+
+ void addDb( const QString& driver, const QString& dbName,
+ const QString& user = QString(), const QString& passwd = QString(),
+ const QString& host = QString(), int port = -1, const QString params = QString() )
+ {
+ QSqlDatabase db;
+
+ if ( !QSqlDatabase::drivers().contains( driver ) ) {
+ qWarning() << "Driver" << driver << "is not installed";
+ return;
+ }
+
+ // construct a stupid unique name
+ QString cName = QString::number( counter++ ) + "_" + driver + "@";
+
+ cName += host.isEmpty() ? dbName : host;
+
+ if ( port > 0 )
+ cName += ":" + QString::number( port );
+
+ db = QSqlDatabase::addDatabase( driver, cName );
+
+ if ( !db.isValid() ) {
+ qWarning( "Could not create database object" );
+ return;
+ }
+
+ db.setDatabaseName( dbName );
+
+ db.setUserName( user );
+ db.setPassword( passwd );
+ db.setHostName( host );
+ db.setPort( port );
+ db.setConnectOptions( params );
+ dbNames.append( cName );
+ }
+
+ void addDbs()
+ {
+// addDb( "QOCI8", "//horsehead.nokia.troll.no:1521/pony.troll.no", "scott", "tiger" ); // Oracle 9i on horsehead
+// addDb( "QOCI8", "//horsehead.nokia.troll.no:1521/ustest.troll.no", "scott", "tiger", "" ); // Oracle 9i on horsehead
+// addDb( "QOCI8", "ICE", "scott", "tiger", "" ); // Oracle 8 on iceblink
+// addDb( "QOCI", "//silence.nokia.troll.no:1521/testdb", "scott", "tiger" ); // Oracle 10g on silence
+// addDb( "QOCI", "//oracle10g-nokia.trolltech.com.au:1521/XE", "scott", "tiger" ); // Oracle 10gexpress on xen
+
+// This requires a local ODBC data source to be configured( pointing to a MySql database )
+// addDb( "QODBC", "mysqlodbc", "troll", "trond" );
+// addDb( "QODBC", "SqlServer", "troll", "trond" );
+// addDb( "QTDS7", "testdb", "troll", "trondk", "horsehead" );
+// addDb( "QODBC", "silencetestdb", "troll", "trond", "silence" );
+// addDb( "QODBC", "horseheadtestdb", "troll", "trondk", "horsehead" );
+
+// addDb( "QMYSQL3", "testdb", "troll", "trond", "horsehead.nokia.troll.no" );
+// addDb( "QMYSQL3", "testdb", "troll", "trond", "horsehead.nokia.troll.no", 3307 );
+// addDb( "QMYSQL3", "testdb", "troll", "trond", "horsehead.nokia.troll.no", 3308, "CLIENT_COMPRESS=1;CLIENT_SSL=1" ); // MySQL 4.1.1
+// addDb( "QMYSQL3", "testdb", "troll", "trond", "horsehead.nokia.troll.no", 3309, "CLIENT_COMPRESS=1;CLIENT_SSL=1" ); // MySQL 5.0.18 Linux
+// addDb( "QMYSQL3", "testdb", "troll", "trond", "iceblink.nokia.troll.no" ); // MySQL 5.0.13 Windows
+// addDb( "QMYSQL3", "testdb", "testuser", "Ee4Gabf6_", "mysql4-nokia.trolltech.com.au" ); // MySQL 4.1.22-2.el4 linux
+// addDb( "QMYSQL3", "testdb", "testuser", "Ee4Gabf6_", "mysql5-nokia.trolltech.com.au" ); // MySQL 5.0.45-7.el5 linux
+
+// addDb( "QPSQL7", "testdb", "troll", "trond", "horsehead.nokia.troll.no" ); // V7.2 NOT SUPPORTED!
+// addDb( "QPSQL7", "testdb", "troll", "trond", "horsehead.nokia.troll.no", 5434 ); // V7.2 NOT SUPPORTED! Multi-byte
+// addDb( "QPSQL7", "testdb", "troll", "trond", "horsehead.nokia.troll.no", 5435 ); // V7.3
+// addDb( "QPSQL7", "testdb", "troll", "trond", "horsehead.nokia.troll.no", 5436 ); // V7.4
+// addDb( "QPSQL7", "testdb", "troll", "trond", "horsehead.nokia.troll.no", 5437 ); // V8.0.3
+// addDb( "QPSQL7", "testdb", "troll", "trond", "silence.nokia.troll.no" ); // V8.2.1, UTF-8
+// addDb( "QPSQL7", "testdb", "testuser", "Ee4Gabf6_", "postgres74-nokia.trolltech.com.au" ); // Version 7.4.19-1.el4_6.1
+// addDb( "QPSQL7", "testdb", "testuser", "Ee4Gabf6_", "postgres81-nokia.trolltech.com.au" ); // Version 8.1.11-1.el5_1.1
+
+// addDb( "QDB2", "testdb", "troll", "trond", "silence" ); // DB2 v9.1 on silence
+
+// yes - interbase really wants the physical path on the host machine.
+// addDb( "QIBASE", "/opt/interbase/qttest.gdb", "SYSDBA", "masterkey", "horsehead.nokia.troll.no" );
+// addDb( "QIBASE", "silence.troll.no:c:\\ibase\\testdb", "SYSDBA", "masterkey", "" ); // InterBase 7.5 on silence
+// addDb( "QIBASE", "silence.troll.no:c:\\ibase\\testdb_ascii", "SYSDBA", "masterkey", "" ); // InterBase 7.5 on silence
+// addDb( "QIBASE", "/opt/firebird/databases/testdb.fdb", "testuser", "Ee4Gabf6_", "firebird1-nokia.trolltech.com.au" ); // Firebird 1.5.5
+// addDb( "QIBASE", "/opt/firebird/databases/testdb.fdb", "testuser", "Ee4Gabf6_", "firebird2-nokia.trolltech.com.au" ); // Firebird 2.1.1
+
+// Anders' local Firebird and InterBase test databases
+// addDb("QIBASE", "localhost:c:\\Firebird\\Firebird_2_0\\TESTDB.FDB", "SYSDBA", "masterkey", ""); // FireBird 2.0
+// addDb("QIBASE", "localhost:c:\\Borland\\InterBase\\TESTDB_UTF8.IB", "SYSDBA", "masterkey", ""); // InterBase 2007 Developer
+
+// use in-memory database to prevent local files
+// addDb("QSQLITE", ":memory:");
+ addDb( "QSQLITE", QDir::toNativeSeparators(QDir::tempPath()+"/foo.db") );
+// addDb( "QSQLITE2", QDir::toNativeSeparators(QDir::tempPath()+"/foo2.db") );
+// addDb( "QODBC3", "DRIVER={SQL SERVER};SERVER=iceblink.nokia.troll.no\\ICEBLINK", "troll", "trond", "" );
+// addDb( "QODBC3", "DRIVER={SQL Native Client};SERVER=silence.nokia.troll.no\\SQLEXPRESS", "troll", "trond", "" );
+
+ }
+
+ void open()
+ {
+ addDbs();
+
+ QStringList::Iterator it = dbNames.begin();
+
+ while ( it != dbNames.end() ) {
+ QSqlDatabase db = QSqlDatabase::database(( *it ), false );
+ qDebug() << "Opening:" << (*it);
+
+ if ( db.isValid() && !db.isOpen() ) {
+ if ( !db.open() ) {
+ qWarning( "tst_Databases: Unable to open %s on %s:\n%s", qPrintable( db.driverName() ), qPrintable( *it ), qPrintable( db.lastError().databaseText() ) );
+ // well... opening failed, so we just ignore the server, maybe it is not running
+ it = dbNames.erase( it );
+ } else {
+ ++it;
+ }
+ }
+ }
+ }
+
+ void close()
+ {
+ for ( QStringList::Iterator it = dbNames.begin(); it != dbNames.end(); ++it ) {
+ {
+ QSqlDatabase db = QSqlDatabase::database(( *it ), false );
+
+ if ( db.isValid() && db.isOpen() )
+ db.close();
+ }
+
+ QSqlDatabase::removeDatabase(( *it ) );
+ }
+
+ dbNames.clear();
+ }
+
+ // for debugging only: outputs the connection as string
+ static QString dbToString( const QSqlDatabase db )
+ {
+ QString res = db.driverName() + "@";
+
+ if ( db.driverName().startsWith( "QODBC" ) || db.driverName().startsWith( "QOCI" ) ) {
+ res += db.databaseName();
+ } else {
+ res += db.hostName();
+ }
+
+ if ( db.port() > 0 ) {
+ res += ":" + QString::number( db.port() );
+ }
+
+ return res;
+ }
+
+ // drop a table only if it exists to prevent warnings
+ static void safeDropTables( QSqlDatabase db, const QStringList& tableNames )
+ {
+ int wasDropped = true;
+ QSqlQuery q( db );
+ QStringList dbtables=db.tables();
+
+ foreach(QString tableName, tableNames)
+ {
+
+ if ( dbtables.contains( tableName, Qt::CaseSensitive ) )
+ wasDropped = q.exec( "drop table " + db.driver()->escapeIdentifier( tableName, QSqlDriver::TableName ) );
+ else if ( dbtables.contains( tableName, Qt::CaseInsensitive ) )
+ wasDropped = q.exec( "drop table " + tableName );
+
+ if ( !wasDropped )
+ qWarning() << "unable to drop table" << tableName << ':' << q.lastError().text();
+ }
+ }
+
+ static void safeDropViews( QSqlDatabase db, const QStringList &viewNames )
+ {
+ if ( isMSAccess( db ) ) // Access is sooo stupid.
+ safeDropTables( db, viewNames );
+
+ QStringList dbtables=db.tables(QSql::Views);
+
+ foreach(QString viewName, viewNames)
+ {
+ if ( dbtables.contains( viewName, Qt::CaseInsensitive ) ) {
+ QSqlQuery q( "drop view " + viewName, db );
+
+ if ( !q.isActive() )
+ qWarning() << "unable to drop view" << viewName << ':' << q.lastError().text();
+ } else if ( db.driverName().startsWith( "QMYSQL" )
+ && dbtables.contains( viewName, Qt::CaseInsensitive ) ) { // MySql is a bit stupid too
+ QSqlQuery q( "drop view " + viewName, db );
+ }
+ }
+ }
+
+ // returns the type name of the blob datatype for the database db.
+ // blobSize is only used if the db doesn't have a generic blob type
+ static QString blobTypeName( QSqlDatabase db, int blobSize = 10000 )
+ {
+ if ( db.driverName().startsWith( "QMYSQL" ) )
+ return "longblob";
+
+ if ( db.driverName().startsWith( "QPSQL" ) )
+ return "bytea";
+
+ if ( db.driverName().startsWith( "QTDS" )
+ || isSqlServer( db )
+ || isMSAccess( db ) )
+ return "image";
+
+ if ( db.driverName().startsWith( "QDB2" ) )
+ return QString( "blob(%1)" ).arg( blobSize );
+
+ if ( db.driverName().startsWith( "QIBASE" ) )
+ return QString( "blob sub_type 0 segment size 4096" );
+
+ if ( db.driverName().startsWith( "QOCI" )
+ || db.driverName().startsWith( "QSQLITE" ) )
+ return "blob";
+
+ qDebug() << "tst_Databases::blobTypeName: Don't know the blob type for" << dbToString( db );
+
+ return "blob";
+ }
+
+ static QString autoFieldName( QSqlDatabase db )
+ {
+ if ( db.driverName().startsWith( "QMYSQL" ) )
+ return "AUTO_INCREMENT";
+ if ( db.driverName().startsWith( "QTDS" ) )
+ return "IDENTITY";
+/* if ( db.driverName().startsWith( "QPSQL" ) )
+ return "SERIAL";*/
+ if ( db.driverName().startsWith( "QDB2" ) )
+ return "GENERATED BY DEFAULT AS IDENTITY";
+
+ return QString();
+ }
+
+ static QByteArray printError( const QSqlError& err )
+ {
+ return QString( "'" + err.driverText() + "' || '" + err.databaseText() + "'" ).toLocal8Bit();
+ }
+
+ static QByteArray printError( const QSqlError& err, const QSqlDatabase& db )
+ {
+ return QString( db.driverName() + ": '" + err.driverText() + "' || '" + err.databaseText() + "'" ).toLocal8Bit();
+ }
+
+ static bool isSqlServer( QSqlDatabase db )
+ {
+ return db.databaseName().contains( "sql server", Qt::CaseInsensitive )
+ || db.databaseName().contains( "sqlserver", Qt::CaseInsensitive )
+ || db.databaseName().contains( "sql native client", Qt::CaseInsensitive );
+ }
+
+ static bool isMSAccess( QSqlDatabase db )
+ {
+ return db.databaseName().contains( "Access Driver", Qt::CaseInsensitive );
+ }
+
+ // -1 on fail, else Oracle version
+ static int getOraVersion( QSqlDatabase db )
+ {
+ int ver = -1;
+ QSqlQuery q( "SELECT banner FROM v$version", db );
+ q.next();
+
+ QRegExp vers( "([0-9]+)\\.[0-9\\.]+[0-9]" );
+
+ if ( vers.indexIn( q.value( 0 ).toString() ) ) {
+ bool ok;
+ ver = vers.cap( 1 ).toInt( &ok );
+
+ if ( !ok )
+ ver = -1;
+ }
+
+ return ver;
+ }
+
+ static QString getMySqlVersion( const QSqlDatabase &db )
+ {
+ QSqlQuery q(db);
+ q.exec( "select version()" );
+ if(q.next())
+ return q.value( 0 ).toString();
+ else
+ return QString();
+ }
+
+ QStringList dbNames;
+ int counter;
+};
+
+#endif
+