7ec818e74a
Change-Id: Ibfcf6b557aed3b0cd2e0ece5cf122819a1acc0c1 Reviewed-by: Volker Hilsheimer <volker.hilsheimer@qt.io>
4759 lines
175 KiB
C++
4759 lines
175 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the test suite of the Qt Toolkit.
|
|
**
|
|
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include <QtTest/QtTest>
|
|
#include <QtSql/QtSql>
|
|
|
|
#include <numeric>
|
|
|
|
#include "../qsqldatabase/tst_databases.h"
|
|
|
|
QString qtest;
|
|
|
|
class tst_QSqlQuery : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
tst_QSqlQuery();
|
|
virtual ~tst_QSqlQuery();
|
|
|
|
public slots:
|
|
void initTestCase();
|
|
void cleanupTestCase();
|
|
void init();
|
|
void cleanup();
|
|
|
|
private slots:
|
|
void value_data() { generic_data(); }
|
|
void value();
|
|
void isValid_data() { generic_data(); }
|
|
void isValid();
|
|
void isActive_data() { generic_data(); }
|
|
void isActive();
|
|
void isSelect_data() { generic_data(); }
|
|
void isSelect();
|
|
void numRowsAffected_data() { generic_data(); }
|
|
void numRowsAffected();
|
|
void size_data() { generic_data(); }
|
|
void size();
|
|
void isNull_data() { generic_data(); }
|
|
void isNull();
|
|
void query_exec_data() { generic_data(); }
|
|
void query_exec();
|
|
void execErrorRecovery_data() { generic_data(); }
|
|
void execErrorRecovery();
|
|
void first_data() { generic_data(); }
|
|
void first();
|
|
void next_data() { generic_data(); }
|
|
void next();
|
|
void prev_data() { generic_data(); }
|
|
void prev();
|
|
void last_data() { generic_data(); }
|
|
void last();
|
|
void seek_data() { generic_data(); }
|
|
void seek();
|
|
void transaction_data() { generic_data(); }
|
|
void transaction();
|
|
void record_data() { generic_data(); }
|
|
void record();
|
|
void record_sqlite_data() { generic_data("QSQLITE"); }
|
|
void record_sqlite();
|
|
void finish_data() { generic_data(); }
|
|
void finish();
|
|
void sqlite_finish_data() { generic_data("QSQLITE"); }
|
|
void sqlite_finish();
|
|
void nextResult_data() { generic_data(); }
|
|
void nextResult();
|
|
|
|
// forwardOnly mode need special treatment
|
|
void forwardOnly_data() { generic_data(); }
|
|
void forwardOnly();
|
|
void forwardOnlyMultipleResultSet_data() { generic_data(); }
|
|
void forwardOnlyMultipleResultSet();
|
|
void psql_forwardOnlyQueryResultsLost_data() { generic_data("QPSQL"); }
|
|
void psql_forwardOnlyQueryResultsLost();
|
|
|
|
// bug specific tests
|
|
void tds_bitField_data() { generic_data("QTDS"); }
|
|
void tds_bitField();
|
|
void oci_nullBlob_data() { generic_data("QOCI"); }
|
|
void oci_nullBlob();
|
|
void blob_data() { generic_data(); }
|
|
void blob();
|
|
void oci_rawField_data() { generic_data("QOCI"); }
|
|
void oci_rawField();
|
|
void precision_data() { generic_data(); }
|
|
void precision();
|
|
void nullResult_data() { generic_data(); }
|
|
void nullResult();
|
|
void joins_data() { generic_data(); }
|
|
void joins();
|
|
void outValues_data() { generic_data(); }
|
|
void outValues();
|
|
void char1Select_data() { generic_data(); }
|
|
void char1Select();
|
|
void char1SelectUnicode_data() { generic_data(); }
|
|
void char1SelectUnicode();
|
|
void synonyms_data() { generic_data(); }
|
|
void synonyms();
|
|
void oraOutValues_data() { generic_data("QOCI"); }
|
|
void oraOutValues();
|
|
void mysql_outValues_data() { generic_data("QMYSQL"); }
|
|
void mysql_outValues();
|
|
void oraClob_data() { generic_data("QOCI"); }
|
|
void oraClob();
|
|
void oraClobBatch_data() { generic_data("QOCI"); }
|
|
void oraClobBatch();
|
|
void oraLong_data() { generic_data("QOCI"); }
|
|
void oraLong();
|
|
void oraOCINumber_data() { generic_data("QOCI"); }
|
|
void oraOCINumber();
|
|
void outValuesDB2_data() { generic_data("QDB2"); }
|
|
void outValuesDB2();
|
|
void storedProceduresIBase_data() {generic_data("QIBASE"); }
|
|
void storedProceduresIBase();
|
|
void oraRowId_data() { generic_data("QOCI"); }
|
|
void oraRowId();
|
|
void prepare_bind_exec_data() { generic_data(); }
|
|
void prepare_bind_exec();
|
|
void prepared_select_data() { generic_data(); }
|
|
void prepared_select();
|
|
void sqlServerLongStrings_data() { generic_data(); }
|
|
void sqlServerLongStrings();
|
|
void invalidQuery_data() { generic_data(); }
|
|
void invalidQuery();
|
|
void batchExec_data() { generic_data(); }
|
|
void batchExec();
|
|
void QTBUG_43874_data() { generic_data(); }
|
|
void QTBUG_43874();
|
|
void oraArrayBind_data() { generic_data("QOCI"); }
|
|
void oraArrayBind();
|
|
void lastInsertId_data() { generic_data(); }
|
|
void lastInsertId();
|
|
void lastQuery_data() { generic_data(); }
|
|
void lastQuery();
|
|
void lastQueryTwoQueries_data() { generic_data(); }
|
|
void lastQueryTwoQueries();
|
|
void bindBool_data() { generic_data(); }
|
|
void bindBool();
|
|
void psql_bindWithDoubleColonCastOperator_data() { generic_data("QPSQL"); }
|
|
void psql_bindWithDoubleColonCastOperator();
|
|
void psql_specialFloatValues_data() { generic_data("QPSQL"); }
|
|
void psql_specialFloatValues();
|
|
void queryOnInvalidDatabase_data() { generic_data(); }
|
|
void queryOnInvalidDatabase();
|
|
void createQueryOnClosedDatabase_data() { generic_data(); }
|
|
void createQueryOnClosedDatabase();
|
|
void seekForwardOnlyQuery_data() { generic_data(); }
|
|
void seekForwardOnlyQuery();
|
|
void reExecutePreparedForwardOnlyQuery_data() { generic_data(); }
|
|
void reExecutePreparedForwardOnlyQuery();
|
|
void blobsPreparedQuery_data() { generic_data(); }
|
|
void blobsPreparedQuery();
|
|
void emptyTableNavigate_data() { generic_data(); }
|
|
void emptyTableNavigate();
|
|
void timeStampParsing_data() { generic_data(); }
|
|
void timeStampParsing();
|
|
void sqliteVirtualTable_data() { generic_data("QSQLITE"); }
|
|
void sqliteVirtualTable();
|
|
void mysql_timeType_data() { generic_data("QMYSQL"); }
|
|
void mysql_timeType();
|
|
|
|
void task_217003_data() { generic_data(); }
|
|
void task_217003();
|
|
|
|
void task_250026_data() { generic_data("QODBC"); }
|
|
void task_250026();
|
|
void crashQueryOnCloseDatabase();
|
|
|
|
void task_233829_data() { generic_data("QPSQL"); }
|
|
void task_233829();
|
|
|
|
void QTBUG_12477_data() { generic_data("QPSQL"); }
|
|
void QTBUG_12477();
|
|
|
|
void sqlServerReturn0_data() { generic_data(); }
|
|
void sqlServerReturn0();
|
|
|
|
void QTBUG_551_data() { generic_data("QOCI"); }
|
|
void QTBUG_551();
|
|
|
|
void QTBUG_5251_data() { generic_data("QPSQL"); }
|
|
void QTBUG_5251();
|
|
void QTBUG_6421_data() { generic_data("QOCI"); }
|
|
void QTBUG_6421();
|
|
void QTBUG_6618_data() { generic_data("QODBC"); }
|
|
void QTBUG_6618();
|
|
void QTBUG_6852_data() { generic_data("QMYSQL"); }
|
|
void QTBUG_6852();
|
|
void QTBUG_5765_data() { generic_data("QMYSQL"); }
|
|
void QTBUG_5765();
|
|
void QTBUG_12186_data() { generic_data("QSQLITE"); }
|
|
void QTBUG_12186();
|
|
void QTBUG_14132_data() { generic_data("QOCI"); }
|
|
void QTBUG_14132();
|
|
void QTBUG_18435_data() { generic_data("QODBC"); }
|
|
void QTBUG_18435();
|
|
void QTBUG_21884_data() { generic_data("QSQLITE"); }
|
|
void QTBUG_21884();
|
|
void QTBUG_16967_data() { generic_data("QSQLITE"); }
|
|
void QTBUG_16967(); //clean close
|
|
void QTBUG_23895_data() { generic_data("QSQLITE"); }
|
|
void QTBUG_23895(); //sqlite boolean type
|
|
void QTBUG_14904_data() { generic_data("QSQLITE"); }
|
|
void QTBUG_14904();
|
|
|
|
void QTBUG_2192_data() { generic_data(); }
|
|
void QTBUG_2192();
|
|
|
|
void QTBUG_36211_data() { generic_data("QPSQL"); }
|
|
void QTBUG_36211();
|
|
|
|
void QTBUG_53969_data() { generic_data("QMYSQL"); }
|
|
void QTBUG_53969();
|
|
|
|
void gisPointDatatype_data() { generic_data("QMYSQL"); }
|
|
void gisPointDatatype();
|
|
|
|
void sqlite_constraint_data() { generic_data("QSQLITE"); }
|
|
void sqlite_constraint();
|
|
|
|
void sqlite_real_data() { generic_data("QSQLITE"); }
|
|
void sqlite_real();
|
|
|
|
void aggregateFunctionTypes_data() { generic_data(); }
|
|
void aggregateFunctionTypes();
|
|
|
|
void integralTypesMysql_data() { generic_data("QMYSQL"); }
|
|
void integralTypesMysql();
|
|
|
|
void QTBUG_57138_data() { generic_data("QSQLITE"); }
|
|
void QTBUG_57138();
|
|
|
|
void dateTime_data();
|
|
void dateTime();
|
|
|
|
private:
|
|
// returns all database connections
|
|
void generic_data(const QString &engine=QString());
|
|
void dropTestTables( QSqlDatabase db );
|
|
void createTestTables( QSqlDatabase db );
|
|
void populateTestTables( QSqlDatabase db );
|
|
|
|
tst_Databases dbs;
|
|
};
|
|
|
|
tst_QSqlQuery::tst_QSqlQuery()
|
|
{
|
|
static QSqlDatabase static_qtest_db = QSqlDatabase();
|
|
qtest = qTableName("qtest", __FILE__, static_qtest_db);
|
|
}
|
|
|
|
tst_QSqlQuery::~tst_QSqlQuery()
|
|
{
|
|
}
|
|
|
|
void tst_QSqlQuery::initTestCase()
|
|
{
|
|
QVERIFY(dbs.open());
|
|
|
|
for ( QStringList::ConstIterator it = dbs.dbNames.begin(); it != dbs.dbNames.end(); ++it ) {
|
|
QSqlDatabase db = QSqlDatabase::database(( *it ) );
|
|
CHECK_DATABASE( db );
|
|
dropTestTables( db ); //in case of leftovers
|
|
createTestTables( db );
|
|
populateTestTables( db );
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::cleanupTestCase()
|
|
{
|
|
for ( QStringList::ConstIterator it = dbs.dbNames.begin(); it != dbs.dbNames.end(); ++it ) {
|
|
QSqlDatabase db = QSqlDatabase::database(( *it ) );
|
|
CHECK_DATABASE( db );
|
|
dropTestTables( db );
|
|
}
|
|
|
|
dbs.close();
|
|
}
|
|
|
|
void tst_QSqlQuery::init()
|
|
{
|
|
}
|
|
|
|
void tst_QSqlQuery::cleanup()
|
|
{
|
|
if (QTest::currentTestFunction() == QLatin1String("crashQueryOnCloseDatabase"))
|
|
return;
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if ( QTest::currentTestFunction() == QLatin1String( "numRowsAffected" )
|
|
|| QTest::currentTestFunction() == QLatin1String( "transactions" )
|
|
|| QTest::currentTestFunction() == QLatin1String( "size" )
|
|
|| QTest::currentTestFunction() == QLatin1String( "isActive" )
|
|
|| QTest::currentTestFunction() == QLatin1String( "lastInsertId" ) ) {
|
|
populateTestTables( db );
|
|
}
|
|
|
|
if (QTest::currentTestFailed() && (dbType == QSqlDriver::Oracle || db.driverName().startsWith("QODBC"))) {
|
|
//since Oracle ODBC totally craps out on error, we init again
|
|
db.close();
|
|
db.open();
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::generic_data(const QString& engine)
|
|
{
|
|
if ( dbs.fillTestTable(engine) == 0 ) {
|
|
if(engine.isEmpty())
|
|
QSKIP( "No database drivers are available in this Qt configuration");
|
|
else
|
|
QSKIP( (QString("No database drivers of type %1 are available in this Qt configuration").arg(engine)).toLocal8Bit());
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::dropTestTables( QSqlDatabase db )
|
|
{
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
QStringList tablenames;
|
|
// drop all the table in case a testcase failed
|
|
tablenames << qtest
|
|
<< qTableName("qtest_null", __FILE__, db)
|
|
<< qTableName("qtest_blob", __FILE__, db)
|
|
<< qTableName("qtest_bittest", __FILE__, db)
|
|
<< qTableName("qtest_nullblob", __FILE__, db)
|
|
<< qTableName("qtest_rawtest", __FILE__, db)
|
|
<< qTableName("qtest_precision", __FILE__, db)
|
|
<< qTableName("qtest_prepare", __FILE__, db)
|
|
<< qTableName("qtestj1", __FILE__, db)
|
|
<< qTableName("qtestj2", __FILE__, db)
|
|
<< qTableName("char1Select", __FILE__, db)
|
|
<< qTableName("char1SU", __FILE__, db)
|
|
<< qTableName("qxmltest", __FILE__, db)
|
|
<< qTableName("qtest_exerr", __FILE__, db)
|
|
<< qTableName("qtest_empty", __FILE__, db)
|
|
<< qTableName("clobby", __FILE__, db)
|
|
<< qTableName("bindtest", __FILE__, db)
|
|
<< qTableName("more_results", __FILE__, db)
|
|
<< qTableName("blobstest", __FILE__, db)
|
|
<< qTableName("oraRowId", __FILE__, db)
|
|
<< qTableName("bug43874", __FILE__, db)
|
|
<< qTableName("bug6421", __FILE__, db).toUpper()
|
|
<< qTableName("bug5765", __FILE__, db)
|
|
<< qTableName("bug6852", __FILE__, db)
|
|
<< qTableName("bug21884", __FILE__, db)
|
|
<< qTableName("bug23895", __FILE__, db)
|
|
<< qTableName("qtest_lockedtable", __FILE__, db)
|
|
<< qTableName("Planet", __FILE__, db)
|
|
<< qTableName("task_250026", __FILE__, db)
|
|
<< qTableName("task_234422", __FILE__, db)
|
|
<< qTableName("test141895", __FILE__, db)
|
|
<< qTableName("qtest_oraOCINumber", __FILE__, db)
|
|
<< qTableName("bug2192", __FILE__, db)
|
|
<< qTableName("tst_record", __FILE__, db);
|
|
|
|
if (dbType == QSqlDriver::PostgreSQL)
|
|
tablenames << qTableName("task_233829", __FILE__, db);
|
|
|
|
if (dbType == QSqlDriver::SQLite)
|
|
tablenames << qTableName("record_sqlite", __FILE__, db);
|
|
|
|
if (dbType == QSqlDriver::MSSqlServer || dbType == QSqlDriver::Oracle)
|
|
tablenames << qTableName("qtest_longstr", __FILE__, db);
|
|
|
|
if (dbType == QSqlDriver::MSSqlServer)
|
|
db.exec("DROP PROCEDURE " + qTableName("test141895_proc", __FILE__, db));
|
|
|
|
if (dbType == QSqlDriver::MySqlServer)
|
|
db.exec("DROP PROCEDURE IF EXISTS "+ qTableName("bug6852_proc", __FILE__, db));
|
|
|
|
tst_Databases::safeDropTables( db, tablenames );
|
|
|
|
if (dbType == QSqlDriver::Oracle) {
|
|
QSqlQuery q( db );
|
|
q.exec("DROP PACKAGE " + qTableName("pkg", __FILE__, db));
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::createTestTables( QSqlDatabase db )
|
|
{
|
|
QSqlQuery q( db );
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::MySqlServer)
|
|
// ### stupid workaround until we find a way to hardcode this
|
|
// in the MySQL server startup script
|
|
q.exec( "set table_type=innodb" );
|
|
else if (dbType == QSqlDriver::PostgreSQL)
|
|
QVERIFY_SQL( q, exec("set client_min_messages='warning'"));
|
|
|
|
if (dbType == QSqlDriver::PostgreSQL)
|
|
QVERIFY_SQL( q, exec( "create table " + qtest + " (id serial NOT NULL, t_varchar varchar(20), t_char char(20), primary key(id)) WITH OIDS" ) );
|
|
else
|
|
QVERIFY_SQL( q, exec( "create table " + qtest + " (id int "+tst_Databases::autoFieldName(db) +" NOT NULL, t_varchar varchar(20), t_char char(20), primary key(id))" ) );
|
|
|
|
if (dbType == QSqlDriver::MSSqlServer || dbType == QSqlDriver::Sybase)
|
|
QVERIFY_SQL(q, exec("create table " + qTableName("qtest_null", __FILE__, db) + " (id int null, t_varchar varchar(20) null)"));
|
|
else
|
|
QVERIFY_SQL(q, exec("create table " + qTableName("qtest_null", __FILE__, db) + " (id int, t_varchar varchar(20))"));
|
|
}
|
|
|
|
void tst_QSqlQuery::populateTestTables( QSqlDatabase db )
|
|
{
|
|
QSqlQuery q( db );
|
|
const QString qtest_null(qTableName( "qtest_null", __FILE__, db));
|
|
q.exec( "delete from " + qtest );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest + " values (1, 'VarChar1', 'Char1')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest + " values (2, 'VarChar2', 'Char2')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest + " values (3, 'VarChar3', 'Char3')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest + " values (4, 'VarChar4', 'Char4')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest + " values (5, 'VarChar5', 'Char5')" ) );
|
|
|
|
q.exec( "delete from " + qtest_null );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_null + " values (0, NULL)" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_null + " values (1, 'n')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_null + " values (2, 'i')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_null + " values (3, NULL)" ) );
|
|
}
|
|
|
|
// There were problems with char fields of size 1
|
|
void tst_QSqlQuery::char1Select()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
{
|
|
QSqlQuery q( db );
|
|
const QString tbl = qTableName("char1Select", __FILE__, db);
|
|
q.exec( "drop table " + tbl);
|
|
QVERIFY_SQL(q, exec("create table " + tbl + " (id char(1))"));
|
|
QVERIFY_SQL(q, exec("insert into " + tbl + " values ('a')"));
|
|
QVERIFY_SQL(q, exec("select * from " + tbl));
|
|
QVERIFY( q.next() );
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::Interbase)
|
|
QCOMPARE( q.value( 0 ).toString().left( 1 ), QString( "a" ) );
|
|
else
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "a" ) );
|
|
|
|
QVERIFY( !q.next() );
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::char1SelectUnicode()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::DB2)
|
|
QSKIP("Needs someone with more Unicode knowledge than I have to fix");
|
|
|
|
if ( db.driver()->hasFeature( QSqlDriver::Unicode ) ) {
|
|
QString uniStr( QChar(0x0915) ); // DEVANAGARI LETTER KA
|
|
QSqlQuery q( db );
|
|
QString createQuery;
|
|
const QString char1SelectUnicode(qTableName("char1SU", __FILE__, db));
|
|
|
|
if (dbType == QSqlDriver::MSSqlServer)
|
|
createQuery = "create table " + char1SelectUnicode + "(id nchar(1))";
|
|
else if (dbType == QSqlDriver::DB2 || dbType == QSqlDriver::Oracle || dbType == QSqlDriver::PostgreSQL)
|
|
createQuery = "create table " + char1SelectUnicode + " (id char(3))";
|
|
else if (dbType == QSqlDriver::Interbase)
|
|
createQuery = "create table " + char1SelectUnicode +
|
|
" (id char(1) character set unicode_fss)";
|
|
else if (dbType == QSqlDriver::MySqlServer)
|
|
createQuery = "create table " + char1SelectUnicode + " (id char(1)) "
|
|
"default character set 'utf8'";
|
|
else
|
|
createQuery = "create table " + char1SelectUnicode + " (id char(1))";
|
|
|
|
QVERIFY_SQL( q, exec( createQuery ) );
|
|
|
|
QVERIFY_SQL( q, prepare( "insert into " + char1SelectUnicode + " values(?)" ) );
|
|
|
|
q.bindValue( 0, uniStr );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from " + char1SelectUnicode ) );
|
|
|
|
QVERIFY( q.next() );
|
|
|
|
if ( !q.value( 0 ).toString().isEmpty() )
|
|
QCOMPARE( q.value( 0 ).toString()[ 0 ].unicode(), uniStr[0].unicode() );
|
|
|
|
QCOMPARE( q.value( 0 ).toString().trimmed(), uniStr );
|
|
|
|
QVERIFY( !q.next() );
|
|
}
|
|
else
|
|
QSKIP( "Database not unicode capable");
|
|
}
|
|
|
|
void tst_QSqlQuery::oraRowId()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString oraRowId(qTableName("oraRowId", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL( q, exec( "select rowid from " + qtest ) );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).type(), QVariant::String );
|
|
QVERIFY( !q.value( 0 ).toString().isEmpty() );
|
|
|
|
QVERIFY_SQL( q, exec( "create table " + oraRowId + " (id char(1))" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "insert into " + oraRowId + " values('a')" ) );
|
|
QVariant v1 = q.lastInsertId();
|
|
QVERIFY( v1.isValid() );
|
|
|
|
QVERIFY_SQL( q, exec( "insert into " + oraRowId + " values('b')" ) );
|
|
QVariant v2 = q.lastInsertId();
|
|
QVERIFY( v2.isValid() );
|
|
|
|
QVERIFY_SQL( q, prepare( "select * from " + oraRowId + " where rowid = ?" ) );
|
|
q.addBindValue( v1 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "a" ) );
|
|
|
|
q.addBindValue( v2 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "b" ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::mysql_outValues()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString hello(qTableName("hello", __FILE__, db)), qtestproc(qTableName("qtestproc", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
|
|
q.exec( "drop function " + hello );
|
|
|
|
QVERIFY_SQL(q, exec("create function " + hello + " (s char(20)) returns varchar(50) READS SQL DATA return concat('Hello ', s)"));
|
|
|
|
QVERIFY_SQL( q, exec( "select " + hello + "('world')" ) );
|
|
QVERIFY_SQL( q, next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Hello world" ) );
|
|
|
|
QVERIFY_SQL( q, prepare( "select " + hello + "('harald')" ) );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY_SQL( q, next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Hello harald" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "drop function " + hello ) );
|
|
|
|
q.exec( "drop procedure " + qtestproc );
|
|
|
|
QVERIFY_SQL( q, exec( "create procedure " + qtestproc + " () "
|
|
"BEGIN select * from " + qtest + " order by id; END" ) );
|
|
QVERIFY_SQL( q, exec( "call " + qtestproc + "()" ) );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 1 ).toString(), QString( "VarChar1" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "drop procedure " + qtestproc ) );
|
|
|
|
QVERIFY_SQL( q, exec( "create procedure " + qtestproc + " (OUT param1 INT) "
|
|
"BEGIN set param1 = 42; END" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "call " + qtestproc + " (@out)" ) );
|
|
QVERIFY_SQL( q, exec( "select @out" ) );
|
|
QCOMPARE( q.record().fieldName( 0 ), QString( "@out" ) );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 42 );
|
|
|
|
QVERIFY_SQL( q, exec( "drop procedure " + qtestproc ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::bindBool()
|
|
{
|
|
// QTBUG-27763: bool value got converted to int 127 by mysql driver because sizeof(bool) < sizeof(int).
|
|
// The problem was the way the bool value from the application was handled. For our purposes here, it
|
|
// doesn't matter whether the column type is BOOLEAN or INT. All DBMSs have INT, and this usually
|
|
// works for this test. Postresql is an exception because its INT type does not accept BOOLEAN
|
|
// values and its BOOLEAN columns do not accept INT values.
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q(db);
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
const QString tableName(qTableName("bindBool", __FILE__, db));
|
|
|
|
q.exec("DROP TABLE " + tableName);
|
|
QString colType = dbType == QSqlDriver::PostgreSQL ? QLatin1String("BOOLEAN") : QLatin1String("INT");
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + " (id INT, flag " + colType + " NOT NULL, PRIMARY KEY(id))"));
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
bool flag = i;
|
|
q.prepare("INSERT INTO " + tableName + " (id, flag) VALUES(:id, :flag)");
|
|
q.bindValue(":id", i);
|
|
q.bindValue(":flag", flag);
|
|
QVERIFY_SQL(q, exec());
|
|
}
|
|
|
|
QVERIFY_SQL(q, exec("SELECT id, flag FROM " + tableName + " ORDER BY id"));
|
|
for (int i = 0; i < 2; ++i) {
|
|
bool flag = i;
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), i);
|
|
QCOMPARE(q.value(1).toBool(), flag);
|
|
}
|
|
QVERIFY_SQL(q, prepare("SELECT flag FROM " + tableName + " WHERE flag = :filter"));
|
|
const bool filter = true;
|
|
q.bindValue(":filter", filter);
|
|
QVERIFY_SQL(q, exec());
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toBool(), filter);
|
|
QFAIL_SQL(q, next());
|
|
QVERIFY_SQL(q, exec("DROP TABLE " + tableName));
|
|
}
|
|
|
|
void tst_QSqlQuery::oraOutValues()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString tst_outValues(qTableName("tst_outValues", __FILE__, db));
|
|
|
|
if ( !db.driver()->hasFeature( QSqlDriver::PreparedQueries ) )
|
|
QSKIP( "Test requires prepared query support");
|
|
|
|
QSqlQuery q( db );
|
|
|
|
q.setForwardOnly( true );
|
|
|
|
/*** outvalue int ***/
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x out int) is\n"
|
|
"begin\n"
|
|
" x := 42;\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
q.addBindValue( 0, QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValue( 0 ).toInt(), 42 );
|
|
|
|
// bind a null value, make sure the OCI driver resets the null flag
|
|
q.addBindValue( QVariant( QVariant::Int ), QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValue( 0 ).toInt(), 42 );
|
|
QVERIFY( !q.boundValue( 0 ).isNull() );
|
|
|
|
/*** outvalue varchar ***/
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x out varchar) is\n"
|
|
"begin\n"
|
|
" x := 'blah';\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
QString s1( "12345" );
|
|
s1.reserve( 512 );
|
|
q.addBindValue( s1, QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValue( 0 ).toString(), QString( "blah" ) );
|
|
|
|
/*** in/outvalue numeric ***/
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x in out numeric) is\n"
|
|
"begin\n"
|
|
" x := x + 10;\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
q.addBindValue( 10, QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValue( 0 ).toInt(), 20 );
|
|
|
|
/*** in/outvalue varchar ***/
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x in out varchar) is\n"
|
|
"begin\n"
|
|
" x := 'homer';\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
q.addBindValue( QString( "maggy" ), QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValue( 0 ).toString(), QString( "homer" ) );
|
|
|
|
/*** in/outvalue varchar ***/
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x in out varchar) is\n"
|
|
"begin\n"
|
|
" x := NULL;\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
q.addBindValue( QString( "maggy" ), QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.boundValue( 0 ).isNull() );
|
|
|
|
/*** in/outvalue int ***/
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x in out int) is\n"
|
|
"begin\n"
|
|
" x := NULL;\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
q.addBindValue( 42, QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.boundValue( 0 ).isNull() );
|
|
|
|
/*** in/outvalue varchar ***/
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x in varchar, y out varchar) is\n"
|
|
"begin\n"
|
|
" y := x||'bubulalakikikokololo';\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?, ?)" ) );
|
|
q.addBindValue( QString( "fifi" ), QSql::In );
|
|
QString out;
|
|
out.reserve( 50 );
|
|
q.addBindValue( out, QSql::Out );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValue( 1 ).toString(), QString( "fifibubulalakikikokololo" ) );
|
|
|
|
/*** in/outvalue date ***/
|
|
QVERIFY_SQL(q, exec("create or replace procedure " + tst_outValues + "(x in date, y out date) is\n"
|
|
"begin\n"
|
|
" y := x;\n"
|
|
"end;\n"));
|
|
QVERIFY(q.prepare("call " + tst_outValues + "(?, ?)"));
|
|
const QDate date = QDate::currentDate();
|
|
q.addBindValue(date, QSql::In);
|
|
q.addBindValue(QVariant(QDate()), QSql::Out);
|
|
QVERIFY_SQL(q, exec());
|
|
QCOMPARE(q.boundValue(1).toDate(), date);
|
|
|
|
/*** in/outvalue timestamp ***/
|
|
QVERIFY_SQL(q, exec("create or replace procedure " + tst_outValues + "(x in timestamp, y out timestamp) is\n"
|
|
"begin\n"
|
|
" y := x;\n"
|
|
"end;\n"));
|
|
QVERIFY(q.prepare("call " + tst_outValues + "(?, ?)"));
|
|
const QDateTime dt = QDateTime::currentDateTime();
|
|
q.addBindValue(dt, QSql::In);
|
|
q.addBindValue(QVariant(QVariant::DateTime), QSql::Out);
|
|
QVERIFY_SQL(q, exec());
|
|
QCOMPARE(q.boundValue(1).toDateTime(), dt);
|
|
}
|
|
|
|
void tst_QSqlQuery::oraClob()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString clobby(qTableName("clobby", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
|
|
// simple short string
|
|
QVERIFY_SQL( q, exec( "create table " + clobby + "(id int primary key, cl clob, bl blob)" ) );
|
|
QVERIFY_SQL( q, prepare( "insert into " + clobby + " (id, cl, bl) values(?, ?, ?)" ) );
|
|
q.addBindValue( 1 );
|
|
q.addBindValue( "bubu" );
|
|
q.addBindValue( QByteArray("bubu") );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY_SQL( q, exec( "select bl, cl from " + clobby + " where id = 1" ) );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "bubu" ) );
|
|
QCOMPARE( q.value( 1 ).toString(), QString( "bubu" ) );
|
|
|
|
// simple short string with binding
|
|
QVERIFY_SQL( q, prepare( "insert into " + clobby + " (id, cl, bl) values(?, ?, ?)" ) );
|
|
q.addBindValue( 2 );
|
|
q.addBindValue( "lala", QSql::Binary );
|
|
q.addBindValue( QByteArray("lala"), QSql::Binary );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY_SQL( q, exec( "select bl, cl from " + clobby + " where id = 2" ) );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "lala" ) );
|
|
QCOMPARE( q.value( 1 ).toString(), QString( "lala" ) );
|
|
|
|
// loooong string
|
|
QString loong;
|
|
loong.fill( QLatin1Char( 'A' ), 25000 );
|
|
QVERIFY_SQL( q, prepare( "insert into " + clobby + " (id, cl, bl) values(?, ?, ?)" ) );
|
|
q.addBindValue( 3 );
|
|
q.addBindValue( loong, QSql::Binary );
|
|
q.addBindValue( loong.toLatin1(), QSql::Binary );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY_SQL( q, exec( "select bl, cl from " + clobby + " where id = 3" ) );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toString().count(), loong.count() );
|
|
QVERIFY( q.value( 0 ).toString() == loong );
|
|
QCOMPARE( q.value( 1 ).toByteArray().count(), loong.toLatin1().count() );
|
|
QVERIFY( q.value( 1 ).toByteArray() == loong.toLatin1() );
|
|
}
|
|
|
|
void tst_QSqlQuery::oraClobBatch()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
const QString clobBatch(qTableName("clobBatch", __FILE__, db));
|
|
tst_Databases::safeDropTables(db, { clobBatch });
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("create table " + clobBatch + "(cl clob)"));
|
|
|
|
const QString longString(USHRT_MAX + 1, QLatin1Char('A'));
|
|
QVERIFY_SQL(q, prepare("insert into " + clobBatch + " (cl) values(:cl)"));
|
|
const QVariantList vars = { longString };
|
|
q.addBindValue(vars);
|
|
QVERIFY_SQL(q, execBatch());
|
|
|
|
QVERIFY_SQL(q, exec("select cl from " + clobBatch));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toString().count(), longString.size());
|
|
QVERIFY(q.value(0).toString() == longString);
|
|
}
|
|
|
|
void tst_QSqlQuery::storedProceduresIBase()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
const auto procName = qTableName("TESTPROC", __FILE__, db);
|
|
q.exec("drop procedure " + procName);
|
|
|
|
QVERIFY_SQL(q, exec("create procedure " + procName +
|
|
" RETURNS (x integer, y varchar(20)) "
|
|
"AS BEGIN "
|
|
" x = 42; "
|
|
" y = 'Hello Anders'; "
|
|
"END" ) );
|
|
|
|
QVERIFY_SQL(q, prepare("execute procedure " + procName));
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
// check for a valid result set
|
|
QSqlRecord rec = q.record();
|
|
QCOMPARE( rec.count(), 2 );
|
|
QCOMPARE( rec.fieldName( 0 ).toUpper(), QString( "X" ) );
|
|
QCOMPARE( rec.fieldName( 1 ).toUpper(), QString( "Y" ) );
|
|
|
|
// the first next shall suceed
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 42 );
|
|
QCOMPARE( q.value( 1 ).toString(), QString( "Hello Anders" ) );
|
|
|
|
// the second next shall fail
|
|
QVERIFY( !q.next() );
|
|
|
|
q.exec("drop procedure " + procName);
|
|
}
|
|
|
|
void tst_QSqlQuery::outValuesDB2()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
if ( !db.driver()->hasFeature( QSqlDriver::PreparedQueries ) )
|
|
QSKIP( "Test requires prepared query support");
|
|
|
|
QSqlQuery q( db );
|
|
|
|
q.setForwardOnly( true );
|
|
|
|
q.exec("drop procedure " + qTableName("tst_outValues", __FILE__, db)); //non-fatal
|
|
QVERIFY_SQL( q, exec( "CREATE PROCEDURE " + qTableName("tst_outValues", __FILE__, db) +
|
|
" (OUT x int, OUT x2 double, OUT x3 char(20))\n"
|
|
"LANGUAGE SQL\n"
|
|
"P1: BEGIN\n"
|
|
" SET x = 42;\n"
|
|
" SET x2 = 4.2;\n"
|
|
" SET x3 = 'Homer';\n"
|
|
"END P1" ) );
|
|
|
|
QVERIFY_SQL(q, prepare("call " + qTableName("tst_outValues", __FILE__, db) + "(?, ?, ?)"));
|
|
|
|
q.addBindValue( 0, QSql::Out );
|
|
q.addBindValue( 0.0, QSql::Out );
|
|
q.addBindValue( "Simpson", QSql::Out );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QCOMPARE( q.boundValue( 0 ).toInt(), 42 );
|
|
QCOMPARE( q.boundValue( 1 ).toDouble(), 4.2 );
|
|
QCOMPARE( q.boundValue( 2 ).toString().trimmed(), QString( "Homer" ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::outValues()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString tst_outValues(qTableName("tst_outValues", __FILE__, db));
|
|
|
|
if ( !db.driver()->hasFeature( QSqlDriver::PreparedQueries ) )
|
|
QSKIP( "Test requires prepared query support");
|
|
|
|
QSqlQuery q( db );
|
|
|
|
q.setForwardOnly( true );
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::Oracle) {
|
|
QVERIFY_SQL( q, exec( "create or replace procedure " + tst_outValues + "(x out int) is\n"
|
|
"begin\n"
|
|
" x := 42;\n"
|
|
"end;\n" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
} else if (dbType == QSqlDriver::DB2) {
|
|
q.exec( "drop procedure " + tst_outValues ); //non-fatal
|
|
QVERIFY_SQL( q, exec( "CREATE PROCEDURE " + tst_outValues + " (OUT x int)\n"
|
|
"LANGUAGE SQL\n"
|
|
"P1: BEGIN\n"
|
|
" SET x = 42;\n"
|
|
"END P1" ) );
|
|
QVERIFY( q.prepare( "call " + tst_outValues + "(?)" ) );
|
|
} else if (dbType == QSqlDriver::MSSqlServer) {
|
|
q.exec( "drop procedure " + tst_outValues ); //non-fatal
|
|
QVERIFY_SQL( q, exec( "create procedure " + tst_outValues + " (@x int out) as\n"
|
|
"begin\n"
|
|
" set @x = 42\n"
|
|
"end\n" ) );
|
|
QVERIFY( q.prepare( "{call " + tst_outValues + "(?)}" ) );
|
|
} else
|
|
QSKIP( "Don't know how to create a stored procedure for this database server, please fix this test");
|
|
|
|
q.addBindValue( 0, QSql::Out );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QCOMPARE( q.boundValue( 0 ).toInt(), 42 );
|
|
}
|
|
|
|
void tst_QSqlQuery::blob()
|
|
{
|
|
static const int BLOBSIZE = 1024 * 10;
|
|
static const int BLOBCOUNT = 2;
|
|
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
if ( !db.driver()->hasFeature( QSqlDriver::BLOB ) )
|
|
QSKIP( "DBMS not BLOB capable");
|
|
|
|
//don' make it too big otherwise sybase and mysql will complain
|
|
QByteArray ba( BLOBSIZE, 0 );
|
|
|
|
for (int i = 0; i < ba.size(); ++i)
|
|
ba[i] = i % 256;
|
|
|
|
QSqlQuery q( db );
|
|
|
|
q.setForwardOnly( true );
|
|
|
|
QString queryString = QString("create table " + qTableName("qtest_blob", __FILE__, db) +
|
|
" (id int not null primary key, t_blob %1)" ).arg( tst_Databases::blobTypeName( db, BLOBSIZE ) );
|
|
QVERIFY_SQL( q, exec( queryString ) );
|
|
|
|
QVERIFY_SQL(q, prepare("insert into " + qTableName("qtest_blob", __FILE__, db) + " (id, t_blob) values (?, ?)"));
|
|
|
|
for (int i = 0; i < BLOBCOUNT; ++i) {
|
|
q.addBindValue( i );
|
|
q.addBindValue( ba );
|
|
QVERIFY_SQL( q, exec() );
|
|
}
|
|
|
|
QVERIFY_SQL(q, exec("select * from " + qTableName("qtest_blob", __FILE__, db)));
|
|
|
|
for (int i = 0; i < BLOBCOUNT; ++i) {
|
|
QVERIFY( q.next() );
|
|
QByteArray res = q.value( 1 ).toByteArray();
|
|
QVERIFY2( res.size() >= ba.size(),
|
|
QString( "array sizes differ, expected %1, got %2" ).arg( ba.size() ).arg( res.size() ).toLatin1() );
|
|
|
|
for (int i2 = 0; i2 < ba.size(); ++i2) {
|
|
if ( res[i2] != ba[i2] )
|
|
QFAIL( QString( "ByteArrays differ at position %1, expected %2, got %3" ).arg(
|
|
i2 ).arg(( int )( unsigned char )ba[i2] ).arg(( int )( unsigned char )res[i2] ).toLatin1() );
|
|
}
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::value()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL( q, exec( "select id, t_varchar, t_char from " + qtest + " order by id" ) );
|
|
int i = 1;
|
|
|
|
while ( q.next() ) {
|
|
QCOMPARE( q.value( 0 ).toInt(), i );
|
|
QCOMPARE( q.value( "id" ).toInt(), i );
|
|
|
|
if (dbType == QSqlDriver::Interbase)
|
|
QVERIFY( q.value( 1 ).toString().startsWith( "VarChar" + QString::number( i ) ) );
|
|
else if ( q.value( 1 ).toString().endsWith(QLatin1Char(' ')))
|
|
QCOMPARE( q.value( 1 ).toString(), ( "VarChar" + QString::number( i ) + " " ) );
|
|
else
|
|
QCOMPARE( q.value( 1 ).toString(), ( "VarChar" + QString::number( i ) ) );
|
|
|
|
if (dbType == QSqlDriver::Interbase)
|
|
QVERIFY( q.value( 2 ).toString().startsWith( "Char" + QString::number( i ) ) );
|
|
else if (!q.value( 2 ).toString().endsWith(QLatin1Char(' ')))
|
|
QCOMPARE( q.value( 2 ).toString(), ( "Char" + QString::number( i ) ) );
|
|
else
|
|
QCOMPARE( q.value( 2 ).toString(), ( "Char" + QString::number( i ) + " " ) );
|
|
|
|
i++;
|
|
}
|
|
}
|
|
|
|
#define SETUP_RECORD_TABLE \
|
|
do { \
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tst_record + " (id integer, extra varchar(50))")); \
|
|
for (int i = 0; i < 3; ++i) \
|
|
QVERIFY_SQL(q, exec(QString("INSERT INTO " + tst_record + " VALUES(%1, 'extra%1')").arg(i))); \
|
|
} while (0)
|
|
|
|
#define CHECK_RECORD \
|
|
do { \
|
|
QVERIFY_SQL(q, exec(QString("select %1.id, %1.t_varchar, %1.t_char, %2.id, %2.extra from %1, %2 where " \
|
|
"%1.id = %2.id order by %1.id").arg(lowerQTest).arg(tst_record))); \
|
|
QCOMPARE(q.record().fieldName(0).toLower(), QString("id")); \
|
|
QCOMPARE(q.record().field(0).tableName().toLower(), lowerQTest); \
|
|
QCOMPARE(q.record().fieldName(1).toLower(), QString("t_varchar")); \
|
|
QCOMPARE(q.record().field(1).tableName().toLower(), lowerQTest); \
|
|
QCOMPARE(q.record().fieldName(2).toLower(), QString("t_char")); \
|
|
QCOMPARE(q.record().field(2).tableName().toLower(), lowerQTest); \
|
|
QCOMPARE(q.record().fieldName(3).toLower(), QString("id")); \
|
|
QCOMPARE(q.record().field(3).tableName().toLower(), tst_record); \
|
|
QCOMPARE(q.record().fieldName(4).toLower(), QString("extra")); \
|
|
QCOMPARE(q.record().field(4).tableName().toLower(), tst_record); \
|
|
} while (0)
|
|
|
|
void tst_QSqlQuery::record()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY( q.record().isEmpty() );
|
|
QVERIFY_SQL( q, exec( "select id, t_varchar, t_char from " + qtest + " order by id" ) );
|
|
QCOMPARE( q.record().fieldName( 0 ).toLower(), QString( "id" ) );
|
|
QCOMPARE( q.record().fieldName( 1 ).toLower(), QString( "t_varchar" ) );
|
|
QCOMPARE( q.record().fieldName( 2 ).toLower(), QString( "t_char" ) );
|
|
QCOMPARE(q.record().value(0), QVariant(q.record().field(0).metaType()));
|
|
QCOMPARE(q.record().value(1), QVariant(q.record().field(1).metaType()));
|
|
QCOMPARE(q.record().value(2), QVariant(q.record().field(2).metaType()));
|
|
|
|
QVERIFY( q.next() );
|
|
QVERIFY( q.next() );
|
|
|
|
QCOMPARE( q.record().fieldName( 0 ).toLower(), QString( "id" ) );
|
|
QCOMPARE( q.value( 0 ).toInt(), 2 );
|
|
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::Oracle)
|
|
QSKIP("Getting the tablename is not supported in Oracle");
|
|
const auto lowerQTest = qtest.toLower();
|
|
for (int i = 0; i < 3; ++i)
|
|
QCOMPARE(q.record().field(i).tableName().toLower(), lowerQTest);
|
|
q.clear();
|
|
const auto tst_record = qTableName("tst_record", __FILE__, db, false).toLower();
|
|
SETUP_RECORD_TABLE;
|
|
CHECK_RECORD;
|
|
q.clear();
|
|
|
|
// Recreate the tables, in a different order
|
|
const QStringList tables = { qtest, tst_record, qTableName("qtest_null", __FILE__, db) };
|
|
tst_Databases::safeDropTables(db, tables);
|
|
SETUP_RECORD_TABLE;
|
|
createTestTables(db);
|
|
populateTestTables(db);
|
|
CHECK_RECORD;
|
|
}
|
|
|
|
void tst_QSqlQuery::isValid()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY( !q.isValid() );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
QVERIFY( q.first() );
|
|
QVERIFY( q.isValid() );
|
|
}
|
|
|
|
void tst_QSqlQuery::isActive()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY( !q.isActive() );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
QVERIFY( q.isActive() );
|
|
QVERIFY( q.last() );
|
|
|
|
if ( !tst_Databases::isMSAccess( db ) )
|
|
// Access is stupid enough to let you scroll over boundaries
|
|
QVERIFY( !q.next() );
|
|
|
|
QVERIFY( q.isActive() );
|
|
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest + " values (41, 'VarChar41', 'Char41')" ) );
|
|
|
|
QVERIFY( q.isActive() );
|
|
|
|
QVERIFY_SQL( q, exec( "update " + qtest + " set id = 42 where id = 41" ) );
|
|
|
|
QVERIFY( q.isActive() );
|
|
|
|
QVERIFY_SQL( q, exec( "delete from " + qtest + " where id = 42" ) );
|
|
|
|
QVERIFY( q.isActive() );
|
|
|
|
QVERIFY_SQL( q, exec( "delete from " + qtest + " where id = 42" ) );
|
|
|
|
QVERIFY( q.isActive() );
|
|
}
|
|
|
|
void tst_QSqlQuery::numRowsAffected()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QCOMPARE( q.numRowsAffected(), -1 );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
int i = 0;
|
|
|
|
while ( q.next() )
|
|
++i;
|
|
|
|
if ( q.numRowsAffected() == -1 || q.numRowsAffected() == 0 )
|
|
QSKIP("Database doesn't support numRowsAffected");
|
|
|
|
if ( q.numRowsAffected() != -1 && q.numRowsAffected() != 0 && q.numRowsAffected() != i ) {
|
|
// the value is undefined for SELECT, this check is just here for curiosity
|
|
qDebug( "Expected numRowsAffected to be -1, 0 or %d, got %d", i, q.numRowsAffected() );
|
|
}
|
|
|
|
QVERIFY_SQL( q, exec( "update " + qtest + " set id = 100 where id = 1" ) );
|
|
|
|
QCOMPARE( q.numRowsAffected(), 1 );
|
|
QCOMPARE( q.numRowsAffected(), 1 ); // yes, we check twice
|
|
|
|
QVERIFY_SQL( q, exec( "update " + qtest + " set id = id + 100" ) );
|
|
QCOMPARE( q.numRowsAffected(), i );
|
|
QCOMPARE( q.numRowsAffected(), i ); // yes, we check twice
|
|
|
|
QVERIFY_SQL( q, prepare( "update " + qtest + " set id = id + :newid" ) );
|
|
q.bindValue(":newid", 100);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.numRowsAffected(), i );
|
|
QCOMPARE( q.numRowsAffected(), i ); // yes, we check twice
|
|
|
|
QVERIFY_SQL( q, prepare( "update " + qtest + " set id = id + :newid where NOT(1 = 1)" ) );
|
|
q.bindValue(":newid", 100);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.numRowsAffected(), 0 );
|
|
QCOMPARE( q.numRowsAffected(), 0 ); // yes, we check twice
|
|
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest + " values (42000, 'homer', 'marge')" ) );
|
|
QCOMPARE( q.numRowsAffected(), 1 );
|
|
QCOMPARE( q.numRowsAffected(), 1 ); // yes, we check twice
|
|
|
|
QSqlQuery q2( db );
|
|
QVERIFY_SQL( q2, exec( "insert into " + qtest + " values (42001, 'homer', 'marge')" ) );
|
|
|
|
if ( !db.driverName().startsWith( "QSQLITE2" ) ) {
|
|
// SQLite 2.x accumulates changed rows in nested queries. See task 33794
|
|
QCOMPARE( q2.numRowsAffected(), 1 );
|
|
QCOMPARE( q2.numRowsAffected(), 1 ); // yes, we check twice
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::size()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QCOMPARE( q.size(), -1 );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
int i = 0;
|
|
|
|
while ( q.next() )
|
|
++i;
|
|
|
|
if ( db.driver()->hasFeature( QSqlDriver::QuerySize ) ) {
|
|
QCOMPARE( q.size(), i );
|
|
QCOMPARE( q.size(), i ); // yes, twice
|
|
} else {
|
|
QCOMPARE( q.size(), -1 );
|
|
QCOMPARE( q.size(), -1 ); // yes, twice
|
|
}
|
|
|
|
QSqlQuery q2( "select * from " + qtest, db );
|
|
|
|
if ( db.driver()->hasFeature( QSqlDriver::QuerySize ) )
|
|
QCOMPARE( q.size(), i );
|
|
else
|
|
QCOMPARE( q.size(), -1 );
|
|
|
|
q2.clear();
|
|
|
|
QVERIFY_SQL( q, exec( "update " + qtest + " set id = 100 where id = 1" ) );
|
|
QCOMPARE( q.size(), -1 );
|
|
QCOMPARE( q.size(), -1 ); // yes, twice
|
|
}
|
|
|
|
void tst_QSqlQuery::isSelect()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
QVERIFY( q.isSelect() );
|
|
|
|
QVERIFY_SQL( q, exec( "update " + qtest + " set id = 1 where id = 1" ) );
|
|
QVERIFY( q.isSelect() == false );
|
|
}
|
|
|
|
void tst_QSqlQuery::first()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY( q.at() == QSql::BeforeFirstRow );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
QVERIFY( q.last() );
|
|
QVERIFY_SQL( q, first() );
|
|
QVERIFY( q.at() == 0 );
|
|
}
|
|
|
|
void tst_QSqlQuery::next()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY( q.at() == QSql::BeforeFirstRow );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
QVERIFY( q.first() );
|
|
QVERIFY( q.next() );
|
|
QVERIFY( q.at() == 1 );
|
|
}
|
|
|
|
void tst_QSqlQuery::prev()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY( q.at() == QSql::BeforeFirstRow );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
QVERIFY( q.first() );
|
|
QVERIFY( q.next() );
|
|
QVERIFY( q.previous() );
|
|
QVERIFY( q.at() == 0 );
|
|
}
|
|
|
|
void tst_QSqlQuery::last()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QCOMPARE( q.at(), int( QSql::BeforeFirstRow ) );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
int i = 0;
|
|
|
|
while ( q.next() )
|
|
i++;
|
|
|
|
QCOMPARE( q.at(), int( QSql::AfterLastRow ) );
|
|
|
|
QVERIFY( q.last() );
|
|
|
|
QSet<int> validReturns(QSet<int>() << -1 << i-1);
|
|
QVERIFY( validReturns.contains(q.at()) );
|
|
|
|
QSqlQuery q2( "select * from " + qtest, db );
|
|
|
|
QVERIFY( q2.last() );
|
|
|
|
QVERIFY( validReturns.contains(q.at()) );
|
|
}
|
|
|
|
void tst_QSqlQuery::seek()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q( db );
|
|
QVERIFY( q.at() == QSql::BeforeFirstRow );
|
|
QVERIFY_SQL( q, exec( QString( "select id from %1 order by id" ).arg( qtest ) ) );
|
|
|
|
// NB! The order of the calls below are important!
|
|
QVERIFY( q.last() );
|
|
QVERIFY( !q.seek( QSql::BeforeFirstRow ) );
|
|
QCOMPARE( q.at(), int( QSql::BeforeFirstRow ) );
|
|
QVERIFY( q.seek( 0 ) );
|
|
QCOMPARE( q.at(), 0 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
|
|
QVERIFY( q.seek( 1 ) );
|
|
QCOMPARE( q.at(), 1 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 2 );
|
|
|
|
QVERIFY( q.seek( 3 ) );
|
|
QCOMPARE( q.at(), 3 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 4 );
|
|
|
|
QVERIFY( q.seek( -2, true ) );
|
|
QCOMPARE( q.at(), 1 );
|
|
QVERIFY( q.seek( 0 ) );
|
|
QCOMPARE( q.at(), 0 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
|
|
QVERIFY(!q.seek(QSql::BeforeFirstRow));
|
|
QCOMPARE(q.at(), int(QSql::BeforeFirstRow));
|
|
QVERIFY(q.seek(1, true));
|
|
QCOMPARE(q.at(), 0);
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
|
|
qint32 count = 1;
|
|
while (q.next()) ++count;
|
|
|
|
QCOMPARE(q.at(), int(QSql::AfterLastRow));
|
|
|
|
if (!q.isForwardOnly()) {
|
|
QVERIFY(q.seek(-1, true));
|
|
QCOMPARE(q.at(), count - 1);
|
|
QCOMPARE(q.value(0).toInt(), count);
|
|
} else {
|
|
QVERIFY(!q.seek(-1, true));
|
|
QCOMPARE(q.at(), int(QSql::AfterLastRow));
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::seekForwardOnlyQuery()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
q.setForwardOnly( false );
|
|
QVERIFY( !q.isForwardOnly() );
|
|
|
|
QVERIFY( q.at() == QSql::BeforeFirstRow );
|
|
QVERIFY_SQL( q, exec( QString( "select id from %1 order by id" ).arg( qtest ) ) );
|
|
|
|
QSqlRecord rec;
|
|
|
|
// NB! The order of the calls below are important!
|
|
QVERIFY( q.seek( 0 ) );
|
|
QCOMPARE( q.at(), 0 );
|
|
rec = q.record();
|
|
QCOMPARE( rec.value( 0 ).toInt(), 1 );
|
|
|
|
QVERIFY( q.seek( 1 ) );
|
|
QCOMPARE( q.at(), 1 );
|
|
rec = q.record();
|
|
QCOMPARE( rec.value( 0 ).toInt(), 2 );
|
|
|
|
// Make a jump!
|
|
QVERIFY( q.seek( 3 ) );
|
|
QCOMPARE( q.at(), 3 );
|
|
rec = q.record();
|
|
QCOMPARE( rec.value( 0 ).toInt(), 4 );
|
|
|
|
// Last record in result set
|
|
QVERIFY( q.seek( 4 ) );
|
|
QCOMPARE( q.at(), 4 );
|
|
rec = q.record();
|
|
QCOMPARE( rec.value( 0 ).toInt(), 5 );
|
|
}
|
|
|
|
// tests the forward only mode;
|
|
void tst_QSqlQuery::forwardOnly()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
q.setForwardOnly( true );
|
|
QVERIFY( q.isForwardOnly() );
|
|
QVERIFY( q.at() == QSql::BeforeFirstRow );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest + " order by id" ) );
|
|
if (!q.isForwardOnly())
|
|
QSKIP("DBMS doesn't support forward-only queries");
|
|
QVERIFY( q.at() == QSql::BeforeFirstRow );
|
|
QVERIFY( q.first() );
|
|
QCOMPARE( q.at(), 0 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.at(), 1 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 2 );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.at(), 2 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 3 );
|
|
|
|
// lets make some mistakes to see how robust it is
|
|
QTest::ignoreMessage( QtWarningMsg, "QSqlQuery::seek: cannot seek backwards in a forward only query" );
|
|
QVERIFY( q.first() == false );
|
|
QCOMPARE( q.at(), 2 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 3 );
|
|
QTest::ignoreMessage( QtWarningMsg, "QSqlQuery::seek: cannot seek backwards in a forward only query" );
|
|
QVERIFY( q.previous() == false );
|
|
QCOMPARE( q.at(), 2 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 3 );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.at(), 3 );
|
|
QCOMPARE( q.value( 0 ).toInt(), 4 );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
int i = 0;
|
|
|
|
while ( q.next() )
|
|
i++;
|
|
|
|
QVERIFY( q.at() == QSql::AfterLastRow );
|
|
|
|
QSqlQuery q2 = q;
|
|
|
|
QVERIFY( q2.isForwardOnly() );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest + " order by id" ) );
|
|
|
|
QVERIFY( q.isForwardOnly() );
|
|
|
|
QVERIFY( q2.isForwardOnly() );
|
|
|
|
QCOMPARE( q.at(), int( QSql::BeforeFirstRow ) );
|
|
|
|
QVERIFY_SQL( q, seek( 3 ) );
|
|
|
|
QCOMPARE( q.at(), 3 );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 4 );
|
|
|
|
QTest::ignoreMessage( QtWarningMsg, "QSqlQuery::seek: cannot seek backwards in a forward only query" );
|
|
|
|
QVERIFY( q.seek( 0 ) == false );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 4 );
|
|
|
|
QCOMPARE( q.at(), 3 );
|
|
|
|
QVERIFY( q.last() );
|
|
|
|
QCOMPARE( q.at(), i-1 );
|
|
|
|
QTest::ignoreMessage( QtWarningMsg, "QSqlQuery::seek: cannot seek backwards in a forward only query" );
|
|
|
|
QVERIFY( q.first() == false );
|
|
|
|
QCOMPARE( q.at(), i-1 );
|
|
|
|
QVERIFY( q.next() == false );
|
|
|
|
QCOMPARE( q.at(), int( QSql::AfterLastRow ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::forwardOnlyMultipleResultSet()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
|
|
if (!db.driver()->hasFeature(QSqlDriver::MultipleResultSets))
|
|
QSKIP("DBMS doesn't support multiple result sets");
|
|
|
|
QSqlQuery q(db);
|
|
q.setForwardOnly(true);
|
|
QVERIFY_SQL(q, exec("select id, t_varchar from " + qtest + " order by id;" // 1.
|
|
"select id, t_varchar, t_char from " + qtest + " where id<4 order by id;" // 2.
|
|
"update " + qtest + " set t_varchar='VarChar555' where id=5;" // 3.
|
|
"select * from " + qtest + " order by id;" // 4.
|
|
"select * from " + qtest + " where id=5 order by id;" // 5.
|
|
"select * from " + qtest + " where id=-1 order by id;" // 6.
|
|
"select * from " + qtest + " order by id")); // 7.
|
|
|
|
if (!q.isForwardOnly())
|
|
QSKIP("DBMS doesn't support forward-only queries");
|
|
|
|
// 1. Result set with 2 columns and 5 rows
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), true);
|
|
|
|
// Test record() of first result set
|
|
QSqlRecord record = q.record();
|
|
QCOMPARE(record.count(), 2);
|
|
QCOMPARE(record.indexOf("id"), 0);
|
|
QCOMPARE(record.indexOf("t_varchar"), 1);
|
|
if (dbType != QSqlDriver::PostgreSQL) { // tableName() is not available in forward-only mode of QPSQL
|
|
QCOMPARE(record.field(0).tableName(), qtest); // BUG: This fails for Microsoft SQL Server 2016 (QODBC), need fix
|
|
QCOMPARE(record.field(1).tableName(), qtest);
|
|
}
|
|
|
|
// Test navigation
|
|
QVERIFY(q.first());
|
|
QCOMPARE(q.at(), 0);
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.at(), 1);
|
|
QCOMPARE(q.value(0).toInt(), 2);
|
|
|
|
QVERIFY(q.seek(3));
|
|
QCOMPARE(q.at(), 3);
|
|
QCOMPARE(q.value(0).toInt(), 4);
|
|
|
|
QTest::ignoreMessage(QtWarningMsg, "QSqlQuery::seek: cannot seek backwards in a forward only query");
|
|
QVERIFY(q.first() == false);
|
|
QCOMPARE(q.at(), 3);
|
|
QCOMPARE(q.value(0).toInt(), 4);
|
|
|
|
QTest::ignoreMessage(QtWarningMsg, "QSqlQuery::seek: cannot seek backwards in a forward only query");
|
|
QVERIFY(q.previous() == false);
|
|
QCOMPARE(q.at(), 3);
|
|
QCOMPARE(q.value(0).toInt(), 4);
|
|
|
|
QTest::ignoreMessage(QtWarningMsg, "QSqlQuery::seek: cannot seek backwards in a forward only query");
|
|
QVERIFY(q.seek(1) == false);
|
|
QCOMPARE(q.at(), 3);
|
|
QCOMPARE(q.value(0).toInt(), 4);
|
|
|
|
QVERIFY(q.last());
|
|
QCOMPARE(q.at(), 4);
|
|
|
|
// Try move after last row
|
|
QVERIFY(q.next() == false);
|
|
QCOMPARE(q.at(), QSql::AfterLastRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
|
|
// 2. Result set with 3 columns and 3 rows
|
|
QVERIFY(q.nextResult());
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), true);
|
|
|
|
// Test record() of second result set
|
|
record = q.record();
|
|
QCOMPARE(record.count(), 3);
|
|
QCOMPARE(record.indexOf("id"), 0);
|
|
QCOMPARE(record.indexOf("t_varchar"), 1);
|
|
QCOMPARE(record.indexOf("t_char"), 2);
|
|
|
|
// Test iteration
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
int index = 0;
|
|
while (q.next()) {
|
|
QCOMPARE(q.at(), index);
|
|
QCOMPARE(q.value(0).toInt(), index+1);
|
|
index++;
|
|
}
|
|
QVERIFY(q.at() == QSql::AfterLastRow);
|
|
QCOMPARE(index, 3);
|
|
|
|
// 3. Update statement
|
|
QVERIFY(q.nextResult());
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), false);
|
|
QCOMPARE(q.numRowsAffected(), 1);
|
|
|
|
// 4. Result set with 5 rows
|
|
QVERIFY(q.nextResult());
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), true);
|
|
|
|
// Test forward seek(n)
|
|
QVERIFY(q.seek(2));
|
|
QCOMPARE(q.at(), 2);
|
|
QCOMPARE(q.value(0).toInt(), 3);
|
|
|
|
// Test value(string)
|
|
QCOMPARE(q.value("id").toInt(), 3);
|
|
QCOMPARE(q.value("t_varchar").toString(), "VarChar3");
|
|
QCOMPARE(q.value("t_char").toString().trimmed(), "Char3");
|
|
|
|
// Next 2 rows of current result set will be
|
|
// discarded by next call of nextResult()
|
|
|
|
// 5. Result set with 1 row
|
|
QVERIFY(q.nextResult());
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), true);
|
|
QVERIFY(q.first());
|
|
QCOMPARE(q.at(), 0);
|
|
QCOMPARE(q.value(0).toInt(), 5);
|
|
QVERIFY(q.next() == false);
|
|
QVERIFY(q.at() == QSql::AfterLastRow);
|
|
|
|
// 6. Result set without rows
|
|
QVERIFY(q.nextResult());
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), true);
|
|
QVERIFY(q.next() == false);
|
|
|
|
// 7. Result set with 5 rows
|
|
QVERIFY(q.nextResult());
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), true);
|
|
|
|
// Just skip it, so we move after last result set.
|
|
QVERIFY(q.nextResult() == false);
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), false);
|
|
|
|
// See if we can execute another query
|
|
QVERIFY_SQL(q, exec("select id from " + qtest + " where id=5"));
|
|
QVERIFY(q.at() == QSql::BeforeFirstRow);
|
|
QCOMPARE(q.isActive(), true);
|
|
QCOMPARE(q.isSelect(), true);
|
|
QVERIFY(q.first());
|
|
QCOMPARE(q.at(), 0);
|
|
QCOMPARE(q.value("id").toInt(), 5);
|
|
QCOMPARE(q.record().count(), 1);
|
|
QCOMPARE(q.record().indexOf("id"), 0);
|
|
}
|
|
|
|
void tst_QSqlQuery::psql_forwardOnlyQueryResultsLost()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery q1(db);
|
|
q1.setForwardOnly(true);
|
|
QVERIFY_SQL(q1, exec("select * from " + qtest + " where id<=3 order by id"));
|
|
if (!q1.isForwardOnly())
|
|
QSKIP("DBMS doesn't support forward-only queries");
|
|
|
|
// Read first row of q1
|
|
QVERIFY(q1.next());
|
|
QCOMPARE(q1.at(), 0);
|
|
QCOMPARE(q1.value(0).toInt(), 1);
|
|
|
|
// Executing another query on the same db connection
|
|
// will cause the query results of q1 to be lost.
|
|
QSqlQuery q2(db);
|
|
q2.setForwardOnly(true);
|
|
QVERIFY_SQL(q2, exec("select * from " + qtest + " where id>3 order by id"));
|
|
|
|
QTest::ignoreMessage(QtWarningMsg, "QPSQLDriver::getResult: Query results lost - "
|
|
"probably discarded on executing another SQL query.");
|
|
|
|
// Reading next row of q1 will not possible.
|
|
QVERIFY(!q1.next());
|
|
QCOMPARE(q1.at(), QSql::AfterLastRow);
|
|
QVERIFY(q1.lastError().type() != QSqlError::NoError);
|
|
|
|
// See if we can read rows from q2
|
|
QVERIFY(q2.seek(1));
|
|
QCOMPARE(q2.at(), 1);
|
|
QCOMPARE(q2.value(0).toInt(), 5);
|
|
}
|
|
|
|
void tst_QSqlQuery::query_exec()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY( !q.isValid() );
|
|
QVERIFY( !q.isActive() );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest ) );
|
|
QVERIFY( q.isActive() );
|
|
QVERIFY( q.next() );
|
|
QVERIFY( q.isValid() );
|
|
}
|
|
|
|
void tst_QSqlQuery::isNull()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL(q, exec("select id, t_varchar from " + qTableName("qtest_null", __FILE__, db) + " order by id"));
|
|
QVERIFY( q.next() );
|
|
QVERIFY( !q.isNull( 0 ) );
|
|
QVERIFY(!q.isNull("id"));
|
|
QVERIFY( q.isNull( 1 ) );
|
|
QVERIFY(q.isNull("t_varchar"));
|
|
QCOMPARE( q.value( 0 ).toInt(), 0 );
|
|
QCOMPARE( q.value( 1 ).toString(), QString() );
|
|
QVERIFY( !q.value( 0 ).isNull() );
|
|
QVERIFY( q.value( 1 ).isNull() );
|
|
|
|
QVERIFY( q.next() );
|
|
QVERIFY( !q.isNull( 0 ) );
|
|
QVERIFY(!q.isNull("id"));
|
|
QVERIFY( !q.isNull( 1 ) );
|
|
QVERIFY(!q.isNull("t_varchar"));
|
|
|
|
// For a non existent field, it should be returning true.
|
|
QVERIFY(q.isNull(2));
|
|
QVERIFY(q.isNull("unknown"));
|
|
}
|
|
|
|
/*! TDS specific BIT field test */
|
|
void tst_QSqlQuery::tds_bitField()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString tableName = qTableName("qtest_bittest", __FILE__, db);
|
|
QSqlQuery q( db );
|
|
|
|
QVERIFY_SQL(q, exec("create table " + tableName + " (bitty bit)"));
|
|
|
|
QVERIFY_SQL(q, exec("insert into " + tableName + " values (0)"));
|
|
|
|
QVERIFY_SQL(q, exec("insert into " + tableName + " values (1)"));
|
|
|
|
QVERIFY_SQL(q, exec("select bitty from " + tableName));
|
|
|
|
QVERIFY( q.next() );
|
|
|
|
QVERIFY( q.value( 0 ).toInt() == 0 );
|
|
|
|
QVERIFY( q.next() );
|
|
|
|
QVERIFY( q.value( 0 ).toInt() == 1 );
|
|
}
|
|
|
|
|
|
/*! Oracle specific NULL BLOB test */
|
|
void tst_QSqlQuery::oci_nullBlob()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString qtest_nullblob(qTableName("qtest_nullblob", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL( q, exec( "create table " + qtest_nullblob + " (id int primary key, bb blob)" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_nullblob + " values (0, EMPTY_BLOB())" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_nullblob + " values (1, NULL)" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_nullblob + " values (2, 'aabbcc00112233445566')" ) );
|
|
// necessary otherwise oracle will bombard you with internal errors
|
|
q.setForwardOnly( true );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest_nullblob + " order by id" ) );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(( int )q.value( 1 ).toByteArray().size(), 0 );
|
|
QVERIFY( !q.isNull( 1 ) );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(( int )q.value( 1 ).toByteArray().size(), 0 );
|
|
QVERIFY( q.isNull( 1 ) );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(( int )q.value( 1 ).toByteArray().size(), 10 );
|
|
QVERIFY( !q.isNull( 1 ) );
|
|
}
|
|
|
|
/* Oracle specific RAW field test */
|
|
void tst_QSqlQuery::oci_rawField()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString qtest_rawtest(qTableName("qtest_rawtest", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
q.setForwardOnly( true );
|
|
QVERIFY_SQL( q, exec( "create table " + qtest_rawtest +
|
|
" (id int, col raw(20))" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_rawtest + " values (0, NULL)" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtest_rawtest + " values (1, '00aa1100ddeeff')" ) );
|
|
QVERIFY_SQL( q, exec( "select col from " + qtest_rawtest + " order by id" ) );
|
|
QVERIFY( q.next() );
|
|
QVERIFY( q.isNull( 0 ) );
|
|
QCOMPARE(( int )q.value( 0 ).toByteArray().size(), 0 );
|
|
QVERIFY( q.next() );
|
|
QVERIFY( !q.isNull( 0 ) );
|
|
QCOMPARE(( int )q.value( 0 ).toByteArray().size(), 7 );
|
|
}
|
|
|
|
// test whether we can fetch values with more than DOUBLE precision
|
|
// note that SQLite highest precision is that of a double, although
|
|
// you can define field with higher precision
|
|
void tst_QSqlQuery::precision()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::Interbase)
|
|
QSKIP("DB unable to store high precision");
|
|
|
|
const auto oldPrecision = db.driver()->numericalPrecisionPolicy();
|
|
db.driver()->setNumericalPrecisionPolicy(QSql::HighPrecision);
|
|
const QString qtest_precision(qTableName("qtest_precision", __FILE__, db));
|
|
static const QLatin1String precStr("1.2345678901234567891");
|
|
|
|
{
|
|
// need a new scope for SQLITE
|
|
QSqlQuery q( db );
|
|
|
|
q.exec("drop table " + qtest_precision);
|
|
if (tst_Databases::isMSAccess(db))
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + qtest_precision + " (col1 number)"));
|
|
else
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + qtest_precision + " (col1 numeric(21, 20))"));
|
|
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + qtest_precision + " (col1) VALUES (" + precStr + ")"));
|
|
QVERIFY_SQL(q, exec("SELECT * FROM " + qtest_precision));
|
|
QVERIFY(q.next());
|
|
const QString val = q.value(0).toString();
|
|
if (!val.startsWith(precStr)) {
|
|
int i = 0;
|
|
while (i < val.size() && precStr[i] != 0 && precStr[i] == val[i].toLatin1())
|
|
i++;
|
|
|
|
// TDS has crappy precisions by default
|
|
if (dbType == QSqlDriver::Sybase) {
|
|
if (i < 18)
|
|
QWARN("TDS didn't return the right precision");
|
|
} else {
|
|
QWARN(QString(tst_Databases::dbToString(db) + " didn't return the right precision (" +
|
|
QString::number(i) + " out of 21), " + val).toUtf8());
|
|
}
|
|
}
|
|
} // SQLITE scope
|
|
db.driver()->setNumericalPrecisionPolicy(oldPrecision);
|
|
}
|
|
|
|
void tst_QSqlQuery::nullResult()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL( q, exec( "select * from " + qtest + " where id > 50000" ) );
|
|
|
|
if ( q.driver()->hasFeature( QSqlDriver::QuerySize ) )
|
|
QCOMPARE( q.size(), 0 );
|
|
|
|
QVERIFY( q.next() == false );
|
|
|
|
QVERIFY( q.first() == false );
|
|
QVERIFY( q.last() == false );
|
|
QVERIFY( q.previous() == false );
|
|
QVERIFY( q.seek( 10 ) == false );
|
|
QVERIFY( q.seek( 0 ) == false );
|
|
}
|
|
|
|
// this test is just an experiment to see whether we can do query-based transactions
|
|
// the real transaction test is in tst_QSqlDatabase
|
|
void tst_QSqlQuery::transaction()
|
|
{
|
|
// query based transaction is not really possible with Qt
|
|
QSKIP( "only tested manually by trained staff");
|
|
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if ( !db.driver()->hasFeature( QSqlDriver::Transactions ) )
|
|
QSKIP( "DBMS not transaction capable");
|
|
|
|
// this is the standard SQL
|
|
QString startTransactionStr( "start transaction" );
|
|
|
|
if (dbType == QSqlDriver::MySqlServer)
|
|
startTransactionStr = "begin work";
|
|
|
|
QSqlQuery q( db );
|
|
|
|
QSqlQuery q2( db );
|
|
|
|
// test a working transaction
|
|
q.exec( startTransactionStr );
|
|
|
|
QVERIFY_SQL( q, exec( "insert into" + qtest + " values (40, 'VarChar40', 'Char40')" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from" + qtest + " where id = 40" ) );
|
|
|
|
QVERIFY( q.next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 40 );
|
|
|
|
QVERIFY_SQL( q, exec( "commit" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from" + qtest + " where id = 40" ) );
|
|
|
|
QVERIFY( q.next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 40 );
|
|
|
|
// test a rollback
|
|
q.exec( startTransactionStr );
|
|
|
|
QVERIFY_SQL( q, exec( "insert into" + qtest + " values (41, 'VarChar41', 'Char41')" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "select * from" + qtest + " where id = 41" ) );
|
|
|
|
QVERIFY( q.next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 41 );
|
|
|
|
if ( !q.exec( "rollback" ) ) {
|
|
if (dbType == QSqlDriver::MySqlServer) {
|
|
qDebug( "MySQL: %s", qPrintable(tst_Databases::printError( q.lastError() ) ));
|
|
QSKIP( "MySQL transaction failed "); //non-fatal
|
|
} else
|
|
QFAIL( "Could not rollback transaction: " + tst_Databases::printError( q.lastError() ) );
|
|
}
|
|
|
|
QVERIFY_SQL( q, exec( "select * from" + qtest + " where id = 41" ) );
|
|
|
|
QVERIFY( q.next() == false );
|
|
|
|
// test concurrent access
|
|
q.exec( startTransactionStr );
|
|
QVERIFY_SQL( q, exec( "insert into" + qtest + " values (42, 'VarChar42', 'Char42')" ) );
|
|
QVERIFY_SQL( q, exec( "select * from" + qtest + " where id = 42" ) );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 42 );
|
|
|
|
QVERIFY_SQL( q2, exec( "select * from" + qtest + " where id = 42" ) );
|
|
|
|
if ( q2.next() )
|
|
qDebug("DBMS '%s' doesn't support query based transactions with concurrent access",
|
|
qPrintable(tst_Databases::dbToString( db )));
|
|
|
|
QVERIFY_SQL( q, exec( "commit" ) );
|
|
|
|
QVERIFY_SQL( q2, exec( "select * from" + qtest + " where id = 42" ) );
|
|
|
|
QVERIFY( q2.next() );
|
|
|
|
QCOMPARE( q2.value( 0 ).toInt(), 42 );
|
|
}
|
|
|
|
void tst_QSqlQuery::joins()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
const QString qtestj1(qTableName("qtestj1", __FILE__, db)), qtestj2(qTableName("qtestj2", __FILE__, db));
|
|
|
|
if (dbType == QSqlDriver::Oracle || dbType == QSqlDriver::Sybase
|
|
|| dbType == QSqlDriver::Interbase || db.driverName().startsWith("QODBC"))
|
|
// Oracle broken beyond recognition - cannot outer join on more than
|
|
// one table.
|
|
QSKIP( "DBMS cannot understand standard SQL");
|
|
|
|
QSqlQuery q( db );
|
|
|
|
QVERIFY_SQL( q, exec( "create table " + qtestj1 + " (id1 int, id2 int)" ) );
|
|
QVERIFY_SQL( q, exec( "create table " + qtestj2 + " (id int, name varchar(20))" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtestj1 + " values (1, 1)" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtestj1 + " values (1, 2)" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtestj2 + " values(1, 'trenton')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + qtestj2 + " values(2, 'marius')" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "select qtestj1.id1, qtestj1.id2, qtestj2.id, qtestj2.name, qtestj3.id, qtestj3.name "
|
|
"from " + qtestj1 + " qtestj1 left outer join " + qtestj2 +
|
|
" qtestj2 on (qtestj1.id1 = qtestj2.id) "
|
|
"left outer join " + qtestj2 + " as qtestj3 on (qtestj1.id2 = qtestj3.id)" ) );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 1 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 2 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 3 ).toString(), QString( "trenton" ) );
|
|
QCOMPARE( q.value( 4 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 5 ).toString(), QString( "trenton" ) );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 1 ).toInt(), 2 );
|
|
QCOMPARE( q.value( 2 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 3 ).toString(), QString( "trenton" ) );
|
|
QCOMPARE( q.value( 4 ).toInt(), 2 );
|
|
QCOMPARE( q.value( 5 ).toString(), QString( "marius" ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::synonyms()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL( q, exec("select a.id, a.t_char, a.t_varchar from " + qtest + " a where a.id = 1") );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), QString( "Char1" ) );
|
|
QCOMPARE( q.value( 2 ).toString().trimmed(), QString( "VarChar1" ) );
|
|
|
|
QSqlRecord rec = q.record();
|
|
QCOMPARE(( int )rec.count(), 3 );
|
|
QCOMPARE( rec.field( 0 ).name().toLower(), QString( "id" ) );
|
|
QCOMPARE( rec.field( 1 ).name().toLower(), QString( "t_char" ) );
|
|
QCOMPARE( rec.field( 2 ).name().toLower(), QString( "t_varchar" ) );
|
|
}
|
|
|
|
// It doesn't make sense to split this into several tests
|
|
void tst_QSqlQuery::prepare_bind_exec()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
const QString qtest_prepare(qTableName("qtest_prepare", __FILE__, db));
|
|
|
|
if (dbType == QSqlDriver::DB2)
|
|
QSKIP("Needs someone with more Unicode knowledge than I have to fix");
|
|
|
|
{
|
|
// new scope for SQLITE
|
|
static const QString utf8str = QString::fromUtf8( "काचं शक्नोम्यत्तुम् । नोपहिनस्ति माम् ॥" );
|
|
|
|
static const QString values[6] = { "Harry", "Trond", "Mark", "Ma?rk", "?", ":id" };
|
|
|
|
bool useUnicode = db.driver()->hasFeature( QSqlDriver::Unicode );
|
|
|
|
QSqlQuery q( db );
|
|
QString createQuery;
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::PostgreSQL)
|
|
QVERIFY_SQL( q, exec("set client_min_messages='warning'"));
|
|
|
|
if (dbType == QSqlDriver::MSSqlServer || dbType == QSqlDriver::Sybase)
|
|
createQuery = "create table " + qtest_prepare + " (id int primary key, name nvarchar(200) null, name2 nvarchar(200) null)";
|
|
else if (dbType == QSqlDriver::MySqlServer && useUnicode)
|
|
createQuery = "create table " + qtest_prepare + " (id int not null primary key, name varchar(200) character set utf8, name2 varchar(200) character set utf8)";
|
|
else
|
|
createQuery = "create table " + qtest_prepare + " (id int not null primary key, name varchar(200), name2 varchar(200))";
|
|
|
|
q.exec("drop table " + qtest_prepare);
|
|
QVERIFY_SQL( q, exec( createQuery ) );
|
|
|
|
QVERIFY( q.prepare( "insert into " + qtest_prepare + " (id, name) values (:id, :name)" ) );
|
|
int i;
|
|
|
|
for ( i = 0; i < 6; ++i ) {
|
|
q.bindValue( ":name", values[i] );
|
|
q.bindValue( ":id", i );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVariantList m = q.boundValues();
|
|
QCOMPARE(( int ) m.count(), 2 );
|
|
QCOMPARE(m.at(0).toInt(), i);
|
|
QCOMPARE(m.at(1).toString(), values[i]);
|
|
}
|
|
|
|
q.bindValue( ":id", 8 );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
if ( useUnicode ) {
|
|
q.bindValue( ":id", 7 );
|
|
q.bindValue( ":name", utf8str );
|
|
QVERIFY_SQL( q, exec() );
|
|
}
|
|
|
|
QVERIFY_SQL( q, exec( "SELECT * FROM " + qtest_prepare + " order by id" ) );
|
|
|
|
for ( i = 0; i < 6; ++i ) {
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), i );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), values[ i ] );
|
|
}
|
|
|
|
if ( useUnicode ) {
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 7 );
|
|
QCOMPARE( q.value( 1 ).toString(), utf8str );
|
|
}
|
|
|
|
QVERIFY_SQL( q, next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 8 );
|
|
QCOMPARE( q.value( 1 ).toString(), values[5] );
|
|
|
|
QVERIFY( q.prepare( "insert into " + qtest_prepare + " (id, name) values (:id, 'Bart')" ) );
|
|
q.bindValue( ":id", 99 );
|
|
QVERIFY_SQL( q, exec() );
|
|
q.bindValue( ":id", 100 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.exec( "select * from " + qtest_prepare + " where id > 98 order by id" ) );
|
|
|
|
for ( i = 99; i <= 100; ++i ) {
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), i );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), QString( "Bart" ) );
|
|
}
|
|
|
|
/*** SELECT stuff ***/
|
|
QVERIFY( q.prepare( "select * from " + qtest_prepare + " where id = :id" ) );
|
|
|
|
for ( i = 0; i < 6; ++i ) {
|
|
q.bindValue( ":id", i );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), i );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), values[ i ] );
|
|
QSqlRecord rInf = q.record();
|
|
QCOMPARE(( int )rInf.count(), 3 );
|
|
QCOMPARE( rInf.field( 0 ).name().toUpper(), QString( "ID" ) );
|
|
QCOMPARE( rInf.field( 1 ).name().toUpper(), QString( "NAME" ) );
|
|
QVERIFY( !q.next() );
|
|
}
|
|
|
|
QVERIFY_SQL( q, exec( "DELETE FROM " + qtest_prepare ) );
|
|
|
|
/*** Below we test QSqlQuery::boundValues() with position arguments.
|
|
* Due to the fact that the name of a positional argument is not
|
|
* specified by the Qt docs, we test that the QList contains
|
|
* the correct values in the same order as QSqlResult::boundValueName
|
|
* returns since it should be in insertion order (i.e. field order). ***/
|
|
QVERIFY( q.prepare( "insert into " + qtest_prepare + " (id, name) values (?, ?)" ) );
|
|
q.bindValue( 0, 0 );
|
|
q.bindValue( 1, values[ 0 ] );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 0);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[0]);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 0);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[0]);
|
|
|
|
q.addBindValue( 1 );
|
|
q.addBindValue( values[ 1 ] );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 1);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[1]);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 1);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[1]);
|
|
|
|
q.addBindValue( 2 );
|
|
q.addBindValue( values[ 2 ] );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 2);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[2]);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 2);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[2]);
|
|
|
|
q.addBindValue( 3 );
|
|
q.addBindValue( values[ 3 ] );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 3);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[3]);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 3);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[3]);
|
|
|
|
q.addBindValue( 4 );
|
|
q.addBindValue( values[ 4 ] );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 4);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[4]);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 4);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[4]);
|
|
|
|
q.bindValue( 1, values[ 5 ] );
|
|
q.bindValue( 0, 5 );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 5);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[5]);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 5);
|
|
QCOMPARE(q.boundValues().at(1).toString(), values[5]);
|
|
|
|
q.bindValue( 0, 6 );
|
|
q.bindValue( 1, QString() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 6);
|
|
QCOMPARE(q.boundValues().at(1).toString(), QString());
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.boundValues().size(), 2 );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 6);
|
|
QCOMPARE(q.boundValues().at(1).toString(), QString());
|
|
|
|
if ( db.driver()->hasFeature( QSqlDriver::Unicode ) ) {
|
|
q.bindValue( 0, 7 );
|
|
q.bindValue( 1, utf8str );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 7);
|
|
QCOMPARE(q.boundValues().at(1).toString(), utf8str);
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE(q.boundValues().at(0).toInt(), 7);
|
|
QCOMPARE(q.boundValues().at(1).toString(), utf8str);
|
|
}
|
|
|
|
QVERIFY_SQL( q, exec( "SELECT * FROM " + qtest_prepare + " order by id" ) );
|
|
|
|
for ( i = 0; i < 6; ++i ) {
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), i );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), values[ i ] );
|
|
}
|
|
|
|
QVERIFY( q.next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 6 );
|
|
QVERIFY( q.value( 1 ).toString().isEmpty() );
|
|
|
|
if ( useUnicode ) {
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 7 );
|
|
QCOMPARE( q.value( 1 ).toString(), utf8str );
|
|
}
|
|
|
|
QVERIFY( q.prepare( "insert into " + qtest_prepare + " (id, name) values (?, 'Bart')" ) );
|
|
|
|
q.bindValue( 0, 99 );
|
|
QVERIFY_SQL( q, exec() );
|
|
q.addBindValue( 100 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.exec( "select * from " + qtest_prepare + " where id > 98 order by id" ) );
|
|
|
|
for ( i = 99; i <= 100; ++i ) {
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), i );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), QString( "Bart" ) );
|
|
}
|
|
|
|
/* insert a duplicate id and make sure the db bails out */
|
|
QVERIFY( q.prepare( "insert into " + qtest_prepare + " (id, name) values (?, ?)" ) );
|
|
|
|
q.addBindValue( 99 );
|
|
|
|
q.addBindValue( "something silly" );
|
|
|
|
QVERIFY( !q.exec() );
|
|
|
|
QVERIFY( q.lastError().isValid() );
|
|
|
|
QVERIFY( !q.isActive() );
|
|
|
|
QVERIFY( q.prepare( "insert into " + qtest_prepare + " (id, name, name2) values (:id, :name, :name)" ) );
|
|
for ( i = 101; i < 103; ++i ) {
|
|
q.bindValue( ":id", i );
|
|
q.bindValue( ":name", "name" );
|
|
QVERIFY( q.exec() );
|
|
}
|
|
|
|
// Test for QTBUG-6420
|
|
QVERIFY( q.exec( "select * from " + qtest_prepare + " where id > 100 order by id" ) );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value(0).toInt(), 101 );
|
|
QCOMPARE( q.value(1).toString(), QString("name") );
|
|
QCOMPARE( q.value(2).toString(), QString("name") );
|
|
|
|
// Test that duplicated named placeholders before the next unique one works correctly - QTBUG-65150
|
|
QVERIFY(q.prepare("insert into " + qtest_prepare + " (id, name, name2) values (:id, :id, :name)"));
|
|
for (i = 104; i < 106; ++i) {
|
|
q.bindValue(":id", i);
|
|
q.bindValue(":name", "name");
|
|
QVERIFY(q.exec());
|
|
}
|
|
QVERIFY(q.exec("select * from " + qtest_prepare + " where id > 103 order by id"));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 104);
|
|
QCOMPARE(q.value(1).toString(), QString("104"));
|
|
QCOMPARE(q.value(2).toString(), QString("name"));
|
|
|
|
// Test that duplicated named placeholders in any order
|
|
QVERIFY(q.prepare("insert into " + qtest_prepare + " (id, name, name2) values (:id, :name, :id)"));
|
|
for (i = 107; i < 109; ++i) {
|
|
q.bindValue(":id", i);
|
|
q.bindValue(":name", "name");
|
|
QVERIFY(q.exec());
|
|
}
|
|
QVERIFY(q.exec("select * from " + qtest_prepare + " where id > 106 order by id"));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 107);
|
|
QCOMPARE(q.value(1).toString(), QString("name"));
|
|
QCOMPARE(q.value(2).toString(), QString("107"));
|
|
|
|
// Test just duplicated placeholders
|
|
QVERIFY(q.prepare("insert into " + qtest_prepare + " (id, name, name2) values (110, :name, :name)"));
|
|
q.bindValue(":name", "name");
|
|
QVERIFY_SQL(q, exec());
|
|
QVERIFY(q.exec("select * from " + qtest_prepare + " where id > 109 order by id"));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 110);
|
|
QCOMPARE(q.value(1).toString(), QString("name"));
|
|
QCOMPARE(q.value(2).toString(), QString("name"));
|
|
} // end of SQLite scope
|
|
}
|
|
|
|
void tst_QSqlQuery::prepared_select()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL( q, prepare( "select a.id, a.t_char, a.t_varchar from " + qtest + " a where a.id = ?" ) );
|
|
|
|
q.bindValue( 0, 1 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.at(), ( int )QSql::BeforeFirstRow );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
|
|
q.bindValue( 0, 2 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.at(), ( int )QSql::BeforeFirstRow );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 2 );
|
|
|
|
q.bindValue( 0, 3 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QCOMPARE( q.at(), ( int )QSql::BeforeFirstRow );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 3 );
|
|
|
|
QVERIFY_SQL( q, prepare( "select a.id, a.t_char, a.t_varchar from " + qtest + " a where a.id = ?" ) );
|
|
QCOMPARE( q.at(), ( int )QSql::BeforeFirstRow );
|
|
QVERIFY( !q.first() );
|
|
}
|
|
|
|
void tst_QSqlQuery::sqlServerLongStrings()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType != QSqlDriver::MSSqlServer)
|
|
QSKIP( "SQL Server specific test");
|
|
|
|
QSqlQuery q( db );
|
|
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + qTableName("qtest_longstr", __FILE__, db) + " (id int primary key, longstring ntext)"));
|
|
|
|
QVERIFY_SQL(q, prepare("INSERT INTO " + qTableName("qtest_longstr", __FILE__, db) + " VALUES (?, ?)"));
|
|
|
|
q.addBindValue( 0 );
|
|
|
|
q.addBindValue( QString::fromLatin1( "bubu" ) );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QString testStr;
|
|
|
|
testStr.fill( QLatin1Char( 'a' ), 85000 );
|
|
|
|
q.addBindValue( 1 );
|
|
|
|
q.addBindValue( testStr );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY_SQL(q, exec("select * from " + qTableName( "qtest_longstr", __FILE__, db)));
|
|
|
|
QVERIFY_SQL( q, next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 0 );
|
|
|
|
QCOMPARE( q.value( 1 ).toString(), QString::fromLatin1( "bubu" ) );
|
|
|
|
QVERIFY_SQL( q, next() );
|
|
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
|
|
QCOMPARE( q.value( 1 ).toString(), testStr );
|
|
}
|
|
|
|
void tst_QSqlQuery::invalidQuery()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
QSqlQuery q( db );
|
|
|
|
QVERIFY( !q.exec() );
|
|
|
|
QVERIFY( !q.exec( "blahfasel" ) );
|
|
QVERIFY( q.lastError().type() != QSqlError::NoError );
|
|
QVERIFY( !q.next() );
|
|
QVERIFY( !q.isActive() );
|
|
|
|
if (dbType != QSqlDriver::Oracle && dbType != QSqlDriver::DB2 && !db.driverName().startsWith("QODBC")) {
|
|
// oracle and db2 just prepares everything without complaining
|
|
if ( db.driver()->hasFeature( QSqlDriver::PreparedQueries ) )
|
|
QVERIFY( !q.prepare( "blahfasel" ) );
|
|
}
|
|
|
|
QVERIFY( !q.exec() );
|
|
|
|
QVERIFY( !q.isActive() );
|
|
QVERIFY( !q.next() );
|
|
}
|
|
|
|
void tst_QSqlQuery::batchExec()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
const QString tableName = qTableName("qtest_batch", __FILE__, db);
|
|
tst_Databases::safeDropTable(db, tableName);
|
|
|
|
const auto dbType = tst_Databases::getDatabaseType(db);
|
|
QString timeStampString;
|
|
if (dbType == QSqlDriver::Interbase)
|
|
timeStampString = QLatin1String("TIMESTAMP");
|
|
else
|
|
timeStampString = QLatin1String("TIMESTAMP (3)");
|
|
|
|
QVERIFY_SQL(q, exec(QStringLiteral("create table ") + tableName +
|
|
QStringLiteral(" (id int, name varchar(20), dt date, num numeric(8, 4), "
|
|
"dtstamp ") + timeStampString +
|
|
QStringLiteral(", extraId int, extraName varchar(20))")));
|
|
|
|
const QVariantList intCol = { 1, 2, QVariant(QVariant::Int) };
|
|
const QVariantList charCol = { QStringLiteral("harald"), QStringLiteral("boris"),
|
|
QVariant(QVariant::String) };
|
|
const QDateTime currentDateTime = QDateTime(QDateTime::currentDateTime());
|
|
const QVariantList dateCol = { currentDateTime.date(), currentDateTime.date().addDays(-1),
|
|
QVariant(QVariant::Date) };
|
|
const QVariantList numCol = { 2.3, 3.4, QVariant(QVariant::Double) };
|
|
const QVariantList timeStampCol = { currentDateTime, currentDateTime.addDays(-1),
|
|
QVariant(QVariant::DateTime) };
|
|
|
|
// Test with positional placeholders
|
|
QVERIFY_SQL(q, prepare(QStringLiteral("insert into ") + tableName +
|
|
QStringLiteral(" (id, name, dt, num, dtstamp, extraId, extraName) values "
|
|
"(?, ?, ?, ?, ?, ?, ?)")));
|
|
q.addBindValue(intCol);
|
|
q.addBindValue( charCol );
|
|
q.addBindValue( dateCol );
|
|
q.addBindValue( numCol );
|
|
q.addBindValue(timeStampCol);
|
|
q.addBindValue(intCol);
|
|
q.addBindValue(charCol);
|
|
|
|
QVERIFY_SQL( q, execBatch() );
|
|
QVERIFY_SQL(q, exec(QStringLiteral("select id, name, dt, num, dtstamp, "
|
|
"extraId, extraName from ") + tableName));
|
|
|
|
for (int i = 0; i < intCol.size(); ++i) {
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), intCol.at(i));
|
|
QCOMPARE(q.value(1).toString(), charCol.at(i));
|
|
QCOMPARE(q.value(2).toDate(), dateCol.at(i));
|
|
QCOMPARE(q.value(3).toDouble(), numCol.at(i));
|
|
if (tst_Databases::getDatabaseType(db) == QSqlDriver::MySqlServer && timeStampCol.at(i).isNull()) {
|
|
QEXPECT_FAIL("", "This appears to be a bug in MySQL as it converts null datetimes to the "
|
|
"current datetime for a timestamp field", Continue);
|
|
}
|
|
QCOMPARE(q.value(4).toDateTime(), timeStampCol.at(i));
|
|
QCOMPARE(q.value(5).toInt(), intCol.at(i));
|
|
QCOMPARE(q.value(6).toString(), charCol.at(i));
|
|
}
|
|
|
|
// Empty table ready for retesting with duplicated named placeholders
|
|
QVERIFY_SQL(q, exec(QStringLiteral("delete from ") + tableName));
|
|
QVERIFY_SQL(q, prepare(QStringLiteral("insert into ") + tableName +
|
|
QStringLiteral(" (id, name, dt, num, dtstamp, extraId, extraName) "
|
|
"values (:id, :name, :dt, :num, :dtstamp, :id, :name)")));
|
|
q.bindValue(":id", intCol);
|
|
q.bindValue(":name", charCol);
|
|
q.bindValue(":dt", dateCol);
|
|
q.bindValue(":num", numCol);
|
|
q.bindValue(":dtstamp", timeStampCol);
|
|
|
|
QVERIFY_SQL(q, execBatch());
|
|
QVERIFY_SQL(q, exec(QStringLiteral("select id, name, dt, num, dtstamp, extraId, extraName from ") +
|
|
tableName));
|
|
|
|
for (int i = 0; i < intCol.size(); ++i) {
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), intCol.at(i));
|
|
QCOMPARE(q.value(1).toString(), charCol.at(i));
|
|
QCOMPARE(q.value(2).toDate(), dateCol.at(i));
|
|
QCOMPARE(q.value(3).toDouble(), numCol.at(i));
|
|
if (tst_Databases::getDatabaseType(db) == QSqlDriver::MySqlServer && timeStampCol.at(i).isNull()) {
|
|
QEXPECT_FAIL("", "This appears to be a bug in MySQL as it converts null datetimes to the "
|
|
"current datetime for a timestamp field", Continue);
|
|
}
|
|
QCOMPARE(q.value(4).toDateTime(), timeStampCol.at(i));
|
|
QCOMPARE(q.value(5).toInt(), intCol.at(i));
|
|
QCOMPARE(q.value(6).toString(), charCol.at(i));
|
|
}
|
|
|
|
// Only test the prepared stored procedure approach where the driver has support
|
|
// for batch operations as this will not work without it
|
|
if (db.driver()->hasFeature(QSqlDriver::BatchOperations)) {
|
|
const QString procName = qTableName("qtest_batch_proc", __FILE__, db);
|
|
QVERIFY_SQL(q, exec("create or replace procedure " + procName +
|
|
" (x in timestamp, y out timestamp) is\n"
|
|
"begin\n"
|
|
" y := x;\n"
|
|
"end;\n"));
|
|
QVERIFY(q.prepare("call " + procName + "(?, ?)"));
|
|
q.addBindValue(timeStampCol, QSql::In);
|
|
QVariantList emptyDateTimes;
|
|
emptyDateTimes.reserve(timeStampCol.size());
|
|
for (int i = 0; i < timeStampCol.size(); i++)
|
|
emptyDateTimes << QVariant(QDateTime());
|
|
q.addBindValue(emptyDateTimes, QSql::Out);
|
|
QVERIFY_SQL(q, execBatch());
|
|
QCOMPARE(q.boundValue(1).toList(), timeStampCol);
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_43874()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery q(db);
|
|
const QString tableName = qTableName("bug43874", __FILE__, db);
|
|
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + " (id INT)"));
|
|
QVERIFY_SQL(q, prepare("INSERT INTO " + tableName + " (id) VALUES (?)"));
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
QVariantList ids;
|
|
ids << i;
|
|
q.addBindValue(ids);
|
|
QVERIFY_SQL(q, execBatch());
|
|
}
|
|
QVERIFY_SQL(q, exec("SELECT id FROM " + tableName + " ORDER BY id"));
|
|
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 0);
|
|
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
}
|
|
|
|
void tst_QSqlQuery::oraArrayBind()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
if ( !db.driver()->hasFeature( QSqlDriver::BatchOperations ) )
|
|
QSKIP( "Database can't do BatchOperations");
|
|
|
|
QSqlQuery q( db );
|
|
|
|
QVERIFY_SQL( q, exec( "CREATE OR REPLACE PACKAGE ora_array_test "
|
|
"IS "
|
|
"TYPE names_type IS TABLE OF VARCHAR(64) NOT NULL INDEX BY BINARY_INTEGER; "
|
|
"names_tab names_type; "
|
|
"PROCEDURE set_name(name_in IN VARCHAR2, row_in in INTEGER); "
|
|
"PROCEDURE get_name(row_in IN INTEGER, str_out OUT VARCHAR2); "
|
|
"PROCEDURE get_table(tbl OUT names_type); "
|
|
"PROCEDURE set_table(tbl IN names_type); "
|
|
"END ora_array_test; " ) );
|
|
|
|
QVERIFY_SQL( q, exec( "CREATE OR REPLACE PACKAGE BODY ora_array_test "
|
|
"IS "
|
|
"PROCEDURE set_name(name_in IN VARCHAR2, row_in in INTEGER) "
|
|
"IS "
|
|
"BEGIN "
|
|
"names_tab(row_in) := name_in; "
|
|
"END set_name; "
|
|
|
|
"PROCEDURE get_name(row_in IN INTEGER, str_out OUT VARCHAR2) "
|
|
"IS "
|
|
"BEGIN "
|
|
"str_out := names_tab(row_in); "
|
|
"END get_name; "
|
|
|
|
"PROCEDURE get_table(tbl OUT names_type) "
|
|
"IS "
|
|
"BEGIN "
|
|
"tbl:=names_tab; "
|
|
"END get_table; "
|
|
|
|
"PROCEDURE set_table(tbl IN names_type) "
|
|
"IS "
|
|
"BEGIN "
|
|
"names_tab := tbl; "
|
|
"END set_table; "
|
|
"END ora_array_test; " ) );
|
|
|
|
QVariantList list;
|
|
|
|
list << QString( "lorem" ) << QString( "ipsum" ) << QString( "dolor" ) << QString( "sit" ) << QString( "amet" );
|
|
|
|
QVERIFY_SQL( q, prepare( "BEGIN "
|
|
"ora_array_test.set_table(?); "
|
|
"END;" ) );
|
|
|
|
q.bindValue( 0, list, QSql::In );
|
|
|
|
QVERIFY_SQL( q, execBatch( QSqlQuery::ValuesAsColumns ) );
|
|
|
|
QVERIFY_SQL( q, prepare( "BEGIN "
|
|
"ora_array_test.get_table(?); "
|
|
"END;" ) );
|
|
|
|
list.clear();
|
|
|
|
list << QString( 64,' ' ) << QString( 64,' ' ) << QString( 64,' ' ) << QString( 64,' ' ) << QString( 64,' ' );
|
|
|
|
q.bindValue( 0, list, QSql::Out );
|
|
|
|
QVERIFY_SQL( q, execBatch( QSqlQuery::ValuesAsColumns ) );
|
|
|
|
QVariantList out_list = q.boundValue( 0 ).toList();
|
|
|
|
QCOMPARE( out_list.at( 0 ).toString(), QString( "lorem" ) );
|
|
|
|
QCOMPARE( out_list.at( 1 ).toString(), QString( "ipsum" ) );
|
|
|
|
QCOMPARE( out_list.at( 2 ).toString(), QString( "dolor" ) );
|
|
|
|
QCOMPARE( out_list.at( 3 ).toString(), QString( "sit" ) );
|
|
|
|
QCOMPARE( out_list.at( 4 ).toString(), QString( "amet" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "DROP PACKAGE ora_array_test" ) );
|
|
}
|
|
|
|
/*
|
|
Tests that QSqlDatabase::record and QSqlQuery::record returns the same thing
|
|
otherwise our models get confused.
|
|
*/
|
|
void tst_QSqlQuery::record_sqlite()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
|
|
QVERIFY_SQL(q, exec("create table " + qTableName("record_sqlite", __FILE__, db) + "(id integer primary key, name varchar, title int)"));
|
|
|
|
QSqlRecord rec = db.record(qTableName("record_sqlite", __FILE__, db));
|
|
|
|
QCOMPARE( rec.count(), 3 );
|
|
QCOMPARE( rec.field( 0 ).metaType().id(), QVariant::Int );
|
|
QCOMPARE( rec.field( 1 ).metaType().id(), QVariant::String );
|
|
QCOMPARE( rec.field( 2 ).metaType().id(), QVariant::Int );
|
|
|
|
/* important - select from an empty table */
|
|
QVERIFY_SQL(q, exec("select id, name, title from " + qTableName("record_sqlite", __FILE__, db)));
|
|
|
|
rec = q.record();
|
|
QCOMPARE( rec.count(), 3 );
|
|
QCOMPARE( rec.field( 0 ).metaType().id(), QVariant::Int );
|
|
QCOMPARE( rec.field( 1 ).metaType().id(), QVariant::String );
|
|
QCOMPARE( rec.field( 2 ).metaType().id(), QVariant::Int );
|
|
}
|
|
|
|
void tst_QSqlQuery::oraLong()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
|
|
QString aLotOfText( 127000, QLatin1Char( 'H' ) );
|
|
|
|
QVERIFY_SQL(q, exec("create table " + qTableName("qtest_longstr", __FILE__, db) + " (id int primary key, astr long)"));
|
|
QVERIFY_SQL(q, prepare("insert into " + qTableName("qtest_longstr", __FILE__, db) + " (id, astr) values (?, ?)"));
|
|
q.addBindValue( 1 );
|
|
q.addBindValue( aLotOfText );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY_SQL(q, exec("select id,astr from " + qTableName("qtest_longstr", __FILE__, db)));
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 1 ).toString(), aLotOfText );
|
|
}
|
|
|
|
void tst_QSqlQuery::execErrorRecovery()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
|
|
const QString tbl = qTableName("qtest_exerr", __FILE__, db);
|
|
q.exec("drop table " + tbl);
|
|
QVERIFY_SQL(q, exec("create table " + tbl + " (id int not null primary key)"));
|
|
QVERIFY_SQL(q, prepare("insert into " + tbl + " values (?)" ));
|
|
|
|
q.addBindValue( 1 );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
q.addBindValue( 1 ); // binding the same pkey - should fail
|
|
QVERIFY( !q.exec() );
|
|
|
|
q.addBindValue( 2 ); // this should work again
|
|
QVERIFY_SQL( q, exec() );
|
|
}
|
|
|
|
void tst_QSqlQuery::lastInsertId()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
if ( !db.driver()->hasFeature( QSqlDriver::LastInsertId ) )
|
|
QSKIP( "Database doesn't support lastInsertId");
|
|
|
|
QSqlQuery q( db );
|
|
|
|
// PostgreSQL >= 8.1 relies on lastval() which does not work if a value is
|
|
// manually inserted to the serial field, so we create a table specifically
|
|
if (tst_Databases::getDatabaseType(db) == QSqlDriver::PostgreSQL) {
|
|
const auto tst_lastInsertId = qTableName("tst_lastInsertId", __FILE__, db);
|
|
tst_Databases::safeDropTable(db, tst_lastInsertId);
|
|
QVERIFY_SQL(q, exec(QStringLiteral("create table ") + tst_lastInsertId +
|
|
QStringLiteral(" (id serial not null, t_varchar "
|
|
"varchar(20), t_char char(20), primary key(id))")));
|
|
QVERIFY_SQL(q, exec(QStringLiteral("insert into ") + tst_lastInsertId +
|
|
QStringLiteral(" (t_varchar, t_char) values "
|
|
"('VarChar41', 'Char41')")));
|
|
} else {
|
|
QVERIFY_SQL(q, exec(QStringLiteral("insert into ") + qtest +
|
|
QStringLiteral(" values (41, 'VarChar41', 'Char41')")));
|
|
}
|
|
QVariant v = q.lastInsertId();
|
|
|
|
QVERIFY( v.isValid() );
|
|
}
|
|
|
|
void tst_QSqlQuery::lastQuery()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QString sql = "select * from " + qtest;
|
|
QVERIFY_SQL( q, exec( sql ) );
|
|
QCOMPARE( q.lastQuery(), sql );
|
|
QCOMPARE( q.executedQuery(), sql );
|
|
}
|
|
|
|
void tst_QSqlQuery::lastQueryTwoQueries()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery q(db);
|
|
|
|
QString sql = QLatin1String("select * from ") + qtest;
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QCOMPARE(q.lastQuery(), sql);
|
|
QCOMPARE(q.executedQuery(), sql);
|
|
|
|
sql = QLatin1String("select id from ") + qtest;
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QCOMPARE(q.lastQuery(), sql);
|
|
QCOMPARE(q.executedQuery(), sql);
|
|
}
|
|
|
|
void tst_QSqlQuery::psql_bindWithDoubleColonCastOperator()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
const QString tablename(qTableName("bindtest", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
|
|
QVERIFY_SQL( q, exec( "create table " + tablename + " (id1 int, id2 int, id3 int, fld1 int, fld2 int)" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + tablename + " values (1, 2, 3, 10, 5)" ) );
|
|
|
|
QVERIFY_SQL( q, prepare( "select sum((fld1 - fld2)::int) from " + tablename + " where id1 = :myid1 and id2 =:myid2 and id3=:myid3" ) );
|
|
q.bindValue( ":myid1", 1 );
|
|
q.bindValue( ":myid2", 2 );
|
|
q.bindValue( ":myid3", 3 );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY_SQL( q, next() );
|
|
|
|
// the positional placeholders are converted to named placeholders in executedQuery()
|
|
if (db.driver()->hasFeature(QSqlDriver::PreparedQueries))
|
|
QCOMPARE(q.executedQuery(), QString("select sum((fld1 - fld2)::int) from " + tablename + " where id1 = :myid1 and id2 =:myid2 and id3=:myid3"));
|
|
else
|
|
QCOMPARE(q.executedQuery(), QString("select sum((fld1 - fld2)::int) from " + tablename + " where id1 = 1 and id2 =2 and id3=3"));
|
|
}
|
|
|
|
void tst_QSqlQuery::psql_specialFloatValues()
|
|
{
|
|
if (!std::numeric_limits<float>::has_quiet_NaN)
|
|
QSKIP("Platform does not have quiet_NaN");
|
|
if (!std::numeric_limits<float>::has_infinity)
|
|
QSKIP("Platform does not have infinity");
|
|
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery query(db);
|
|
const QString tableName = qTableName("floattest", __FILE__, db);
|
|
QVERIFY_SQL( query, exec("create table " + tableName + " (value float)" ) );
|
|
QVERIFY_SQL(query, prepare("insert into " + tableName + " values(:value)") );
|
|
|
|
QVariantList data;
|
|
data << QVariant(double(42.42))
|
|
<< QVariant(std::numeric_limits<double>::quiet_NaN())
|
|
<< QVariant(std::numeric_limits<double>::infinity())
|
|
<< QVariant(float(42.42))
|
|
<< QVariant(std::numeric_limits<float>::quiet_NaN())
|
|
<< QVariant(std::numeric_limits<float>::infinity());
|
|
|
|
foreach (const QVariant &v, data) {
|
|
query.bindValue(":value", v);
|
|
QVERIFY_SQL( query, exec() );
|
|
}
|
|
|
|
QVERIFY_SQL( query, exec("drop table " + tableName) );
|
|
}
|
|
|
|
/* For task 157397: Using QSqlQuery with an invalid QSqlDatabase
|
|
does not set the last error of the query.
|
|
This test function will output some warnings, that's ok.
|
|
*/
|
|
void tst_QSqlQuery::queryOnInvalidDatabase()
|
|
{
|
|
{
|
|
QTest::ignoreMessage( QtWarningMsg, "QSqlDatabase: INVALID driver not loaded" );
|
|
QSqlDatabase db = QSqlDatabase::addDatabase( "INVALID", "invalidConnection" );
|
|
QVERIFY2( db.lastError().isValid(),
|
|
qPrintable( QString( "db.lastError().isValid() should be true!" ) ) );
|
|
|
|
QTest::ignoreMessage( QtWarningMsg, "QSqlQuery::exec: database not open" );
|
|
QSqlQuery query( "SELECT 1 AS ID", db );
|
|
QVERIFY2( query.lastError().isValid(),
|
|
qPrintable( QString( "query.lastError().isValid() should be true!" ) ) );
|
|
}
|
|
|
|
QSqlDatabase::removeDatabase( "invalidConnection" );
|
|
|
|
{
|
|
QSqlDatabase db = QSqlDatabase::database( "this connection does not exist" );
|
|
QTest::ignoreMessage( QtWarningMsg, "QSqlQuery::exec: database not open" );
|
|
QSqlQuery query( "SELECT 1 AS ID", db );
|
|
QVERIFY2( query.lastError().isValid(),
|
|
qPrintable( QString( "query.lastError().isValid() should be true!" ) ) );
|
|
}
|
|
}
|
|
|
|
/* For task 159138: Error on instantiating a sql-query before explicitly
|
|
opening the database. This is something we don't support, so this isn't
|
|
really a bug. However some of the drivers are nice enough to support it.
|
|
*/
|
|
void tst_QSqlQuery::createQueryOnClosedDatabase()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
// Only supported by these drivers
|
|
|
|
if (dbType != QSqlDriver::PostgreSQL && dbType != QSqlDriver::Oracle
|
|
&& dbType != QSqlDriver::MySqlServer && dbType != QSqlDriver::DB2)
|
|
QSKIP( "Test is specific for PostgreSQL, Oracle, MySql and DB2");
|
|
|
|
db.close();
|
|
|
|
QSqlQuery q( db );
|
|
db.open();
|
|
QVERIFY_SQL( q, exec( QString( "select * from %1 where id = 1" ).arg( qtest ) ) );
|
|
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), QLatin1String( "VarChar1" ) );
|
|
QCOMPARE( q.value( 2 ).toString().trimmed(), QLatin1String( "Char1" ) );
|
|
|
|
db.close();
|
|
QVERIFY2( !q.exec( QString( "select * from %1 where id = 1" ).arg( qtest ) ),
|
|
qPrintable( QString( "This can't happen! The query should not have been executed!" ) ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::reExecutePreparedForwardOnlyQuery()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
q.setForwardOnly( true );
|
|
|
|
QVERIFY_SQL( q, prepare( QString( "SELECT id, t_varchar, t_char FROM %1 WHERE id = :id" ).arg( qtest ) ) );
|
|
q.bindValue( ":id", 1 );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
// Do something, like iterate over the result, or skip to the end
|
|
QVERIFY_SQL( q, last() );
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
/* This was broken with SQLite because the cache size was set to 0 in the 2nd execute.
|
|
When forwardOnly is set we don't cahce the entire result, but we do cache the current row
|
|
but this requires the cache size to be equal to the column count.
|
|
*/
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QCOMPARE( q.value( 1 ).toString().trimmed(), QString( "VarChar1" ) );
|
|
QCOMPARE( q.value( 2 ).toString().trimmed(), QString( "Char1" ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::finish()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
QVERIFY_SQL( q, prepare( "SELECT id FROM " + qtest + " WHERE id = ?" ) );
|
|
|
|
int id = 4;
|
|
q.bindValue( 0, id );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.isActive() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), id );
|
|
|
|
q.finish();
|
|
QVERIFY( !q.isActive() ); // query is now inactive
|
|
QCOMPARE( q.boundValue( 0 ).toInt(), id ); // bound values are retained
|
|
|
|
QVERIFY_SQL( q, exec() ); // no prepare necessary
|
|
QVERIFY( q.isActive() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), id );
|
|
|
|
q.finish();
|
|
QVERIFY( !q.isActive() );
|
|
|
|
QVERIFY_SQL( q, exec( "SELECT id FROM " + qtest + " WHERE id = 1" ) );
|
|
QVERIFY( q.isActive() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1 );
|
|
QCOMPARE( q.record().count(), 1 );
|
|
}
|
|
|
|
void tst_QSqlQuery::sqlite_finish()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
if ( db.databaseName().startsWith( ':' ) )
|
|
QSKIP( "This test requires a database on the filesystem, not in-memory");
|
|
|
|
{
|
|
QSqlDatabase db2 = QSqlDatabase::addDatabase( "QSQLITE", "sqlite_finish_sqlite" );
|
|
db2.setDatabaseName( db.databaseName() );
|
|
QVERIFY_SQL( db2, open() );
|
|
|
|
const QString tableName(qTableName("qtest_lockedtable", __FILE__, db));
|
|
QSqlQuery q( db );
|
|
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
q.exec( "CREATE TABLE " + tableName + " (pk_id INTEGER PRIMARY KEY, whatever TEXT)" );
|
|
q.exec( "INSERT INTO " + tableName + " values(1, 'whatever')" );
|
|
q.exec( "INSERT INTO " + tableName + " values(2, 'whatever more')" );
|
|
|
|
// This creates a read-lock in the database
|
|
QVERIFY_SQL( q, exec( "SELECT * FROM " + tableName + " WHERE pk_id = 1 or pk_id = 2" ) );
|
|
QVERIFY_SQL( q, next() );
|
|
|
|
// The DELETE will fail because of the read-lock
|
|
QSqlQuery q2( db2 );
|
|
QVERIFY( !q2.exec( "DELETE FROM " + tableName + " WHERE pk_id=2" ) );
|
|
QCOMPARE( q2.numRowsAffected(), -1 );
|
|
|
|
// The DELETE will succeed now because finish() removes the lock
|
|
q.finish();
|
|
QVERIFY_SQL( q2, exec( "DELETE FROM " + tableName + " WHERE pk_id=2" ) );
|
|
QCOMPARE( q2.numRowsAffected(), 1 );
|
|
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
}
|
|
|
|
QSqlDatabase::removeDatabase( "sqlite_finish_sqlite" );
|
|
}
|
|
|
|
void tst_QSqlQuery::nextResult()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (!db.driver()->hasFeature(QSqlDriver::MultipleResultSets))
|
|
QSKIP("DBMS does not support multiple result sets");
|
|
|
|
QSqlQuery q( db );
|
|
const QString tableName(qTableName("more_results", __FILE__, db));
|
|
|
|
QVERIFY_SQL( q, exec( "CREATE TABLE " + tableName + " (id integer, text varchar(20), num numeric(6, 3), empty varchar(10));" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "INSERT INTO " + tableName + " VALUES(1, 'one', 1.1, '');" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "INSERT INTO " + tableName + " VALUES(2, 'two', 2.2, '');" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "INSERT INTO " + tableName + " VALUES(3, 'three', 3.3, '');" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "INSERT INTO " + tableName + " VALUES(4, 'four', 4.4, '');" ) );
|
|
|
|
QStringList tstStrings;
|
|
|
|
tstStrings << "one" << "two" << "three" << "four";
|
|
|
|
// Query that returns only one result set, nothing special about this
|
|
QVERIFY_SQL( q, exec( QString( "SELECT * FROM %1;" ).arg( tableName ) ) );
|
|
|
|
QVERIFY( q.next() ); // Move to first row of the result set
|
|
|
|
QVERIFY( !q.nextResult() ); // No more result sets are available
|
|
|
|
QVERIFY( !q.isActive() ); // So the query is no longer active
|
|
|
|
QVERIFY( !q.next() ); // ... and no data is available as the call
|
|
|
|
// to nextResult() discarded the result set
|
|
|
|
// Query that returns two result sets (batch sql)
|
|
// When working with multiple result sets SQL Server insists on non-scrollable cursors
|
|
if (db.driverName().startsWith("QODBC"))
|
|
q.setForwardOnly( true );
|
|
|
|
QVERIFY_SQL( q, exec( "SELECT id FROM " + tableName + "; SELECT text, num FROM " + tableName + ';' ) );
|
|
|
|
QCOMPARE( q.record().count(), 1 ); // Check that the meta data is as expected
|
|
|
|
QCOMPARE( q.record().field( 0 ).name().toUpper(), QString( "ID" ) );
|
|
|
|
QCOMPARE( q.record().field( 0 ).metaType().id(), QVariant::Int );
|
|
|
|
QVERIFY( q.nextResult() ); // Discards first result set and move to the next
|
|
|
|
QCOMPARE( q.record().count(), 2 ); // New meta data should be available
|
|
|
|
QCOMPARE( q.record().field( 0 ).name().toUpper(), QString( "TEXT" ) );
|
|
|
|
QCOMPARE( q.record().field( 0 ).metaType().id(), QVariant::String );
|
|
|
|
QCOMPARE( q.record().field( 1 ).name().toUpper(), QString( "NUM" ) );
|
|
QCOMPARE(q.record().field(1).metaType().id(), QVariant::Double);
|
|
|
|
QVERIFY( q.next() ); // Move to first row of the second result set
|
|
|
|
QFAIL_SQL(q, nextResult()); // No more result sets after this
|
|
|
|
QVERIFY( !q.isActive() ); // So the query is no longer active
|
|
|
|
QVERIFY( !q.next() ); // ... and no data is available as the call to
|
|
|
|
// nextResult() discarded the result set
|
|
|
|
// Query that returns one result set, a count of affected rows and then another result set
|
|
QString query1 = QString( "SELECT id, text, num, empty FROM %1 WHERE id <= 3" ).arg( tableName );
|
|
|
|
QString query2 = QString( "UPDATE %1 SET empty = 'Yatta!'" ).arg( tableName );
|
|
|
|
QString query3 = QString( "SELECT id, empty FROM %1 WHERE id <=2" ).arg( tableName );
|
|
|
|
QVERIFY_SQL( q, exec( QString( "%1; %2; %3;" ).arg( query1 ).arg( query2 ).arg( query3 ) ) );
|
|
|
|
// Check result set returned by first statement
|
|
QVERIFY( q.isSelect() ); // The first statement is a select
|
|
|
|
for ( int i = 0; i < 3; i++ ) {
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1+i );
|
|
QCOMPARE( q.value( 1 ).toString(), tstStrings.at( i ) );
|
|
QCOMPARE( q.value( 2 ).toDouble(), 1.1*( i+1 ) );
|
|
QVERIFY( q.value( 3 ).toString().isEmpty() );
|
|
}
|
|
|
|
QVERIFY_SQL( q, nextResult() );
|
|
|
|
QVERIFY( !q.isSelect() ); // The second statement isn't a SELECT
|
|
QVERIFY( !q.next() ); // ... so no result set is available
|
|
QCOMPARE( q.numRowsAffected(), 4 ); // 4 rows was affected by the UPDATE
|
|
|
|
// Check result set returned by third statement
|
|
QVERIFY_SQL( q, nextResult() );
|
|
QVERIFY( q.isSelect() ); // The third statement is a SELECT
|
|
|
|
for ( int i = 0; i < 2; i++ ) {
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), 1+i );
|
|
QCOMPARE( q.value( 1 ).toString(), QString( "Yatta!" ) );
|
|
}
|
|
|
|
// Stored procedure with multiple result sets
|
|
const QString procName(qTableName("proc_more_res", __FILE__, db));
|
|
|
|
if (dbType == QSqlDriver::PostgreSQL)
|
|
q.exec(QString("DROP FUNCTION %1(refcursor, refcursor);").arg(procName));
|
|
else
|
|
q.exec(QString("DROP PROCEDURE %1;").arg(procName));
|
|
|
|
if (dbType == QSqlDriver::MySqlServer)
|
|
QVERIFY_SQL( q, exec( QString( "CREATE PROCEDURE %1()"
|
|
"\nBEGIN"
|
|
"\nSELECT id, text FROM %2;"
|
|
"\nSELECT empty, num, text, id FROM %3;"
|
|
"\nEND" ).arg( procName ).arg( tableName ).arg( tableName ) ) );
|
|
else if (dbType == QSqlDriver::DB2)
|
|
QVERIFY_SQL( q, exec( QString( "CREATE PROCEDURE %1()"
|
|
"\nRESULT SETS 2"
|
|
"\nLANGUAGE SQL"
|
|
"\np1:BEGIN"
|
|
"\nDECLARE cursor1 CURSOR WITH RETURN FOR SELECT id, text FROM %2;"
|
|
"\nDECLARE cursor2 CURSOR WITH RETURN FOR SELECT empty, num, text, id FROM %3;"
|
|
"\nOPEN cursor1;"
|
|
"\nOPEN cursor2;"
|
|
"\nEND p1" ).arg( procName ).arg( tableName ).arg( tableName ) ) );
|
|
else if (dbType == QSqlDriver::PostgreSQL)
|
|
QVERIFY_SQL(q, exec(QString("CREATE FUNCTION %1(ref1 refcursor, ref2 refcursor)"
|
|
"\nRETURNS SETOF refcursor AS $$"
|
|
"\nBEGIN"
|
|
"\nOPEN ref1 FOR SELECT id, text FROM %2;"
|
|
"\nRETURN NEXT ref1;"
|
|
"\nOPEN ref2 FOR SELECT empty, num, text, id FROM %2;"
|
|
"\nRETURN NEXT ref2;"
|
|
"\nEND;"
|
|
"\n$$ LANGUAGE plpgsql").arg(procName).arg(tableName)));
|
|
else
|
|
QVERIFY_SQL( q, exec( QString( "CREATE PROCEDURE %1"
|
|
"\nAS"
|
|
"\nSELECT id, text FROM %2"
|
|
"\nSELECT empty, num, text, id FROM %3" ).arg( procName ).arg( tableName ).arg( tableName ) ) );
|
|
|
|
if (dbType == QSqlDriver::MySqlServer || dbType == QSqlDriver::DB2) {
|
|
q.setForwardOnly( true );
|
|
QVERIFY_SQL( q, exec( QString( "CALL %1()" ).arg( procName ) ) );
|
|
} else if (dbType == QSqlDriver::PostgreSQL) {
|
|
// Returning multiple result sets from PostgreSQL stored procedure:
|
|
// http://sqlines.com/postgresql/how-to/return_result_set_from_stored_procedure
|
|
QVERIFY_SQL(q, exec(QString("BEGIN;"
|
|
"SELECT %1('cur1', 'cur2');"
|
|
"FETCH ALL IN cur1;"
|
|
"FETCH ALL IN cur2;"
|
|
"COMMIT;").arg(procName)));
|
|
} else {
|
|
QVERIFY_SQL( q, exec( QString( "EXEC %1" ).arg( procName ) ) );
|
|
}
|
|
|
|
if (dbType == QSqlDriver::PostgreSQL) {
|
|
// First result set - start of transaction
|
|
QVERIFY(!q.isSelect());
|
|
QCOMPARE(q.numRowsAffected(), 0);
|
|
QVERIFY(q.nextResult());
|
|
// Second result set contains cursor names
|
|
QVERIFY(q.isSelect());
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toString(), "cur1");
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toString(), "cur2");
|
|
QVERIFY(q.nextResult());
|
|
}
|
|
|
|
for ( int i = 0; i < 4; i++ ) {
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toInt(), i+1 );
|
|
QCOMPARE( q.value( 1 ).toString(), tstStrings.at( i ) );
|
|
}
|
|
|
|
QVERIFY_SQL( q, nextResult() );
|
|
|
|
QVERIFY_SQL( q, isActive() );
|
|
|
|
for ( int i = 0; i < 4; i++ ) {
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Yatta!" ) );
|
|
QCOMPARE( q.value( 1 ).toDouble(), 1.1*( 1+i ) );
|
|
QCOMPARE( q.value( 2 ).toString(), tstStrings.at( i ) );
|
|
QCOMPARE( q.value( 3 ).toInt(), 1+i );
|
|
}
|
|
|
|
// MySQL also counts the CALL itself as a result
|
|
if (dbType == QSqlDriver::MySqlServer) {
|
|
QVERIFY( q.nextResult() );
|
|
QVERIFY( !q.isSelect() ); // ... but it's not a select
|
|
QCOMPARE( q.numRowsAffected(), 0 ); // ... and no rows are affected (at least not with this procedure)
|
|
}
|
|
if (dbType == QSqlDriver::PostgreSQL) {
|
|
// Last result set - commit transaction
|
|
QVERIFY(q.nextResult());
|
|
QVERIFY(!q.isSelect());
|
|
QCOMPARE(q.numRowsAffected(), 0);
|
|
}
|
|
|
|
QVERIFY( !q.nextResult() );
|
|
|
|
QVERIFY( !q.isActive() );
|
|
|
|
if (dbType == QSqlDriver::PostgreSQL)
|
|
q.exec(QString("DROP FUNCTION %1(refcursor, refcursor);").arg(procName));
|
|
else
|
|
q.exec(QString("DROP PROCEDURE %1;").arg(procName));
|
|
}
|
|
|
|
|
|
// For task 190311. Problem: Truncation happens on the 2nd execution if that BLOB is larger
|
|
// than the BLOB on the 1st execution. This was only for MySQL, but the test is general
|
|
// enough to be run with all backends.
|
|
void tst_QSqlQuery::blobsPreparedQuery()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
if ( !db.driver()->hasFeature( QSqlDriver::BLOB ) || !db.driver()->hasFeature( QSqlDriver::PreparedQueries ) )
|
|
QSKIP( "DBMS does not support BLOBs or prepared queries");
|
|
|
|
const QString tableName(qTableName("blobstest", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
q.setForwardOnly( true ); // This is needed to make the test work with DB2.
|
|
QString shortBLOB( "abc" );
|
|
QString longerBLOB( "abcdefghijklmnopqrstuvxyz¿äëïöü¡ " );
|
|
|
|
// In PostgreSQL a BLOB is not called a BLOB, but a BYTEA! :-)
|
|
// ... and in SQL Server it can be called a lot, but IMAGE will do.
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
QString typeName( "BLOB" );
|
|
if (dbType == QSqlDriver::PostgreSQL)
|
|
typeName = "BYTEA";
|
|
else if (dbType == QSqlDriver::MSSqlServer)
|
|
typeName = "IMAGE";
|
|
|
|
QVERIFY_SQL( q, exec( QString( "CREATE TABLE %1(id INTEGER, data %2)" ).arg( tableName ).arg( typeName ) ) );
|
|
q.prepare( QString( "INSERT INTO %1(id, data) VALUES(:id, :data)" ).arg( tableName ) );
|
|
q.bindValue( ":id", 1 );
|
|
q.bindValue( ":data", shortBLOB );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
q.bindValue( ":id", 2 );
|
|
q.bindValue( ":data", longerBLOB );
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
// Two executions and result sets
|
|
q.prepare( QString( "SELECT data FROM %1 WHERE id = ?" ).arg( tableName ) );
|
|
q.bindValue( 0, QVariant( 1 ) );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toString(), shortBLOB );
|
|
|
|
q.bindValue( 0, QVariant( 2 ) );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toString().toUtf8(), longerBLOB.toUtf8() );
|
|
|
|
// Only one execution and result set
|
|
q.prepare( QString( "SELECT id, data FROM %1 ORDER BY id" ).arg( tableName ) );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 1 ).toString(), shortBLOB );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 1 ).toString(), longerBLOB );
|
|
}
|
|
|
|
// There were problems with navigating past the end of a table returning an error on mysql
|
|
void tst_QSqlQuery::emptyTableNavigate()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
{
|
|
QSqlQuery q( db );
|
|
const QString tbl = qTableName("qtest_empty", __FILE__, db);
|
|
q.exec("drop table " + tbl);
|
|
QVERIFY_SQL(q, exec("create table " + tbl + " (id char(10))"));
|
|
QVERIFY_SQL(q, prepare("select * from " + tbl));
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( !q.next() );
|
|
QCOMPARE( q.lastError().isValid(), false );
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::timeStampParsing()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
const QString tableName(qTableName("timeStampParsing", __FILE__, db));
|
|
tst_Databases::safeDropTable(db, tableName);
|
|
QSqlQuery q(db);
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::PostgreSQL) {
|
|
QVERIFY_SQL(q, exec(QStringLiteral("CREATE TABLE ") + tableName + QStringLiteral("("
|
|
"id serial NOT NULL, "
|
|
"datefield timestamp, primary key(id));")));
|
|
} else if (dbType == QSqlDriver::MySqlServer) {
|
|
QVERIFY_SQL(q, exec(QStringLiteral("CREATE TABLE ") + tableName + QStringLiteral("("
|
|
"id integer NOT NULL AUTO_INCREMENT,"
|
|
"datefield timestamp, primary key(id));")));
|
|
} else if (dbType == QSqlDriver::Interbase) {
|
|
// Since there is no auto-increment feature in Interbase we allow it to be null
|
|
QVERIFY_SQL(q, exec(QStringLiteral("CREATE TABLE ") + tableName + QStringLiteral("("
|
|
"id integer,"
|
|
"datefield timestamp);")));
|
|
} else {
|
|
QVERIFY_SQL(q, exec(QStringLiteral("CREATE TABLE ") + tableName + QStringLiteral("("
|
|
"\"id\" integer NOT NULL PRIMARY KEY AUTOINCREMENT,"
|
|
"\"datefield\" timestamp);")));
|
|
}
|
|
QVERIFY_SQL(q, exec(
|
|
QStringLiteral("INSERT INTO ") + tableName + QStringLiteral(" (datefield) VALUES (current_timestamp);"
|
|
)));
|
|
QVERIFY_SQL(q, exec(QStringLiteral("SELECT * FROM ") + tableName));
|
|
while (q.next())
|
|
QVERIFY(q.value(1).toDateTime().isValid());
|
|
}
|
|
|
|
void tst_QSqlQuery::task_217003()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q( db );
|
|
const QString Planet(qTableName( "Planet", __FILE__, db));
|
|
|
|
q.exec("drop table " + Planet);
|
|
QVERIFY_SQL( q, exec( "create table " + Planet + " (Name varchar(20))" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + Planet + " VALUES ('Mercury')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + Planet + " VALUES ('Venus')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + Planet + " VALUES ('Earth')" ) );
|
|
QVERIFY_SQL( q, exec( "insert into " + Planet + " VALUES ('Mars')" ) );
|
|
|
|
QVERIFY_SQL( q, exec( "SELECT Name FROM " + Planet ) );
|
|
QVERIFY_SQL( q, seek( 3 ) );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Mars" ) );
|
|
QVERIFY_SQL( q, seek( 1 ) );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Venus" ) );
|
|
QVERIFY_SQL( q, exec( "SELECT Name FROM " + Planet ) );
|
|
QVERIFY_SQL( q, seek( 3 ) );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Mars" ) );
|
|
QVERIFY_SQL( q, seek( 0 ) );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Mercury" ) );
|
|
QVERIFY_SQL( q, seek( 1 ) );
|
|
QCOMPARE( q.value( 0 ).toString(), QString( "Venus" ) );
|
|
}
|
|
|
|
void tst_QSqlQuery::task_250026()
|
|
{
|
|
QString data258, data1026;
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q( db );
|
|
|
|
const QString tableName(qTableName("task_250026", __FILE__, db));
|
|
|
|
if ( !q.exec( "create table " + tableName + " (longfield varchar(1100))" ) ) {
|
|
qDebug() << "Error" << q.lastError();
|
|
QSKIP( "Db doesn't support \"1100\" as a size for fields");
|
|
}
|
|
|
|
data258.fill( 'A', 258 );
|
|
data1026.fill( 'A', 1026 );
|
|
QVERIFY_SQL( q, prepare( "insert into " + tableName + "(longfield) VALUES (:longfield)" ) );
|
|
q.bindValue( ":longfield", data258 );
|
|
QVERIFY_SQL( q, exec() );
|
|
q.bindValue( ":longfield", data1026 );
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY_SQL( q, exec( "select * from " + tableName ) );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toString().length(), data258.length() );
|
|
QVERIFY_SQL( q, next() );
|
|
QCOMPARE( q.value( 0 ).toString().length(), data1026.length() );
|
|
}
|
|
|
|
void tst_QSqlQuery::crashQueryOnCloseDatabase()
|
|
{
|
|
for (const auto &dbName : qAsConst(dbs.dbNames)) {
|
|
QSqlDatabase clonedDb = QSqlDatabase::cloneDatabase(
|
|
QSqlDatabase::database(dbName), "crashTest");
|
|
qDebug() << "Testing crash in sqlquery dtor for driver" << clonedDb.driverName();
|
|
QVERIFY(clonedDb.open());
|
|
QSqlQuery q(clonedDb);
|
|
clonedDb.close();
|
|
QSqlDatabase::removeDatabase("crashTest");
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::task_233829()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q( db );
|
|
const QString tableName(qTableName("task_233829", __FILE__, db));
|
|
QVERIFY_SQL(q,exec("CREATE TABLE " + tableName + "(dbl1 double precision,dbl2 double precision) without oids;"));
|
|
|
|
QString queryString("INSERT INTO " + tableName +"(dbl1, dbl2) VALUES(?,?)");
|
|
|
|
double k = 0.0;
|
|
QVERIFY_SQL(q,prepare(queryString));
|
|
q.bindValue(0,0.0 / k); // nan
|
|
q.bindValue(1,0.0 / k); // nan
|
|
QVERIFY_SQL(q,exec());
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_12477()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
if (!db.driverName().startsWith("QPSQL"))
|
|
QSKIP("PostgreSQL specific test");
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("SELECT 1::bit, '10101010000111101101'::varbit, "
|
|
"'10101111011'::varbit(15), '22222.20'::numeric(16,2), "
|
|
"'333333'::numeric(18), '444444'::numeric"));
|
|
QVERIFY_SQL(q, next());
|
|
QSqlRecord r = q.record();
|
|
QSqlField f;
|
|
|
|
f = r.field(0);
|
|
QCOMPARE(f.length(), 1);
|
|
QCOMPARE(f.precision(), -1);
|
|
|
|
f = r.field(1);
|
|
QCOMPARE(f.length(), -1);
|
|
QCOMPARE(f.precision(), -1);
|
|
|
|
f = r.field(2);
|
|
QCOMPARE(f.length(), 15);
|
|
QCOMPARE(f.precision(), -1);
|
|
|
|
f = r.field(3);
|
|
QCOMPARE(f.length(), 16);
|
|
QCOMPARE(f.precision(), 2);
|
|
|
|
f = r.field(4);
|
|
QCOMPARE(f.length(), 18);
|
|
QCOMPARE(f.precision(), 0);
|
|
|
|
f = r.field(5);
|
|
QCOMPARE(f.length(), -1);
|
|
QCOMPARE(f.precision(), -1);
|
|
}
|
|
|
|
void tst_QSqlQuery::sqlServerReturn0()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType != QSqlDriver::MSSqlServer)
|
|
QSKIP("SQL Server specific test");
|
|
|
|
const QString tableName(qTableName("test141895", __FILE__, db)), procName(qTableName("test141895_proc", __FILE__, db));
|
|
QSqlQuery q( db );
|
|
q.exec("DROP TABLE " + tableName);
|
|
q.exec("DROP PROCEDURE " + procName);
|
|
QVERIFY_SQL(q, exec("CREATE TABLE "+tableName+" (id integer)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" (id) VALUES (1)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" (id) VALUES (2)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" (id) VALUES (2)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" (id) VALUES (3)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" (id) VALUES (1)"));
|
|
QVERIFY_SQL(q, exec("CREATE PROCEDURE "+procName+
|
|
" AS "
|
|
"SELECT * FROM "+tableName+" WHERE ID = 2 "
|
|
"RETURN 0"));
|
|
|
|
QVERIFY_SQL(q, exec("{CALL " + procName + QLatin1Char('}')));
|
|
|
|
QVERIFY_SQL(q, next());
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_551()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q(db);
|
|
const QString pkgname(qTableName("pkg", __FILE__, db));
|
|
QVERIFY_SQL(q, exec("CREATE OR REPLACE PACKAGE "+pkgname+" IS \n\
|
|
\n\
|
|
TYPE IntType IS TABLE OF INTEGER INDEX BY BINARY_INTEGER;\n\
|
|
TYPE VCType IS TABLE OF VARCHAR2(60) INDEX BY BINARY_INTEGER;\n\
|
|
PROCEDURE P (Inp IN IntType, Outp OUT VCType);\n\
|
|
END "+ pkgname + QLatin1Char(';')));
|
|
|
|
QVERIFY_SQL(q, exec("CREATE OR REPLACE PACKAGE BODY "+pkgname+" IS\n\
|
|
PROCEDURE P (Inp IN IntType, Outp OUT VCType)\n\
|
|
IS\n\
|
|
BEGIN\n\
|
|
Outp(1) := '1. Value is ' ||TO_CHAR(Inp(1));\n\
|
|
Outp(2) := '2. Value is ' ||TO_CHAR(Inp(2));\n\
|
|
Outp(3) := '3. Value is ' ||TO_CHAR(Inp(3));\n\
|
|
END p;\n\
|
|
END " + pkgname + QLatin1Char(';')));
|
|
|
|
QVariantList inLst, outLst, res_outLst;
|
|
|
|
q.prepare("begin "+pkgname+".p(:inp, :outp); end;");
|
|
|
|
QString StVal;
|
|
StVal.reserve(60);
|
|
|
|
// loading arrays
|
|
for (int Cnt=0; Cnt < 3; Cnt++) {
|
|
inLst << Cnt;
|
|
outLst << StVal;
|
|
}
|
|
|
|
q.bindValue(":inp", inLst);
|
|
q.bindValue(":outp", outLst, QSql::Out);
|
|
|
|
QVERIFY_SQL(q, execBatch(QSqlQuery::ValuesAsColumns) );
|
|
res_outLst = qvariant_cast<QVariantList>(q.boundValues().at(1));
|
|
|
|
QCOMPARE(res_outLst[0].toString(), QLatin1String("1. Value is 0"));
|
|
QCOMPARE(res_outLst[1].toString(), QLatin1String("2. Value is 1"));
|
|
QCOMPARE(res_outLst[2].toString(), QLatin1String("3. Value is 2"));
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_12186()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
|
|
// make sure that query.boundValues() returns the values in the right order
|
|
// even for more than 16 placeholders
|
|
QSqlQuery query(db);
|
|
query.prepare("INSERT INTO person (col1, col2, col3, col4, col5, col6, col7, col8, col9, col10, col11, col12, col13, col14, col15, col16, col17, col18) "
|
|
"VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)");
|
|
|
|
QList<QVariant> values;
|
|
|
|
for (int i = 0; i < 18; ++i)
|
|
values << i;
|
|
|
|
foreach (QVariant v, values)
|
|
query.bindValue(v.toInt(), v);
|
|
|
|
QCOMPARE(query.boundValues(), values);
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_14132()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q(db);
|
|
const QString procedureName(qTableName("procedure", __FILE__, db));
|
|
QVERIFY_SQL(q, exec("CREATE OR REPLACE PROCEDURE "+ procedureName + " (outStr OUT varchar2) \n\
|
|
is \n\
|
|
begin \n\
|
|
outStr := 'OUTSTRING'; \n\
|
|
end;"));
|
|
QString placeholder = "XXXXXXXXX";
|
|
QVERIFY(q.prepare("CALL "+procedureName+"(?)"));
|
|
q.addBindValue(placeholder, QSql::Out);
|
|
QVERIFY_SQL(q, exec());
|
|
QCOMPARE(q.boundValue(0).toString(), QLatin1String("OUTSTRING"));
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_18435()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType != QSqlDriver::MSSqlServer || !db.driverName().startsWith("QODBC"))
|
|
QSKIP("SQL Server specific test");
|
|
|
|
QSqlQuery q(db);
|
|
QString procName(qTableName("qtbug_18435_proc", __FILE__, db));
|
|
|
|
q.exec("DROP PROCEDURE " + procName);
|
|
const QString stmt =
|
|
"CREATE PROCEDURE " + procName + " @key nvarchar(50) OUTPUT AS\n"
|
|
"BEGIN\n"
|
|
" SET NOCOUNT ON\n"
|
|
" SET @key = 'TEST'\n"
|
|
"END\n";
|
|
|
|
QVERIFY_SQL(q, exec(stmt));
|
|
QVERIFY_SQL(q, prepare("{CALL "+ procName +"(?)}"));
|
|
const QString testStr = "0123";
|
|
q.bindValue(0, testStr, QSql::Out);
|
|
QVERIFY_SQL(q, exec());
|
|
QCOMPARE(q.boundValue(0).toString(), QLatin1String("TEST"));
|
|
|
|
QVERIFY_SQL(q, exec("DROP PROCEDURE " + procName));
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_5251()
|
|
{
|
|
// Since QSqlTableModel will escape the identifiers, we need to escape
|
|
// them for databases that are case sensitive
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString timetest(qTableName("timetest", __FILE__, db));
|
|
tst_Databases::safeDropTable(db, timetest);
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec(QStringLiteral("CREATE TABLE ") + timetest + QStringLiteral(" (t TIME)")));
|
|
QVERIFY_SQL(q, exec(QStringLiteral("INSERT INTO ") + timetest +
|
|
QStringLiteral(" VALUES ('1:2:3.666')")));
|
|
|
|
QSqlTableModel timetestModel(0,db);
|
|
timetestModel.setEditStrategy(QSqlTableModel::OnManualSubmit);
|
|
timetestModel.setTable(timetest);
|
|
QVERIFY_SQL(timetestModel, select());
|
|
|
|
QCOMPARE(timetestModel.record(0).field(0).value().toTime().toString("HH:mm:ss.zzz"), QString("01:02:03.666"));
|
|
QVERIFY_SQL(timetestModel,setData(timetestModel.index(0, 0), QTime(0,12,34,500)));
|
|
QCOMPARE(timetestModel.record(0).field(0).value().toTime().toString("HH:mm:ss.zzz"), QString("00:12:34.500"));
|
|
QVERIFY_SQL(timetestModel, submitAll());
|
|
QCOMPARE(timetestModel.record(0).field(0).value().toTime().toString("HH:mm:ss.zzz"), QString("00:12:34.500"));
|
|
|
|
QVERIFY_SQL(q, exec(QStringLiteral("UPDATE ") + timetest +
|
|
QStringLiteral(" SET t = '0:11:22.33'")));
|
|
QVERIFY_SQL(timetestModel, select());
|
|
QCOMPARE(timetestModel.record(0).field(0).value().toTime().toString("HH:mm:ss.zzz"), QString("00:11:22.330"));
|
|
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_6421()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlQuery q(db);
|
|
const QString tableName(qTableName("bug6421", __FILE__, db).toUpper());
|
|
|
|
QVERIFY_SQL(q, exec("create table "+tableName+"(COL1 char(10), COL2 char(10), COL3 char(10))"));
|
|
QVERIFY_SQL(q, exec("create index INDEX1 on "+tableName+" (COL1 desc)"));
|
|
QVERIFY_SQL(q, exec("create index INDEX2 on "+tableName+" (COL2 desc)"));
|
|
QVERIFY_SQL(q, exec("create index INDEX3 on "+tableName+" (COL3 desc)"));
|
|
q.setForwardOnly(true);
|
|
QVERIFY_SQL(q, exec("select COLUMN_EXPRESSION from ALL_IND_EXPRESSIONS where TABLE_NAME='" + tableName + QLatin1Char('\'')));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toString(), QLatin1String("\"COL1\""));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toString(), QLatin1String("\"COL2\""));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toString(), QLatin1String("\"COL3\""));
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_6618()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType != QSqlDriver::MSSqlServer)
|
|
QSKIP("SQL Server specific test");
|
|
|
|
QSqlQuery q(db);
|
|
q.exec("drop procedure " + qTableName("tst_raiseError", __FILE__, db)); //non-fatal
|
|
QString errorString;
|
|
for (int i=0;i<110;i++)
|
|
errorString+="reallylong";
|
|
errorString+=" error";
|
|
QVERIFY_SQL(q, exec("create procedure " + qTableName("tst_raiseError", __FILE__, db) + " as\n"
|
|
"begin\n"
|
|
" raiserror('" + errorString + "', 16, 1)\n"
|
|
"end\n" ));
|
|
q.exec("{call " + qTableName("tst_raiseError", __FILE__, db) + QLatin1Char('}'));
|
|
QVERIFY(q.lastError().text().contains(errorString));
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_6852()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q(db);
|
|
const QString tableName(qTableName("bug6852", __FILE__, db)), procName(qTableName("bug6852_proc", __FILE__, db));
|
|
|
|
QVERIFY_SQL(q, exec("DROP PROCEDURE IF EXISTS "+procName));
|
|
QVERIFY_SQL(q, exec("CREATE TABLE "+tableName+"(\n"
|
|
"MainKey INT NOT NULL,\n"
|
|
"OtherTextCol VARCHAR(45) NOT NULL,\n"
|
|
"PRIMARY KEY(`MainKey`))"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" VALUES(0, \"Disabled\")"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" VALUES(5, \"Error Only\")"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" VALUES(10, \"Enabled\")"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO "+tableName+" VALUES(15, \"Always\")"));
|
|
QVERIFY_SQL(q, exec("CREATE PROCEDURE "+procName+"()\n"
|
|
"READS SQL DATA\n"
|
|
"BEGIN\n"
|
|
" SET @st = 'SELECT MainKey, OtherTextCol from "+tableName+"';\n"
|
|
" PREPARE stmt from @st;\n"
|
|
" EXECUTE stmt;\n"
|
|
"END;"));
|
|
|
|
QVERIFY_SQL(q, exec("CALL "+procName+"()"));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 0);
|
|
QCOMPARE(q.value(1).toString(), QLatin1String("Disabled"));
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_5765()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
QSqlQuery q(db);
|
|
const QString tableName(qTableName("bug5765", __FILE__, db));
|
|
|
|
QVERIFY_SQL(q, exec("CREATE TABLE "+tableName+"(testval TINYINT(1) DEFAULT 0)"));
|
|
q.prepare("INSERT INTO "+tableName+" SET testval = :VALUE");
|
|
q.bindValue(":VALUE", 1);
|
|
QVERIFY_SQL(q, exec());
|
|
q.bindValue(":VALUE", 12);
|
|
QVERIFY_SQL(q, exec());
|
|
q.bindValue(":VALUE", 123);
|
|
QVERIFY_SQL(q, exec());
|
|
QString sql="select testval from "+tableName;
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 12);
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 123);
|
|
QVERIFY_SQL(q, prepare(sql));
|
|
QVERIFY_SQL(q, exec());
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 12);
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 123);
|
|
}
|
|
|
|
/**
|
|
* This test case tests multiple statements in one execution.
|
|
* Sqlite driver doesn't support multiple statement at one time.
|
|
* If more than one statement is given, the exec or prepare function
|
|
* return failure to the client.
|
|
*/
|
|
void tst_QSqlQuery::QTBUG_21884()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery q(db);
|
|
|
|
QStringList stList;
|
|
QString tableName(qTableName("bug21884", __FILE__, db));
|
|
stList << "create table " + tableName + "(id integer primary key, note string)";
|
|
stList << "select * from " + tableName + QLatin1Char(';');
|
|
stList << "select * from " + tableName + "; \t\n\r";
|
|
stList << "drop table " + tableName;
|
|
|
|
|
|
foreach (const QString& st, stList) {
|
|
QVERIFY_SQL(q, exec(st));
|
|
}
|
|
|
|
foreach (const QString& st, stList) {
|
|
QVERIFY_SQL(q, prepare(st));
|
|
QVERIFY_SQL(q, exec());
|
|
}
|
|
|
|
stList.clear();
|
|
stList << "create table " + tableName + "(id integer primary key); select * from " + tableName;
|
|
stList << "create table " + tableName + "(id integer primary key); syntax error!;";
|
|
stList << "create table " + tableName + "(id integer primary key);;";
|
|
stList << "create table " + tableName + "(id integer primary key);\'\"\a\b\b\v";
|
|
|
|
foreach (const QString&st , stList) {
|
|
QVERIFY2(!q.prepare(st), qPrintable(QString("the statement is expected to fail! ") + st));
|
|
QVERIFY2(!q.exec(st), qPrintable(QString("the statement is expected to fail! ") + st));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This test case test sqlite driver close function. Sqlite driver should close cleanly
|
|
* even if there is still outstanding prepared statement.
|
|
*/
|
|
void tst_QSqlQuery::QTBUG_16967()
|
|
{
|
|
QSqlQuery q2;
|
|
QFETCH(QString, dbName);
|
|
{
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
db.close();
|
|
QCOMPARE(db.lastError().type(), QSqlError::NoError);
|
|
}
|
|
{
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
QSqlQuery q(db);
|
|
q2 = q;
|
|
q.prepare("CREATE TABLE t1 (id INTEGER PRIMARY KEY, str TEXT);");
|
|
db.close();
|
|
QCOMPARE(db.lastError().type(), QSqlError::NoError);
|
|
}
|
|
{
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
QSqlQuery q(db);
|
|
q2 = q;
|
|
q2.prepare("CREATE TABLE t1 (id INTEGER PRIMARY KEY, str TEXT);");
|
|
q2.exec();
|
|
db.close();
|
|
QCOMPARE(db.lastError().type(), QSqlError::NoError);
|
|
}
|
|
{
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
QSqlQuery q(db);
|
|
q2 = q;
|
|
q.exec("INSERT INTO t1 (id, str) VALUES(1, \"test1\");");
|
|
db.close();
|
|
QCOMPARE(db.lastError().type(), QSqlError::NoError);
|
|
}
|
|
{
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
QSqlQuery q(db);
|
|
q2 = q;
|
|
q.exec("SELECT * FROM t1;");
|
|
db.close();
|
|
QCOMPARE(db.lastError().type(), QSqlError::NoError);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* In SQLite when a boolean value is bound to a placeholder, it should be converted
|
|
* into integer 0/1 rather than text "false"/"true". According to documentation,
|
|
* SQLite does not have separate Boolean storage class. Instead, Boolean values are
|
|
* stored as integers.
|
|
*/
|
|
void tst_QSqlQuery::QTBUG_23895()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery q(db);
|
|
|
|
QString tableName(qTableName("bug23895", __FILE__, db));
|
|
q.prepare("create table " + tableName + "(id integer primary key, val1 bool, val2 boolean)");
|
|
QVERIFY_SQL(q, exec());
|
|
q.prepare("insert into " + tableName + "(id, val1, val2) values(?, ?, ?);");
|
|
q.addBindValue(1);
|
|
q.addBindValue(true);
|
|
q.addBindValue(false);
|
|
QVERIFY_SQL(q, exec());
|
|
|
|
QString sql="select * from " + tableName;
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QVERIFY_SQL(q, next());
|
|
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Int);
|
|
QCOMPARE(q.value(0).metaType().id(), QVariant::LongLong);
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
QCOMPARE(q.record().field(1).metaType().id(), QVariant::Bool);
|
|
QCOMPARE(q.value(1).metaType().id(), QVariant::LongLong);
|
|
QCOMPARE(q.value(1).toBool(), true);
|
|
QCOMPARE(q.record().field(2).metaType().id(), QVariant::Bool);
|
|
QCOMPARE(q.value(2).metaType().id(), QVariant::LongLong);
|
|
QCOMPARE(q.value(2).toBool(), false);
|
|
|
|
q.prepare("insert into " + tableName + "(id, val1, val2) values(?, ?, ?);");
|
|
q.addBindValue(2);
|
|
q.addBindValue(false);
|
|
q.addBindValue(false);
|
|
QVERIFY_SQL(q, exec());
|
|
|
|
sql="select * from " + tableName + " where val1";
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
QVERIFY(!q.next());
|
|
|
|
sql="select * from " + tableName + " where not val2";
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.value(0).toInt(), 2);
|
|
QVERIFY(!q.next());
|
|
}
|
|
|
|
/**
|
|
* Test for aliases with dots
|
|
*/
|
|
void tst_QSqlQuery::QTBUG_14904()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery q(db);
|
|
|
|
QString tableName(qTableName("bug14904", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
q.prepare("create table " + tableName + "(val1 bool)");
|
|
QVERIFY_SQL(q, exec());
|
|
q.prepare("insert into " + tableName + "(val1) values(?);");
|
|
q.addBindValue(true);
|
|
QVERIFY_SQL(q, exec());
|
|
|
|
QString sql="select val1 AS value1 from " + tableName;
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QVERIFY_SQL(q, next());
|
|
|
|
QCOMPARE(q.record().indexOf("value1"), 0);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Bool);
|
|
QCOMPARE(q.value(0).toBool(), true);
|
|
|
|
sql="select val1 AS 'value.one' from " + tableName;
|
|
QVERIFY_SQL(q, exec(sql));
|
|
QVERIFY_SQL(q, next());
|
|
QCOMPARE(q.record().indexOf("value.one"), 0); // was -1 before bug fix
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Bool);
|
|
QCOMPARE(q.value(0).toBool(), true);
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_2192()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
{
|
|
const QString tableName(qTableName("bug2192", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec(QString("CREATE TABLE " + tableName + " (dt %1)").arg(tst_Databases::dateTimeTypeName(db))));
|
|
|
|
QDateTime dt = QDateTime(QDate(2012, 7, 4), QTime(23, 59, 59, 999));
|
|
QVERIFY_SQL(q, prepare("INSERT INTO " + tableName + " (dt) VALUES (?)"));
|
|
q.bindValue(0, dt);
|
|
QVERIFY_SQL(q, exec());
|
|
|
|
QVERIFY_SQL(q, exec("SELECT dt FROM " + tableName));
|
|
QVERIFY_SQL(q, next());
|
|
|
|
// Check if retrieved value preserves reported precision
|
|
int precision = qMax(0, q.record().field("dt").precision());
|
|
qint64 diff = qAbs(q.value(0).toDateTime().msecsTo(dt));
|
|
qint64 keep = qMin(1000LL, qRound64(qPow(10.0, precision)));
|
|
QVERIFY(diff <= 1000 - keep);
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_36211()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
if (tst_Databases::getDatabaseType(db) == QSqlDriver::PostgreSQL) {
|
|
const QString tableName(qTableName("bug36211", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec(QString("CREATE TABLE %1 (dtwtz timestamptz, dtwotz timestamp)").arg(tableName)));
|
|
|
|
#if QT_CONFIG(timezone)
|
|
QTimeZone l_tzBrazil("America/Sao_Paulo");
|
|
QTimeZone l_tzChina("Asia/Shanghai");
|
|
QVERIFY(l_tzBrazil.isValid());
|
|
QVERIFY(l_tzChina.isValid());
|
|
QDateTime dt = QDateTime(QDate(2014, 10, 30), QTime(14, 12, 02, 357));
|
|
QVERIFY_SQL(q, prepare("INSERT INTO " + tableName + " (dtwtz, dtwotz) VALUES (:dt, :dt)"));
|
|
q.bindValue(":dt", dt);
|
|
QVERIFY_SQL(q, exec());
|
|
q.bindValue(":dt", dt.toTimeZone(l_tzBrazil));
|
|
QVERIFY_SQL(q, exec());
|
|
q.bindValue(":dt", dt.toTimeZone(l_tzChina));
|
|
QVERIFY_SQL(q, exec());
|
|
|
|
QVERIFY_SQL(q, exec("SELECT dtwtz, dtwotz FROM " + tableName));
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
QVERIFY_SQL(q, next());
|
|
|
|
for (int j = 0; j < 2; ++j) {
|
|
// Check if retrieved value preserves reported precision
|
|
int precision = qMax(0, q.record().field(j).precision());
|
|
qint64 diff = qAbs(q.value(j).toDateTime().msecsTo(dt));
|
|
qint64 keep = qMin(1000LL, qRound64(qPow(10.0, precision)));
|
|
QVERIFY(diff <= 1000 - keep);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_53969()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QList<int> values = QList<int>() << 10 << 20 << 127 << 128 << 1, tableValues;
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
tableValues.reserve(values.size());
|
|
if (tst_Databases::getDatabaseType(db) == QSqlDriver::MySqlServer) {
|
|
const QString tableName(qTableName("bug53969", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec(QString("CREATE TABLE %1 (id INT AUTO_INCREMENT PRIMARY KEY, "
|
|
"test_number TINYINT(3) UNSIGNED)")
|
|
.arg(tableName)));
|
|
|
|
QVERIFY_SQL(q, prepare("INSERT INTO " + tableName + " (test_number) VALUES (:value)"));
|
|
|
|
QList<int>::iterator begin = values.begin(), end = values.end(), it;
|
|
for (it = begin; it != end; ++it) {
|
|
q.bindValue(":value", *it);
|
|
QVERIFY_SQL(q, exec());
|
|
}
|
|
|
|
QVERIFY_SQL(q, prepare("SELECT test_number FROM " + tableName));
|
|
QVERIFY_SQL(q, exec());
|
|
|
|
while (q.next()) {
|
|
bool ok;
|
|
tableValues.push_back(q.value(0).toUInt(&ok));
|
|
QVERIFY(ok);
|
|
}
|
|
QCOMPARE(values, tableValues);
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::gisPointDatatype()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery sqlQuery(db);
|
|
const auto tableName = qTableName("qtbug72140", __FILE__, db);
|
|
tst_Databases::safeDropTable(db, tableName);
|
|
QString sqlCommand = QStringLiteral("CREATE TABLE %1 (`lonlat_point` POINT NULL) ENGINE = InnoDB;").arg(tableName);
|
|
QVERIFY(sqlQuery.exec(sqlCommand));
|
|
sqlCommand = QStringLiteral("INSERT INTO %1(lonlat_point) VALUES(ST_GeomFromText('POINT(1 1)'));").arg(tableName);
|
|
QVERIFY(sqlQuery.exec(sqlCommand));
|
|
sqlCommand = QStringLiteral("SELECT * FROM %1;").arg(tableName);
|
|
QVERIFY(sqlQuery.exec(sqlCommand));
|
|
QCOMPARE(sqlQuery.record().field(0).metaType().id(), QVariant::Type::ByteArray);
|
|
QVERIFY(sqlQuery.next());
|
|
}
|
|
|
|
void tst_QSqlQuery::oraOCINumber()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
const QString qtest_oraOCINumber(qTableName("qtest_oraOCINumber", __FILE__, db));
|
|
|
|
QSqlQuery q( db );
|
|
q.setForwardOnly( true );
|
|
QVERIFY_SQL( q, exec( "create table " + qtest_oraOCINumber +
|
|
" (col1 number(20), col2 number(20))" ) );
|
|
QVERIFY(q.prepare("insert into " + qtest_oraOCINumber + " values (?, ?)"));
|
|
QVariantList col1Values;
|
|
QVariantList col2Values;
|
|
col1Values << (qulonglong)(1)
|
|
<< (qulonglong)(0)
|
|
<< (qulonglong)(INT_MAX)
|
|
<< (qulonglong)(UINT_MAX)
|
|
<< (qulonglong)(LONG_MAX)
|
|
<< (qulonglong)(ULONG_MAX)
|
|
<< (qulonglong)(LLONG_MAX)
|
|
<< (qulonglong)(ULLONG_MAX);
|
|
|
|
col2Values << (qlonglong)(1)
|
|
<< (qlonglong)(0)
|
|
<< (qlonglong)(-1)
|
|
<< (qlonglong)(LONG_MAX)
|
|
<< (qlonglong)(LONG_MIN)
|
|
<< (qlonglong)(ULONG_MAX)
|
|
<< (qlonglong)(LLONG_MAX)
|
|
<< (qlonglong)(LLONG_MIN);
|
|
|
|
q.addBindValue(col1Values);
|
|
q.addBindValue(col2Values);
|
|
QVERIFY(q.execBatch());
|
|
QString sqlStr = "select * from " + qtest_oraOCINumber + " where col1 = :bindValue0 AND col2 = :bindValue1";
|
|
QVERIFY(q.prepare(sqlStr));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(1), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(1), QSql::InOut);
|
|
|
|
QVERIFY_SQL( q, exec() );
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), qulonglong(1));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(1));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(0), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(0), QSql::InOut);
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), (qulonglong)(0));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(0));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(INT_MAX), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(-1), QSql::InOut);
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), (qulonglong)(INT_MAX));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(-1));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(UINT_MAX), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(LONG_MAX), QSql::InOut);
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), (qulonglong)(UINT_MAX));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(LONG_MAX));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(LONG_MAX), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(LONG_MIN), QSql::InOut);
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), (qulonglong)(LONG_MAX));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(LONG_MIN));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(ULONG_MAX), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(ULONG_MAX), QSql::InOut);
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), (qulonglong)(ULONG_MAX));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(ULONG_MAX));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(LLONG_MAX), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(LLONG_MAX), QSql::InOut);
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), (qulonglong)(LLONG_MAX));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(LLONG_MAX));
|
|
|
|
q.bindValue(":bindValue0", (qulonglong)(ULLONG_MAX), QSql::InOut);
|
|
q.bindValue(":bindValue1", (qlonglong)(LLONG_MIN), QSql::InOut);
|
|
QVERIFY_SQL( q, exec() );
|
|
|
|
QVERIFY( q.next() );
|
|
QCOMPARE(q.boundValue( 0 ).toULongLong(), (qulonglong)(ULLONG_MAX));
|
|
QCOMPARE(q.boundValue( 1 ).toLongLong(), (qlonglong)(LLONG_MIN));
|
|
|
|
}
|
|
|
|
void tst_QSqlQuery::sqlite_constraint()
|
|
{
|
|
QFETCH( QString, dbName );
|
|
QSqlDatabase db = QSqlDatabase::database( dbName );
|
|
CHECK_DATABASE( db );
|
|
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType != QSqlDriver::SQLite)
|
|
QSKIP("Sqlite3 specific test");
|
|
|
|
QSqlQuery q(db);
|
|
const QString trigger(qTableName("test_constraint", __FILE__, db));
|
|
|
|
QVERIFY_SQL(q, exec("CREATE TEMP TRIGGER "+trigger+" BEFORE DELETE ON "+qtest+
|
|
"\nFOR EACH ROW "
|
|
"\nBEGIN"
|
|
"\n SELECT RAISE(ABORT, 'Raised Abort successfully');"
|
|
"\nEND;"
|
|
));
|
|
|
|
QVERIFY(!q.exec("DELETE FROM "+qtest));
|
|
QCOMPARE(q.lastError().databaseText(), QLatin1String("Raised Abort successfully"));
|
|
}
|
|
|
|
void tst_QSqlQuery::sqlite_real()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
const QString tableName(qTableName("sqliterealtype", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + " (id INTEGER, realVal REAL)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id, realVal) VALUES (1, 2.3)"));
|
|
QVERIFY_SQL(q, exec("SELECT realVal FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDouble(), 2.3);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
|
|
q.prepare("INSERT INTO " + tableName + " (id, realVal) VALUES (?, ?)");
|
|
QVariant var((double)5.6);
|
|
q.addBindValue(4);
|
|
q.addBindValue(var);
|
|
QVERIFY_SQL(q, exec());
|
|
|
|
QVERIFY_SQL(q, exec("SELECT realVal FROM " + tableName + " WHERE ID=4"));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDouble(), 5.6);
|
|
}
|
|
|
|
void tst_QSqlQuery::aggregateFunctionTypes()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
QVariant::Type intType = QVariant::Int;
|
|
QVariant::Type sumType = intType;
|
|
QVariant::Type countType = intType;
|
|
// QPSQL uses LongLong for manipulation of integers
|
|
const QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType == QSqlDriver::PostgreSQL || dbType == QSqlDriver::Interbase) {
|
|
sumType = countType = QVariant::LongLong;
|
|
} else if (dbType == QSqlDriver::Oracle) {
|
|
intType = sumType = countType = QVariant::Double;
|
|
} else if (dbType == QSqlDriver::MySqlServer) {
|
|
sumType = QVariant::Double;
|
|
countType = QVariant::LongLong;
|
|
}
|
|
{
|
|
const QString tableName(qTableName("numericFunctionsWithIntValues", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + " (id INTEGER)"));
|
|
|
|
// First test without any entries
|
|
QVERIFY_SQL(q, exec("SELECT SUM(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
if (dbType == QSqlDriver::SQLite)
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Invalid);
|
|
else
|
|
QCOMPARE(q.record().field(0).metaType().id(), sumType);
|
|
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id) VALUES (1)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id) VALUES (2)"));
|
|
|
|
QVERIFY_SQL(q, exec("SELECT SUM(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 3);
|
|
QCOMPARE(q.record().field(0).metaType().id(), sumType);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT AVG(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
if (dbType == QSqlDriver::SQLite || dbType == QSqlDriver::PostgreSQL || dbType == QSqlDriver::MySqlServer
|
|
|| dbType == QSqlDriver::Oracle) {
|
|
QCOMPARE(q.value(0).toDouble(), 1.5);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
} else {
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
QCOMPARE(q.record().field(0).metaType().id(), (dbType == QSqlDriver::Interbase ? QVariant::LongLong : QVariant::Int));
|
|
}
|
|
|
|
QVERIFY_SQL(q, exec("SELECT COUNT(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 2);
|
|
QCOMPARE(q.record().field(0).metaType().id(), countType);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT MIN(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 1);
|
|
QCOMPARE(q.record().field(0).metaType().id(), intType);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT MAX(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 2);
|
|
QCOMPARE(q.record().field(0).metaType().id(), intType);
|
|
}
|
|
{
|
|
const QString tableName(qTableName("numericFunctionsWithDoubleValues", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + " (id REAL)"));
|
|
|
|
// First test without any entries
|
|
QVERIFY_SQL(q, exec("SELECT SUM(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
if (dbType == QSqlDriver::SQLite)
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Invalid);
|
|
else
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id) VALUES (1.5)"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id) VALUES (2.5)"));
|
|
|
|
QVERIFY_SQL(q, exec("SELECT SUM(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDouble(), 4.0);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT AVG(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDouble(), 2.0);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT COUNT(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toInt(), 2);
|
|
QCOMPARE(q.record().field(0).metaType().id(), countType);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT MIN(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDouble(), 1.5);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT MAX(id) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDouble(), 2.5);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
|
|
QString field = "id";
|
|
|
|
// PSQL does not have the round() function with real type
|
|
if (dbType == QSqlDriver::PostgreSQL) {
|
|
field += "::NUMERIC";
|
|
}
|
|
|
|
QVERIFY_SQL(q, exec("SELECT ROUND(" + field + ", 1) FROM " + tableName + " WHERE id=1.5"));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDouble(), 1.5);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
|
|
QVERIFY_SQL(q, exec("SELECT ROUND(" + field + ", 0) FROM " + tableName + " WHERE id=2.5"));
|
|
QVERIFY(q.next());
|
|
if (dbType == QSqlDriver::MySqlServer)
|
|
QCOMPARE(q.value(0).toDouble(), 2.0);
|
|
else
|
|
QCOMPARE(q.value(0).toDouble(), 3.0);
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Double);
|
|
}
|
|
{
|
|
const QString tableName(qTableName("stringFunctions", __FILE__, db));
|
|
tst_Databases::safeDropTable( db, tableName );
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + " (id INTEGER, txt VARCHAR(50))"));
|
|
|
|
QVERIFY_SQL(q, exec("SELECT MAX(txt) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
if (dbType == QSqlDriver::SQLite)
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::Invalid);
|
|
else
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::String);
|
|
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id, txt) VALUES (1, 'lower')"));
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id, txt) VALUES (2, 'upper')"));
|
|
|
|
QVERIFY_SQL(q, exec("SELECT MAX(txt) FROM " + tableName));
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toString(), QLatin1String("upper"));
|
|
QCOMPARE(q.record().field(0).metaType().id(), QVariant::String);
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
void runIntegralTypesMysqlTest(QSqlDatabase &db, const QString &tableName, const QString &type,
|
|
bool withPreparedStatement, const QList<T> &values)
|
|
{
|
|
QList<QVariant> variantValues;
|
|
variantValues.reserve(values.size());
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("DROP TABLE IF EXISTS " + tableName));
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + " (id " + type + ')'));
|
|
|
|
if (withPreparedStatement) {
|
|
QVERIFY_SQL(q, prepare("INSERT INTO " + tableName + " (id) VALUES (?)"));
|
|
}
|
|
for (int i = 0; i < values.size(); ++i) {
|
|
const T v = values.at(i);
|
|
if (withPreparedStatement) {
|
|
q.bindValue(0, v);
|
|
QVERIFY_SQL(q, exec());
|
|
} else {
|
|
QVERIFY_SQL(q, exec("INSERT INTO " + tableName + " (id) VALUES (" + QString::number(v) + QLatin1Char(')')));
|
|
}
|
|
variantValues.append(QVariant::fromValue(v));
|
|
}
|
|
|
|
// ensure we can read them back properly
|
|
if (withPreparedStatement) {
|
|
QVERIFY_SQL(q, prepare("SELECT id FROM " + tableName));
|
|
QVERIFY_SQL(q, exec());
|
|
} else {
|
|
QVERIFY_SQL(q, exec("SELECT id FROM " + tableName));
|
|
}
|
|
QList<T> actualValues;
|
|
QList<QVariant> actualVariantValues;
|
|
actualValues.reserve(values.size());
|
|
while (q.next()) {
|
|
QVariant value = q.value(0);
|
|
actualVariantValues << value;
|
|
actualValues << value.value<T>();
|
|
QVERIFY(actualVariantValues.last().metaType().id() != qMetaTypeId<char>());
|
|
QVERIFY(actualVariantValues.last().metaType().id() != qMetaTypeId<signed char>());
|
|
QVERIFY(actualVariantValues.last().metaType().id() != qMetaTypeId<unsigned char>());
|
|
}
|
|
QCOMPARE(actualValues, values);
|
|
QCOMPARE(actualVariantValues, variantValues);
|
|
}
|
|
|
|
template<typename T>
|
|
void runIntegralTypesMysqlTest(QSqlDatabase &db, const QString &tableName,
|
|
const QString &type, const bool withPreparedStatement,
|
|
const T min = std::numeric_limits<T>::min(),
|
|
const T max = std::numeric_limits<T>::max())
|
|
{
|
|
// insert some values
|
|
const int steps = 20;
|
|
const T increment = (max / steps - min / steps);
|
|
QList<T> values;
|
|
values.reserve(steps);
|
|
T v = min;
|
|
for (int i = 0; i < steps; ++i, v += increment)
|
|
values.append(v);
|
|
runIntegralTypesMysqlTest(db, tableName, type, withPreparedStatement, values);
|
|
}
|
|
|
|
void tst_QSqlQuery::integralTypesMysql()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
const QList<bool> boolValues = QList<bool>() << false << true;
|
|
for (int i = 0; i < 2; ++i) {
|
|
const bool withPreparedStatement = (i == 1);
|
|
runIntegralTypesMysqlTest<bool>(db, "tinyInt1Test", "TINYINT(1)", withPreparedStatement, boolValues);
|
|
runIntegralTypesMysqlTest<bool>(db, "unsignedTinyInt1Test", "TINYINT(1) UNSIGNED", withPreparedStatement, boolValues);
|
|
runIntegralTypesMysqlTest<qint8>(db, "tinyIntTest", "TINYINT", withPreparedStatement);
|
|
runIntegralTypesMysqlTest<quint8>(db, "unsignedTinyIntTest", "TINYINT UNSIGNED", withPreparedStatement);
|
|
runIntegralTypesMysqlTest<qint16>(db, "smallIntTest", "SMALLINT", withPreparedStatement);
|
|
runIntegralTypesMysqlTest<quint16>(db, "unsignedSmallIntTest", "SMALLINT UNSIGNED", withPreparedStatement);
|
|
runIntegralTypesMysqlTest<qint32>(db, "mediumIntTest", "MEDIUMINT", withPreparedStatement, -(1 << 23), (1 << 23) - 1);
|
|
runIntegralTypesMysqlTest<quint32>(db, "unsignedMediumIntTest", "MEDIUMINT UNSIGNED", withPreparedStatement, 0, (1 << 24) - 1);
|
|
runIntegralTypesMysqlTest<qint32>(db, "intTest", "INT", withPreparedStatement);
|
|
runIntegralTypesMysqlTest<quint32>(db, "unsignedIntTest", "INT UNSIGNED", withPreparedStatement);
|
|
runIntegralTypesMysqlTest<qint64>(db, "bigIntTest", "BIGINT", withPreparedStatement);
|
|
runIntegralTypesMysqlTest<quint64>(db, "unsignedBigIntTest", "BIGINT UNSIGNED", withPreparedStatement);
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::QTBUG_57138()
|
|
{
|
|
QDateTime utc = QDateTime(QDate(2150, 1, 5), QTime(14, 0, 0, 123), Qt::UTC);
|
|
QDateTime localtime = QDateTime(QDate(2150, 1, 5), QTime(14, 0, 0, 123), Qt::LocalTime);
|
|
QDateTime tzoffset = QDateTime(QDate(2150, 1, 5), QTime(14, 0, 0, 123), Qt::OffsetFromUTC, 3600);
|
|
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlQuery create(db);
|
|
QString tableName = qTableName("qtbug57138", __FILE__, db);
|
|
|
|
QVERIFY_SQL(create, exec("create table " + tableName + " (id int, dt_utc datetime, dt_lt datetime, dt_tzoffset datetime)"));
|
|
QVERIFY_SQL(create, prepare("insert into " + tableName + " (id, dt_utc, dt_lt, dt_tzoffset) values (?, ?, ?, ?)"));
|
|
|
|
create.addBindValue(0);
|
|
create.addBindValue(utc);
|
|
create.addBindValue(localtime);
|
|
create.addBindValue(tzoffset);
|
|
|
|
QVERIFY_SQL(create, exec());
|
|
|
|
QSqlQuery q(db);
|
|
q.prepare("SELECT dt_utc, dt_lt, dt_tzoffset FROM " + tableName + " WHERE id = ?");
|
|
q.addBindValue(0);
|
|
|
|
QVERIFY_SQL(q, exec());
|
|
QVERIFY(q.next());
|
|
|
|
QCOMPARE(q.value(0).toDateTime(), utc);
|
|
QCOMPARE(q.value(1).toDateTime(), localtime);
|
|
QCOMPARE(q.value(2).toDateTime(), tzoffset);
|
|
}
|
|
|
|
void tst_QSqlQuery::dateTime_data()
|
|
{
|
|
if (dbs.dbNames.isEmpty())
|
|
QSKIP( "No database drivers are available in this Qt configuration");
|
|
|
|
QTest::addColumn<QString>("dbName");
|
|
QTest::addColumn<QString>("tableName");
|
|
QTest::addColumn<QString>("createTableString");
|
|
QTest::addColumn<QList<QDateTime> >("initialDateTimes");
|
|
QTest::addColumn<QList<QDateTime> >("expectedDateTimes");
|
|
|
|
#if QT_CONFIG(timezone)
|
|
// Using time zones which are highly unlikely to be the same as the testing machine's one
|
|
// as it could pass as a result despite it.
|
|
// +8.5 hours from UTC to North Korea
|
|
const QTimeZone afterUTCTimeZone(30600);
|
|
// -8 hours from UTC to Belize
|
|
const QTimeZone beforeUTCTimeZone(-28800);
|
|
const QTimeZone utcTimeZone("UTC");
|
|
|
|
const QDateTime dtWithAfterTZ(QDate(2015, 5, 18), QTime(4, 26, 30, 500), afterUTCTimeZone);
|
|
const QDateTime dtWithBeforeTZ(QDate(2015, 5, 18), QTime(4, 26, 30, 500), beforeUTCTimeZone);
|
|
const QDateTime dtWithUTCTZ(QDate(2015, 5, 18), QTime(4, 26, 30, 500), utcTimeZone);
|
|
#endif
|
|
const QDateTime dt(QDate(2015, 5, 18), QTime(4, 26, 30));
|
|
const QDateTime dtWithMS(QDate(2015, 5, 18), QTime(4, 26, 30, 500));
|
|
const QList<QDateTime> dateTimes = {
|
|
dt, dtWithMS,
|
|
#if QT_CONFIG(timezone)
|
|
dtWithAfterTZ, dtWithBeforeTZ, dtWithUTCTZ
|
|
#endif
|
|
};
|
|
const QList<QDateTime> expectedDateTimesLocalTZ = {
|
|
dt, dtWithMS,
|
|
#if QT_CONFIG(timezone)
|
|
dtWithAfterTZ.toLocalTime(), dtWithBeforeTZ.toLocalTime(), dtWithUTCTZ.toLocalTime()
|
|
#endif
|
|
};
|
|
const QList<QDateTime> expectedTimeStampDateTimes = {
|
|
dt, dtWithMS,
|
|
#if QT_CONFIG(timezone)
|
|
dtWithMS, dtWithMS, dtWithMS
|
|
#endif
|
|
};
|
|
const QList<QDateTime> expectedDateTimes = {
|
|
dt, dt,
|
|
#if QT_CONFIG(timezone)
|
|
dt, dt, dt
|
|
#endif
|
|
};
|
|
|
|
for (const QString &dbName : qAsConst(dbs.dbNames)) {
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
if (!db.isValid())
|
|
continue;
|
|
const QString tableNameTSWithTimeZone(qTableName("dateTimeTSWithTimeZone", __FILE__, db));
|
|
const QString tableNameTSWithLocalTimeZone(qTableName("dateTimeTSWithLocalTimeZone", __FILE__, db));
|
|
const QString tableNameTS(qTableName("dateTimeTS", __FILE__, db));
|
|
const QString tableNameDate(qTableName("dateTimeDate", __FILE__, db));
|
|
QTest::newRow(QString(dbName + " timestamp with time zone").toLatin1())
|
|
<< dbName << tableNameTSWithTimeZone
|
|
<< QStringLiteral(" (dt TIMESTAMP WITH TIME ZONE)")
|
|
<< dateTimes << dateTimes;
|
|
QTest::newRow(QString(dbName + " timestamp with local time zone").toLatin1())
|
|
<< dbName << tableNameTSWithTimeZone
|
|
<< QStringLiteral(" (dt TIMESTAMP WITH LOCAL TIME ZONE)")
|
|
<< dateTimes << expectedDateTimesLocalTZ;
|
|
QTest::newRow(QString(dbName + "timestamp").toLatin1())
|
|
<< dbName << tableNameTS << QStringLiteral(" (dt TIMESTAMP(3))")
|
|
<< dateTimes << expectedTimeStampDateTimes;
|
|
QTest::newRow(QString(dbName + "date").toLatin1())
|
|
<< dbName << tableNameDate << QStringLiteral(" (dt DATE)")
|
|
<< dateTimes << expectedDateTimes;
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::dateTime()
|
|
{
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
|
|
QSqlDriver::DbmsType dbType = tst_Databases::getDatabaseType(db);
|
|
if (dbType != QSqlDriver::Oracle)
|
|
QSKIP("Implemented only for Oracle");
|
|
|
|
QFETCH(QString, tableName);
|
|
QFETCH(QString, createTableString);
|
|
QFETCH(QList<QDateTime>, initialDateTimes);
|
|
QFETCH(QList<QDateTime>, expectedDateTimes);
|
|
|
|
tst_Databases::safeDropTable(db, tableName);
|
|
|
|
QSqlQuery q(db);
|
|
QVERIFY_SQL(q, exec("CREATE TABLE " + tableName + createTableString));
|
|
for (const QDateTime &dt : qAsConst(initialDateTimes)) {
|
|
QVERIFY_SQL(q, prepare("INSERT INTO " + tableName + " values(:dt)"));
|
|
q.bindValue(":dt", dt);
|
|
QVERIFY_SQL(q, exec());
|
|
}
|
|
QVERIFY_SQL(q, exec("SELECT * FROM " + tableName));
|
|
for (const QDateTime &dt : qAsConst(expectedDateTimes)) {
|
|
QVERIFY(q.next());
|
|
QCOMPARE(q.value(0).toDateTime(), dt);
|
|
}
|
|
}
|
|
|
|
void tst_QSqlQuery::sqliteVirtualTable()
|
|
{
|
|
// Virtual tables can behave differently when it comes to prepared
|
|
// queries, so we need to check these explicitly
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
const auto tableName = qTableName("sqliteVirtual", __FILE__, db);
|
|
QSqlQuery qry(db);
|
|
QVERIFY_SQL(qry, exec("create virtual table " + tableName + " using fts3(id, name)"));
|
|
|
|
// Delibrately malform the query to try and provoke a potential crash situation
|
|
QVERIFY_SQL(qry, prepare("select * from " + tableName + " where name match '?'"));
|
|
qry.addBindValue("Andy");
|
|
QVERIFY(!qry.exec());
|
|
|
|
QVERIFY_SQL(qry, prepare("insert into " + tableName + "(id, name) VALUES (?, ?)"));
|
|
qry.addBindValue(1);
|
|
qry.addBindValue("Andy");
|
|
QVERIFY_SQL(qry, exec());
|
|
|
|
QVERIFY_SQL(qry, exec("select * from " + tableName));
|
|
QVERIFY(qry.next());
|
|
QCOMPARE(qry.value(0).toInt(), 1);
|
|
QCOMPARE(qry.value(1).toString(), "Andy");
|
|
|
|
QVERIFY_SQL(qry, prepare("insert into " + tableName + "(id, name) values (:id, :name)"));
|
|
qry.bindValue(":id", 2);
|
|
qry.bindValue(":name", "Peter");
|
|
QVERIFY_SQL(qry, exec());
|
|
|
|
QVERIFY_SQL(qry, prepare("select * from " + tableName + " where name match ?"));
|
|
qry.addBindValue("Peter");
|
|
QVERIFY_SQL(qry, exec());
|
|
QVERIFY(qry.next());
|
|
QCOMPARE(qry.value(0).toInt(), 2);
|
|
QCOMPARE(qry.value(1).toString(), "Peter");
|
|
}
|
|
|
|
void tst_QSqlQuery::mysql_timeType()
|
|
{
|
|
// The TIME data type is different to the standard with MySQL as it has a range of
|
|
// '-838:59:59' to '838:59:59'.
|
|
QFETCH(QString, dbName);
|
|
QSqlDatabase db = QSqlDatabase::database(dbName);
|
|
CHECK_DATABASE(db);
|
|
const auto tableName = qTableName("mysqlTimeType", __FILE__, db);
|
|
tst_Databases::safeDropTables(db, { tableName });
|
|
QSqlQuery qry(db);
|
|
QVERIFY_SQL(qry, exec("create table " + tableName + " (t time(6))"));
|
|
|
|
// MySQL will convert days into hours and add them together so 17 days 11 hours becomes 419 hours
|
|
const QStringList timeData = { "-838:59:59.000000", "-123:45:56.789", "000:00:00.0", "123:45:56.789",
|
|
"838:59:59.000000", "15:50", "12", "1213", "0 1:2:3", "17 11:22:33" };
|
|
const QStringList resultTimeData = { "-838:59:59.000000", "-123:45:56.789000", "00:00:00.000000",
|
|
"123:45:56.789000", "838:59:59.000000", "15:50:00.000000", "00:00:12.000000", "00:12:13.000000",
|
|
"01:02:03.000000", "419:22:33.000000" };
|
|
for (const QString &time : timeData)
|
|
QVERIFY_SQL(qry, exec("insert into " + tableName + " (t) VALUES ('" + time + "')"));
|
|
|
|
QVERIFY_SQL(qry, exec("select * from " + tableName));
|
|
for (const QString &time : qAsConst(resultTimeData)) {
|
|
QVERIFY(qry.next());
|
|
QCOMPARE(qry.value(0).toString(), time);
|
|
}
|
|
|
|
QVERIFY_SQL(qry, exec("delete from " + tableName));
|
|
for (const QString &time : timeData) {
|
|
QVERIFY_SQL(qry, prepare("insert into " + tableName + " (t) VALUES (:time)"));
|
|
qry.bindValue(0, time);
|
|
QVERIFY_SQL(qry, exec());
|
|
}
|
|
QVERIFY_SQL(qry, exec("select * from " + tableName));
|
|
for (const QString &time : resultTimeData) {
|
|
QVERIFY(qry.next());
|
|
QCOMPARE(qry.value(0).toString(), time);
|
|
}
|
|
|
|
QVERIFY_SQL(qry, exec("delete from " + tableName));
|
|
const QList<QTime> qTimeBasedData = { QTime(), QTime(1, 2, 3, 4), QTime(0, 0, 0, 0), QTime(23,59,59,999) };
|
|
for (const QTime &time : qTimeBasedData) {
|
|
QVERIFY_SQL(qry, prepare("insert into " + tableName + " (t) VALUES (:time)"));
|
|
qry.bindValue(0, time);
|
|
QVERIFY_SQL(qry, exec());
|
|
}
|
|
QVERIFY_SQL(qry, exec("select * from " + tableName));
|
|
for (const QTime &time : qTimeBasedData) {
|
|
QVERIFY(qry.next());
|
|
QCOMPARE(qry.value(0).toTime(), time);
|
|
}
|
|
}
|
|
|
|
QTEST_MAIN( tst_QSqlQuery )
|
|
#include "tst_qsqlquery.moc"
|