Update contrib.
1 // Copyright (c) 2006-2010 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 ///////////////////////////////////////////////////////////////////////////////////////
23 RTest TheTest(_L("t_sqlprivcage test"));
25 _LIT(KTestDir, "c:\\test\\");
26 _LIT(KTestDb1, "c:\\private\\21212124\\t_sqlprivcage_1.db");
27 _LIT(KTestDb2, "c:\\private\\21212124\\t_sqlprivcage_2.db");
28 _LIT(KTestDbZ, "z:\\private\\21212124\\t_sqldb1.db");//Created outside this test
29 _LIT(KTestDb, "\\private\\21212124\\t_sqlprivcage_3.db");
30 TParse TheFileNameParse;
32 static RCriticalSection ThreadCritSect;
33 static RCriticalSection MainCritSect;
35 _LIT(KPanicCategory, "TransFail");
36 const TInt KPanicCode = 1111;
38 ///////////////////////////////////////////////////////////////////////////////////////
40 //Deletes all created test files.
41 void DeleteTestFiles()
43 if(TheFileNameParse.FullName().Length() > 0)
45 (void)RSqlDatabase::Delete(TheFileNameParse.FullName());
47 (void)RSqlDatabase::Delete(KTestDb2);
48 (void)RSqlDatabase::Delete(KTestDb1);
51 ///////////////////////////////////////////////////////////////////////////////////////
52 ///////////////////////////////////////////////////////////////////////////////////////
53 //Test macros and functions
54 void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
62 TName name = th.Name();
63 RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
67 RDebug::Print(_L("*** Line %d\r\n"), aLine);
69 TheTest(EFalse, aLine);
72 void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
74 if(aValue != aExpected)
80 TName name = th.Name();
81 RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
85 RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
87 TheTest(EFalse, aLine);
90 #define TEST(arg) ::Check1((arg), __LINE__)
91 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
92 #define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
93 #define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
95 ///////////////////////////////////////////////////////////////////////////////////////
97 //Creates file session instance and the test directory
100 TInt err = TheFs.Connect();
101 TEST2(err, KErrNone);
103 err = TheFs.CreatePrivatePath(EDriveC);
104 TEST(err == KErrNone || err == KErrAlreadyExists);
106 err = TheFs.MkDir(KTestDir);
107 TEST(err == KErrNone || err == KErrAlreadyExists);
110 ///////////////////////////////////////////////////////////////////////////////////////
113 @SYMTestCaseID SYSLIB-SQL-CT-1764
114 @SYMTestCaseDesc The test creates a database in the test application's private data cage.
115 Then the test does some operations with the created private database:
116 create table, insert records, select records, transactions, delete database.
117 The tests verifies that the SQL server can create a database in the application's
118 private data cage and can operate with the database.
119 @SYMTestPriority High
120 @SYMTestActions SQL, Private database test.
121 @SYMTestExpectedResults Test must not fail
125 void SimpleDbOpTest()
127 //Create private database
129 TInt err = db.Create(KTestDb1);
130 TEST2(err, KErrNone);
132 //Execute some operations with the private database
133 err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
135 err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1);INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
138 //Check written records
140 err = stmt.Prepare(db, _L8("SELECT * FROM A"));
141 TEST2(err, KErrNone);
143 TEST2(err, KSqlAtRow);
144 TEST2(stmt.ColumnInt(0), 1);
146 TEST2(err, KSqlAtRow);
147 TEST2(stmt.ColumnInt(0), 2);
149 TEST2(err, KSqlAtRow);
150 TEST2(stmt.ColumnInt(0), 3);
152 TEST2(err, KSqlAtEnd);
156 //Open private database
157 err = db.Open(KTestDb1);
158 TEST2(err, KErrNone);
160 //Execute a DELETE transaction
161 err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
164 err = db.Exec(_L8("DELETE FROM A WHERE ID > 1"));
167 err = db.Exec(_L8("COMMIT TRANSACTION"));
171 err = stmt.Prepare(db, _L8("SELECT * FROM A"));
172 TEST2(err, KErrNone);
174 TEST2(err, KSqlAtRow);
175 TEST2(stmt.ColumnInt(0), 1);
177 TEST2(err, KSqlAtEnd);
180 //Open private database which is on drive Z and with the same name
182 err = db2.Open(KTestDbZ);
183 TEST2(err, KErrNone);
184 //An attempt to write to a read-only database
185 err = db2.Exec(_L("INSERT INTO A VALUES(6)"));
186 TheTest.Printf(_L(" === Read-only private database. RSqlDatabase::Exec() returned err=%d\r\n"), err);
187 TEST(err != KErrNone);
189 err = stmt.Prepare(db2, _L8("SELECT * FROM A"));
190 TEST2(err, KErrNone);
192 TEST2(err, KSqlAtRow);
193 TEST2(stmt.ColumnInt(0), 1);
195 TEST2(err, KSqlAtEnd);
200 err = RSqlDatabase::Delete(KTestDb1);
201 TEST2(err, KErrNone);
203 //Create private database on drive different than C:
204 for(TInt drvNum=EDriveD;drvNum<=EDriveZ;++drvNum)
206 TDriveUnit drvUnit(drvNum);
207 TPtrC drvName = drvUnit.Name();
208 TheFileNameParse.Set(KTestDb, &drvName, 0);
209 //Check if it is possible to create application's private data cage on drvNum drive.
210 err = TheFs.CreatePrivatePath(drvNum);
211 if(err == KErrNone || err == KErrAlreadyExists)
213 (void)RSqlDatabase::Delete(TheFileNameParse.FullName());
214 err = db.Create(TheFileNameParse.FullName());
217 //Execute some operations with the private database
218 err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
220 err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
222 err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1);INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
224 err = db.Exec(_L8("COMMIT TRANSACTION"));
227 err = RSqlDatabase::Delete(TheFileNameParse.FullName());
228 TEST2(err, KErrNone);
232 TheFileNameParse.Set(KNullDesC, 0, 0);
235 //An attempt to create/open "C:[21212122]BBDb2.db" - this test has no enough rights to do that.
236 //...open as a non-secure database
237 err = db.Open(_L("C:[21212122]BBDb2.db"));
238 TEST2(err, KErrPermissionDenied);
239 //...create as a non-secure database
240 err = db.Create(_L("C:[21212122]BBDb2.db"));
241 TEST2(err, KErrArgument);//secure database name, no security policy
242 //Very long private database name
243 err = db.Create(_L("c:\\private\\21212124\\hhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh.db"));
244 TEST2(err, KErrBadName);
245 //Private database name without drive letter
246 err = db.Create(_L("\\private\\21212124\\dbname_no_drive_letter.db"));
247 TEST2(err, KErrBadName);
248 //Zero length private database name
249 err = db.Create(_L(""));
250 TEST2(err, KErrBadName);
251 //Private database + very long config string
252 _LIT8(KVeryLongConfig, "jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj");
253 err = db.Create(KTestDb2, &KVeryLongConfig);
254 TEST2(err, KErrArgument);
255 //...create as a secure database
256 RSqlSecurityPolicy dbSecurity;
257 TSecurityPolicy policy(TSecurityPolicy::EAlwaysPass);
258 err = dbSecurity.Create(policy);
259 TEST2(err, KErrNone);
260 err = db.Create(_L("C:[21212122]BBDb2.db"), dbSecurity);
261 TEST2(err, KErrPermissionDenied);
264 //An attempt to delete "C:[21212122]BBDb2.db" - this test has no enough rights to do that.
265 err = RSqlDatabase::Delete(_L("C:[21212122]BBDb2.db"));
266 TEST2(err, KErrPermissionDenied);
270 @SYMTestCaseID SYSLIB-SQL-CT-1765
271 @SYMTestCaseDesc The test creates two databases in the test application's private data cage.
272 Then the test inserts some records in both databases using separate RSqlDatabase objects.
273 The test closes both databases, then reopens the first one and attaches th second one.
274 Again, the test inserts some records in both databases, using single RSqlDatabase object.
275 The test reads the inserted records and verifies their column values.
276 The tests verifies that the SQL server can create a database in the application's
277 private data cage, can operate with the database and can attach private databases.
278 @SYMTestPriority High
279 @SYMTestActions SQL, Attach private database test.
280 @SYMTestExpectedResults Test must not fail
284 void AttachDbOpTest()
286 //Create private database 1
288 TInt err = db1.Create(KTestDb1);
289 TEST2(err, KErrNone);
290 //Execute some operations with private database 1
291 err = db1.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
293 err = db1.Exec(_L8("INSERT INTO A(ID) VALUES(1)"));
297 //Create private database 2
299 err = db2.Create(KTestDb2);
300 TEST2(err, KErrNone);
301 //Execute some operations with private database 2
302 err = db2.Exec(_L8("CREATE TABLE A(ID INTEGER, T TEXT)"));
304 err = db2.Exec(_L8("INSERT INTO A(ID, T) VALUES(1, 'NAME-NAME-NAME')"));
310 err = db.Open(KTestDb1);
311 TEST2(err, KErrNone);
312 //An attempt to attach a database with zero length name
313 err = db.Attach(_L(""), _L("Db2"));
314 TEST2(err, KErrBadName);
316 err = db.Attach(KTestDb2, _L("Db2"));
317 TEST2(err, KErrNone);
319 //Insert some records
320 err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
322 err = db.Exec(_L8("INSERT INTO Main.A(ID) VALUES(2);INSERT INTO Db2.A(ID, T) VALUES(2, 'AAA');"));
324 err = db.Exec(_L8("COMMIT TRANSACTION"));
327 err = db.Detach(_L("Db2"));
328 TEST2(err, KErrNone);
331 //Verify inserted data in database 2
332 err = db.Open(KTestDb2);
333 TEST2(err, KErrNone);
336 err = stmt.Prepare(db, _L8("SELECT * FROM A"));
337 TEST2(err, KErrNone);
340 TEST2(err, KSqlAtRow);
341 TEST2(stmt.ColumnInt(0), 1);
343 err = stmt.ColumnText(1, text);
344 TEST2(err, KErrNone);
345 TEST(text == _L("NAME-NAME-NAME"));
348 TEST2(err, KSqlAtRow);
349 TEST2(stmt.ColumnInt(0), 2);
350 err = stmt.ColumnText(1, text);
351 TEST2(err, KErrNone);
352 TEST(text == _L("AAA"));
355 TEST2(err, KSqlAtEnd);
359 //Open database 1, attach read-only database 2
360 err = db.Open(KTestDb1);
361 TEST2(err, KErrNone);
362 //Make database 2 read-only.
363 err = TheFs.SetAtt(KTestDb2, KEntryAttReadOnly, 0);
364 TEST2(err, KErrNone);
366 err = db.Attach(KTestDb2, _L("Db2"));
367 TEST2(err, KErrNone);
369 err = db.Exec(_L("INSERT INTO Db2.A(ID, T) VALUES(3, 'AAA')"));
370 TPtrC errmsg = db.LastErrorMessage();
371 TheTest.Printf(_L(" === Read-only private attached database. Msg=%S, err=%d\r\n"), &errmsg, err);
372 TEST(err != KErrNone);
373 TSqlScalarFullSelectQuery q(db);
375 TRAP(err, q.SelectTextL(_L("SELECT T FROM Db2.A WHERE ID=2"), text2));
376 TEST2(err, KErrNone);
377 TEST(text2 == _L("AAA"));
379 err = db.Detach(_L("Db2"));
380 TEST2(err, KErrNone);
381 err = TheFs.SetAtt(KTestDb2, 0, KEntryAttReadOnly);
382 TEST2(err, KErrNone);
385 err = RSqlDatabase::Delete(KTestDb2);
386 TEST2(err, KErrNone);
387 err = RSqlDatabase::Delete(KTestDb1);
388 TEST2(err, KErrNone);
392 @SYMTestCaseID SYSLIB-SQL-CT-1766
393 @SYMTestCaseDesc The test creates a database in the test application's private data cage.
394 Then the test creates two RSqlDatabase obejcts connecting them to the same
395 private database. The test inserts some records using both connections.
396 The test verifies that the inserted records are in the database file and
397 verifies the column values.
398 The test verifies that it is possible to make more than one connection to the
399 same private database and operate with the database using the database connections.
400 @SYMTestPriority High
401 @SYMTestActions SQL, Two database connections to the same private database test.
402 @SYMTestExpectedResults Test must not fail
408 //Create private database
410 TInt err = db1.Create(KTestDb1);
411 TEST2(err, KErrNone);
413 //Make a second connection with the test database
415 err = db2.Open(KTestDb1);
416 TEST2(err, KErrNone);
418 //Execute some operations with the private database
419 err = db1.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
421 err = db1.Exec(_L8("INSERT INTO A(ID) VALUES(1);"));
423 err = db2.Exec(_L8("INSERT INTO A(ID) VALUES(2);"));
426 //Verify inserted data
428 err = stmt.Prepare(db2, _L8("SELECT * FROM A"));
429 TEST2(err, KErrNone);
432 TEST2(err, KSqlAtRow);
433 TEST2(stmt.ColumnInt(0), 1);
435 TEST2(err, KSqlAtRow);
436 TEST2(stmt.ColumnInt(0), 2);
438 TEST2(err, KSqlAtEnd);
443 err = RSqlDatabase::Delete(KTestDb1);
444 TEST2(err, KErrNone);
447 //Test thread function.
448 //The test thread opens a database, begins a transaction and then simulates a crash within the transaction.
449 TInt ThreadFunc1(void*)
453 CTrapCleanup* tc = CTrapCleanup::New();
458 TInt err = db.Open(KTestDb1);
459 TTEST2(err, KErrNone);
461 RDebug::Print(_L("---:WorkThread: Begin transaction. Exec SQL...\r\n"));
463 //Begin a transaction
464 err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
467 //Execute INSERT sql statements
468 err = db.Exec(_L8("INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
471 RDebug::Print(_L("---:WorkThread: Notify the main thread about the SQL statement execution\r\n"));
472 MainCritSect.Signal();
474 RDebug::Print(_L("---:WorkThread: Wait for permisson to continue...\r\n"));
475 ThreadCritSect.Wait();
477 User::SetJustInTime(EFalse); // disable debugger panic handling
479 //Panic current thread without commiting the transaction (crash simulation)
480 RDebug::Print(_L("---:WorkThread: Panic!\r\n"));
481 User::Panic(KPanicCategory, KPanicCode);
491 @SYMTestCaseID SYSLIB-SQL-CT-1767
492 @SYMTestCaseDesc The test creates a database in the test application's private data cage.
493 The test does some operations with the database leaving it in state A.
494 Then the test creates a test thread and runs the thread. The test thread
495 opens a connection to the database, begins a transaction, inserts some data
496 and then simulates a crash within the transaction (kills the thread).
497 When the main thread takes the execution control, it reopens the database connection
498 and verifies that the database is in the same state A as it was before.
499 @SYMTestPriority High
500 @SYMTestActions SQL, Private database - transaction recovery test.
501 @SYMTestExpectedResults Test must not fail
507 //Create private database
509 TInt err = db.Create(KTestDb1);
510 TEST2(err, KErrNone);
512 //Execute some operations with the private database
513 err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
515 err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1)"));
518 //Check written records
520 err = stmt.Prepare(db, _L8("SELECT * FROM A"));
521 TEST2(err, KErrNone);
523 TEST2(err, KSqlAtRow);
524 TEST2(stmt.ColumnInt(0), 1);
526 TEST2(err, KSqlAtEnd);
531 //Run a test thread which will begin a transaction and then simulate a crash within the transaction
532 TEST2(ThreadCritSect.CreateLocal(), KErrNone);
533 ThreadCritSect.Wait();
534 TEST2(MainCritSect.CreateLocal(), KErrNone);
536 RDebug::Print(_L("+++:MainThread: Create the worker thread\r\n"));
537 _LIT(KThreadName, "WorkThrd");
539 TEST2(thread.Create(KThreadName, &ThreadFunc1, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
540 TRequestStatus status;
541 thread.Logon(status);
542 TEST2(status.Int(), KRequestPending);
544 RDebug::Print(_L("+++:MainThread: Wait SQL statement(s) to be executed...\r\n"));
546 RDebug::Print(_L("+++:MainThread: Notify the worker thread to simulate a crash...\r\n"));
547 ThreadCritSect.Signal();
548 User::WaitForRequest(status);
549 User::SetJustInTime(ETrue); // enable debugger panic handling
550 TEST2(thread.ExitType(), EExitPanic);
551 TEST2(thread.ExitReason(), KPanicCode);
553 MainCritSect.Close();
554 ThreadCritSect.Close();
556 //Reopen the test database. The failed transaction must be rolled back.
557 err = db.Open(KTestDb1);
558 TEST2(err, KErrNone);
559 //Verify that the database content is the same as before the failed transaction
560 err = stmt.Prepare(db, _L8("SELECT * FROM A"));
561 TEST2(err, KErrNone);
563 TEST2(err, KSqlAtRow);
564 TEST2(stmt.ColumnInt(0), 1);
566 TEST2(err, KSqlAtEnd);
570 err = RSqlDatabase::Delete(KTestDb1);
571 TEST2(err, KErrNone);
576 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1764 Simple private db operations "));
579 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1765 Private db operations - attach database "));
582 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1766 Private db operations - 2 database connections "));
585 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1767 Private db operations - transaction recovery "));
593 CTrapCleanup* tc = CTrapCleanup::New();
610 User::Heap().Check();