os/persistentdata/persistentstorage/sql/TEST/t_sqldbconfigfile.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2008-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 <stdlib.h>
    18 #include <bautils.h>
    19 #include <hal.h>
    20 #include <sqldb.h>
    21 #include "sqlite3.h"
    22 #include "SqliteSymbian.h"
    23 #include "SqlResourceTester.h"
    24 
    25 //In order to be able to compile the test, the following variables are defined (used inside the OS porting layer, 
    26 //when _SQLPROFILER macro is defined)
    27 #ifdef _SQLPROFILER
    28 TInt TheSqlSrvProfilerFileRead = 0;
    29 TInt TheSqlSrvProfilerFileWrite = 0;
    30 TInt TheSqlSrvProfilerFileSync = 0;
    31 TInt TheSqlSrvProfilerFileSetSize = 0;
    32 #endif
    33 
    34 /////////////////////////////////////////////////////////////////////////////////////////////////
    35 /// This test works only if the whole SQL component is built with SYSLIBS_TEST macro defined! ///
    36 /////////////////////////////////////////////////////////////////////////////////////////////////
    37 
    38 RTest TheTest(_L("t_sqldbconfigfile test"));
    39 
    40 #ifdef SYSLIBS_TEST	
    41 
    42 extern TBool IsStatementSupported(const TDesC& aStatementIn, const TDesC& aDbName, TDes& aStatementOut);
    43 
    44 RFs TheFs;
    45 RSqlDatabase TheDb;
    46 sqlite3 *TheDbHandle = NULL;
    47 
    48 _LIT(KCfgDb1, "c:[1111C1C1]a.db"); // shared, secure db
    49 _LIT(KCfgDb2, "c:[1111C1C1]b.db"); // shared, secure db (no config file for it)
    50 _LIT(KCfgDb3, "c:\\private\\1111C1C1\\c.db"); // private, secure db
    51 _LIT(KCfgDb4, "c:\\test\\d.db"); // public db
    52 _LIT(KCfgDb5, "c:[1111C1C1]e.db"); // shared, secure db (config file created before db is created)
    53 
    54 _LIT(KCfgDb1ConfigFilePath, "c:\\private\\10281e17\\cfg[1111C1C1]a.db.0%d"); // config file version %d for a.db
    55 _LIT(KCfgDb3ConfigFileV01Path, "c:\\private\\10281e17\\cfgc.db.01"); // config file v01 for c.db (unsupported)
    56 _LIT(KCfgDb4ConfigFileV01Path, "c:\\private\\10281e17\\cfgd.db.01"); // config file v01 for d.db (unsupported)
    57 _LIT(KCfgDb5ConfigFileV01Path, "c:\\private\\10281e17\\cfg[1111C1C1]e.db.01"); // config file v01 for e.db
    58 _LIT(KCfgDb1CorruptConfigFilePath, "c:\\private\\10281e17\\cfg[1111C1C1]a.db.invalidextension"); // invalid config file name syntax
    59 
    60 // config file valid contents (used for v02 and others)
    61 _LIT8(KCfgConfigFileValidStmt, "CREATE INDEX newIdx ON table1(i3)  ;"); 
    62 // config file v03 empty contents
    63 _LIT8(KCfgDb1ConfigFileV03EmptyStmt, ""); 
    64 // config file v04 unsupported contents
    65 _LIT8(KCfgDb1ConfigFileV04UnsupportedStmt, "DELETE FROM table1"); 
    66 // config file v05 only whitespace contents
    67 _LIT8(KCfgDb1ConfigFileV05OnlyWhitespaceStmt, " \r\n   \r\n"); 
    68 // config file v06 invalid schema contents
    69 _LIT8(KCfgDb1ConfigFileV06InvalidSchemaStmt, "CREATE INDEX thisIdx ON table999(i3)"); 
    70 // config file v07 unsupported comment style
    71 _LIT8(KCfgDb1ConfigFileV07InvalidCommentedStmt, "CREATE INDEX ind1 ON table1(i2) // create an index"); 
    72 // config file v08 sequence of different statements
    73 _LIT8(KCfgDb1ConfigFileV08SeqStmt, ";  CREATE INDEX IdxFirst ON table1(i3)\r\n;  DELETE FROM table1;INSERT INTO table1 (i1,i2,i3) values(6,7,8);;CREATE INDEX IdxSecond ON table1(i1);");
    74 // config file v09 whitespace before and after statement
    75 _LIT8(KCfgDb1ConfigFileV09WhitespacePreAndPostStmt, "  CREATE INDEX intIdx ON table1(i1)       "); 
    76 // config file v10 valid contents
    77 _LIT8(KCfgDb1ConfigFileV10ValidStmt, "CREATE INDEX i3Index ON table1(i3)\n"); 
    78 // config file v11 valid contents (also tests that any amount spaces and tabs are allowed between 'CREATE' and 'INDEX')
    79 _LIT8(KCfgDb1ConfigFileV11ValidStmt, "CREATE     INDEX i1Index ON table1(i1);\nCREATE	INDEX i2Index ON table1(i2)");
    80 // config file v12 invalid stmt plus valid stmt
    81 _LIT8(KCfgDb1ConfigFileV12InvalidPlusValidStmt, "CREATE UNIQUE INDEX uniqueIdx ON table1(i1);CREATE INDEX v12Idx ON table1(i2)");
    82 // config file v13 supported SQL comment style
    83 _LIT8(KCfgDb1ConfigFileV13SQLCommentStmt, "CREATE INDEX v13Idx ON table1(i1) -- this is an SQL comment");
    84 // config file v14 supported 'C' comment style
    85 _LIT8(KCfgDb1ConfigFileV14CCommentStmt, "CREATE INDEX v14Idx ON table1(i3) /* this is a C comment */;");
    86 
    87 //KLongDbName1 is "long" enough to allow "-journal" to be added at the end.
    88 //Also, on simulators/emulators the file names have a prefix with length of about 30 characters
    89 //(something like \epoc\winscw\c\....)
    90 _LIT(KLongDbName1,  "c:[1111C1C1]a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a.db");
    91 _LIT(KLongCfgName1, "c:\\private\\10281e17\\cfg[1111C1C1]a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a123456789a.db.01");
    92 
    93 _LIT(KSqlSrvPrivatePath, "\\private\\10281e17\\");
    94 _LIT(KResetCollationDllSql, "UPDATE symbian_settings SET CollationDllName='hjagafsff'");
    95 
    96 _LIT(KAttachDb1, "Db1");
    97 _LIT(KAttachDb2, "Db2");
    98 _LIT(KAttachDb5, "Db5");
    99 
   100 ///////////////////////////////////////////////////////////////////////////////////////
   101 // Destroy functions
   102 
   103 void KillSqlServer()
   104 	{
   105 	_LIT(KSqlSrvName, "sqlsrv.exe");
   106 	TFullName name;
   107 	//RDebug::Print(_L("Find and kill \"%S\" process.\n"), &aProcessName);
   108 	TBuf<64> pattern(KSqlSrvName);
   109 	TInt length = pattern.Length();
   110 	pattern += _L("*");
   111 	TFindProcess procFinder(pattern);
   112 
   113 	while (procFinder.Next(name) == KErrNone)
   114 		{
   115 		if (name.Length() > length)
   116 			{//If found name is a string containing aProcessName string.
   117 			TChar c(name[length]);
   118 			if (c.IsAlphaDigit() ||
   119 				c == TChar('_') ||
   120 				c == TChar('-'))
   121 				{
   122 				// If the found name is other valid application name
   123 				// starting with aProcessName string.
   124 				//RDebug::Print(_L(":: Process name: \"%S\".\n"), &name);
   125 				continue;
   126 				}
   127 			}
   128 		RProcess proc;
   129 		if (proc.Open(name) == KErrNone)
   130 			{
   131 			proc.Kill(0);
   132 			//RDebug::Print(_L("\"%S\" process killed.\n"), &name);
   133 			}
   134 		proc.Close();
   135 		}
   136 	}
   137 
   138 void DeleteCfgFilesAndDbs()
   139 	{
   140  	(void)RSqlDatabase::Delete(KLongDbName1);
   141  	(void)RSqlDatabase::Delete(KCfgDb1);
   142  	(void)RSqlDatabase::Delete(KCfgDb2);	
   143  	(void)RSqlDatabase::Delete(KCfgDb3);
   144  	(void)RSqlDatabase::Delete(KCfgDb4);
   145  	(void)RSqlDatabase::Delete(KCfgDb5);
   146  	(void)TheFs.Delete(KLongCfgName1);
   147  	(void)TheFs.Delete(KCfgDb3ConfigFileV01Path);
   148  	(void)TheFs.Delete(KCfgDb4ConfigFileV01Path);
   149  	(void)TheFs.Delete(KCfgDb5ConfigFileV01Path);
   150  	CFileMan* fileMan = 0;	
   151  	TRAPD(err, fileMan = CFileMan::NewL(TheFs));
   152  	if(KErrNone == err)
   153  		{
   154  		(void)fileMan->Delete(_L("c:\\private\\10281e17\\cfg[1111C1C1]a.db.*"));
   155 		delete fileMan;			
   156  		}
   157   	}
   158 
   159 void DestroyTestEnv()
   160 	{
   161 	if(TheDbHandle)
   162 		{
   163 		sqlite3_close(TheDbHandle);	
   164 		TheDbHandle = NULL;
   165 		}
   166 	TheDb.Close();
   167 	KillSqlServer();
   168 	DeleteCfgFilesAndDbs();
   169 	TheFs.Close();
   170 	}
   171 	
   172 ///////////////////////////////////////////////////////////////////////////////////////
   173 // Test macros and functions
   174 
   175 void Check(TInt aValue, TInt aLine)
   176 	{
   177 	if(!aValue)
   178 		{
   179 		DestroyTestEnv();
   180 		TheTest(EFalse, aLine);
   181 		}
   182 	}
   183 void Check(TInt aValue, TInt aExpected, TInt aLine)
   184 	{
   185 	if(aValue != aExpected)
   186 		{
   187 		DestroyTestEnv();
   188 		TheTest.Printf(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
   189 		TheTest(EFalse, aLine);
   190 		}
   191 	}
   192 #define TEST(arg) ::Check((arg), __LINE__)
   193 #define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
   194 
   195 // OOM test functions
   196 
   197 static TInt TheHandleCount1;
   198 static TInt TheHandleCount2;
   199 static TInt TheAllocatedCellsCount;
   200 
   201 void MarkHandles()
   202 	{
   203 	RThread().HandleCount(TheHandleCount1, TheHandleCount2);
   204 	}
   205 
   206 void CheckHandles()
   207 	{
   208 	TInt endHandleCount1;
   209 	TInt endHandleCount2;
   210 
   211 	RThread().HandleCount(endHandleCount1, endHandleCount2);
   212 
   213 	TEST2(TheHandleCount1, endHandleCount1);
   214 	TEST2(TheHandleCount2, endHandleCount2);
   215 	}
   216 
   217 void MarkAllocatedCells()
   218 	{
   219 	TheAllocatedCellsCount = User::CountAllocCells();
   220 	}
   221 
   222 void CheckAllocatedCells()
   223 	{
   224 	TInt allocatedCellsCount = User::CountAllocCells();
   225 	TEST2(allocatedCellsCount, TheAllocatedCellsCount);
   226 	}
   227 
   228 ///////////////////////////////////////////////////////////////////////////////////////
   229 // Set up functions
   230 
   231 RSqlSecurityPolicy CreateSecurityPolicy()
   232 	{
   233 	const TSecurityPolicy KDefaultPolicy(TSecurityPolicy::EAlwaysPass);
   234 	RSqlSecurityPolicy policy;
   235 	TInt err = policy.Create(KDefaultPolicy);
   236 	TEST2(err, KErrNone);
   237 	err = policy.SetDbPolicy(RSqlSecurityPolicy::ESchemaPolicy, KDefaultPolicy);
   238     TEST2(err, KErrNone);
   239     err = policy.SetDbPolicy(RSqlSecurityPolicy::EWritePolicy, KDefaultPolicy);
   240     TEST2(err, KErrNone);
   241     err = policy.SetDbPolicy(RSqlSecurityPolicy::EReadPolicy, KDefaultPolicy);
   242     TEST2(err, KErrNone);
   243     return policy;
   244 	}
   245 	
   246 void DoCreateCfgFile(const TDesC& aFileName, const TDesC8& aData)
   247     {
   248     RFile file;
   249     TInt err = file.Create(TheFs, aFileName, EFileRead | EFileWrite);
   250     TEST2(err, KErrNone);
   251     err = file.Write(aData); 
   252     file.Close();   
   253     TEST2(err, KErrNone);
   254     }
   255 
   256 // Create v01 of the test config files
   257 void CreateCfgFiles()
   258 	{
   259 	TFileName fileName;
   260 	TInt v1 = 1;
   261 	fileName.Format(KCfgDb1ConfigFilePath, v1);
   262 
   263 	// config file v01 contents for db1 (also tests that upper and lower case statements are both supported)
   264 	_LIT8(KCfgDb1ConfigFile01Stmts, "CREATE INDEX idx ON table1(i1);CREATE INDEX idx2 ON table1(i2);INSERT INTO table1 (i1,i2,i3) values(5,8,9);DELETE FROM table1;create index multiidx ON TABLE1(i2,i3)");
   265     DoCreateCfgFile(fileName, KCfgDb1ConfigFile01Stmts);
   266     
   267     // config file v01 contents for db3 (will not be processed as db3 is not a shared, secure db)
   268     _LIT8(KCfgDb3ConfigFile01Stmts, "CREATE INDEX idx ON table3(i1)");
   269     DoCreateCfgFile(KCfgDb3ConfigFileV01Path, KCfgDb3ConfigFile01Stmts);
   270     
   271     // config file v01 contents for db4 (will not be processed as db4 is not a shared, secure db)
   272     _LIT8(KCfgDb4ConfigFile01Stmts, "CREATE INDEX idx ON table4(i1)");
   273     DoCreateCfgFile(KCfgDb4ConfigFileV01Path, KCfgDb4ConfigFile01Stmts);
   274     
   275     // create the config file for Db5 before the database has been created
   276     // config file v01 contents for db5 (will eventually be processed after db5 itself is created)
   277     _LIT8(KCfgDb5ConfigFile01Stmts, "CREATE INDEX idx ON table5(i1);\r\n");
   278     DoCreateCfgFile(KCfgDb5ConfigFileV01Path, KCfgDb5ConfigFile01Stmts);
   279 	}
   280 
   281 void DoCreateCfgDb(const TDesC& aFileName, const TDesC& aCreateTblSql, const TDesC& aInsertSql, 
   282                    RSqlSecurityPolicy* aPolicy = NULL)
   283     {
   284     TInt err = aPolicy ? TheDb.Create(aFileName, *aPolicy) : TheDb.Create(aFileName);
   285     TEST2(err, KErrNone);
   286     err = TheDb.Exec(aCreateTblSql);
   287     TEST(err >= 0);
   288     err = TheDb.Exec(aInsertSql);
   289     TEST(err == 1);
   290     TheDb.Close();
   291     }
   292 
   293 // Create the test databases
   294 void CreateCfgDbs()
   295 	{
   296     RSqlSecurityPolicy securityPolicy = CreateSecurityPolicy();
   297     DoCreateCfgDb(KCfgDb1, _L("CREATE TABLE table1(i1 INTEGER, i2 INTEGER, i3 INTEGER)"),
   298                            _L("INSERT INTO table1 (i1,i2,i3) values(1,2,3)"), &securityPolicy);
   299     DoCreateCfgDb(KCfgDb2, _L("CREATE TABLE table2(i1 INTEGER, i2 INTEGER, i3 INTEGER)"),
   300                            _L("INSERT INTO table2 (i1,i2,i3) values(4,5,6)"), &securityPolicy);
   301     securityPolicy.Close();
   302     DoCreateCfgDb(KCfgDb3, _L("CREATE TABLE table3(i1 INTEGER, i2 INTEGER)"),
   303                            _L("INSERT INTO table3 (i1,i2) values(7,8)"));
   304     DoCreateCfgDb(KCfgDb4, _L("CREATE TABLE table4(i1 INTEGER, i2 INTEGER, i3 INTEGER)"),
   305                            _L("INSERT INTO table4 (i1,i2,i3) values(9,10,11)"));
   306  	}
   307 
   308 void CreateCfgFilesAndDbs()
   309 	{
   310 	CreateCfgFiles();
   311 	CreateCfgDbs();
   312 	
   313 	// Must now kill the SQL Server so that the config files 
   314 	// created above are found and processed when it restarts
   315 	KillSqlServer();
   316  	}
   317  	
   318 // Create the Db5 test database (a config file for it already exists)
   319  void CreateDb5()
   320 	{
   321  	RSqlSecurityPolicy securityPolicy = CreateSecurityPolicy();
   322     DoCreateCfgDb(KCfgDb5, _L("CREATE TABLE table5(i1 INTEGER, i2 INTEGER, i3 INTEGER)"),
   323                            _L("INSERT INTO table5 (i1,i2,i3) values(1,2,3)"), &securityPolicy);
   324  	securityPolicy.Close();
   325 	}
   326 
   327 void SetupTestEnv()
   328     {
   329 	TInt err = TheFs.Connect();
   330 	TEST2(err, KErrNone);
   331 
   332 	_LIT(KTestDir, "c:\\test\\");
   333 	err = TheFs.MkDir(KTestDir);
   334 	TEST(err == KErrNone || err == KErrAlreadyExists);
   335 
   336 	err = TheFs.CreatePrivatePath(EDriveC);
   337 	TEST(err == KErrNone || err == KErrAlreadyExists);
   338 
   339 	// Create the cfg dbs and config files
   340  	DeleteCfgFilesAndDbs(); // just incase any previous files are lingering
   341  	CreateCfgFilesAndDbs();
   342 	}
   343 
   344 ///////////////////////////////////////////////////////////////////////////////////////
   345 //
   346 TInt CalcTimeMs(TUint32 aStartTicks, TUint32 aEndTicks)
   347 	{
   348 	static TInt freq = 0;
   349 	if(freq == 0)
   350 		{
   351 		TEST2(HAL::Get(HAL::EFastCounterFrequency, freq), KErrNone);
   352 		}
   353 	TInt64 diffTicks = (TInt64)aEndTicks - (TInt64)aStartTicks;
   354 	if(diffTicks < 0)
   355 		{
   356 		diffTicks = KMaxTUint32 + diffTicks + 1;
   357 		}
   358 	const TInt KMicroSecIn1Sec = 1000000;
   359 	TInt32 us = (diffTicks * KMicroSecIn1Sec) / freq;
   360 	return us / 1000;
   361 	}
   362 
   363 ///////////////////////////////////////////////////////////////////////////////////////
   364 // Config file replacement functions
   365 
   366 void UpgradeDbConfigFile(TInt aCurrentVersion)
   367 	{
   368     TInt newVersion = aCurrentVersion + 1;
   369     TEST(newVersion != 0 && newVersion != 1 && newVersion != 10 && newVersion != 11 && newVersion < 15);
   370 	
   371     KillSqlServer();
   372 		
   373 	TFileName fname;
   374 	fname.Format(KCfgDb1ConfigFilePath, aCurrentVersion);
   375 	TInt err = TheFs.Delete(fname);
   376 	TEST2(err, KErrNone);
   377 	
   378 	if(newVersion == 12)
   379 	    {
   380         // also delete version 10 of file
   381 	    fname.Format(KCfgDb1ConfigFilePath, 10);
   382         err = TheFs.Delete(fname);
   383         TEST2(err, KErrNone);   
   384 	    }
   385 
   386 	const TPtrC8 stmts[] = {KNullDesC8(), KNullDesC8(), 
   387 	        KCfgConfigFileValidStmt(), 
   388 	        KCfgDb1ConfigFileV03EmptyStmt(),
   389 	        KCfgDb1ConfigFileV04UnsupportedStmt(),
   390 	        KCfgDb1ConfigFileV05OnlyWhitespaceStmt(),
   391 	        KCfgDb1ConfigFileV06InvalidSchemaStmt(),
   392 	        KCfgDb1ConfigFileV07InvalidCommentedStmt(),
   393 	        KCfgDb1ConfigFileV08SeqStmt(),
   394 	        KCfgDb1ConfigFileV09WhitespacePreAndPostStmt(),
   395 	        KNullDesC8(), KNullDesC8(),
   396 	        KCfgDb1ConfigFileV12InvalidPlusValidStmt(),
   397 	        KCfgDb1ConfigFileV13SQLCommentStmt(),
   398 	        KCfgDb1ConfigFileV14CCommentStmt()};
   399 
   400     fname.Format(KCfgDb1ConfigFilePath, newVersion);
   401 	DoCreateCfgFile(fname, stmts[newVersion]);
   402 	}	
   403 	
   404 void DowngradeDbConfigFile(TInt aCurrentVersion)
   405 	{
   406 	TEST(aCurrentVersion > 1);
   407 	
   408 	KillSqlServer();
   409 	
   410 	TFileName fname;
   411 	fname.Format(KCfgDb1ConfigFilePath, aCurrentVersion);
   412 	TInt err = TheFs.Delete(fname);
   413 	TEST2(err, KErrNone);
   414 	
   415     fname.Format(KCfgDb1ConfigFilePath, aCurrentVersion - 1);
   416     DoCreateCfgFile(fname, KCfgConfigFileValidStmt);
   417 	}
   418 	
   419 void CreateCorruptDbConfigFile(TInt aCurrentVersion)
   420 	{
   421 	KillSqlServer();
   422 	
   423 	TFileName fname;
   424 	fname.Format(KCfgDb1ConfigFilePath, aCurrentVersion);
   425 	TInt err = TheFs.Delete(fname);
   426 	TEST2(err, KErrNone);
   427 
   428     DoCreateCfgFile(KCfgDb1CorruptConfigFilePath, KCfgConfigFileValidStmt);
   429 	}	
   430 	
   431 void CreateTwoVersionsOfConfigFile()
   432 	{
   433 	KillSqlServer();
   434 	
   435 	TInt err = TheFs.Delete(KCfgDb1CorruptConfigFilePath);
   436 	TEST2(err, KErrNone);
   437 
   438     TInt nextVersion = 10;
   439     TFileName fname;
   440     fname.Format(KCfgDb1ConfigFilePath, nextVersion);
   441     DoCreateCfgFile(fname, KCfgDb1ConfigFileV10ValidStmt);
   442 	
   443 	++nextVersion;
   444     fname.Format(KCfgDb1ConfigFilePath, nextVersion);
   445     DoCreateCfgFile(fname, KCfgDb1ConfigFileV11ValidStmt);
   446 	}
   447  
   448 ///////////////////////////////////////////////////////////////////////////////////////
   449 // DoDbCfgTests() functions
   450 
   451 TInt GetDbCfgVersion(const TDesC& aDbName)
   452     {
   453     // Note: We have to use SQLite directly to access the settings
   454     // table as the SQL Server denies permission to access this table
   455     // as it is in a shared, secure database
   456     
   457     TParse parse;
   458     TInt err = parse.Set(aDbName, &KSqlSrvPrivatePath, 0);
   459     TEST2(err, KErrNone);
   460     
   461     TBuf8<KMaxFileName + 1> dbFileName;
   462     dbFileName.Copy(parse.FullName());
   463     
   464     sqlite3 *dbHandle = NULL;
   465     TInt rc = sqlite3_open((const char*)dbFileName.PtrZ(), &dbHandle);
   466     TEST2(rc, SQLITE_OK);
   467     
   468     _LIT(KGetDbCfgVersionSql, "SELECT Reserved AS DbCfgVersion FROM symbian_settings");
   469     TBuf<100> queryBuf;
   470     queryBuf.Copy(KGetDbCfgVersionSql);
   471     
   472     sqlite3_stmt* stmtHandle = NULL;
   473     const void* stmtTailZ = NULL;
   474     rc = sqlite3_prepare16_v2(dbHandle, queryBuf.PtrZ(), -1, &stmtHandle, &stmtTailZ);
   475     TEST2(rc, SQLITE_OK);
   476     
   477     rc = sqlite3_step(stmtHandle);
   478     TEST2(rc, SQLITE_ROW);
   479     
   480     TInt version = sqlite3_column_int(stmtHandle, 0);
   481     
   482     rc = sqlite3_step(stmtHandle);
   483     TEST2(rc, SQLITE_DONE);
   484     
   485     sqlite3_finalize(stmtHandle);
   486     sqlite3_close(dbHandle);
   487     
   488     return version;
   489     }
   490 
   491 TBool GuessSystemSettingsTable(const TDesC& aDbName, TInt aVersion)
   492 	{
   493 	TInt dbCfgVersion = GetDbCfgVersion(aDbName);
   494 	return dbCfgVersion == aVersion; 
   495 	}
   496 
   497 void CheckSystemSettingsTable(const TDesC& aDbName, TInt aVersion)
   498 	{
   499     TInt dbCfgVersion = GetDbCfgVersion(aDbName);
   500     TEST2(dbCfgVersion, aVersion); 
   501 	}
   502 
   503 void CheckRecordCount(const TDesC& aTable)
   504 	{
   505 	// There should always be only 1 record in the table
   506 	// in each database as INSERT and DELETE statements are
   507 	// not supported in the config files
   508 	TBuf<50> sql;
   509 	sql.Format(_L("SELECT COUNT(*) FROM %S"), &aTable);
   510 	TSqlScalarFullSelectQuery q(TheDb);
   511 	TInt count = 0;
   512     TRAPD(err, count = q.SelectIntL(sql));
   513     TEST2(err, KErrNone);
   514     TEST2(count, 1);
   515 	}
   516 	
   517 TInt ExpectedDb1IndexCount(TInt aExpectedStoredVersion)
   518 	{
   519 	TEST(aExpectedStoredVersion >= 0 && aExpectedStoredVersion < 15);
   520 	const TInt KIdxCnt[] = {
   521 	       -1, 
   522 	        3,             // Only files 01 - 04, 09 and 11 will be successfully processed and so stored in the settings table
   523 	        4,             // 1 more index should be added to db1 based on config file 02
   524 	        4, 4, 4, 4, 4, // no more indices should be added to db1 based on config file 03 - 07 
   525 	        6,             // 2 more indices should be added to db1 based on config file 08 
   526 	        7,             // 1 more index should be added to db1 based on config file 09 
   527 	        -1, 
   528 	        9,             // 2 more indices should be added to db1 based on config file 11 
   529 	        10,            // 1 more index should be added to db1 based on config file 12
   530 	        11,            // 1 more index should be added to db1 based on config file 13
   531 	        12};           // 1 more index should be added to db1 based on config file 14
   532 	return KIdxCnt[aExpectedStoredVersion];
   533 	}
   534 
   535 void CheckIndexCount(TInt aExpectedIndexCount)
   536 	{
   537 	_LIT(KDbCheckNumIndices, "SELECT COUNT(*) FROM sqlite_master WHERE type = 'index'");
   538     TSqlScalarFullSelectQuery q(TheDb);
   539     TInt count = 0;
   540     TRAPD(err, count = q.SelectIntL(KDbCheckNumIndices));
   541     TEST2(err, KErrNone);
   542     TEST2(count, aExpectedIndexCount);
   543 	}
   544 
   545 void DoCfgOpenTest(TInt aExpectedStoredVersion, const TDesC& aDbFileName, const TDesC& aTblName, TInt aExpectedIndexCnt)
   546     {
   547     TInt err = TheDb.Open(aDbFileName);
   548     TEST2(err, KErrNone);
   549     CheckRecordCount(aTblName);
   550     CheckIndexCount(aExpectedIndexCnt);
   551     TheDb.Close();
   552     // Check that the ops in the specified config file have been applied.
   553     CheckSystemSettingsTable(aDbFileName, aExpectedStoredVersion);
   554     }
   555 
   556 void DoCfgOpenTests(TInt aExpectedStoredVersion)
   557 	{	
   558 	// Open a shared, secure database - config ops should be applied on it.
   559     // There should still be only 1 record and 3 indices in the table.
   560 	TheTest.Printf(_L("===CfgOpen: Open shared, secure database\r\n"));
   561 	DoCfgOpenTest(aExpectedStoredVersion, KCfgDb1, _L("table1"), ExpectedDb1IndexCount(aExpectedStoredVersion));
   562 		
   563 	// Open again the same shared, secure database - no config should occur (it has already been done).
   564 	TheTest.Printf(_L("===CfgOpen: Open shared, secure database again\r\n"));
   565     DoCfgOpenTest(aExpectedStoredVersion, KCfgDb1, _L("table1"), ExpectedDb1IndexCount(aExpectedStoredVersion));
   566 	
   567 	// Open a shared, secure database - no config should occur (there is no config file for this database)
   568 	TheTest.Printf(_L("===CfgOpen: Open shared, secure database (that has no config file)\r\n"));
   569     DoCfgOpenTest(0, KCfgDb2, _L("table2"), 0);
   570 	
   571 	// Open a private, secure database - no config should occur
   572 	TheTest.Printf(_L("===CfgOpen: Open private, secure database\r\n"));
   573     DoCfgOpenTest(0, KCfgDb3, _L("table3"), 0);
   574 	
   575 	// Open a public database - no config should occur
   576 	TheTest.Printf(_L("===CfgOpen: Open public database\r\n"));
   577     DoCfgOpenTest(0, KCfgDb4, _L("table4"), 0);
   578 	}
   579 	
   580 void DoUpgradedCfgOpenTest()
   581 	{
   582 	// Upgrade the config file for KCfgDb1, i.e. replace v1 file with v2 file
   583 	UpgradeDbConfigFile(1);
   584 	DoCfgOpenTests(2);	
   585 	}
   586 
   587 void DoBadCfgOpenTests()
   588 	{
   589 	// Test an empty config file
   590 	UpgradeDbConfigFile(2); // the current file version is 02, replace it with 03
   591 	DoCfgOpenTests(3); // version 03 will be successfully processed and stored in the settings table
   592 	
   593 	// Test a config file with unsupported operations (these will be ignored)
   594 	UpgradeDbConfigFile(3); // the current file version is 03, replace it with 04
   595 	DoCfgOpenTests(4); // version 04 will be successfully processed (the unsupported operations are ignored) and stored in the settings table
   596 
   597 	// Test a config file with only whitespace in it
   598 	UpgradeDbConfigFile(4); // the current file version is 04, replace it with 05
   599 	DoCfgOpenTests(5); // version 05 will be successfully processed (the whitespace is ignored) and stored in the settings table
   600 
   601 	// Test a config file with operations on an invalid table
   602 	UpgradeDbConfigFile(5); // the current file version is 05, replace it with 06
   603 	DoCfgOpenTests(6); // version 06 will be successfully processed (the failed-to-execute operation will be ignored) and stored in the settings table
   604 
   605 	// Test a config file that contains an invalid comment style
   606 	UpgradeDbConfigFile(6); // the current file version is 06, replace it with 07
   607 	DoCfgOpenTests(7); // version 07 will be successfully processed (the line with invalid comment syntax will be ignored) and stored in the settings table
   608 
   609 	// Test a config file that contains a sequence of statements as one statement (this is currently unsupported)
   610 	UpgradeDbConfigFile(7); // the current file version is 07, replace it with 08
   611 	DoCfgOpenTests(8); // version 08 will be successfully processed (the line with a sequence of statements is ignored) and stored in the settings table	
   612 
   613 	// Test a config file that contains whitespace before and after the SQL statement
   614 	UpgradeDbConfigFile(8); // the current file version is 08, replace it with 09
   615 	DoCfgOpenTests(9); // version 09 will be successfully processed and stored in the settings table	
   616 
   617 	// Test a config file that has a lower extension number
   618 	DowngradeDbConfigFile(9); // the current file version is 09, replace it with 08
   619 	DoCfgOpenTests(9); // version 08 will NOT be processed (as it is a lower version than that stored), and 09 will remain stored in the settings table
   620 
   621 	// Test a config file that has an invalid extension
   622 	CreateCorruptDbConfigFile(8); // the current file version is 08, replace it with a file with an invalid extension
   623 	DoCfgOpenTests(9); // the invalid file will NOT be processed, and 09 will remain stored in the settings table			
   624 	
   625 	// Test two versions of the config file (two versions should not be present at the same time)
   626 	CreateTwoVersionsOfConfigFile(); // the current file has an invalid extension, delete it and create version 10 and version 11
   627 	DoCfgOpenTests(11); // only version 11 will be processed (as it is the highest version), and 11 will be stored in the settings table
   628 	
   629 	// Test a config file that contains an invalid statement and a valid statement
   630 	UpgradeDbConfigFile(11); // the current file versions are 10 and 11, replace them with 12
   631 	DoCfgOpenTests(12); // version 12 will be successfully processed (the invalid statement will be ignored and the valid statement executed) and stored in the settings table
   632 	
   633 	// Test a config file that contains a SQL style comment
   634 	UpgradeDbConfigFile(12); // the current file version is 12, replace it with 13
   635 	DoCfgOpenTests(13); // version 13 will be successfully processed (the SQL style comment will be ignored) and stored in the settings table
   636 	
   637 	// Test a config file that contains a 'C' style comment
   638 	UpgradeDbConfigFile(13); // the current file version is 13, replace it with 14
   639 	DoCfgOpenTests(14); // version 14 will be successfully processed (the 'C' style comment will be ignored) and stored in the settings table
   640 	}
   641 		
   642 void DoNewDbCfgOpenTest()
   643 	{
   644 	// Create Db5 - a config file already exists for this database
   645 	CreateDb5();
   646 	
   647 	// Open the shared, secure database Db5 - config ops should be applied on it.
   648 	// There should now be 1 index in the table.
   649 	TheTest.Printf(_L("===NewDbCfg: Open shared, secure database\r\n"));
   650     DoCfgOpenTest(1, KCfgDb5, _L("table5"), 1);
   651 		
   652 	// Open again the same shared, secure database - no config should occur (it has already been done)
   653 	TheTest.Printf(_L("===NewDbCfg: Open shared, secure database again\r\n"));
   654     DoCfgOpenTest(1, KCfgDb5, _L("table5"), 1);
   655 	}
   656 	
   657 void DoCfgAttachTests(TInt aExpectedStoredVersion = 0)
   658 	{			
   659 	// Open a private, secure database - no config should occur
   660 	TheTest.Printf(_L("===CfgAttach: Open private, secure database\r\n"));
   661 	TInt err = TheDb.Open(KCfgDb3);
   662 	TEST2(err, KErrNone);
   663 	// Attach a shared, secure database - the db1 config file should not be processed
   664 	TheTest.Printf(_L("===CfgAttach: Attach shared, secure database\r\n"));
   665 	err = TheDb.Attach(KCfgDb1, KAttachDb1);
   666 	TEST2(err, KErrNone);
   667 	TheDb.Close();
   668 	CheckSystemSettingsTable(KCfgDb1, aExpectedStoredVersion); // check that the config file has not been processed for db1
   669 	CheckSystemSettingsTable(KCfgDb3, 0);
   670 	
   671 	// Open a public database - no config should occur
   672 	TheTest.Printf(_L("===CfgAttach: Open public database\r\n"));
   673     err = TheDb.Open(KCfgDb4);
   674 	TEST2(err, KErrNone);
   675 	// Attach a shared, secure database - no config should occur (there is no config file for this database)
   676 	TheTest.Printf(_L("===CfgAttach: Attach shared, secure database (that has no config file)\r\n"));
   677 	err = TheDb.Attach(KCfgDb2, KAttachDb2);
   678 	TEST2(err, KErrNone);
   679 	TheDb.Close();
   680 	CheckSystemSettingsTable(KCfgDb2, 0);
   681 	CheckSystemSettingsTable(KCfgDb4, 0);
   682 	}
   683 	
   684 void DoUpgradedCfgAttachTest()
   685 	{
   686 	// Upgrade the config file for KCfgDb1, i.e. replace v1 file with v2 file
   687 	UpgradeDbConfigFile(1);	
   688 	DoCfgAttachTests(2);
   689 	}
   690 	
   691 void DoBadCfgAttachTests()
   692 	{
   693 	// Test an empty config file
   694 	UpgradeDbConfigFile(2); // the current file version is 02, replace it with 03
   695 	DoCfgAttachTests(3); // version 03 will not be processed so no update to settings table
   696 	
   697 	// Test a config file with unsupported operations
   698 	UpgradeDbConfigFile(3); // the current file version is 03, replace it with 04
   699 	DoCfgAttachTests(4); // version 04 will not be processed so no update to settings table
   700 	
   701 	// Test a config file with only whitespace in it
   702 	UpgradeDbConfigFile(4); // the current file version is 04, replace it with 05
   703 	DoCfgAttachTests(5); // version 05 will not be processed so no update to settings table
   704 	
   705 	// Test a config file with operations on an invalid table
   706 	UpgradeDbConfigFile(5); // the current file version is 05, replace it with 06
   707 	DoCfgAttachTests(6); // version 06 will not be processed so no update to settings table
   708 	
   709 	// Test a config file that contains an invalid comment style
   710 	UpgradeDbConfigFile(6); // the current file version is 06, replace it with 07
   711 	DoCfgAttachTests(7); // version 07 will not be processed so no update to settings table
   712 	
   713 	// Test a config file that contains a sequence of statements as one statement (this is currently unsupported)
   714 	UpgradeDbConfigFile(7); // the current file version is 07, replace it with 08
   715 	DoCfgAttachTests(8); // version 08 will not be processed so no update to settings table
   716 	
   717 	// Test a config file that contains whitespace before and after the SQL statement
   718 	UpgradeDbConfigFile(8); // the current file version is 08, replace it with 09
   719 	DoCfgAttachTests(9); // version 09 will not be processed so no update to settings table
   720 	
   721 	// Test a config file that has a lower extension number
   722 	DowngradeDbConfigFile(9); // the current file version is 09, replace it with 08
   723 	DoCfgAttachTests(9); // version 08 will not be processed so no update to settings table
   724 	
   725 	// Test a config file that has an invalid extension
   726 	CreateCorruptDbConfigFile(8); // the current file version is 08, replace it with a file with an invalid extension
   727 	DoCfgAttachTests(9); // the invalid file will not be processed so no update to settings table
   728 	
   729 	// Test two versions of the config file (two versions should not be present at the same time)
   730 	CreateTwoVersionsOfConfigFile(); // the current file has an invalid extension, delete it and create version 10 and version 11
   731 	DoCfgAttachTests(11); // neither version 10 or 11 will be processed so no update to settings table
   732 		
   733 	// Test a config file that contains an invalid statement and a valid statement
   734 	UpgradeDbConfigFile(11); // the current file versions are 10 and 11, replace them with 12
   735 	DoCfgAttachTests(12); // version 12 will not be processed so no update to settings table
   736 
   737 	// Test a config file that contains a SQL style comment
   738 	UpgradeDbConfigFile(12); // the current file version is 12, replace it with 13
   739 	DoCfgAttachTests(13); // version 13 will not be processed so no update to settings table
   740 	
   741 	// Test a config file that contains a 'C' style comment
   742 	UpgradeDbConfigFile(13); // the current file version is 13, replace it with 14
   743 	DoCfgAttachTests(14); // version 14 will not be processed so no update to settings table
   744 	}
   745 	
   746 void DoNewDbCfgAttachTest()
   747 	{
   748 	// Create Db5 - a config file already exists for this
   749 	CreateDb5();
   750 	
   751 	// Open a private, secure database - no config should occur
   752 	TheTest.Printf(_L("===NewDbCfgAttach: Open private, secure database\r\n"));
   753 	TInt err = TheDb.Open(KCfgDb3);
   754 	TEST2(err, KErrNone);
   755     CheckRecordCount(_L("table3"));
   756     CheckIndexCount(0); 
   757 	
   758 	// Attach a shared, secure database - the db5 config file should not be processed
   759 	TheTest.Printf(_L("===NewDbCfgAttach: Attach shared, secure database\r\n"));
   760 	err = TheDb.Attach(KCfgDb5, KAttachDb5);
   761 	TEST2(err, KErrNone);
   762     CheckRecordCount(_L("table5"));
   763     CheckIndexCount(0); // there should still be no indices in the table
   764 	TheDb.Close();
   765 	CheckSystemSettingsTable(KCfgDb5, 1); // check that the config file has been processed for db1
   766 	CheckSystemSettingsTable(KCfgDb3, 0);
   767 	}
   768 	
   769 void ResetStoredCollationDll()
   770 	{
   771 	// Note: We have to use SQLite directly to access the settings
   772 	// table as the SQL Server denies permission to access this table	
   773 	// as it is in a shared, secure database
   774 	
   775 	TParse parse;
   776 	parse.Set(KCfgDb1, &KSqlSrvPrivatePath, 0);
   777 	
   778 	TBuf8<KMaxFileName + 1> dbFileName;
   779 	dbFileName.Copy(parse.FullName());
   780 	
   781 	sqlite3 *dbHandle = NULL;
   782 	TInt rc = sqlite3_open((const char*)dbFileName.PtrZ(), &dbHandle);
   783 	TEST2(rc, SQLITE_OK);
   784 	
   785 	TBuf<100> queryBuf;
   786 	queryBuf.Append(KResetCollationDllSql());	
   787 	
   788 	sqlite3_stmt* stmtHandle = NULL;
   789 	const void* stmtTailZ = NULL;
   790 	rc = sqlite3_prepare16_v2(dbHandle, queryBuf.PtrZ(), -1, &stmtHandle, &stmtTailZ);
   791 	TEST2(rc, SQLITE_OK);
   792 	
   793 	rc = sqlite3_step(stmtHandle);
   794 	TEST2(rc, SQLITE_DONE);
   795 	
   796 	sqlite3_finalize(stmtHandle);
   797 	sqlite3_close(dbHandle);
   798 	}
   799 	
   800 void CheckCollationDllUpdated()
   801 	{
   802 	// Note: We have to use SQLite directly to access the settings
   803 	// table as the SQL Server denies permission to access this table
   804 	// as it is in a shared, secure database
   805 	
   806 	TParse parse;
   807 	parse.Set(KCfgDb1, &KSqlSrvPrivatePath, 0);
   808 	
   809 	TBuf8<KMaxFileName + 1> dbFileName;
   810 	dbFileName.Copy(parse.FullName());
   811 	
   812 	sqlite3 *dbHandle = NULL;
   813 	TInt rc = sqlite3_open((const char*)dbFileName.PtrZ(), &dbHandle);
   814 	TEST2(rc, SQLITE_OK);
   815 	
   816 	_LIT(KGetCollationDllSql, "SELECT CollationDllName FROM symbian_settings");
   817     TBuf<100> queryBuf;
   818 	queryBuf.Append(KGetCollationDllSql());	
   819 	
   820 	sqlite3_stmt* stmtHandle = NULL;
   821 	const void* stmtTailZ = NULL;
   822 	rc = sqlite3_prepare16_v2(dbHandle, queryBuf.PtrZ(), -1, &stmtHandle, &stmtTailZ);
   823 	TEST2(rc, SQLITE_OK);
   824 	
   825 	rc = sqlite3_step(stmtHandle);
   826 	TEST2(rc, SQLITE_ROW);
   827 	
   828 	const TUint16* collationDllName = (const TUint16*)sqlite3_column_text16(stmtHandle, 0);
   829 	TInt collationDllNameLen  = sqlite3_column_bytes(stmtHandle, 0) / sizeof(TUint16);
   830 	TPtrC ptr(collationDllName, collationDllNameLen);
   831 	
   832 	rc = sqlite3_step(stmtHandle);
   833 	TEST2(rc, SQLITE_DONE);
   834 	
   835 	sqlite3_finalize(stmtHandle);
   836 	sqlite3_close(dbHandle);
   837 	
   838 	_LIT(KTestCollationDllName, "hjagafsff");//The same as the used in KResetCollationDllSql statement
   839 	TEST(ptr != KTestCollationDllName);
   840 	}
   841 
   842 /**
   843 @SYMTestCaseID			SYSLIB-SQL-UT-4010
   844 @SYMTestCaseDesc		New database configuration files feature - unit test.
   845 						The test opens several test databases, and the database configuration
   846 						file for one of the shared, secure databases is processed. This is
   847 						repeated for a variety of versions of the database configuration file, 
   848 						both valid and invalid. The same tests are repeated for when attaching, 
   849 						rather than opening, the test databases. The test also verifies that
   850 						reindexing occurs if necessary when a database is opened or attached,
   851 						regardless of whether or not database configuration is also required.
   852 						Finally, the test also verifies that a config file that exists before
   853 						the database itself is created will be processed when the database is
   854 						opened for the first time.				
   855 @SYMTestPriority		High
   856 @SYMTestActions			New database configuration files feature - unit test.
   857 @SYMTestExpectedResults The test must not fail
   858 @SYMCR 					LMAN-79SJ7L
   859 */
   860 void DoDbCfgTests()
   861 	{
   862 	// Do 'open' tests for new db config file feature
   863 	DoCfgOpenTests(1);         // open the test databases
   864  	DoUpgradedCfgOpenTest();  // upgrade the config file for db1 and reopen the test databases
   865  	DoBadCfgOpenTests();      // corrupt the config file for db1 (in a variety of ways) and reopen the test databases
   866  	DoNewDbCfgOpenTest();		  // create a db for which a config file already exists and then open the db
   867  
   868  	// Recreate the original dbs and config files
   869  	DeleteCfgFilesAndDbs();
   870  	CreateCfgFilesAndDbs();
   871 
   872  	// Do 'attach' tests for new db config file feature
   873 	DoCfgAttachTests(1);		// attach the test databases
   874  	DoUpgradedCfgAttachTest(); // upgrade the config file for db1 and reattach the test databases
   875  	DoBadCfgAttachTests();     // corrupt the config file for db1 (in a variety of ways) and reattach the test databases	
   876     DoNewDbCfgAttachTest();	   // create a db for which a config file already exists and then attach the db
   877 	
   878 	// Recreate the original dbs and config files
   879  	DeleteCfgFilesAndDbs();
   880  	CreateCfgFilesAndDbs();
   881  	
   882  	// Do the test that causes both reindexing and db configuration to occur when db1 is opened
   883 	ResetStoredCollationDll();
   884 	DoCfgOpenTests(1);
   885 	CheckCollationDllUpdated();
   886 	
   887 	// Recreate the original dbs and config files
   888  	DeleteCfgFilesAndDbs();
   889  	CreateCfgFilesAndDbs();
   890  	
   891  	// Do the test that causes reindexing to occur when db1 is attached (no db configuration occurs for attach)
   892 	ResetStoredCollationDll();
   893 	DoCfgAttachTests(1);
   894 	CheckCollationDllUpdated(); // db1's stored collation dll name should now match db3's (to which Db1 is attached) stored collation name
   895 	}	
   896 		
   897 /**
   898 @SYMTestCaseID			SYSLIB-SQL-UT-4013
   899 @SYMTestCaseDesc		New database configuration files feature - OOM test.
   900 						The test opens a shared secure database for which there
   901 						exists a database configuration file that is to be processed.
   902 						The open call is executed within an OOM loop, which causes
   903 						an OOM error to occur at different stages of the open call. 
   904 						The test verifies that OOM errors are handled correctly and 
   905 						that on the final loop - when no OOM errors occur - the database is 
   906 						successfully opened and the database configuration file is processed.	
   907 @SYMTestPriority		High
   908 @SYMTestActions			New database configuration files feature - OOM test.
   909 @SYMTestExpectedResults The test must not fail
   910 @SYMCR 					LMAN-79SJ7L
   911 */
   912 void DoDbCfgOOMTests()
   913 	{
   914 	// Recreate the original dbs and config files
   915 	DeleteCfgFilesAndDbs();
   916 	CreateCfgFilesAndDbs();
   917 	
   918 	// The server is stopped at the end of CreateCfgFilesAndDbs().
   919 	// Before the test loop below begins we attach Db1 so that the 
   920 	// server starts up again and stores Db1's security policy in
   921 	// CSqlServer::iSecurityMap. Otherwise there will be unmatching 
   922 	// allocated cells in the test below. Do not call Detach() before 
   923 	// Close(). We 'attach' Db1 here rather than 'open' it because 
   924 	// open is the API that we want to test in the while loop
   925 	TInt err = TheDb.Open(KCfgDb3);
   926 	TEST2(err, KErrNone);
   927 	err = TheDb.Attach(KCfgDb1, KAttachDb1);
   928 	TEST2(err, KErrNone);
   929 	TheDb.Close();
   930 	
   931 	err = KErrNoMemory;
   932 	TInt failingAllocationNo = 0;
   933 	while(err == KErrNoMemory)
   934 		{		
   935 		MarkHandles();
   936 		MarkAllocatedCells();
   937 		
   938 		__UHEAP_MARK;
   939 
   940 		TheTest.Printf(_L("%d\r"), failingAllocationNo + 1);
   941 
   942 		// Set failure rate
   943 		TSqlResourceTester::SetDbHeapFailure(RHeap::EDeterministic, ++failingAllocationNo);
   944 		err = TheDb.Open(KCfgDb1);
   945 		if(err != KErrNoMemory)
   946 			{
   947 			TEST2(err, KErrNone);
   948 			TheDb.Close();
   949 			if(GuessSystemSettingsTable(KCfgDb1, 0))
   950 				{
   951 				err = KErrNoMemory;
   952 				}
   953 			}
   954 		
   955 		// Reset failure rate
   956 		TSqlResourceTester::SetDbHeapFailure(RHeap::ENone, 0);
   957 		
   958 		TheDb.Close();
   959 		
   960 		__UHEAP_MARKEND;
   961 
   962 		CheckAllocatedCells();	    	
   963 		CheckHandles();	    	
   964 		}
   965 	TEST2(err, KErrNone);
   966 	TheTest.Printf(_L("\r\n"));
   967 	CheckSystemSettingsTable(KCfgDb1, 1); // check that the settings table has been updated with v01 of the config file
   968 	}
   969 	
   970 /**
   971 @SYMTestCaseID			SYSLIB-SQL-UT-4014
   972 @SYMTestCaseDesc		New database configuration files feature - File I/O failures test.
   973 						The test opens a shared secure database for which there
   974 						exists a database configuration file that is to be processed.
   975 						The open call is executed within a file I/O failure loop, which
   976 						causes a selection of file I/O errors to occur at different stages 
   977 						of the open call. 
   978 						The test verifies that file I/O errors are handled correctly and 
   979 						that on the final loop - when no file I/O errors occur - the database 
   980 						is successfully opened and the database configuration file is processed.
   981 						NOTE: This test also acts as the test case for defect DEF116688.
   982 @SYMTestPriority		High
   983 @SYMTestActions			New database configuration files feature - File I/O failures test.
   984 @SYMTestExpectedResults The test must not fail
   985 @SYMCR					LMAN-79SJ7L
   986 */
   987 void DoDbCfgFileIOFailuresTests()
   988 	{
   989 	// Recreate the original dbs and config files
   990 	DeleteCfgFilesAndDbs();
   991 	CreateCfgFilesAndDbs();
   992 	
   993 	// The server is stopped at the end of CreateCfgFilesAndDbs().
   994 	// Before the test loop below begins we start the server again
   995 	// by opening Db3) so that any server start up file i/o ops will 
   996 	// succeed. We don't open Db1 here because opening Db1 is what 
   997 	// we want to test in the while loop
   998 	TInt err = TheDb.Open(KCfgDb3);
   999 	TEST2(err, KErrNone);
  1000 	TheDb.Close();
  1001 
  1002 	TBool isFinished = EFalse;
  1003 	err = KErrNotFound;
  1004 	TInt iter = 0;
  1005 	TheTest.Printf(_L("Iteration\r\n"));
  1006 	for(TInt cnt = 0; !isFinished; ++cnt)
  1007 		{	
  1008 		for(TInt fsError = KErrNotFound; fsError >= KErrUnderflow; --fsError) // errors -1 to -10 will be generated
  1009 			{
  1010 			TheTest.Printf(_L("%d/%d   \r"), ++iter, fsError);
  1011 			(void)TheFs.SetErrorCondition(fsError, cnt); // set error code and how soon it is to occur
  1012 			err = TheDb.Open(KCfgDb1);
  1013 			(void)TheFs.SetErrorCondition(KErrNone);
  1014 			TheDb.Close();
  1015 			if(err != KErrNone)
  1016 				{
  1017 				// An error has occured. We know that this means that the database 
  1018 				// configuration part of the open call wasn't reached (because the
  1019 				// database configuration part returns KErrNone even if it fails).
  1020 				// But check anyway that the database content is still the same as
  1021 				// before, i.e. the settings table has not been updated
  1022 				CheckSystemSettingsTable(KCfgDb1, 0);
  1023 				}
  1024 			else
  1025 				{
  1026 				// Either the database configuration file for Db1 has been successfully
  1027 				// processed or it failed part-way through being processed (KErrNone is 
  1028 				// returned in this case too).
  1029 				// If it was the former then the the settings table will have 
  1030 				// been updated to store v01 of the config file
  1031 				if(GuessSystemSettingsTable(KCfgDb1, 1))
  1032 					{
  1033 					isFinished = ETrue;	
  1034 					break;
  1035 					}
  1036 				}
  1037 			}
  1038 		}
  1039 	TheTest.Printf(_L("\r\n"));
  1040 	TEST2(err, KErrNone);		
  1041 	// Check that the database configuration was allowed to be successfully applied in one of the loops
  1042 	CheckSystemSettingsTable(KCfgDb1, 1); // check that the settings table has been updated with v01 of the config file		
  1043 	}
  1044 	
  1045 /**
  1046 @SYMTestCaseID			SYSLIB-SQL-UT-4015
  1047 @SYMTestCaseDesc		New database configuration files feature - Performance test.
  1048 						The test measures:
  1049 						- the startup time of the SQL Server when 0 and 4 database 
  1050 						configuration files exist
  1051 						- the time taken to open a shared, secure database when a 
  1052 						database configuration file does and does not exist for it
  1053 						- the time taken to open a shared, secure database when a 
  1054 						database configuration file exists for it but it has already
  1055 						been processed
  1056 						- the time taken to attach a shared, secure database when a 
  1057 						database configuration file does and does not exist for it
  1058 @SYMTestPriority		High
  1059 @SYMTestActions			New database configuration files feature - Performance test.
  1060 @SYMTestExpectedResults The test must not fail
  1061 @SYMCR 					LMAN-79SJ7L
  1062 */
  1063 void DoDbCfgPerfTests()
  1064 	{
  1065 	// Recreate the original dbs
  1066 	DeleteCfgFilesAndDbs();
  1067 	CreateCfgDbs(); 
  1068 	KillSqlServer();
  1069 
  1070 	// Measure the start up time of the server when
  1071 	// there are no database configuration files to be cached.
  1072 	// Open Db4 (public db):
  1073 	// No reindexing required
  1074 	// Database configuration is not considered
  1075 	TUint32 start = User::FastCounter();
  1076 	TInt err = TheDb.Open(KCfgDb4);
  1077 	TUint32 end = User::FastCounter();
  1078 	TEST2(err, KErrNone);
  1079     CheckRecordCount(_L("table4"));
  1080     CheckIndexCount(0);
  1081 	TheDb.Close();
  1082 	CheckSystemSettingsTable(KCfgDb4, 0);
  1083 	TInt ms = CalcTimeMs(start, end);
  1084 	TheTest.Printf(_L("Execution time: Server startup (via Open call) - no database config files to cache: %d ms\r\n"), ms);
  1085 	
  1086 	// Measure the time to open Db1 when no database
  1087 	// configuration file exists for it.
  1088 	// Open Db1 (shared, secure db):
  1089 	// No reindexing required
  1090 	// Database configuration is considered but no config file is found
  1091 	start = User::FastCounter();
  1092 	err = TheDb.Open(KCfgDb1);
  1093 	end = User::FastCounter();
  1094 	TEST2(err, KErrNone);
  1095     CheckRecordCount(_L("table1"));
  1096     CheckIndexCount(0);
  1097 	TheDb.Close();
  1098 	CheckSystemSettingsTable(KCfgDb1, 0);
  1099 	ms = CalcTimeMs(start, end);
  1100 	TheTest.Printf(_L("Execution time: Open shared, secure Db1 - no config file is found: %d ms\r\n"), ms);
  1101 	
  1102 	// Measure the time to attach Db1 (database configuration will not be considered).
  1103 	// Attach Db1 (shared, secure db):
  1104 	// No reindexing required
  1105 	// Database configuration is not considered
  1106 	TheDb.Open(KCfgDb3);
  1107 	TEST2(err, KErrNone);
  1108 	start = User::FastCounter();	
  1109 	err = TheDb.Attach(KCfgDb1, KAttachDb1);
  1110 	end = User::FastCounter();
  1111 	TEST2(err, KErrNone);
  1112 	TheDb.Close();
  1113 	CheckSystemSettingsTable(KCfgDb3, 0);
  1114 	CheckSystemSettingsTable(KCfgDb1, 0);
  1115 	ms = CalcTimeMs(start, end);
  1116 	TheTest.Printf(_L("Execution time: Attach shared, secure Db1 - database config is not considered: %d ms\r\n"), ms);
  1117 	
  1118 	// Create the 4 version 01 config files now
  1119 	CreateCfgFiles(); 
  1120 	KillSqlServer(); // stop the server so that the files are found when it is restarted
  1121 	
  1122 	// Measure the start up time of the server when
  1123 	// there are 4 database configuration files to be cached.
  1124 	// Open Db4 (public db):
  1125 	// No reindexing required
  1126 	// Database configuration is not considered
  1127 	start = User::FastCounter();
  1128 	err = TheDb.Open(KCfgDb4);
  1129 	end = User::FastCounter();
  1130 	TEST2(err, KErrNone);
  1131     CheckRecordCount(_L("table4"));
  1132     CheckIndexCount(0);
  1133 	TheDb.Close();
  1134 	CheckSystemSettingsTable(KCfgDb4, 0);
  1135 	ms = CalcTimeMs(start, end);
  1136 	TheTest.Printf(_L("Execution time: Server startup (via Open call) - 4 database config files to cache: %d ms\r\n"), ms);
  1137 	
  1138 	// Measure the time to open Db1 when a database
  1139 	// configuration file exists for it.
  1140 	// Open Db1 (shared, secure db):
  1141 	// No reindexing required
  1142 	// Database configuration is considered, a file is found and config is applied (3 indices are created)
  1143 	start = User::FastCounter();
  1144 	err = TheDb.Open(KCfgDb1);
  1145 	end = User::FastCounter();
  1146 	TEST2(err, KErrNone);
  1147     CheckRecordCount(_L("table1"));
  1148     CheckIndexCount(3);
  1149 	TheDb.Close();
  1150 	CheckSystemSettingsTable(KCfgDb1, 1);
  1151 	ms = CalcTimeMs(start, end);
  1152 	TheTest.Printf(_L("Execution time: Open shared, secure Db1 - config file is found and applied: %d ms\r\n"), ms);
  1153 	
  1154 	// Measure the time to open Db1 when a database
  1155 	// configuration file exists for it but is has already been processed.
  1156 	// Open Db1 (shared, secure db):
  1157 	// No reindexing required
  1158 	// Database configuration is considered, a file is found but it has already been processed
  1159 	start = User::FastCounter();
  1160 	err = TheDb.Open(KCfgDb1);
  1161 	end = User::FastCounter();
  1162 	TEST2(err, KErrNone);
  1163     CheckRecordCount(_L("table1"));
  1164     CheckIndexCount(3);
  1165 	TheDb.Close();
  1166 	CheckSystemSettingsTable(KCfgDb1, 1);
  1167 	ms = CalcTimeMs(start, end);
  1168 	TheTest.Printf(_L("Execution time: Open shared, secure Db1 - config file is found but already processed: %d ms\r\n"), ms);	
  1169 	
  1170 	// Measure the time to attach Db1 (database configuration will not be considered).
  1171 	// Attach Db1 (shared, secure db):
  1172 	// No reindexing required
  1173 	// Database configuration is not considered
  1174 	TheDb.Open(KCfgDb3);
  1175 	TEST2(err, KErrNone);
  1176 	start = User::FastCounter();	
  1177 	err = TheDb.Attach(KCfgDb1, KAttachDb1);
  1178 	end = User::FastCounter();
  1179 	TEST2(err, KErrNone);
  1180 	TheDb.Close();
  1181 	CheckSystemSettingsTable(KCfgDb3, 0);
  1182 	CheckSystemSettingsTable(KCfgDb1, 1);
  1183 	ms = CalcTimeMs(start, end);
  1184 	TheTest.Printf(_L("Execution time: Attach shared, secure Db1 - database config is not considered: %d ms\r\n"), ms);
  1185 	}
  1186 
  1187 void TestStatements()
  1188 	{
  1189 	_LIT(KDbName, "attachDb");
  1190 	TBuf<200> buf;
  1191 
  1192 	// supported statements
  1193 	TBool rc = IsStatementSupported(_L("CREATE INDEX idx ON tbl(ColA)"), KDbName, buf); 
  1194 	TEST(rc);
  1195 	rc = IsStatementSupported(_L("CREATE INDEX IF NOT EXISTS idx ON tbl(ColA)"), KDbName, buf); 
  1196 	TEST(rc);
  1197 
  1198 	// unsupported statements
  1199 	rc = IsStatementSupported(_L("CREATE UNIQUE INDEX idx ON tbl(ColA)"), KDbName, buf); 
  1200 	TEST(!rc);
  1201 	rc = IsStatementSupported(_L("CREATE idx ON tbl(ColA)"), KDbName, buf); 
  1202 	TEST(!rc);
  1203 	}
  1204 
  1205 /**
  1206 @SYMTestCaseID			SYSLIB-SQL-UT-4030
  1207 @SYMTestCaseDesc		Test IsStatementSupportedLC function.
  1208 						This function checks either SQL statement from DB configuration file is supported or not.
  1209 						In the case of supported statement, function allocates buffer and copies statement into that buffer.
  1210 						Call IsStatementSupportedLC on several supported and unsupported SQL statements.
  1211 @SYMTestPriority		High
  1212 @SYMTestActions			Test IsStatementSupportedLC function.
  1213 @SYMTestExpectedResults The test must not fail
  1214 @SYMDEF					DEF118058
  1215 */
  1216 void DoIsStatementSupportedTests()
  1217 	{
  1218 	TestStatements();
  1219 	}
  1220 
  1221 void DoLongDbNameTest()
  1222 	{
  1223 	//Create the database	
  1224  	RSqlSecurityPolicy securityPolicy = CreateSecurityPolicy();
  1225  	
  1226  	TInt err = TheDb.Create(KLongDbName1, securityPolicy);
  1227  	TEST2(err, KErrNone);
  1228  	err = TheDb.Exec(_L("CREATE TABLE table1(i1 INTEGER, i2 INTEGER, i3 INTEGER)"));
  1229  	TEST(err >= 0);
  1230  	err = TheDb.Exec(_L("INSERT INTO table1 (i1,i2,i3) values(1,2,3)"));
  1231  	TEST(err == 1);
  1232  	TheDb.Close();
  1233  	//Kill the server (to reload config file names at the server startup)
  1234  	KillSqlServer();
  1235 	///////////////////////////////////////////////////////////////////////
  1236 	TheTest.Printf(_L("Open a database with a long name\r\n"));
  1237 	//Create cfg file
  1238  	RFile file;
  1239 	TFileName fileName;
  1240 	err = file.Create(TheFs, KLongCfgName1, EFileRead | EFileWrite);
  1241 	TEST2(err, KErrNone);
  1242 	TPtrC8 pDb1((const TUint8*)KCfgConfigFileValidStmt().Ptr(), KCfgConfigFileValidStmt().Length());
  1243 	err = file.Write(pDb1);	
  1244 	file.Close();	
  1245 	TEST2(err, KErrNone);
  1246 	//Open the database
  1247 	err = TheDb.Open(KLongDbName1);
  1248 	TEST2(err, KErrNone);
  1249     CheckRecordCount(_L("table1"));
  1250     CheckIndexCount(1); // there should now be 1 index in the table
  1251 	TheDb.Close();
  1252 	const TInt KVersion = 1;
  1253 	CheckSystemSettingsTable(KLongDbName1, KVersion); // check that the ops in the specified config file have been applied
  1254 	///////////////////////////////////////////////////////////////////////
  1255 	TheTest.Printf(_L("Attach a database with a long logical name\r\n"));
  1256 	//Attached database test
  1257 	//Recreate the database
  1258  	(void)RSqlDatabase::Delete(KLongDbName1);
  1259  	err = TheDb.Create(KLongDbName1, securityPolicy);
  1260  	TEST2(err, KErrNone);
  1261  	err = TheDb.Exec(_L("CREATE TABLE table1(i1 INTEGER, i2 INTEGER, i3 INTEGER)"));
  1262  	TEST(err >= 0);
  1263  	err = TheDb.Exec(_L("INSERT INTO table1 (i1,i2,i3) values(1,2,3)"));
  1264  	TEST(err == 1);
  1265  	TheDb.Close();
  1266  	//Kill the server (to reload config file names at the server startup)
  1267  	KillSqlServer();
  1268 	//Open the main database
  1269  	err = TheDb.Open(KCfgDb1);
  1270  	securityPolicy.Close();
  1271 	TEST2(err, KErrNone);
  1272 	//Attach a database with a very long logical name
  1273 	TFileName attachDbName;
  1274 	attachDbName.SetLength(KMaxFileName);
  1275 	attachDbName.Fill(TChar('A'));
  1276  	err = TheDb.Attach(KLongDbName1, attachDbName);
  1277 	TEST2(err, KErrNone);
  1278  	err = TheDb.Detach(attachDbName);
  1279 	TEST2(err, KErrNone);
  1280  	TheDb.Close();
  1281 	CheckSystemSettingsTable(KLongDbName1, KVersion); // check that the ops in the specified config file have been applied
  1282 	//Cleanup
  1283  	(void)RSqlDatabase::Delete(KLongDbName1);
  1284 	}
  1285 
  1286 void DoTests()
  1287     {
  1288     TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4010 Database config files test"));
  1289     DoDbCfgTests();
  1290     
  1291     TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4013 Database config files OOM test"));
  1292     DoDbCfgOOMTests();
  1293     
  1294     TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4014 Database config files file I/O error simulation test"));
  1295     DoDbCfgFileIOFailuresTests();
  1296     
  1297     TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4015 Database config files performance test"));
  1298     DoDbCfgPerfTests();
  1299 
  1300     TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4030 Database config files - supported SQl statements test"));
  1301     DoIsStatementSupportedTests();
  1302 
  1303     TheTest.Next(_L("'Long database name' tests"));
  1304     DoLongDbNameTest();
  1305     }
  1306 
  1307 #endif	//SYSLIBS_TEST
  1308 
  1309 TInt E32Main()
  1310 	{
  1311 	TheTest.Title();
  1312 	
  1313 	CTrapCleanup* tc = CTrapCleanup::New();
  1314 	TheTest(tc != NULL);
  1315 	
  1316 	__UHEAP_MARK;
  1317 
  1318 #ifdef SYSLIBS_TEST	
  1319 	TheTest.Start(_L("Setting up the test environment"));
  1320 	SetupTestEnv();
  1321 	sqlite3SymbianLibInit();
  1322 	DoTests();
  1323  	DestroyTestEnv();
  1324 	sqlite3SymbianLibFinalize();
  1325 	CloseSTDLIB();
  1326 	TheTest.End();
  1327 #else
  1328  	TheTest.Start(_L("This test works only if the whole SQL component is built with SYSLIBS_TEST macro defined!"));
  1329 	TheTest.End();
  1330 #endif	
  1331 	
  1332 	__UHEAP_MARKEND;
  1333 	
  1334 	TheTest.Close();
  1335 	
  1336 	delete tc;
  1337 	
  1338 	User::Heap().Check();
  1339 	return KErrNone;
  1340 	}