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