Update contrib.
1 // Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
20 ///////////////////////////////////////////////////////////////////////////////////////
22 RTest TheTest(_L("t_sqlmulti test"));
24 _LIT(KTestDir, "c:\\test\\");
25 _LIT(KTestDbName1, "c:\\test\\t_sqlmulti.db");
27 ///////////////////////////////////////////////////////////////////////////////////////
29 void DeleteTestFiles()
31 RSqlDatabase::Delete(KTestDbName1);
34 ///////////////////////////////////////////////////////////////////////////////////////
35 ///////////////////////////////////////////////////////////////////////////////////////
36 //Test macros and functions
37 void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
45 TName name = th.Name();
46 RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
50 RDebug::Print(_L("*** Line %d\r\n"), aLine);
52 TheTest(EFalse, aLine);
55 void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
57 if(aValue != aExpected)
63 TName name = th.Name();
64 RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
68 RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
70 TheTest(EFalse, aLine);
73 #define TEST(arg) ::Check1((arg), __LINE__)
74 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
75 #define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
76 #define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
78 ///////////////////////////////////////////////////////////////////////////////////////
83 TInt err = fs.Connect();
86 err = fs.MkDir(KTestDir);
87 TEST(err == KErrNone || err == KErrAlreadyExists);
92 ///////////////////////////////////////////////////////////////////////////////////////
95 @SYMTestCaseID SYSLIB-SQL-CT-1612
96 @SYMTestCaseDesc Two connections to the same database in the same thread. Create a test database
97 and insert some records from both connections. Verify that all records were inserted
100 @SYMTestActions Testing SQL engine behaviour when having mutiple connections to the same database
102 @SYMTestExpectedResults Test must not fail
106 void TestMultiConnSameThread()
110 TInt err = db1.Create(KTestDbName1);
111 TEST2(err, KErrNone);
113 //Create test database
114 RDebug::Print(_L("###Create test database\r\n"));
115 _LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER PRIMARY KEY AUTOINCREMENT, Data INTEGER)");
116 err = db1.Exec(KCreateSql);
121 err = db2.Open(KTestDbName1);
122 TEST2(err, KErrNone);
124 //Insert some records using both connections
125 RDebug::Print(_L("###Insert some records\r\n"));
126 const TInt KRecNum = 100;
127 _LIT8(KInsertSql, "INSERT INTO A(Data) VALUES(");
128 for(TInt i=0;i<KRecNum;++i)
130 TBuf8<100> sql(KInsertSql);
131 sql.AppendNum((TInt64)i + 1);
132 sql.Append(_L(");"));
133 err = (i%2) ? db1.Exec(sql) : db2.Exec(sql);
136 TPtrC msg = (i%2) ? db1.LastErrorMessage() : db2.LastErrorMessage();
137 RDebug::Print(_L("##Db Error msg: \"%S\"\n\r"), &msg);
142 //Check the database content
143 RDebug::Print(_L("###Check the database content\r\n"));
144 _LIT8(KSelectSql, "SELECT * FROM A");
146 err = stmt.Prepare(db1, KSelectSql);
147 TEST2(err, KErrNone);
149 for(TInt j=0;j<KRecNum;++j)
152 TEST2(err, KSqlAtRow);
154 TInt id = stmt.ColumnInt(0);
155 TInt data = stmt.ColumnInt(1);
164 RDebug::Print(_L("###Delete the test database\r\n"));
165 (void)RSqlDatabase::Delete(KTestDbName1);
168 ///////////////////////////////////////////////////////////////////////////////////////
173 RSqlDatabase::TIsolationLevel iIsolationLevel;
178 TInt ThreadFunc(void* aData)
182 CTrapCleanup* tc = CTrapCleanup::New();
185 TThreadData* data = static_cast<TThreadData*> (aData);
189 TInt err = db.Open(KTestDbName1);
190 TTEST2(err, KErrNone);
192 err = db.SetIsolationLevel(data->iIsolationLevel);
193 TTEST2(err, KErrNone);
195 if(data->iTransType == 1)
197 _LIT8(KBeginTrans, "BEGIN");
198 err = db.Exec(KBeginTrans);
202 _LIT8(KInsertSql, "INSERT INTO A(Id) VALUES(");
203 for(TInt id=data->iLowRecNo;id<=data->iHighRecNo;++id)
205 TBuf8<128> sql(KInsertSql);
206 sql.AppendNum((TInt64)id);
209 const TInt KAttempts = 20;
210 for(TInt i=0;i<KAttempts&&err==KSqlErrBusy;++i)
213 if(err == KSqlErrBusy)
216 TName name = th.Name();
217 RDebug::Print(_L("!!!Database locked, Thread: %S, Attempt %d, column value %d\r\n"), &name, i + 1, id);
218 User::After(1000000);
224 if(data->iTransType == 1)
226 _LIT8(KCommitTrans, "COMMIT");
227 err = db.Exec(KCommitTrans);
240 @SYMTestCaseID SYSLIB-SQL-CT-1613
241 @SYMTestCaseDesc Multiple connections to the same database from different threads.
242 Each thread inserts set of record to the same table. Verify that all expected records
243 and their column values meet the expectations.
244 @SYMTestPriority High
245 @SYMTestActions Testing SQL engine behaviour when having mutiple connections to the same database
246 from different threads.
247 @SYMTestExpectedResults Test must not fail
251 void TestMultiConnDiffThread()
253 //Create a test database
254 RDebug::Print(_L("+++:MainThread: Create test database\r\n"));
256 TInt err = db.Create(KTestDbName1);
257 TEST2(err, KErrNone);
259 //Create a test table
260 RDebug::Print(_L("+++:MainThread: Create a table in the test database\r\n"));
261 _LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER PRIMARY KEY)");
262 err = db.Exec(KCreateSql);
265 const TInt KThreadCnt = 4;
266 const TInt KRange = 100;
268 const TInt KIsolationLevelCnt = 2;
269 TPtrC KIsolationLevelName[KIsolationLevelCnt] = {_L("Read Uncommitted"), _L("Serializable")};
270 const RSqlDatabase::TIsolationLevel KIsolationLevels[KIsolationLevelCnt] = {
271 RSqlDatabase::EReadUncommitted, RSqlDatabase::ESerializable};
273 const TInt KTransTypeCnt = 2;
276 // - doing each per thread database operation in a single transaction;
277 // - doing all per thread database operations in a single transaction;
278 for(TInt transType=0;transType<KTransTypeCnt;++transType)
280 //For both supported isolation levels: read uncommitted and serializable
281 for(TInt isolLevel=0;isolLevel<KIsolationLevelCnt;++isolLevel)
286 RDebug::Print(_L("+++:MainThread: Test: thread count %d, records %d, trans type %d, isolation level: %S\r\n"),
287 KThreadCnt, KRange, transType, &KIsolationLevelName[isolLevel]);
289 RThread thread[KThreadCnt];
290 TRequestStatus status[KThreadCnt];
291 TThreadData data[KThreadCnt];
293 //Create the test threads and run them. Each thread establishes a connection with the test database
294 //and attempts to write set of records in the test table.
296 for(j=0;j<KThreadCnt;++j,low=high+1,high+=KRange)
298 data[j].iTransType = transType;
299 data[j].iIsolationLevel = KIsolationLevels[isolLevel];
300 data[j].iLowRecNo = low;
301 data[j].iHighRecNo = high;
303 _LIT(KThreadName,"Thr-");
304 TBuf<32> threadName(KThreadName);
305 threadName.AppendNum((TInt64)j + 1);
307 TEST2(thread[j].Create(threadName, &ThreadFunc, 0x2000, 0x1000, 0x10000, (void*)&data[j], EOwnerThread), KErrNone);
308 thread[j].Logon(status[j]);
309 TEST2(status[j].Int(), KRequestPending);
313 User::After(2000000);
314 //Wait until threads finish the database operations and close them.
315 for(j=0;j<KThreadCnt;++j)
317 User::WaitForRequest(status[j]);
318 TEST(thread[j].ExitType() != EExitPanic);
322 //Check that all records which are esupposed to be in the database, are there.
323 RDebug::Print(_L("+++:MainThread: Check that all records have been written\r\n"));
324 _LIT8(KSelectSql1, "SELECT COUNT(*) FROM A;");
326 err = stmt.Prepare(db, KSelectSql1);
327 TEST2(err, KErrNone);
329 TEST2(err, KSqlAtRow);
330 TInt cnt = stmt.ColumnInt(0);
331 TEST2(cnt, KThreadCnt * KRange);
334 //Check that all records have expected column values.
335 RDebug::Print(_L("+++:MainThread: Check that all records have expected column values\r\n"));
336 _LIT8(KSelectSql2, "SELECT * FROM A;");
337 err = stmt.Prepare(db, KSelectSql2);
338 TEST2(err, KErrNone);
339 for(TInt k=0;k<(KThreadCnt*KRange);++k)
342 TEST2(err, KSqlAtRow);
343 TInt val = stmt.ColumnInt(0);
344 TEST(val > 0 && val <= (KThreadCnt * KRange));
348 //Prepare for the next test run - delete all records.
349 RDebug::Print(_L("+++:MainThread: Delete all records\r\n"));
350 _LIT8(KDeleteSql, "DELETE FROM A");
351 err = db.Exec(KDeleteSql);
353 }//end of "for(TInt isolLevel=0;isolLevel<KIsolationLevelCnt;++isolLevel)"
354 }//end of "for(TInt transType=0;transType<KTransTypeCnt;++transType)"
357 RDebug::Print(_L("+++:MainThread: Delete the test database\r\n"));
358 (void)RSqlDatabase::Delete(KTestDbName1);
361 ///////////////////////////////////////////////////////////////////////////////////////
363 static RCriticalSection UpdateThreadCrS;
364 static RCriticalSection MainThreadCrS;
365 const TInt KInitialValue = 10;
366 const TInt KUpdatedValue = 20;
368 TInt UpdateThreadFunc(void*)
372 CTrapCleanup* tc = CTrapCleanup::New();
376 TInt err = db.Open(KTestDbName1);
377 TTEST2(err, KErrNone);
379 RDebug::Print(_L("---:UpdThread: Set the isolation level to \"Read uncommitted\"\r\n"));
380 err = db.SetIsolationLevel(RSqlDatabase::EReadUncommitted);
381 TTEST2(err, KErrNone);
383 RDebug::Print(_L("---:UpdThread: Begin a write transaction\r\n"));
384 _LIT8(KBeginTransSql, "BEGIN IMMEDIATE TRANSACTION");
385 err = db.Exec(KBeginTransSql);
388 RDebug::Print(_L("---:UpdThread: Update the record\r\n"));
389 _LIT8(KUpdateSql, "UPDATE A SET Id = ");
390 TBuf8<64> sql(KUpdateSql);
391 sql.AppendNum((TInt64)KUpdatedValue);
395 RDebug::Print(_L("---:UpdThread: Notify the main thread about the update\r\n"));
396 MainThreadCrS.Signal();
398 RDebug::Print(_L("---:UpdThread: Wait for permisson to continue...\r\n"));
399 UpdateThreadCrS.Wait();
401 RDebug::Print(_L("---:UpdThread: Rollback the update\r\n"));
402 _LIT8(KRollBackTransSql, "ROLLBACK TRANSACTION");
403 err = db.Exec(KRollBackTransSql);
406 RDebug::Print(_L("---:UpdThread: Notify the main thread about the rollback\r\n"));
407 MainThreadCrS.Signal();
418 @SYMTestCaseID SYSLIB-SQL-CT-1614
419 @SYMTestCaseDesc Verifying that when having 2 database connections in different threads, both set
420 the isolation level to "Read Uncommitted", the reading thread can make "dirty read"
421 operations (can read the updated but not committed yet record values made by the
423 @SYMTestPriority High
424 @SYMTestActions Testing "Read Uncommitted" database isolation level.
425 @SYMTestExpectedResults Test must not fail
429 void TestIsolationLevel()
431 RDebug::Print(_L("+++:MainThread: Create critical sections\r\n"));
432 TEST2(UpdateThreadCrS.CreateLocal(), KErrNone);
433 UpdateThreadCrS.Wait();
434 TEST2(MainThreadCrS.CreateLocal(), KErrNone);
435 MainThreadCrS.Wait();
437 RDebug::Print(_L("+++:MainThread: Create test database\r\n"));
439 TInt err = db.Create(KTestDbName1);
440 TEST2(err, KErrNone);
442 RDebug::Print(_L("+++:MainThread: Set the isolation level to \"Read uncommitted\"\r\n"));
443 err = db.SetIsolationLevel(RSqlDatabase::EReadUncommitted);
444 TEST2(err, KErrNone);
446 RDebug::Print(_L("+++:MainThread: Create a table in the test database\r\n"));
447 _LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER)");
448 err = db.Exec(KCreateSql);
451 RDebug::Print(_L("+++:MainThread: Insert one record in the table\r\n"));
452 _LIT8(KInsertSql, "INSERT INTO A(Id) VALUES(");
453 TBuf8<64> sql(KInsertSql);
454 sql.AppendNum((TInt64)KInitialValue);
459 RDebug::Print(_L("+++:MainThread: Create the \"update\" thread\r\n"));
460 _LIT(KThreadName, "UpdTh");
462 TEST2(thread.Create(KThreadName, &UpdateThreadFunc, 0x2000, 0x1000, 0x10000, NULL, EOwnerThread), KErrNone);
463 TRequestStatus status;
464 thread.Logon(status);
465 TEST2(status.Int(), KRequestPending);
468 RDebug::Print(_L("+++:MainThread: Wait for record update completion...\r\n"));
469 MainThreadCrS.Wait();
471 RDebug::Print(_L("+++:MainThread: Read the record and check the data...\r\n"));
472 _LIT8(KSelectSql, "SELECT * FROM A");
474 err = stmt.Prepare(db, KSelectSql);
475 TEST2(err, KErrNone);
477 TEST2(err, KSqlAtRow);
478 TInt val = stmt.ColumnInt(0);
479 TEST(val == KUpdatedValue);
482 RDebug::Print(_L("+++:MainThread: Notify the update thread that it can rollback\r\n"));
483 UpdateThreadCrS.Signal();
485 RDebug::Print(_L("+++:MainThread: Wait for rollback completion...\r\n"));
486 MainThreadCrS.Wait();
488 RDebug::Print(_L("+++:MainThread: Read the record and check the data...\r\n"));
489 err = stmt.Prepare(db, KSelectSql);
490 TEST2(err, KErrNone);
492 TEST2(err, KSqlAtRow);
493 val = stmt.ColumnInt(0);
494 TEST2(val, KInitialValue);
497 User::WaitForRequest(status);
501 RDebug::Print(_L("+++:MainThread: Delete the test database\r\n"));
502 (void)RSqlDatabase::Delete(KTestDbName1);
504 RDebug::Print(_L("+++:MainThread: Close critical sections\r\n"));
505 MainThreadCrS.Close();
506 UpdateThreadCrS.Close();
509 ///////////////////////////////////////////////////////////////////////////////////////
513 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1612 Multiple connections, the same thread "));
514 TestMultiConnSameThread();
516 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1613 Multiple connections, different threads "));
517 TestMultiConnDiffThread();
519 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1614 Isolation level "));
520 TestIsolationLevel();
527 CTrapCleanup* tc = CTrapCleanup::New();
533 TRAPD(err, DoTestsL());
535 TEST2(err, KErrNone);
544 User::Heap().Check();