os/persistentdata/persistentstorage/sql/TEST/t_sqlfserr.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 2007-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 #include <stdio.h>
    20 #include <stdlib.h>
    21 #include <string.h> 
    22 #include "sqlite3.h"
    23 #include "SqliteSymbian.h"
    24 
    25 ///////////////////////////////////////////////////////////////////////////////////////
    26 
    27 RTest TheTest(_L("t_sqlfserr test"));
    28 _LIT(KTestDir, "c:\\test\\");
    29 _LIT(KTestDbName, "c:\\test\\t_fserr.db");
    30 _LIT(KPrivateTestDbName, "c:\\private\\212A2C27\\t_fserr2.db");
    31 _LIT(KSecureTestDbName, "c:[212A2C27]t_fserr3.db");
    32 
    33 TFileName TheRmvMediaDbFileName;//The name of the file used for tests on a removable media
    34 RFs TheFs;
    35 RSqlDatabase TheDb;
    36 
    37 //The next constants are used in the "blob write" test
    38 const TInt KWriteCnt = 9;
    39 const TInt KBlobSize = 397 * KWriteCnt;
    40 _LIT(KAttachDb, "AttachDb");
    41 
    42 //In order to be able to compile the test, the following variables are defined (used inside the OS porting layer, when _SQLPROFILER macro is defined)
    43 #ifdef _SQLPROFILER
    44 TInt TheSqlSrvProfilerFileRead = 0;
    45 TInt TheSqlSrvProfilerFileWrite = 0;
    46 TInt TheSqlSrvProfilerFileSync = 0;
    47 TInt TheSqlSrvProfilerFileSetSize = 0;
    48 #endif
    49 
    50 ///////////////////////////////////////////////////////////////////////////////////////
    51 
    52 TBool FileExists(const TDesC& aFileName)
    53 	{
    54 	TEntry entry;
    55 	return TheFs.Entry(aFileName, entry) == KErrNone;
    56 	}
    57 
    58 void DestroyTestEnv()
    59 	{
    60 	TheDb.Close();
    61     (void)RSqlDatabase::Delete(KSecureTestDbName);
    62     (void)RSqlDatabase::Delete(KPrivateTestDbName);
    63 	(void)RSqlDatabase::Delete(KTestDbName);
    64 	(void)RSqlDatabase::Delete(TheRmvMediaDbFileName);
    65 	TheFs.Close();
    66 	sqlite3SymbianLibFinalize();
    67 	CloseSTDLIB();
    68 	}
    69 
    70 ///////////////////////////////////////////////////////////////////////////////////////
    71 ///////////////////////////////////////////////////////////////////////////////////////
    72 //Test macros and functions
    73 void Check(TInt aValue, TInt aLine)
    74 	{
    75 	if(!aValue)
    76 		{
    77 		DestroyTestEnv();
    78 		TheTest(EFalse, aLine);
    79 		}
    80 	}
    81 void Check(TInt aValue, TInt aExpected, TInt aLine)
    82 	{
    83 	if(aValue != aExpected)
    84 		{
    85 		DestroyTestEnv();
    86 		RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
    87 		TheTest(EFalse, aLine);
    88 		}
    89 	}
    90 #define TEST(arg) ::Check((arg), __LINE__)
    91 #define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
    92 
    93 void SqliteCheck(sqlite3* aDbHandle, TInt aValue, TInt aExpected, TInt aLine)
    94 	{
    95 	if(aValue != aExpected)
    96 		{
    97 		RDebug::Print(_L("*** SQLITE: Expected error: %d, got: %d\r\n"), aExpected, aValue);
    98 		if(aDbHandle)
    99 			{
   100 			const char* errMsg = sqlite3_errmsg(aDbHandle);
   101 			TPtrC8 ptr8((const TUint8*)errMsg, strlen(errMsg));
   102 			TBuf<200> buf;
   103 			buf.Copy(ptr8);
   104 			RDebug::Print(_L("*** SQLITE error message: \"%S\"\r\n"), &buf);
   105 			}
   106 		DestroyTestEnv();
   107 		TheTest(EFalse, aLine);
   108 		}
   109 	}
   110 #define SQLITE_TEST(aDbHandle, aValue, aExpected) ::SqliteCheck(aDbHandle, aValue, aExpected, __LINE__)
   111 
   112 ///////////////////////////////////////////////////////////////////////////////////////
   113 
   114 void SetupTestEnv()
   115     {
   116 	TInt err = TheFs.Connect();
   117 	TEST2(err, KErrNone);
   118 
   119 	err = TheFs.MkDir(KTestDir);
   120 	TEST(err == KErrNone || err == KErrAlreadyExists);
   121 
   122 	sqlite3SymbianLibInit();
   123 	}
   124 
   125 TBool CheckRecord(const TDesC& aDbName, TInt aId, const TDesC& aExpectedName, TBool aOpenDb = ETrue)
   126 	{
   127 	if(aOpenDb)
   128 		{
   129 		TEST2(TheDb.Open(aDbName), KErrNone);
   130 		}
   131 	TBuf<64> sql;
   132 	sql.Copy(_L("SELECT Name FROM A WHERE Id="));
   133 	sql.AppendNum(aId);
   134 	TSqlScalarFullSelectQuery q(TheDb);
   135 	TBuf<20> name;
   136 	TRAPD(err, (void)q.SelectTextL(sql, name));
   137 	TEST2(err, KErrNone);
   138 	if(aOpenDb)
   139 		{
   140 		TheDb.Close();
   141 		}
   142 	return name == aExpectedName;
   143 	}
   144 
   145 ///////////////////////////////////////////////////////////////////////////////////////
   146 
   147 /**
   148 @SYMTestCaseID			SYSLIB-SQL-UT-3419
   149 @SYMTestCaseDesc		Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   150 						The test creates a test database with one table, inserts one record.
   151 						Then the test attempts to update the existing record while simulating file I/O failures.
   152 						After each test iteration the database content is tested and is expected to be the same
   153 						as it was before the test. RSqlDatabase::Exec() is used for the update operation.
   154 @SYMTestPriority		High
   155 @SYMTestActions			Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   156 @SYMTestExpectedResults The test must not fail
   157 @SYMDEF					DEF103859
   158 */
   159 void AlterDatabaseTest()
   160 	{
   161 	(void)RSqlDatabase::Delete(KTestDbName);
   162 	TInt err = TheDb.Create(KTestDbName);
   163 	TEST2(err, KErrNone);
   164 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
   165 	TEST(err >= 0);
   166 	TheDb.Close();
   167 	err = KErrNotFound;
   168 	for(TInt cnt=0;err<KErrNone;++cnt)
   169 		{
   170 		TheTest.Printf(_L("%d \r"), cnt);		
   171 		for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
   172 			{
   173 			//Preprocessing
   174 			TEST2(TheDb.Open(KTestDbName), KErrNone);
   175 			(void)TheDb.Exec(_L("DELETE FROM A WHERE Id=1"));
   176 			err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
   177 			TEST2(err, 1);
   178 			//The test
   179 			(void)TheFs.SetErrorCondition(fsError, cnt);
   180 			err = TheDb.Exec(_L("UPDATE A SET Name='Name2' WHERE Id=1"));
   181 			(void)TheFs.SetErrorCondition(KErrNone);
   182 			if(err < 1)
   183 				{
   184 				TheDb.Close();//close the database to recover from the last error
   185 				//check the database content - all bets are off in a case of an I/O error. 
   186 				//The existing record might have been updated.
   187 				TEST(CheckRecord(KTestDbName, 1, _L("Name")) || CheckRecord(KTestDbName, 1, _L("Name2")));
   188 				}
   189 			else
   190 				{
   191 				TEST2(err, 1);
   192 				//check the database content has been modified by the operation. 
   193 				TEST(CheckRecord(KTestDbName, 1, _L("Name2"), EFalse));
   194 				TheDb.Close();
   195 				}
   196 			}
   197 		}
   198 	(void)TheFs.SetErrorCondition(KErrNone);
   199 	TEST2(err, 1);
   200 	//check the database content (transaction durability). 
   201 	TEST(CheckRecord(KTestDbName, 1, _L("Name2")));
   202 	err = RSqlDatabase::Delete(KTestDbName);
   203 	TEST2(err, KErrNone);
   204 	TheTest.Printf(_L("\r\n"));
   205 	}
   206 
   207 /**
   208 @SYMTestCaseID			SYSLIB-SQL-UT-3420
   209 @SYMTestCaseDesc		Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   210 						The test creates a test database with one table, inserts one record.
   211 						Then the test attempts to update the existing record while simulating file I/O failures.
   212 						After each test iteration the database content is tested and is expected to be the same
   213 						as it was before the test. RSqlStatement::Exec() is used for the update operation.
   214 @SYMTestPriority		High
   215 @SYMTestActions			Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   216 @SYMTestExpectedResults The test must not fail
   217 @SYMDEF					DEF103859
   218 */
   219 void AlterDatabaseTest2()
   220 	{
   221 	(void)RSqlDatabase::Delete(KTestDbName);
   222 	TInt err = TheDb.Create(KTestDbName);
   223 	TEST2(err, KErrNone);
   224 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
   225 	TEST(err >= 0);
   226 	TheDb.Close();
   227 	err = KErrNotFound;
   228 	for(TInt cnt=0;err<KErrNone;++cnt)
   229 		{
   230 		TheTest.Printf(_L("%d \r"), cnt);		
   231 		for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
   232 			{
   233 			//Preprocessing
   234 			TEST2(TheDb.Open(KTestDbName), KErrNone);
   235 			(void)TheDb.Exec(_L("DELETE FROM A WHERE Id=1"));
   236 			err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
   237 			TEST2(err, 1);
   238 			//The test
   239 			(void)TheFs.SetErrorCondition(fsError, cnt);
   240 			RSqlStatement stmt;
   241 			err = stmt.Prepare(TheDb, _L("UPDATE A SET Name='Name2' WHERE Id=1"));
   242 			if(err == KErrNone)
   243 				{
   244 				err = stmt.Exec();	
   245 				}
   246 			(void)TheFs.SetErrorCondition(KErrNone);
   247 			stmt.Close();
   248 			if(err < 1)
   249 				{
   250 				TheDb.Close();//close the database to recover from the last error
   251 				//check the database content - all bets are off in a case of an I/O error. 
   252 				//The existing record might have been updated.
   253 				TEST(CheckRecord(KTestDbName, 1, _L("Name")) || CheckRecord(KTestDbName, 1, _L("Name2")));
   254 				}
   255 			else
   256 				{
   257 				TEST2(err, 1);
   258 				//check the database content has been modified by the operation. 
   259 				TEST(CheckRecord(KTestDbName, 1, _L("Name2"), EFalse));
   260 				TheDb.Close();
   261 				}
   262 			}
   263 		}
   264 	(void)TheFs.SetErrorCondition(KErrNone);
   265 	TEST2(err, 1);
   266 	//check the database content has been modified by the operation. 
   267 	TEST(CheckRecord(KTestDbName, 1, _L("Name2")));
   268 	err = RSqlDatabase::Delete(KTestDbName);
   269 	TEST2(err, KErrNone);
   270 	TheTest.Printf(_L("\r\n"));
   271 	}
   272 
   273 void CreateTestSecurityPolicy(RSqlSecurityPolicy& aSecurityPolicy)
   274     {
   275     TSecurityPolicy alwaysPassPolicy(TSecurityPolicy::EAlwaysPass);
   276     TInt err = aSecurityPolicy.Create(alwaysPassPolicy);
   277     TEST2(err, KErrNone);
   278 
   279     err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::ESchemaPolicy, alwaysPassPolicy);
   280     TEST2(err, KErrNone);
   281     err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::EWritePolicy, alwaysPassPolicy);
   282     TEST2(err, KErrNone);
   283     err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::EReadPolicy, alwaysPassPolicy);
   284     TEST2(err, KErrNone);
   285     }
   286 
   287 //Creates public shared, private secure and public secure databases.
   288 void DoCreateTestDatabases(const TPtrC aDbName[], TInt aCount)
   289     {
   290     TEST(aCount > 0);
   291     for(TInt i=0;i<aCount;++i)
   292         {
   293         TheTest.Printf(_L("Database: \"%S\"\r\n"), &aDbName[i]);       
   294         (void)RSqlDatabase::Delete(aDbName[i]);
   295         TInt err = KErrGeneral;
   296         if(i == (aCount - 1))
   297             {
   298             RSqlSecurityPolicy policy;
   299             CreateTestSecurityPolicy(policy);
   300             err = TheDb.Create(aDbName[i], policy);
   301             policy.Close();
   302             }
   303         else
   304             {
   305             err = TheDb.Create(aDbName[i]);
   306             }
   307         TEST2(err, KErrNone);
   308         err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
   309         TEST(err >= 0);
   310         err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
   311         TEST2(err, 1);
   312         TheDb.Close();
   313         }
   314     }
   315 
   316 /**
   317 @SYMTestCaseID			SYSLIB-SQL-UT-3421
   318 @SYMTestCaseDesc		Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   319 						The test creates a test database with one table, inserts one record.
   320 						Then the test attempts to open the database while simulating file I/O failures.
   321 						At the end of the test the database content is tested and is expected to be the same
   322 						as it was before the test. RSqlStatement::Open() is used in the test.
   323 @SYMTestPriority		High
   324 @SYMTestActions			Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   325 @SYMTestExpectedResults The test must not fail
   326 @SYMDEF					DEF103859
   327 */
   328 void OpenDatabaseTest()
   329 	{
   330     TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
   331     const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
   332     DoCreateTestDatabases(dbName, KDbNameCnt);
   333     for(TInt k=0;k<KDbNameCnt;++k)
   334         {	
   335         TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);       
   336         TInt err = KErrNotFound;
   337         for(TInt cnt=0;err<KErrNone;++cnt)
   338             {		
   339             TheTest.Printf(_L("%d \r"), cnt);		
   340             for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
   341                 {
   342                 (void)TheFs.SetErrorCondition(fsError, cnt);
   343                 err = TheDb.Open(dbName[k]);
   344                 (void)TheFs.SetErrorCondition(KErrNone);
   345                 if(err != KErrNone)
   346                     {
   347                     TheDb.Close();//close the database to recover from the last error
   348                     //check the database content is still the same as before the "open" call
   349                     TEST(CheckRecord(dbName[k], 1, _L("Name")));
   350                     }
   351                 else
   352                     {
   353                     TEST2(err, KErrNone);
   354                     //check the database content is still the same as before the operation, without closing the database
   355                     TEST(CheckRecord(dbName[k], 1, _L("Name"), EFalse));
   356                     TheDb.Close();
   357                     }
   358                 }
   359             }
   360         (void)TheFs.SetErrorCondition(KErrNone);
   361         TEST2(err, KErrNone);
   362         //check the database content is the same as before the operation, after reopening the database.
   363         TEST(CheckRecord(dbName[k], 1, _L("Name")));
   364         err = RSqlDatabase::Delete(dbName[k]);
   365         TEST2(err, KErrNone);
   366         TheTest.Printf(_L("\r\n"));
   367         }//end of: for(TInt k=0;k<KDbNameCnt;++k)
   368 	}
   369 
   370 /**
   371 @SYMTestCaseID			SYSLIB-SQL-UT-3434
   372 @SYMTestCaseDesc		Test for DEF104820 "SQL, RSqlDatabase::Create() does not delete the file if fails".
   373 						Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   374 						Then the test attempts to create a database while simulating file I/O failures.
   375 						When the test succeeds, the test verifies that the database file does exist.
   376 @SYMTestPriority		High
   377 @SYMTestActions			Test for DEF104820 "SQL, RSqlDatabase::Create() does not delete the file if fails".
   378 						Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
   379 @SYMTestExpectedResults The test must not fail
   380 @SYMDEF					DEF103859
   381 */
   382 void CreateDatabaseTest()
   383 	{
   384 	RSqlSecurityPolicy policy;
   385 	CreateTestSecurityPolicy(policy);
   386 	
   387 	TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
   388 	const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
   389 	for(TInt k=0;k<KDbNameCnt;++k)
   390 	    {
   391         TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);       
   392         TInt err = -1;
   393         for(TInt cnt=0;err<KErrNone;++cnt)
   394             {		
   395             TheTest.Printf(_L("%d \r"), cnt);		
   396             for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
   397                 {
   398                 //Ideally, the database should be deleted by the SQL server, if RSqlDatabase::Create() fails.
   399                 //But SetErrorCondition() makes the error persistent, so the SQL server will fail to delete the file.
   400                 //This is the reason, RSqlDatabase::Delete()to be used, before simulating file I/O error.
   401                 (void)RSqlDatabase::Delete(dbName[k]);
   402                 (void)TheFs.SetErrorCondition(fsError, cnt);
   403                 err = (k == (KDbNameCnt - 1)) ? TheDb.Create(dbName[k], policy) : TheDb.Create(dbName[k]);
   404                 (void)TheFs.SetErrorCondition(KErrNone);
   405                 TheDb.Close();
   406                 //If err != KErrNone, the database file should have been already deleted by the server and here is 
   407                 //the place to check that. But since the file I/O failure simulation makes the file I/O error 
   408                 //persistent, the file cannot be deleted by the server, because the "file delete" operation also fails.
   409                 }
   410             }
   411         (void)TheFs.SetErrorCondition(KErrNone);
   412         TheDb.Close();
   413         TEST2(err, KErrNone);
   414         if( k != (KDbNameCnt - 1))
   415             {
   416             TEST(FileExists(dbName[k]));
   417             }
   418         err = RSqlDatabase::Delete(dbName[k]);
   419         TEST2(err, KErrNone);
   420         TheTest.Printf(_L("\r\n"));
   421 	    }
   422 	policy.Close();
   423 	}
   424 
   425 /**
   426 @SYMTestCaseID          PDS-SQL-UT-4189
   427 @SYMTestCaseDesc        Test for DEF145125 "SQL, low code coverage".
   428                         The test creates public shared, private secure and public secure test databases.
   429                         Then the test opens the publich shared database and attempts to attach one of the other two
   430                         in a file I/O error simulation loop.
   431 @SYMTestPriority        High
   432 @SYMTestActions         Test for DEF145125 - "SQL, low code coverage".
   433 @SYMTestExpectedResults The test must not fail
   434 @SYMDEF                 DEF145125 
   435 */
   436 void AttachDatabaseTest()
   437     {
   438     TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
   439     const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
   440     DoCreateTestDatabases(dbName, KDbNameCnt);
   441     for(TInt k=1;k<KDbNameCnt;++k)
   442         {
   443         TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);       
   444         TInt err = KErrGeneral;
   445         for(TInt cnt=0;err<KErrNone;++cnt)
   446             {
   447             TheTest.Printf(_L("%d \r"), cnt);       
   448             for(TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
   449                 {
   450                 err = TheDb.Open(KTestDbName);
   451                 TEST2(err, KErrNone);
   452                 (void)TheFs.SetErrorCondition(fsError, cnt);
   453                 err = TheDb.Attach(dbName[k], _L("DB2"));
   454                 (void)TheFs.SetErrorCondition(KErrNone);
   455                 (void)TheDb.Detach(_L("DB2"));
   456                 TheDb.Close();//close the database to recover from the last error
   457                 }
   458             }
   459         TEST2(err, KErrNone);
   460         err = RSqlDatabase::Delete(dbName[k]);
   461         TEST2(err, KErrNone);
   462         TheTest.Printf(_L("\r\n"));
   463         }
   464     }
   465 
   466 /**
   467 @SYMTestCaseID          PDS-SQL-UT-4190
   468 @SYMTestCaseDesc        Test for DEF145125 "SQL, low code coverage".
   469                         The tests attempts to delete a database in a file I/O error simulation loop.
   470 @SYMTestPriority        High
   471 @SYMTestActions         Test for DEF145125 - "SQL, low code coverage".
   472 @SYMTestExpectedResults The test must not fail
   473 @SYMDEF                 DEF145125 
   474 */
   475 void DeleteDatabaseTest()
   476     {
   477     TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
   478     const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
   479     DoCreateTestDatabases(dbName, KDbNameCnt);
   480     for(TInt k=0;k<KDbNameCnt;++k)
   481         {   
   482         TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);       
   483         TInt err = KErrGeneral;
   484         for(TInt cnt=1;err<KErrNone;++cnt)
   485             {
   486             TheTest.Printf(_L("%d \r"), cnt);
   487             (void)TheFs.SetErrorCondition(KErrGeneral, cnt);
   488             err = RSqlDatabase::Delete(dbName[k]);
   489             (void)TheFs.SetErrorCondition(KErrNone);
   490             }
   491         TEST2(err, KErrNone);
   492         err = RSqlDatabase::Delete(KTestDbName);
   493         TEST2(err, KErrNotFound);
   494         }    
   495     }
   496 
   497 /**
   498 @SYMTestCaseID			SYSLIB-SQL-UT-3462
   499 @SYMTestCaseDesc		Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
   500 						The test creates a test database with one table, inserts one record.
   501 						Then the test attempts to retrieve the existing record while simulating file I/O failures.
   502 						After each iteration, the database content is tested, that it has not been modified by the operation.
   503 @SYMTestPriority		High
   504 @SYMTestActions			Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
   505 @SYMTestExpectedResults The test must not fail
   506 @SYMDEF					DEF105434
   507 */
   508 void SelectRecordTest()
   509 	{
   510 	(void)RSqlDatabase::Delete(KTestDbName);
   511 	TInt err = TheDb.Create(KTestDbName);
   512 	TEST2(err, KErrNone);
   513 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER, Name TEXT)"));
   514 	TEST(err >= 0);
   515 	err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
   516 	TEST2(err, 1);
   517 	TheDb.Close();
   518 	err = -1;
   519 	for(TInt cnt=0;err<KErrNone;++cnt)
   520 		{		
   521 		TheTest.Printf(_L("%d \r"), cnt);		
   522 		err = TheDb.Open(KTestDbName);
   523 		TEST2(err, KErrNone);
   524 		RSqlStatement stmt;
   525 		(void)TheFs.SetErrorCondition(KErrGeneral, cnt);
   526 		err = stmt.Prepare(TheDb, _L("SELECT * FROM A WHERE Id=?"));
   527 		if(err == KErrNone)
   528 			{
   529 			err = stmt.BindInt(0, 1);
   530 			if(err == KErrNone)
   531 				{
   532 				err = stmt.Next();
   533 				TEST(err == KSqlAtRow || err < 0);
   534 				if(err == KSqlAtRow)
   535 					{
   536 					TInt id = stmt.ColumnInt(0);
   537 					TEST2(id, 1);
   538 					TPtrC name;
   539 					err = stmt.ColumnText(1, name);
   540 					TEST2(err, KErrNone);
   541 					TEST(name == _L("Name"));
   542 					}
   543 				}
   544 			}
   545 		(void)TheFs.SetErrorCondition(KErrNone);
   546 		stmt.Close();
   547 		TheDb.Close();
   548 		//check the database content is the same as before the operation
   549 		TEST(CheckRecord(KTestDbName, 1, _L("Name")));
   550 		}
   551 	(void)TheFs.SetErrorCondition(KErrNone);
   552 	TEST(err >= 0);
   553 	TheDb.Close();
   554 	//check the database content is the same as before the operation, after reopening the database.
   555 	TEST(CheckRecord(KTestDbName, 1, _L("Name")));
   556 	err = RSqlDatabase::Delete(KTestDbName);
   557 	TEST2(err, KErrNone);
   558 	TheTest.Printf(_L("\r\n"));
   559 	}
   560 
   561 /**
   562 @SYMTestCaseID			SYSLIB-SQL-UT-3463
   563 @SYMTestCaseDesc		Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
   564 						The test creates a test database with one table, inserts one record.
   565 						Then the test attempts to insert another while simulating file I/O failures.
   566 						After each iteration, the database content is tested, that it has not been modified by the operation.
   567 						If the operation succeeds, the database content is tested again to check that the inserted record is there.
   568 @SYMTestPriority		High
   569 @SYMTestActions			Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
   570 @SYMTestExpectedResults The test must not fail
   571 @SYMDEF					DEF105434
   572 */
   573 void InsertRecordTest()
   574 	{
   575 	(void)RSqlDatabase::Delete(KTestDbName);
   576 	TInt err = TheDb.Create(KTestDbName);
   577 	TEST2(err, KErrNone);
   578 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
   579 	TEST(err >= 0);
   580 	err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
   581 	TEST2(err, 1);
   582 	TheDb.Close();
   583 	err = -1;
   584 	for(TInt cnt=0;err<KErrNone;++cnt)
   585 		{		
   586 		TheTest.Printf(_L("%d \r"), cnt);		
   587 		err = TheDb.Open(KTestDbName);
   588 		TEST2(err, KErrNone);
   589 		RSqlStatement stmt;
   590 		(void)TheFs.SetErrorCondition(KErrGeneral, cnt);
   591 		err = stmt.Prepare(TheDb, _L("INSERT INTO A(Id,Name) VALUES(2, 'Name2')"));
   592 		if(err == KErrNone)
   593 			{
   594 			err = TheDb.Exec(_L("BEGIN TRANSACTION"));
   595 			if(err == KErrNone)
   596 				{
   597 				err = stmt.Exec();
   598 				TEST(err == 1 || err < 0);
   599 				if(err == 1)
   600 					{
   601 					err = TheDb.Exec(_L("COMMIT TRANSACTION"));
   602 					}
   603 				}
   604 			}
   605 		(void)TheFs.SetErrorCondition(KErrNone);
   606 		stmt.Close();
   607 		if(err < 1)
   608 			{
   609 			TheDb.Close();//close the database to recover from the last error
   610 			//check that the database contains the "name" record that has been inserted before the file I/O failure test.
   611 			TEST(CheckRecord(KTestDbName, 1, _L("Name")));
   612 			}
   613 		else
   614 			{
   615 			TEST2(err, 1);
   616 			//check the database content has been modified by the operation, without closing the database.
   617 			TEST(CheckRecord(KTestDbName, 1, _L("Name"), EFalse));
   618 			TEST(CheckRecord(KTestDbName, 2, _L("Name2"), EFalse));
   619 			TheDb.Close();
   620 			}
   621 		}
   622 	(void)TheFs.SetErrorCondition(KErrNone);
   623 	TEST2(err, 1);
   624 	//check the database content (transaction durability).
   625 	TEST(CheckRecord(KTestDbName, 1, _L("Name")));
   626 	TEST(CheckRecord(KTestDbName, 2, _L("Name2")));
   627 	(void)RSqlDatabase::Delete(KTestDbName);
   628 	TheTest.Printf(_L("\r\n"));
   629 	}
   630 
   631 /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   632 //////////////////////////////////         Removable media robustness test      /////////////////////////////////////////
   633 /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   634 
   635 _LIT8(KNameColData, "A123456789012345678901234567890");
   636 _LIT8(KUpdatedNameColData, "1234");
   637 
   638 //TRemovableMediaTest class is used for testing the SQLITE behaviour when the database file is created on a removable media where
   639 //the cluster size is bigger than the page size and in case of a power failure is not guaranteed that the content
   640 //of the last updated cluster will be preserved.
   641 class TRemovableMediaTest
   642 	{
   643 	enum {KTestRecCnt = 200};
   644 	enum {KMinCachePageSize = 512};
   645 	
   646 public:	
   647 	void Run();
   648 	
   649 private:	
   650 	TInt GetRemovableMediaDriveNo();
   651 	TInt ClusterSize(TInt aDriveNo);
   652 	void CreateDatabase(TInt aDriveNo, TInt aCachePageSize);
   653 	void CheckRecord(sqlite3_stmt* aStmt, TInt aRecId);
   654 	void VerifyDatabase();
   655 	void DoTest();
   656 
   657 	};
   658 
   659 //Returns the number of the removable media drive, KErrNotFound otherwise.
   660 TInt TRemovableMediaTest::GetRemovableMediaDriveNo()
   661 	{
   662 	for(TInt driveNo=EDriveA;driveNo<=EDriveZ;++driveNo)
   663 		{
   664 		TDriveInfo driveInfo;
   665 		TInt err = TheFs.Drive(driveInfo, driveNo);
   666 		if(err == KErrNone)
   667 			{
   668 			_LIT(KType1, "Not present");
   669 			_LIT(KType2, "Unknown");
   670 			_LIT(KType3, "Floppy");
   671 			_LIT(KType4, "Hard disk");
   672 			_LIT(KType5, "CD ROM");
   673 			_LIT(KType6, "RAM disk");
   674 			_LIT(KType7, "Flash");
   675 			_LIT(KType8, "ROM drive");
   676 			_LIT(KType9, "Remote drive");
   677 			_LIT(KType10,"NAND flash");
   678 			_LIT(KType11,"Rotating media");
   679 			_LIT(KYes, "Yes");
   680 			_LIT(KNo,  "No ");
   681 			TPtrC KMediaTypeNames[] = {KType1(), KType2(), KType3(), KType4(), KType5(), KType6(), KType7(), KType8(), KType9(), KType10(), KType11()};
   682 			TheTest.Printf(_L("Drive: %C:, %S, Removable: %S\r\n"), 'A' + driveNo, &KMediaTypeNames[driveInfo.iType], 
   683 																	driveInfo.iDriveAtt & KDriveAttRemovable ? &KYes : &KNo);
   684 			if(driveInfo.iDriveAtt & KDriveAttRemovable)
   685 				{
   686 				TheTest.Printf(_L("Removable drive to test: %C:\r\n"), 'A' + driveNo);
   687 				return driveNo;	
   688 				}
   689 			}
   690 		}
   691 	return KErrNotFound;
   692 	}
   693 
   694 //Get the cluster size of aDriveNo drive	
   695 TInt TRemovableMediaTest::ClusterSize(TInt aDriveNo)
   696 	{
   697 	__ASSERT_DEBUG((TUint)aDriveNo <= EDriveZ, User::Invariant());
   698 	TVolumeIOParamInfo volIoParams;
   699 	TInt err = TheFs.VolumeIOParam(aDriveNo, volIoParams);
   700 	return (err == KErrNone) ? volIoParams.iClusterSize : err;
   701 	}
   702 
   703 //Create a test database on aDriveNo with aCachePageSize page size.
   704 //Insert KTestRecCnt records.
   705 void TRemovableMediaTest::CreateDatabase(TInt aDriveNo, TInt aCachePageSize)
   706 	{
   707 	__ASSERT_DEBUG((TUint)aDriveNo <= EDriveZ, User::Invariant());
   708 	__ASSERT_DEBUG(aCachePageSize > 0, User::Invariant());
   709 	TDriveUnit drvUnit(aDriveNo);
   710 	_LIT(KDbName, "\\flashmedia.db");
   711 	TParse parse;
   712 	parse.Set(drvUnit.Name(), &KDbName, 0);
   713 	TheRmvMediaDbFileName.Copy(parse.FullName());
   714 	TBuf8<KMaxFileName + 1> dbFileName8;
   715 	dbFileName8.Copy(TheRmvMediaDbFileName);
   716 	(void)TheFs.Delete(TheRmvMediaDbFileName);
   717 	
   718 	sqlite3* dbHandle = NULL;
   719 	TInt rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
   720 	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   721 	__ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
   722 	
   723 	TBuf8<40> config;
   724 	config.Copy(_L8("PRAGMA PAGE_SIZE="));
   725 	config.AppendNum(aCachePageSize);
   726 	rc = sqlite3_exec(dbHandle, (const char*)config.PtrZ(), 0, 0, 0);
   727 	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   728 
   729 	rc = sqlite3_exec(dbHandle, "CREATE TABLE A(Id INTEGER,Name TEXT)", 0, 0, 0);
   730 	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   731 	rc = sqlite3_exec(dbHandle, "BEGIN", 0, 0, 0);
   732 	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   733 	for(TInt recid=1;recid<=KTestRecCnt;++recid)
   734 		{
   735 		
   736 		TBuf8<100> sql;
   737 		sql.Copy(_L8("INSERT INTO A VALUES("));
   738 		sql.AppendNum(recid);
   739 		sql.Append(_L8(",'"));
   740 		sql.Append(KNameColData);
   741 		sql.Append(_L8("')"));
   742 		rc = sqlite3_exec(dbHandle, (const char*)sql.PtrZ(), 0, 0, 0);
   743 		SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   744 		}
   745 	rc = sqlite3_exec(dbHandle, "COMMIT", 0, 0, 0);
   746 	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   747 	sqlite3_close(dbHandle);
   748 	}
   749 
   750 //Checks the content of a single record
   751 void TRemovableMediaTest::CheckRecord(sqlite3_stmt* aStmt, TInt aRecId)
   752 	{
   753 	__ASSERT_DEBUG(aStmt != NULL, User::Invariant());
   754 	TInt id = sqlite3_column_int(aStmt, 0);
   755 	TEST2(id, aRecId);
   756 	const TUint8* text = (const TUint8*)sqlite3_column_text(aStmt, 1);
   757 	TPtrC8 name(text, User::StringLength(text));
   758 	TEST(KNameColData() == name || KUpdatedNameColData() == name);
   759 	}
   760 
   761 //Verifies that the database content is either the same as it was before the UPDATE operation or 
   762 //it has been updated with the new data.
   763 void TRemovableMediaTest::VerifyDatabase()
   764 	{
   765 	TBuf8<KMaxFileName + 1> dbFileName8;
   766 	dbFileName8.Copy(TheRmvMediaDbFileName);
   767 	
   768 	sqlite3* dbHandle = NULL;
   769 	TInt rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
   770 	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   771 	__ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
   772 	
   773 	sqlite3_stmt* stmtHandle = NULL;
   774 	rc = sqlite3_prepare(dbHandle, "SELECT Id,Name FROM A", -1, &stmtHandle, 0);
   775 	SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   776 	__ASSERT_DEBUG(stmtHandle != NULL, User::Invariant());
   777 	
   778 	for(TInt recid=1;recid<=KTestRecCnt;++recid)
   779 		{
   780 		rc = sqlite3_step(stmtHandle);
   781 		SQLITE_TEST(dbHandle, rc, SQLITE_ROW);
   782 		CheckRecord(stmtHandle, recid);
   783 		}
   784 	rc = sqlite3_step(stmtHandle);
   785 	SQLITE_TEST(dbHandle, rc, SQLITE_DONE);
   786 	
   787 	sqlite3_finalize(stmtHandle);
   788 	sqlite3_close(dbHandle);
   789 	}
   790 
   791 //Simulates a file system error in a loop.
   792 //Attempts to update single record in a transaction.
   793 //If the UPDATE operation fails - verifies the database content on each iteration.
   794 //Note: pages are stored at the moment, not clusters. The database operations are not more robust if
   795 //      clusters are stored in a case of a removable media.
   796 void TRemovableMediaTest::DoTest()
   797 	{
   798 	TheTest.Printf(_L("Update 1 record in a file I/o simulation loop\r\n"));		
   799 	TInt rc = -1;
   800 	TBuf8<KMaxFileName + 1> dbFileName8;
   801 	dbFileName8.Copy(TheRmvMediaDbFileName);
   802 	for(TInt cnt=0;rc!=SQLITE_OK;++cnt)
   803 		{		
   804 		TheTest.Printf(_L("%d \r"), cnt);		
   805 		sqlite3* dbHandle = NULL;
   806 		rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
   807 		SQLITE_TEST(dbHandle, rc, SQLITE_OK);
   808 		__ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
   809 		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
   810 		rc = sqlite3_exec(dbHandle, "BEGIN IMMEDIATE", 0, 0, 0);
   811 		if(rc == SQLITE_OK)
   812 			{
   813 			rc = sqlite3_exec(dbHandle, "UPDATE A SET Name='1234' WHERE Id=1", 0, 0, 0);
   814 			if(rc == SQLITE_OK)
   815 				{
   816 				TInt cnt = sqlite3_changes(dbHandle);
   817 				TEST2(cnt, 1);
   818 				rc = sqlite3_exec(dbHandle, "COMMIT", 0, 0, 0);
   819 				}
   820 			}
   821 		(void)TheFs.SetErrorCondition(KErrNone);
   822 		sqlite3_close(dbHandle);
   823 		if(rc != SQLITE_OK)
   824 			{
   825 			VerifyDatabase();
   826 			}
   827 		}
   828 	TEST2(rc, SQLITE_OK);
   829 	VerifyDatabase();
   830 	}
   831 	
   832 void TRemovableMediaTest::Run()
   833 	{
   834 	TInt driveNo = GetRemovableMediaDriveNo();
   835 	if(driveNo == KErrNotFound)
   836 		{
   837 		TheTest.Printf(_L("No removable media discovered. Test case not executed.\r\n"));	
   838 		return;
   839 		}
   840 	TInt clusterSize = ClusterSize(driveNo);
   841 	if(clusterSize < 0)
   842 		{
   843 		TheTest.Printf(_L("Error %d retrieving the cluster size of drive %C. Test case not executed.\r\n"), clusterSize, 'A' + driveNo);
   844 		return;
   845 		}
   846 	if(clusterSize <= KMinCachePageSize)
   847 		{
   848 		TheTest.Printf(_L("Cluster size: %d. No appropriate cache page size found. Test case not executed.\r\n"), clusterSize);	
   849 		return;
   850 		}
   851 		
   852 	TheTest.Printf(_L("Cluster size: %d. Cache page size %d.\r\nBegin test.\r\n"), clusterSize, KMinCachePageSize);	
   853 	CreateDatabase(driveNo, KMinCachePageSize);
   854 	DoTest();
   855 	(void)TheFs.Delete(TheRmvMediaDbFileName);
   856 	TheTest.Printf(_L("End test.\r\n"));
   857 	}
   858 
   859 /**
   860 @SYMTestCaseID			SYSLIB-SQL-UT-3516
   861 @SYMTestCaseDesc		Removable media robustness test
   862 						The test creates a test database with a table with some records. Then the test verifies
   863 						that the database content cannot be corrupted by file I/O failures during database updates, 
   864 						when the database file is on a removable media and the media cluster size is bigger than the 
   865 						database page size.
   866 @SYMTestPriority		High
   867 @SYMTestActions			Removable media robustness test
   868 @SYMTestExpectedResults The test must not fail
   869 @SYMREQ					REQ7913
   870 */
   871 void RemovableMediaRobustnessTest()
   872 	{
   873 	TRemovableMediaTest removableMediaTest;
   874 	removableMediaTest.Run();
   875 	}
   876 
   877 /**
   878 @SYMTestCaseID			SYSLIB-SQL-UT-4044
   879 @SYMTestCaseDesc		RSqlDatabase::Size(TSize&), file I/O error simulation test.
   880 						The test creates a database and executes RSqldatabase::Size(TSize&)
   881 						during a file I/O error simulation. The database should not be corrupted 
   882 						by the call.
   883 @SYMTestPriority		High
   884 @SYMTestActions			RSqlDatabase::Size(TSize&), file I/O error simulation test.
   885 @SYMTestExpectedResults Test must not fail
   886 @SYMREQ					REQ10407
   887 */
   888 void SizeTest()
   889 	{
   890 	(void)RSqlDatabase::Delete(KTestDbName);
   891 	TInt err = TheDb.Create(KTestDbName);
   892 	TEST2(err, KErrNone);
   893 	err = TheDb.Exec(_L("BEGIN;CREATE TABLE A(Id INTEGER,Data BLOB);INSERT INTO A VALUES(1, x'11223344');COMMIT;"));
   894 	TEST(err >= 0);
   895 	RSqlDatabase::TSize size1 = {-1, -1};
   896 	err = TheDb.Size(size1);
   897 	TEST2(err, KErrNone);
   898 	TEST(size1.iSize > 0);
   899 	TEST2(size1.iFree, 0);
   900 	TheDb.Close();
   901 	//"File I/O" error simulation loop
   902 	err = KErrCorrupt;
   903 	for(TInt cnt=0;err<KErrNone;++cnt)
   904 		{
   905 		TheTest.Printf(_L("%d \r"), cnt);		
   906 		TEST2(TheDb.Open(KTestDbName), KErrNone);
   907 		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
   908 		RSqlDatabase::TSize size2 = {-1, -1};
   909 		err = TheDb.Size(size2);
   910 		(void)TheFs.SetErrorCondition(KErrNone);
   911 		TheDb.Close();
   912 		if(err == KErrNone)
   913 			{
   914 			TEST(size2.iSize == size1.iSize);
   915 			TEST(size2.iFree == size1.iFree);
   916 			break;
   917 			}
   918 		else
   919 			{
   920 			//check the database content - all bets are off in a case of an I/O error. 
   921 			TEST2(TheDb.Open(KTestDbName), KErrNone);
   922 			TSqlScalarFullSelectQuery q(TheDb);
   923 			TInt recCnt = 0;
   924 			TRAPD(err2, recCnt = q.SelectIntL(_L8("SELECT COUNT(*) FROM A")));
   925 			TheDb.Close();
   926 			TEST2(err2, KErrNone);
   927 			TEST2(recCnt, 1);
   928 			}
   929 		}
   930 	(void)RSqlDatabase::Delete(KTestDbName);
   931 	}
   932 
   933 /**
   934 @SYMTestCaseID			SYSLIB-SQL-UT-4045
   935 @SYMTestCaseDesc		RSqlDatabase::Compact(), file I/O error simulation test.
   936 						The test creates a database and executes RSqlDatabase::Compact()
   937 						during a file I/O error simulation. The database should not be corrupted 
   938 						by the call.
   939 @SYMTestPriority		High
   940 @SYMTestActions			RSqlDatabase::Compact(), file I/O error simulation test.
   941 @SYMTestExpectedResults Test must not fail
   942 @SYMREQ					REQ10405
   943 */
   944 void CompactTest()
   945 	{
   946 	(void)RSqlDatabase::Delete(KTestDbName);
   947 	_LIT8(KConfig, "compaction=manual");
   948 	TInt err = TheDb.Create(KTestDbName, &KConfig);
   949 	TEST2(err, KErrNone);
   950 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
   951 	TEST(err >= 0);
   952 	//Insert records
   953 	err = TheDb.Exec(_L8("BEGIN"));
   954 	TEST(err >= 0);
   955 	const TInt KRecLen = 1000;
   956 	TBuf8<KRecLen> sqlfmt;
   957 	sqlfmt.Copy(_L8("INSERT INTO A VALUES(%d,x'"));
   958 	for(TInt j=0;j<(KRecLen-50);++j)
   959 		{
   960 		sqlfmt.Append(_L8("A"));
   961 		}
   962 	sqlfmt.Append(_L8("')"));
   963 	const TInt KRecCount = 100;	
   964 	for(TInt i=0;i<KRecCount;++i)
   965 		{
   966 		TBuf8<KRecLen> sql;
   967 		sql.Format(sqlfmt, i + 1);
   968 		err = TheDb.Exec(sql);
   969 		TEST2(err, 1);
   970 		}
   971 	err = TheDb.Exec(_L8("COMMIT"));
   972 	TEST(err >= 0);
   973 	//Free some space
   974 	const TInt KDeletedRecCnt = KRecCount - 10;
   975 	err = TheDb.Exec(_L8("DELETE FROM A WHERE Id > 10"));
   976 	TEST(err >= 0);
   977 	//Get the database size
   978 	RSqlDatabase::TSize size;
   979 	err = TheDb.Size(size);
   980 	TEST2(err, KErrNone);
   981 	TheDb.Close();
   982 	TEST(size.iSize > 0);
   983 	TEST(size.iFree > 0);
   984 	//"File I/O" error simulation loop
   985 	err = KErrCorrupt;
   986 	for(TInt cnt=0;err<KErrNone;++cnt)
   987 		{
   988 		TheTest.Printf(_L("%d \r"), cnt);		
   989 		TEST2(TheDb.Open(KTestDbName), KErrNone);
   990 		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
   991 		err = TheDb.Compact(RSqlDatabase::EMaxCompaction);
   992 		(void)TheFs.SetErrorCondition(KErrNone);
   993 		TheDb.Close();
   994 		if(err == KErrNone)
   995 			{
   996 			break;
   997 			}
   998 		else
   999 			{
  1000 			//check the database content - all bets are off in a case of an I/O error. 
  1001 			//The database maight have been compacted, so - no check for that.
  1002 			TEST2(TheDb.Open(KTestDbName), KErrNone);
  1003 			TSqlScalarFullSelectQuery q(TheDb);
  1004 			TInt recCnt = 0;
  1005 			TRAPD(err2, recCnt = q.SelectIntL(_L8("SELECT COUNT(*) FROM A")));
  1006 			TheDb.Close();
  1007 			TEST2(err2, KErrNone);
  1008 			TEST2(recCnt, (KRecCount - KDeletedRecCnt));
  1009 			}
  1010 		}
  1011 	TheTest.Printf(_L("\r\n"));
  1012 	//Check that the database has been really compacted
  1013 	TEST2(TheDb.Open(KTestDbName), KErrNone);
  1014 	RSqlDatabase::TSize size2;
  1015 	err = TheDb.Size(size2);
  1016 	TEST2(err, KErrNone);
  1017 	TheDb.Close();
  1018 	(void)RSqlDatabase::Delete(KTestDbName);
  1019 	TEST(size.iSize > size2.iSize);
  1020 	TEST2(size2.iFree, 0);
  1021 	}
  1022 
  1023 void DoBlobWriteStreamTestL(TBool aAttachDb)
  1024 	{
  1025 	RSqlBlobWriteStream strm;
  1026 	CleanupClosePushL(strm);
  1027 	if(aAttachDb)
  1028 		{
  1029 		strm.OpenL(TheDb, _L("A"), _L("Data"), 1, KAttachDb);
  1030 		}
  1031 	else
  1032 		{
  1033 		strm.OpenL(TheDb, _L("A"), _L("Data"), 1);
  1034 		}
  1035 
  1036 	TBuf8<KBlobSize / KWriteCnt> data;
  1037 	data.SetLength(KBlobSize / KWriteCnt);
  1038 	data.Fill(0xA5);
  1039 	
  1040 	for(TInt i=0;i<KWriteCnt;++i)
  1041 		{
  1042 		strm.WriteL(data);
  1043 		}
  1044 	
  1045 	strm.CommitL();
  1046 	
  1047 	CleanupStack::PopAndDestroy(&strm);
  1048 	}
  1049 
  1050 /**
  1051 @SYMTestCaseID			SYSLIB-SQL-UT-4089
  1052 @SYMTestCaseDesc		RSqlBlobWriteStream::WriteL(), file I/O error simulation test.
  1053 						The test creates a database and executes RSqlBlobWriteStream::WriteL()
  1054 						during a file I/O error simulation. The database should not be corrupted 
  1055 						by the call.
  1056 @SYMTestPriority		High
  1057 @SYMTestActions			RSqlBlobWriteStream::WriteL(), file I/O error simulation test.
  1058 @SYMTestExpectedResults Test must not fail
  1059 @SYMREQ					REQ5792
  1060                         REQ10418
  1061 */
  1062 void BlobWriteStreamTest(TBool aAttachDb)
  1063 	{
  1064 	(void)RSqlDatabase::Delete(KTestDbName);
  1065 	TInt err = TheDb.Create(KTestDbName);
  1066 	TEST2(err, KErrNone);
  1067 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
  1068 	TEST2(err, 1);
  1069 	TBuf8<100> sql;
  1070 	sql.Format(_L8("INSERT INTO A VALUES(1, zeroblob(%d))"), KBlobSize);
  1071 	err = TheDb.Exec(sql);
  1072 	TEST2(err, 1);
  1073 	TheDb.Close();
  1074 	
  1075 	err = KErrCorrupt;
  1076 	for(TInt cnt=0;err<KErrNone;++cnt)
  1077 		{
  1078 		TheTest.Printf(_L("%d \r"), cnt);		
  1079 		TEST2(TheDb.Open(KTestDbName), KErrNone);
  1080 		if(aAttachDb)
  1081 			{
  1082 			TEST2(TheDb.Attach(KTestDbName, KAttachDb), KErrNone);	
  1083 			}
  1084 		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
  1085 		TRAP(err, DoBlobWriteStreamTestL(aAttachDb));
  1086 		(void)TheFs.SetErrorCondition(KErrNone);
  1087 		if(aAttachDb)
  1088 			{
  1089 			TEST2(TheDb.Detach(KAttachDb), KErrNone);	
  1090 			}
  1091 		TheDb.Close();
  1092 		}
  1093 	TheTest.Printf(_L("\r\n"));
  1094 
  1095 	TEST2(TheDb.Open(KTestDbName), KErrNone);
  1096 	
  1097 	RSqlStatement stmt;
  1098 	err = stmt.Prepare(TheDb, _L8("SELECT * FROM A"));
  1099 	TEST2(err, KErrNone);
  1100 	err = stmt.Next();
  1101 	TEST2(err, KSqlAtRow);
  1102 	TPtrC8 data;
  1103 	err = stmt.ColumnBinary(1, data);
  1104 	TEST2(err, KErrNone);
  1105 	TEST2(data.Length(), KBlobSize);
  1106 	for(TInt j=0;j<KBlobSize;++j)
  1107 		{
  1108 		TUint8 d = data[j];
  1109 		TEST2(d, 0xA5);	
  1110 		}
  1111 	stmt.Close();
  1112 	
  1113 	TheDb.Close();
  1114 	(void)RSqlDatabase::Delete(KTestDbName);
  1115 	}
  1116 
  1117 void DoBlobReadStreamTestL(TBool aAttachDb, TDes8& aDes)
  1118 	{
  1119 	RSqlBlobReadStream strm;
  1120 	CleanupClosePushL(strm);
  1121 	if(aAttachDb)
  1122 		{
  1123 		strm.OpenL(TheDb, _L("A"), _L("Data"), 1, KAttachDb);
  1124 		}
  1125 	else
  1126 		{
  1127 		strm.OpenL(TheDb, _L("A"), _L("Data"), 1);
  1128 		}
  1129 
  1130 	TBuf8<KBlobSize / KWriteCnt> data;
  1131 	aDes.SetLength(0);
  1132 	
  1133 	for(TInt i=0;i<KWriteCnt;++i)
  1134 		{
  1135 		strm.ReadL(data);
  1136 		aDes.Append(data);
  1137 		}
  1138 	
  1139 	CleanupStack::PopAndDestroy(&strm);
  1140 	}
  1141 
  1142 /**
  1143 @SYMTestCaseID			SYSLIB-SQL-UT-4090
  1144 @SYMTestCaseDesc		RSqlBlobReadStream::ReadL(), file I/O error simulation test.
  1145 						The test creates a database and executes RSqlBlobReadStream::ReadL()
  1146 						during a file I/O error simulation. The database should not be corrupted 
  1147 						by the call.
  1148 @SYMTestPriority		High
  1149 @SYMTestActions			RSqlBlobReadStream::ReadL(), file I/O error simulation test.
  1150 @SYMTestExpectedResults Test must not fail
  1151 @SYMREQ					REQ5792
  1152                         REQ10410
  1153                         REQ10411
  1154 */
  1155 void BlobReadStreamTest(TBool aAttachDb)
  1156 	{
  1157 	(void)RSqlDatabase::Delete(KTestDbName);
  1158 	TInt err = TheDb.Create(KTestDbName);
  1159 	TEST2(err, KErrNone);
  1160 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
  1161 	TEST2(err, 1);
  1162 	TBuf8<100> sql;
  1163 	sql.Format(_L8("INSERT INTO A VALUES(1, zeroblob(%d))"), KBlobSize);
  1164 	err = TheDb.Exec(sql);
  1165 	TEST2(err, 1);
  1166 	TRAP(err, DoBlobWriteStreamTestL(EFalse));
  1167 	TEST2(err, KErrNone);
  1168 	TheDb.Close();
  1169 	
  1170 	HBufC8* buf = HBufC8::New(KBlobSize);
  1171 	TEST(buf != NULL);
  1172 	TPtr8 bufptr = buf->Des();
  1173 	
  1174 	err = KErrCorrupt;
  1175 	for(TInt cnt=0;err<KErrNone;++cnt)
  1176 		{
  1177 		TheTest.Printf(_L("%d \r"), cnt);		
  1178 		TEST2(TheDb.Open(KTestDbName), KErrNone);
  1179 		if(aAttachDb)
  1180 			{
  1181 			TEST2(TheDb.Attach(KTestDbName, KAttachDb), KErrNone);	
  1182 			}
  1183 		(void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
  1184 		TRAP(err, DoBlobReadStreamTestL(aAttachDb, bufptr));
  1185 		(void)TheFs.SetErrorCondition(KErrNone);
  1186 		if(aAttachDb)
  1187 			{
  1188 			TEST2(TheDb.Detach(KAttachDb), KErrNone);	
  1189 			}
  1190 		TheDb.Close();
  1191 		}
  1192 	TheTest.Printf(_L("\r\n"));
  1193 
  1194 	TEST2(bufptr.Length(), KBlobSize);
  1195 	for(TInt j=0;j<KBlobSize;++j)
  1196 		{
  1197 		TUint8 d = bufptr[j];
  1198 		TEST2(d, 0xA5);	
  1199 		}
  1200 		
  1201 	delete buf;
  1202 		
  1203 	(void)RSqlDatabase::Delete(KTestDbName);
  1204 	}
  1205 
  1206 /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  1207 
  1208 void DoTests()
  1209 	{
  1210 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3419 Alter database during file I/O error "));
  1211 	AlterDatabaseTest();
  1212 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3420 Alter database during file I/O error (using statement object) "));
  1213 	AlterDatabaseTest2();
  1214 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3421 Open database during file I/O error "));
  1215 	OpenDatabaseTest();
  1216 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3434 Create database during file I/O error "));
  1217 	CreateDatabaseTest();
  1218     TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-UT-4189 Attach database during file I/O error "));
  1219     AttachDatabaseTest();
  1220     TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-UT-4190 Delete database during file I/O error "));
  1221     DeleteDatabaseTest();
  1222 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3462 Select record test during file I/O error "));
  1223 	SelectRecordTest();
  1224 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3463 Insert record test during file I/O error "));
  1225 	InsertRecordTest();
  1226 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3516 Removable Media robustness test "));
  1227 	RemovableMediaRobustnessTest();
  1228 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4044 Database size test during file I/O error"));	
  1229 	SizeTest();
  1230 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4045 Compact database test during file I/O error"));	
  1231 	CompactTest();
  1232 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4089 RSqlBlobWriteStream::WriteL() test during file I/O error"));	
  1233 	BlobWriteStreamTest(EFalse);
  1234 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4089 RSqlBlobWriteStream::WriteL()+attached database test during file I/O error"));	
  1235 	BlobWriteStreamTest(ETrue);
  1236 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4090 RSqlBlobReadStream::ReadL() test during file I/O error"));	
  1237 	BlobReadStreamTest(EFalse);
  1238 	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4090 RSqlBlobReadStream::ReadL()+attached database test during file I/O error"));	
  1239 	BlobReadStreamTest(ETrue);
  1240 	}
  1241 
  1242 TInt E32Main()
  1243 	{
  1244 	TheTest.Title();
  1245 	
  1246 	CTrapCleanup* tc = CTrapCleanup::New();
  1247 	
  1248 	__UHEAP_MARK;
  1249 	
  1250 	SetupTestEnv();
  1251 	DoTests();
  1252 	DestroyTestEnv();
  1253 	
  1254 	__UHEAP_MARKEND;
  1255 	
  1256 	TheTest.End();
  1257 	TheTest.Close();
  1258 	
  1259 	delete tc;
  1260 	
  1261 	User::Heap().Check();
  1262 	return KErrNone;
  1263 	}