os/persistentdata/persistentstorage/sql/TEST/t_sqlprivcage.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include <e32test.h>
    17 #include <bautils.h>
    18 #include <sqldb.h>
    19 
    20 ///////////////////////////////////////////////////////////////////////////////////////
    21 
    22 static RFs TheFs;
    23 RTest TheTest(_L("t_sqlprivcage test"));
    24 
    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;
    31 
    32 static RCriticalSection ThreadCritSect;
    33 static RCriticalSection MainCritSect;
    34 
    35 _LIT(KPanicCategory, "TransFail");
    36 const TInt KPanicCode = 1111;
    37 
    38 ///////////////////////////////////////////////////////////////////////////////////////
    39 
    40 //Deletes all created test files.
    41 void DeleteTestFiles()
    42 	{
    43 	if(TheFileNameParse.FullName().Length() > 0)
    44 		{
    45 		(void)RSqlDatabase::Delete(TheFileNameParse.FullName());
    46 		}
    47 	(void)RSqlDatabase::Delete(KTestDb2);
    48 	(void)RSqlDatabase::Delete(KTestDb1);
    49 	}
    50 
    51 ///////////////////////////////////////////////////////////////////////////////////////
    52 ///////////////////////////////////////////////////////////////////////////////////////
    53 //Test macros and functions
    54 void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
    55 	{
    56 	if(!aValue)
    57 		{
    58 		DeleteTestFiles();
    59 		if(aPrintThreadName)
    60 			{
    61 			RThread th;
    62 			TName name = th.Name();
    63 			RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
    64 			}
    65 		else
    66 			{
    67 			RDebug::Print(_L("*** Line %d\r\n"), aLine);
    68 			}
    69 		TheTest(EFalse, aLine);
    70 		}
    71 	}
    72 void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
    73 	{
    74 	if(aValue != aExpected)
    75 		{
    76 		DeleteTestFiles();
    77 		if(aPrintThreadName)
    78 			{
    79 			RThread th;
    80 			TName name = th.Name();
    81 			RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
    82 			}
    83 		else
    84 			{
    85 			RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
    86 			}
    87 		TheTest(EFalse, aLine);
    88 		}
    89 	}
    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)
    94 
    95 ///////////////////////////////////////////////////////////////////////////////////////
    96 
    97 //Creates file session instance and the test directory
    98 void CreateTestEnv()
    99     {
   100 	TInt err = TheFs.Connect();
   101 	TEST2(err, KErrNone);
   102 
   103 	err = TheFs.CreatePrivatePath(EDriveC);
   104 	TEST(err == KErrNone || err == KErrAlreadyExists);
   105 
   106 	err = TheFs.MkDir(KTestDir);
   107 	TEST(err == KErrNone || err == KErrAlreadyExists);
   108 	}
   109 
   110 ///////////////////////////////////////////////////////////////////////////////////////
   111 
   112 /**
   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
   122 @SYMREQ					REQ5792
   123                         REQ5793
   124 */	
   125 void SimpleDbOpTest()
   126 	{
   127 	//Create private database
   128 	RSqlDatabase db;
   129 	TInt err = db.Create(KTestDb1);
   130 	TEST2(err, KErrNone);
   131 	
   132 	//Execute some operations with the private database
   133 	err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
   134 	TEST(err >= 0);	
   135 	err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1);INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
   136 	TEST(err >= 0);	
   137 
   138 	//Check written records
   139 	RSqlStatement stmt;
   140 	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
   141 	TEST2(err, KErrNone);
   142 	err = stmt.Next();
   143 	TEST2(err, KSqlAtRow);
   144 	TEST2(stmt.ColumnInt(0), 1);
   145 	err = stmt.Next();
   146 	TEST2(err, KSqlAtRow);
   147 	TEST2(stmt.ColumnInt(0), 2);
   148 	err = stmt.Next();
   149 	TEST2(err, KSqlAtRow);
   150 	TEST2(stmt.ColumnInt(0), 3);
   151 	err = stmt.Next();
   152 	TEST2(err, KSqlAtEnd);
   153 	stmt.Close();
   154 
   155 	db.Close();
   156 	//Open private database
   157 	err = db.Open(KTestDb1);
   158 	TEST2(err, KErrNone);
   159 
   160 	//Execute a DELETE transaction
   161 	err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
   162 	TEST(err >= 0);	
   163 
   164 	err = db.Exec(_L8("DELETE FROM A WHERE ID > 1"));
   165 	TEST(err >= 0);	
   166 
   167 	err = db.Exec(_L8("COMMIT TRANSACTION"));
   168 	TEST(err >= 0);	
   169 
   170 	//Check records left
   171 	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
   172 	TEST2(err, KErrNone);
   173 	err = stmt.Next();
   174 	TEST2(err, KSqlAtRow);
   175 	TEST2(stmt.ColumnInt(0), 1);
   176 	err = stmt.Next();
   177 	TEST2(err, KSqlAtEnd);
   178 	stmt.Close();
   179 
   180 	//Open private database which is on drive Z and with the same name
   181 	RSqlDatabase db2;
   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);
   188 	//Check records
   189 	err = stmt.Prepare(db2, _L8("SELECT * FROM A"));
   190 	TEST2(err, KErrNone);
   191 	err = stmt.Next();
   192 	TEST2(err, KSqlAtRow);
   193 	TEST2(stmt.ColumnInt(0), 1);
   194 	err = stmt.Next();
   195 	TEST2(err, KSqlAtEnd);
   196 	stmt.Close();
   197 	db2.Close();
   198 	
   199 	db.Close();
   200 	err = RSqlDatabase::Delete(KTestDb1);
   201 	TEST2(err, KErrNone);
   202 
   203 	//Create private database on drive different than C:
   204 	for(TInt drvNum=EDriveD;drvNum<=EDriveZ;++drvNum)
   205 		{
   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)
   212 			{
   213 			(void)RSqlDatabase::Delete(TheFileNameParse.FullName());
   214 			err = db.Create(TheFileNameParse.FullName());
   215 			if(err == KErrNone)
   216 				{
   217 				//Execute some operations with the private database
   218 				err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
   219 				TEST(err >= 0);	
   220 				err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
   221 				TEST(err >= 0);	
   222 				err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1);INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
   223 				TEST(err >= 0);	
   224 				err = db.Exec(_L8("COMMIT TRANSACTION"));
   225 				TEST(err >= 0);	
   226 				db.Close();
   227 				err = RSqlDatabase::Delete(TheFileNameParse.FullName());
   228 				TEST2(err, KErrNone);
   229 				break;
   230 				}
   231 			}
   232 		TheFileNameParse.Set(KNullDesC, 0, 0);
   233 		}
   234 		
   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);
   262 	dbSecurity.Close();
   263 
   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);
   267 	}
   268 
   269 /**
   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
   281 @SYMREQ					REQ5792
   282                         REQ5793
   283 */	
   284 void AttachDbOpTest()
   285 	{
   286 	//Create private database 1
   287 	RSqlDatabase db1;
   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)"));
   292 	TEST(err >= 0);	
   293 	err = db1.Exec(_L8("INSERT INTO A(ID) VALUES(1)"));
   294 	TEST2(err, 1);
   295 	db1.Close();
   296 
   297 	//Create private database 2
   298 	RSqlDatabase db2;
   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)"));
   303 	TEST(err >= 0);	
   304 	err = db2.Exec(_L8("INSERT INTO A(ID, T) VALUES(1, 'NAME-NAME-NAME')"));
   305 	TEST2(err, 1);
   306 	db2.Close();
   307 
   308 	//Open database 1
   309 	RSqlDatabase db;
   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);
   315     //Attach database 2
   316 	err = db.Attach(KTestDb2, _L("Db2"));
   317 	TEST2(err, KErrNone);
   318 
   319 	//Insert some records
   320 	err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
   321 	TEST(err >= 0);	
   322 	err = db.Exec(_L8("INSERT INTO Main.A(ID) VALUES(2);INSERT INTO Db2.A(ID, T) VALUES(2, 'AAA');"));
   323 	TEST(err >= 0);	
   324 	err = db.Exec(_L8("COMMIT TRANSACTION"));
   325 	TEST(err >= 0);	
   326 	
   327 	err = db.Detach(_L("Db2"));
   328 	TEST2(err, KErrNone);
   329 	db.Close();
   330 	
   331 	//Verify inserted data in database 2
   332 	err = db.Open(KTestDb2);
   333 	TEST2(err, KErrNone);
   334 	
   335 	RSqlStatement stmt;
   336 	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
   337 	TEST2(err, KErrNone);
   338 	
   339 	err = stmt.Next();
   340 	TEST2(err, KSqlAtRow);
   341 	TEST2(stmt.ColumnInt(0), 1);
   342 	TPtrC text;
   343 	err = stmt.ColumnText(1, text);
   344 	TEST2(err, KErrNone);
   345 	TEST(text == _L("NAME-NAME-NAME"));
   346 
   347 	err = stmt.Next();
   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"));
   353 	
   354 	err = stmt.Next();
   355 	TEST2(err, KSqlAtEnd);
   356 	stmt.Close();
   357 	db.Close();
   358 		
   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);
   365     //Attach database 2
   366     err = db.Attach(KTestDb2, _L("Db2"));
   367     TEST2(err, KErrNone);
   368     //
   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);
   374     TBuf<20> text2;
   375     TRAP(err, q.SelectTextL(_L("SELECT T FROM Db2.A WHERE ID=2"), text2));
   376     TEST2(err, KErrNone);
   377     TEST(text2 == _L("AAA"));
   378     //
   379     err = db.Detach(_L("Db2"));
   380     TEST2(err, KErrNone);
   381     err = TheFs.SetAtt(KTestDb2, 0, KEntryAttReadOnly);
   382     TEST2(err, KErrNone);
   383     db.Close();
   384     
   385 	err = RSqlDatabase::Delete(KTestDb2);
   386 	TEST2(err, KErrNone);
   387 	err = RSqlDatabase::Delete(KTestDb1);
   388 	TEST2(err, KErrNone);
   389 	}
   390 
   391 /**
   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
   403 @SYMREQ					REQ5792
   404                         REQ5793
   405 */	
   406 void TwoDbOpTest()
   407 	{
   408 	//Create private database
   409 	RSqlDatabase db1;
   410 	TInt err = db1.Create(KTestDb1);
   411 	TEST2(err, KErrNone);
   412 	
   413 	//Make a second connection with the test database
   414 	RSqlDatabase db2;
   415 	err = db2.Open(KTestDb1);
   416 	TEST2(err, KErrNone);
   417 	
   418 	//Execute some operations with the private database
   419 	err = db1.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
   420 	TEST(err >= 0);	
   421 	err = db1.Exec(_L8("INSERT INTO A(ID) VALUES(1);"));
   422 	TEST2(err, 1);
   423 	err = db2.Exec(_L8("INSERT INTO A(ID) VALUES(2);"));
   424 	TEST2(err, 1);
   425 
   426 	//Verify inserted data
   427 	RSqlStatement stmt;
   428 	err = stmt.Prepare(db2, _L8("SELECT * FROM A"));
   429 	TEST2(err, KErrNone);
   430 	
   431 	err = stmt.Next();
   432 	TEST2(err, KSqlAtRow);
   433 	TEST2(stmt.ColumnInt(0), 1);
   434 	err = stmt.Next();
   435 	TEST2(err, KSqlAtRow);
   436 	TEST2(stmt.ColumnInt(0), 2);
   437 	err = stmt.Next();
   438 	TEST2(err, KSqlAtEnd);
   439 	stmt.Close();
   440 
   441 	db2.Close();
   442 	db1.Close();
   443 	err = RSqlDatabase::Delete(KTestDb1);
   444 	TEST2(err, KErrNone);
   445 	}
   446 
   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*)
   450 	{
   451 	__UHEAP_MARK;
   452 	
   453 	CTrapCleanup* tc = CTrapCleanup::New();
   454 	TTEST(tc != NULL);
   455 
   456 	//Open test database
   457 	RSqlDatabase db;
   458 	TInt err = db.Open(KTestDb1);
   459 	TTEST2(err, KErrNone);
   460 
   461 	RDebug::Print(_L("---:WorkThread: Begin transaction. Exec SQL...\r\n"));
   462 	
   463 	//Begin a transaction
   464 	err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
   465 	TTEST(err >= 0);	
   466 
   467 	//Execute INSERT sql statements
   468 	err = db.Exec(_L8("INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
   469 	TTEST(err >= 0);	
   470 	
   471 	RDebug::Print(_L("---:WorkThread: Notify the main thread about the SQL statement execution\r\n"));
   472 	MainCritSect.Signal();
   473 
   474 	RDebug::Print(_L("---:WorkThread: Wait for permisson to continue...\r\n"));
   475 	ThreadCritSect.Wait();
   476 
   477 	User::SetJustInTime(EFalse);	// disable debugger panic handling
   478 
   479 	//Panic current thread without commiting the transaction (crash simulation)
   480 	RDebug::Print(_L("---:WorkThread: Panic!\r\n"));
   481 	User::Panic(KPanicCategory, KPanicCode);
   482 
   483 	delete tc;	
   484 	
   485 	__UHEAP_MARKEND;
   486 	
   487 	return KErrNone;		
   488 	}
   489 
   490 /**
   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
   502 @SYMREQ					REQ5792
   503                         REQ5793
   504 */	
   505 void TransDbOpTest()
   506 	{
   507 	//Create private database
   508 	RSqlDatabase db;
   509 	TInt err = db.Create(KTestDb1);
   510 	TEST2(err, KErrNone);
   511 	
   512 	//Execute some operations with the private database
   513 	err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
   514 	TEST(err >= 0);	
   515 	err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1)"));
   516 	TEST2(err, 1);
   517 
   518 	//Check written records
   519 	RSqlStatement stmt;
   520 	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
   521 	TEST2(err, KErrNone);
   522 	err = stmt.Next();
   523 	TEST2(err, KSqlAtRow);
   524 	TEST2(stmt.ColumnInt(0), 1);
   525 	err = stmt.Next();
   526 	TEST2(err, KSqlAtEnd);
   527 	stmt.Close();
   528 
   529 	db.Close();
   530 
   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);
   535 	MainCritSect.Wait();
   536 	RDebug::Print(_L("+++:MainThread: Create the worker thread\r\n"));
   537 	_LIT(KThreadName, "WorkThrd");
   538 	RThread thread;
   539 	TEST2(thread.Create(KThreadName, &ThreadFunc1, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
   540 	TRequestStatus status;
   541 	thread.Logon(status);
   542 	TEST2(status.Int(), KRequestPending);
   543 	thread.Resume();
   544 	RDebug::Print(_L("+++:MainThread: Wait SQL statement(s) to be executed...\r\n"));
   545 	MainCritSect.Wait();
   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);
   552 	thread.Close();
   553 	MainCritSect.Close();
   554 	ThreadCritSect.Close();
   555 
   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);
   562 	err = stmt.Next();
   563 	TEST2(err, KSqlAtRow);
   564 	TEST2(stmt.ColumnInt(0), 1);
   565 	err = stmt.Next();
   566 	TEST2(err, KSqlAtEnd);
   567 	stmt.Close();
   568 	db.Close();
   569 
   570 	err = RSqlDatabase::Delete(KTestDb1);
   571 	TEST2(err, KErrNone);
   572 	}
   573 	
   574 void DoTests()
   575 	{
   576 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1764 Simple private db operations "));
   577 	SimpleDbOpTest();
   578 
   579 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1765 Private db operations - attach database "));
   580 	AttachDbOpTest();
   581 
   582 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1766 Private db operations - 2 database connections "));
   583 	TwoDbOpTest();
   584 
   585 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1767 Private db operations - transaction recovery "));
   586 	TransDbOpTest();
   587 	}
   588 
   589 TInt E32Main()
   590 	{
   591 	TheTest.Title();
   592 	
   593 	CTrapCleanup* tc = CTrapCleanup::New();
   594 	
   595 	__UHEAP_MARK;
   596 	
   597 	CreateTestEnv();
   598 	DeleteTestFiles();
   599 	DoTests();
   600 	DeleteTestFiles();
   601 	TheFs.Close();
   602 
   603 	__UHEAP_MARKEND;
   604 	
   605 	TheTest.End();
   606 	TheTest.Close();
   607 	
   608 	delete tc;
   609 
   610 	User::Heap().Check();
   611 	return KErrNone;
   612 	}