os/persistentdata/persistentstorage/sql/TEST/t_sqlprivcage.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2006-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 <bautils.h>
sl@0
    18
#include <sqldb.h>
sl@0
    19
sl@0
    20
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    21
sl@0
    22
static RFs TheFs;
sl@0
    23
RTest TheTest(_L("t_sqlprivcage test"));
sl@0
    24
sl@0
    25
_LIT(KTestDir, "c:\\test\\");
sl@0
    26
_LIT(KTestDb1, "c:\\private\\21212124\\t_sqlprivcage_1.db");
sl@0
    27
_LIT(KTestDb2, "c:\\private\\21212124\\t_sqlprivcage_2.db");
sl@0
    28
_LIT(KTestDbZ, "z:\\private\\21212124\\t_sqldb1.db");//Created outside this test
sl@0
    29
_LIT(KTestDb,  "\\private\\21212124\\t_sqlprivcage_3.db");
sl@0
    30
TParse TheFileNameParse;
sl@0
    31
sl@0
    32
static RCriticalSection ThreadCritSect;
sl@0
    33
static RCriticalSection MainCritSect;
sl@0
    34
sl@0
    35
_LIT(KPanicCategory, "TransFail");
sl@0
    36
const TInt KPanicCode = 1111;
sl@0
    37
sl@0
    38
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    39
sl@0
    40
//Deletes all created test files.
sl@0
    41
void DeleteTestFiles()
sl@0
    42
	{
sl@0
    43
	if(TheFileNameParse.FullName().Length() > 0)
sl@0
    44
		{
sl@0
    45
		(void)RSqlDatabase::Delete(TheFileNameParse.FullName());
sl@0
    46
		}
sl@0
    47
	(void)RSqlDatabase::Delete(KTestDb2);
sl@0
    48
	(void)RSqlDatabase::Delete(KTestDb1);
sl@0
    49
	}
sl@0
    50
sl@0
    51
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    52
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    53
//Test macros and functions
sl@0
    54
void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
sl@0
    55
	{
sl@0
    56
	if(!aValue)
sl@0
    57
		{
sl@0
    58
		DeleteTestFiles();
sl@0
    59
		if(aPrintThreadName)
sl@0
    60
			{
sl@0
    61
			RThread th;
sl@0
    62
			TName name = th.Name();
sl@0
    63
			RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
sl@0
    64
			}
sl@0
    65
		else
sl@0
    66
			{
sl@0
    67
			RDebug::Print(_L("*** Line %d\r\n"), aLine);
sl@0
    68
			}
sl@0
    69
		TheTest(EFalse, aLine);
sl@0
    70
		}
sl@0
    71
	}
sl@0
    72
void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
sl@0
    73
	{
sl@0
    74
	if(aValue != aExpected)
sl@0
    75
		{
sl@0
    76
		DeleteTestFiles();
sl@0
    77
		if(aPrintThreadName)
sl@0
    78
			{
sl@0
    79
			RThread th;
sl@0
    80
			TName name = th.Name();
sl@0
    81
			RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
sl@0
    82
			}
sl@0
    83
		else
sl@0
    84
			{
sl@0
    85
			RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
sl@0
    86
			}
sl@0
    87
		TheTest(EFalse, aLine);
sl@0
    88
		}
sl@0
    89
	}
sl@0
    90
#define TEST(arg) ::Check1((arg), __LINE__)
sl@0
    91
#define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
sl@0
    92
#define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
sl@0
    93
#define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
sl@0
    94
sl@0
    95
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    96
sl@0
    97
//Creates file session instance and the test directory
sl@0
    98
void CreateTestEnv()
sl@0
    99
    {
sl@0
   100
	TInt err = TheFs.Connect();
sl@0
   101
	TEST2(err, KErrNone);
sl@0
   102
sl@0
   103
	err = TheFs.CreatePrivatePath(EDriveC);
sl@0
   104
	TEST(err == KErrNone || err == KErrAlreadyExists);
sl@0
   105
sl@0
   106
	err = TheFs.MkDir(KTestDir);
sl@0
   107
	TEST(err == KErrNone || err == KErrAlreadyExists);
sl@0
   108
	}
sl@0
   109
sl@0
   110
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   111
sl@0
   112
/**
sl@0
   113
@SYMTestCaseID			SYSLIB-SQL-CT-1764
sl@0
   114
@SYMTestCaseDesc		The test creates a database in the test application's private data cage.
sl@0
   115
						Then the test does some operations with the created private database:
sl@0
   116
						create table, insert records, select records, transactions, delete database.
sl@0
   117
						The tests verifies that the SQL server can create a database in the application's
sl@0
   118
						private data cage and can operate with the database.
sl@0
   119
@SYMTestPriority		High
sl@0
   120
@SYMTestActions			SQL, Private database test.
sl@0
   121
@SYMTestExpectedResults Test must not fail
sl@0
   122
@SYMREQ					REQ5792
sl@0
   123
                        REQ5793
sl@0
   124
*/	
sl@0
   125
void SimpleDbOpTest()
sl@0
   126
	{
sl@0
   127
	//Create private database
sl@0
   128
	RSqlDatabase db;
sl@0
   129
	TInt err = db.Create(KTestDb1);
sl@0
   130
	TEST2(err, KErrNone);
sl@0
   131
	
sl@0
   132
	//Execute some operations with the private database
sl@0
   133
	err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
sl@0
   134
	TEST(err >= 0);	
sl@0
   135
	err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1);INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
sl@0
   136
	TEST(err >= 0);	
sl@0
   137
sl@0
   138
	//Check written records
sl@0
   139
	RSqlStatement stmt;
sl@0
   140
	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
sl@0
   141
	TEST2(err, KErrNone);
sl@0
   142
	err = stmt.Next();
sl@0
   143
	TEST2(err, KSqlAtRow);
sl@0
   144
	TEST2(stmt.ColumnInt(0), 1);
sl@0
   145
	err = stmt.Next();
sl@0
   146
	TEST2(err, KSqlAtRow);
sl@0
   147
	TEST2(stmt.ColumnInt(0), 2);
sl@0
   148
	err = stmt.Next();
sl@0
   149
	TEST2(err, KSqlAtRow);
sl@0
   150
	TEST2(stmt.ColumnInt(0), 3);
sl@0
   151
	err = stmt.Next();
sl@0
   152
	TEST2(err, KSqlAtEnd);
sl@0
   153
	stmt.Close();
sl@0
   154
sl@0
   155
	db.Close();
sl@0
   156
	//Open private database
sl@0
   157
	err = db.Open(KTestDb1);
sl@0
   158
	TEST2(err, KErrNone);
sl@0
   159
sl@0
   160
	//Execute a DELETE transaction
sl@0
   161
	err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
sl@0
   162
	TEST(err >= 0);	
sl@0
   163
sl@0
   164
	err = db.Exec(_L8("DELETE FROM A WHERE ID > 1"));
sl@0
   165
	TEST(err >= 0);	
sl@0
   166
sl@0
   167
	err = db.Exec(_L8("COMMIT TRANSACTION"));
sl@0
   168
	TEST(err >= 0);	
sl@0
   169
sl@0
   170
	//Check records left
sl@0
   171
	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
sl@0
   172
	TEST2(err, KErrNone);
sl@0
   173
	err = stmt.Next();
sl@0
   174
	TEST2(err, KSqlAtRow);
sl@0
   175
	TEST2(stmt.ColumnInt(0), 1);
sl@0
   176
	err = stmt.Next();
sl@0
   177
	TEST2(err, KSqlAtEnd);
sl@0
   178
	stmt.Close();
sl@0
   179
sl@0
   180
	//Open private database which is on drive Z and with the same name
sl@0
   181
	RSqlDatabase db2;
sl@0
   182
	err = db2.Open(KTestDbZ);
sl@0
   183
	TEST2(err, KErrNone);
sl@0
   184
	//An attempt to write to a read-only database
sl@0
   185
	err = db2.Exec(_L("INSERT INTO A VALUES(6)"));
sl@0
   186
	TheTest.Printf(_L(" === Read-only private database. RSqlDatabase::Exec() returned err=%d\r\n"), err);
sl@0
   187
	TEST(err != KErrNone);
sl@0
   188
	//Check records
sl@0
   189
	err = stmt.Prepare(db2, _L8("SELECT * FROM A"));
sl@0
   190
	TEST2(err, KErrNone);
sl@0
   191
	err = stmt.Next();
sl@0
   192
	TEST2(err, KSqlAtRow);
sl@0
   193
	TEST2(stmt.ColumnInt(0), 1);
sl@0
   194
	err = stmt.Next();
sl@0
   195
	TEST2(err, KSqlAtEnd);
sl@0
   196
	stmt.Close();
sl@0
   197
	db2.Close();
sl@0
   198
	
sl@0
   199
	db.Close();
sl@0
   200
	err = RSqlDatabase::Delete(KTestDb1);
sl@0
   201
	TEST2(err, KErrNone);
sl@0
   202
sl@0
   203
	//Create private database on drive different than C:
sl@0
   204
	for(TInt drvNum=EDriveD;drvNum<=EDriveZ;++drvNum)
sl@0
   205
		{
sl@0
   206
		TDriveUnit drvUnit(drvNum);
sl@0
   207
		TPtrC drvName = drvUnit.Name();
sl@0
   208
		TheFileNameParse.Set(KTestDb, &drvName, 0);
sl@0
   209
		//Check if it is possible to create application's private data cage on drvNum drive.	
sl@0
   210
		err = TheFs.CreatePrivatePath(drvNum);
sl@0
   211
		if(err == KErrNone || err == KErrAlreadyExists)
sl@0
   212
			{
sl@0
   213
			(void)RSqlDatabase::Delete(TheFileNameParse.FullName());
sl@0
   214
			err = db.Create(TheFileNameParse.FullName());
sl@0
   215
			if(err == KErrNone)
sl@0
   216
				{
sl@0
   217
				//Execute some operations with the private database
sl@0
   218
				err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
sl@0
   219
				TEST(err >= 0);	
sl@0
   220
				err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
sl@0
   221
				TEST(err >= 0);	
sl@0
   222
				err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1);INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
sl@0
   223
				TEST(err >= 0);	
sl@0
   224
				err = db.Exec(_L8("COMMIT TRANSACTION"));
sl@0
   225
				TEST(err >= 0);	
sl@0
   226
				db.Close();
sl@0
   227
				err = RSqlDatabase::Delete(TheFileNameParse.FullName());
sl@0
   228
				TEST2(err, KErrNone);
sl@0
   229
				break;
sl@0
   230
				}
sl@0
   231
			}
sl@0
   232
		TheFileNameParse.Set(KNullDesC, 0, 0);
sl@0
   233
		}
sl@0
   234
		
sl@0
   235
	//An attempt to create/open "C:[21212122]BBDb2.db" - this test has no enough rights to do that.
sl@0
   236
	//...open as a non-secure database
sl@0
   237
	err = db.Open(_L("C:[21212122]BBDb2.db"));
sl@0
   238
	TEST2(err, KErrPermissionDenied);
sl@0
   239
	//...create as a non-secure database
sl@0
   240
	err = db.Create(_L("C:[21212122]BBDb2.db"));
sl@0
   241
	TEST2(err, KErrArgument);//secure database name, no security policy
sl@0
   242
    //Very long private database name
sl@0
   243
    err = db.Create(_L("c:\\private\\21212124\\hhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh.db"));
sl@0
   244
    TEST2(err, KErrBadName);
sl@0
   245
    //Private database name without drive letter
sl@0
   246
    err = db.Create(_L("\\private\\21212124\\dbname_no_drive_letter.db"));
sl@0
   247
    TEST2(err, KErrBadName);
sl@0
   248
    //Zero length private database name
sl@0
   249
    err = db.Create(_L(""));
sl@0
   250
    TEST2(err, KErrBadName);
sl@0
   251
    //Private database + very long config string
sl@0
   252
    _LIT8(KVeryLongConfig, "jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj");
sl@0
   253
    err = db.Create(KTestDb2, &KVeryLongConfig);
sl@0
   254
    TEST2(err, KErrArgument);
sl@0
   255
	//...create as a secure database
sl@0
   256
	RSqlSecurityPolicy dbSecurity;
sl@0
   257
	TSecurityPolicy policy(TSecurityPolicy::EAlwaysPass);
sl@0
   258
	err = dbSecurity.Create(policy);
sl@0
   259
	TEST2(err, KErrNone);
sl@0
   260
	err = db.Create(_L("C:[21212122]BBDb2.db"), dbSecurity);
sl@0
   261
	TEST2(err, KErrPermissionDenied);
sl@0
   262
	dbSecurity.Close();
sl@0
   263
sl@0
   264
	//An attempt to delete "C:[21212122]BBDb2.db" - this test has no enough rights to do that.
sl@0
   265
	err = RSqlDatabase::Delete(_L("C:[21212122]BBDb2.db"));
sl@0
   266
	TEST2(err, KErrPermissionDenied);
sl@0
   267
	}
sl@0
   268
sl@0
   269
/**
sl@0
   270
@SYMTestCaseID			SYSLIB-SQL-CT-1765
sl@0
   271
@SYMTestCaseDesc		The test creates two databases in the test application's private data cage.
sl@0
   272
						Then the test inserts some records in both databases using separate RSqlDatabase objects.
sl@0
   273
						The test closes both databases, then reopens the first one and attaches th second one.
sl@0
   274
						Again, the test inserts some records in both databases, using single RSqlDatabase object.
sl@0
   275
						The test reads the inserted records and verifies their column values.
sl@0
   276
						The tests verifies that the SQL server can create a database in the application's
sl@0
   277
						private data cage, can operate with the database and can attach private databases.
sl@0
   278
@SYMTestPriority		High
sl@0
   279
@SYMTestActions			SQL, Attach private database test.
sl@0
   280
@SYMTestExpectedResults Test must not fail
sl@0
   281
@SYMREQ					REQ5792
sl@0
   282
                        REQ5793
sl@0
   283
*/	
sl@0
   284
void AttachDbOpTest()
sl@0
   285
	{
sl@0
   286
	//Create private database 1
sl@0
   287
	RSqlDatabase db1;
sl@0
   288
	TInt err = db1.Create(KTestDb1);
sl@0
   289
	TEST2(err, KErrNone);
sl@0
   290
	//Execute some operations with private database 1
sl@0
   291
	err = db1.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
sl@0
   292
	TEST(err >= 0);	
sl@0
   293
	err = db1.Exec(_L8("INSERT INTO A(ID) VALUES(1)"));
sl@0
   294
	TEST2(err, 1);
sl@0
   295
	db1.Close();
sl@0
   296
sl@0
   297
	//Create private database 2
sl@0
   298
	RSqlDatabase db2;
sl@0
   299
	err = db2.Create(KTestDb2);
sl@0
   300
	TEST2(err, KErrNone);
sl@0
   301
	//Execute some operations with private database 2
sl@0
   302
	err = db2.Exec(_L8("CREATE TABLE A(ID INTEGER, T TEXT)"));
sl@0
   303
	TEST(err >= 0);	
sl@0
   304
	err = db2.Exec(_L8("INSERT INTO A(ID, T) VALUES(1, 'NAME-NAME-NAME')"));
sl@0
   305
	TEST2(err, 1);
sl@0
   306
	db2.Close();
sl@0
   307
sl@0
   308
	//Open database 1
sl@0
   309
	RSqlDatabase db;
sl@0
   310
	err = db.Open(KTestDb1);
sl@0
   311
	TEST2(err, KErrNone);
sl@0
   312
	//An attempt to attach a database with zero length name
sl@0
   313
    err = db.Attach(_L(""), _L("Db2"));
sl@0
   314
    TEST2(err, KErrBadName);
sl@0
   315
    //Attach database 2
sl@0
   316
	err = db.Attach(KTestDb2, _L("Db2"));
sl@0
   317
	TEST2(err, KErrNone);
sl@0
   318
sl@0
   319
	//Insert some records
sl@0
   320
	err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
sl@0
   321
	TEST(err >= 0);	
sl@0
   322
	err = db.Exec(_L8("INSERT INTO Main.A(ID) VALUES(2);INSERT INTO Db2.A(ID, T) VALUES(2, 'AAA');"));
sl@0
   323
	TEST(err >= 0);	
sl@0
   324
	err = db.Exec(_L8("COMMIT TRANSACTION"));
sl@0
   325
	TEST(err >= 0);	
sl@0
   326
	
sl@0
   327
	err = db.Detach(_L("Db2"));
sl@0
   328
	TEST2(err, KErrNone);
sl@0
   329
	db.Close();
sl@0
   330
	
sl@0
   331
	//Verify inserted data in database 2
sl@0
   332
	err = db.Open(KTestDb2);
sl@0
   333
	TEST2(err, KErrNone);
sl@0
   334
	
sl@0
   335
	RSqlStatement stmt;
sl@0
   336
	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
sl@0
   337
	TEST2(err, KErrNone);
sl@0
   338
	
sl@0
   339
	err = stmt.Next();
sl@0
   340
	TEST2(err, KSqlAtRow);
sl@0
   341
	TEST2(stmt.ColumnInt(0), 1);
sl@0
   342
	TPtrC text;
sl@0
   343
	err = stmt.ColumnText(1, text);
sl@0
   344
	TEST2(err, KErrNone);
sl@0
   345
	TEST(text == _L("NAME-NAME-NAME"));
sl@0
   346
sl@0
   347
	err = stmt.Next();
sl@0
   348
	TEST2(err, KSqlAtRow);
sl@0
   349
	TEST2(stmt.ColumnInt(0), 2);
sl@0
   350
	err = stmt.ColumnText(1, text);
sl@0
   351
	TEST2(err, KErrNone);
sl@0
   352
	TEST(text == _L("AAA"));
sl@0
   353
	
sl@0
   354
	err = stmt.Next();
sl@0
   355
	TEST2(err, KSqlAtEnd);
sl@0
   356
	stmt.Close();
sl@0
   357
	db.Close();
sl@0
   358
		
sl@0
   359
    //Open database 1, attach read-only database 2
sl@0
   360
    err = db.Open(KTestDb1);
sl@0
   361
    TEST2(err, KErrNone);
sl@0
   362
    //Make database 2 read-only.
sl@0
   363
    err = TheFs.SetAtt(KTestDb2, KEntryAttReadOnly, 0);
sl@0
   364
    TEST2(err, KErrNone);
sl@0
   365
    //Attach database 2
sl@0
   366
    err = db.Attach(KTestDb2, _L("Db2"));
sl@0
   367
    TEST2(err, KErrNone);
sl@0
   368
    //
sl@0
   369
    err = db.Exec(_L("INSERT INTO Db2.A(ID, T) VALUES(3, 'AAA')"));
sl@0
   370
    TPtrC errmsg = db.LastErrorMessage();
sl@0
   371
    TheTest.Printf(_L(" === Read-only private attached database. Msg=%S, err=%d\r\n"), &errmsg, err);
sl@0
   372
    TEST(err != KErrNone);
sl@0
   373
    TSqlScalarFullSelectQuery q(db);
sl@0
   374
    TBuf<20> text2;
sl@0
   375
    TRAP(err, q.SelectTextL(_L("SELECT T FROM Db2.A WHERE ID=2"), text2));
sl@0
   376
    TEST2(err, KErrNone);
sl@0
   377
    TEST(text2 == _L("AAA"));
sl@0
   378
    //
sl@0
   379
    err = db.Detach(_L("Db2"));
sl@0
   380
    TEST2(err, KErrNone);
sl@0
   381
    err = TheFs.SetAtt(KTestDb2, 0, KEntryAttReadOnly);
sl@0
   382
    TEST2(err, KErrNone);
sl@0
   383
    db.Close();
sl@0
   384
    
sl@0
   385
	err = RSqlDatabase::Delete(KTestDb2);
sl@0
   386
	TEST2(err, KErrNone);
sl@0
   387
	err = RSqlDatabase::Delete(KTestDb1);
sl@0
   388
	TEST2(err, KErrNone);
sl@0
   389
	}
sl@0
   390
sl@0
   391
/**
sl@0
   392
@SYMTestCaseID			SYSLIB-SQL-CT-1766
sl@0
   393
@SYMTestCaseDesc		The test creates a database in the test application's private data cage.
sl@0
   394
						Then the test creates two RSqlDatabase obejcts connecting them to the same 
sl@0
   395
						private database. The test inserts some records using both connections.
sl@0
   396
						The test verifies that the inserted records are in the database file and
sl@0
   397
						verifies the column values.
sl@0
   398
						The test verifies that it is possible to make more than one connection to the
sl@0
   399
						same private database and operate with the database using the database connections.
sl@0
   400
@SYMTestPriority		High
sl@0
   401
@SYMTestActions			SQL, Two database connections to the same private database test.
sl@0
   402
@SYMTestExpectedResults Test must not fail
sl@0
   403
@SYMREQ					REQ5792
sl@0
   404
                        REQ5793
sl@0
   405
*/	
sl@0
   406
void TwoDbOpTest()
sl@0
   407
	{
sl@0
   408
	//Create private database
sl@0
   409
	RSqlDatabase db1;
sl@0
   410
	TInt err = db1.Create(KTestDb1);
sl@0
   411
	TEST2(err, KErrNone);
sl@0
   412
	
sl@0
   413
	//Make a second connection with the test database
sl@0
   414
	RSqlDatabase db2;
sl@0
   415
	err = db2.Open(KTestDb1);
sl@0
   416
	TEST2(err, KErrNone);
sl@0
   417
	
sl@0
   418
	//Execute some operations with the private database
sl@0
   419
	err = db1.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
sl@0
   420
	TEST(err >= 0);	
sl@0
   421
	err = db1.Exec(_L8("INSERT INTO A(ID) VALUES(1);"));
sl@0
   422
	TEST2(err, 1);
sl@0
   423
	err = db2.Exec(_L8("INSERT INTO A(ID) VALUES(2);"));
sl@0
   424
	TEST2(err, 1);
sl@0
   425
sl@0
   426
	//Verify inserted data
sl@0
   427
	RSqlStatement stmt;
sl@0
   428
	err = stmt.Prepare(db2, _L8("SELECT * FROM A"));
sl@0
   429
	TEST2(err, KErrNone);
sl@0
   430
	
sl@0
   431
	err = stmt.Next();
sl@0
   432
	TEST2(err, KSqlAtRow);
sl@0
   433
	TEST2(stmt.ColumnInt(0), 1);
sl@0
   434
	err = stmt.Next();
sl@0
   435
	TEST2(err, KSqlAtRow);
sl@0
   436
	TEST2(stmt.ColumnInt(0), 2);
sl@0
   437
	err = stmt.Next();
sl@0
   438
	TEST2(err, KSqlAtEnd);
sl@0
   439
	stmt.Close();
sl@0
   440
sl@0
   441
	db2.Close();
sl@0
   442
	db1.Close();
sl@0
   443
	err = RSqlDatabase::Delete(KTestDb1);
sl@0
   444
	TEST2(err, KErrNone);
sl@0
   445
	}
sl@0
   446
sl@0
   447
//Test thread function.
sl@0
   448
//The test thread opens a database, begins a transaction and then simulates a crash within the transaction.
sl@0
   449
TInt ThreadFunc1(void*)
sl@0
   450
	{
sl@0
   451
	__UHEAP_MARK;
sl@0
   452
	
sl@0
   453
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   454
	TTEST(tc != NULL);
sl@0
   455
sl@0
   456
	//Open test database
sl@0
   457
	RSqlDatabase db;
sl@0
   458
	TInt err = db.Open(KTestDb1);
sl@0
   459
	TTEST2(err, KErrNone);
sl@0
   460
sl@0
   461
	RDebug::Print(_L("---:WorkThread: Begin transaction. Exec SQL...\r\n"));
sl@0
   462
	
sl@0
   463
	//Begin a transaction
sl@0
   464
	err = db.Exec(_L8("BEGIN IMMEDIATE TRANSACTION"));
sl@0
   465
	TTEST(err >= 0);	
sl@0
   466
sl@0
   467
	//Execute INSERT sql statements
sl@0
   468
	err = db.Exec(_L8("INSERT INTO A(ID) VALUES(2);INSERT INTO A(ID) VALUES(3);"));
sl@0
   469
	TTEST(err >= 0);	
sl@0
   470
	
sl@0
   471
	RDebug::Print(_L("---:WorkThread: Notify the main thread about the SQL statement execution\r\n"));
sl@0
   472
	MainCritSect.Signal();
sl@0
   473
sl@0
   474
	RDebug::Print(_L("---:WorkThread: Wait for permisson to continue...\r\n"));
sl@0
   475
	ThreadCritSect.Wait();
sl@0
   476
sl@0
   477
	User::SetJustInTime(EFalse);	// disable debugger panic handling
sl@0
   478
sl@0
   479
	//Panic current thread without commiting the transaction (crash simulation)
sl@0
   480
	RDebug::Print(_L("---:WorkThread: Panic!\r\n"));
sl@0
   481
	User::Panic(KPanicCategory, KPanicCode);
sl@0
   482
sl@0
   483
	delete tc;	
sl@0
   484
	
sl@0
   485
	__UHEAP_MARKEND;
sl@0
   486
	
sl@0
   487
	return KErrNone;		
sl@0
   488
	}
sl@0
   489
sl@0
   490
/**
sl@0
   491
@SYMTestCaseID			SYSLIB-SQL-CT-1767
sl@0
   492
@SYMTestCaseDesc		The test creates a database in the test application's private data cage.
sl@0
   493
						The test does some operations with the database leaving it in state A.
sl@0
   494
						Then the test creates a test thread and runs the thread. The test thread
sl@0
   495
						opens a connection to the database, begins a transaction, inserts some data
sl@0
   496
						and then simulates a crash within the transaction (kills the thread).
sl@0
   497
						When the main thread takes the execution control, it reopens the database connection
sl@0
   498
						and verifies that the database is in the same state A as it was before.
sl@0
   499
@SYMTestPriority		High
sl@0
   500
@SYMTestActions			SQL, Private database - transaction recovery test.
sl@0
   501
@SYMTestExpectedResults Test must not fail
sl@0
   502
@SYMREQ					REQ5792
sl@0
   503
                        REQ5793
sl@0
   504
*/	
sl@0
   505
void TransDbOpTest()
sl@0
   506
	{
sl@0
   507
	//Create private database
sl@0
   508
	RSqlDatabase db;
sl@0
   509
	TInt err = db.Create(KTestDb1);
sl@0
   510
	TEST2(err, KErrNone);
sl@0
   511
	
sl@0
   512
	//Execute some operations with the private database
sl@0
   513
	err = db.Exec(_L8("CREATE TABLE A(ID INTEGER)"));
sl@0
   514
	TEST(err >= 0);	
sl@0
   515
	err = db.Exec(_L8("INSERT INTO A(ID) VALUES(1)"));
sl@0
   516
	TEST2(err, 1);
sl@0
   517
sl@0
   518
	//Check written records
sl@0
   519
	RSqlStatement stmt;
sl@0
   520
	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
sl@0
   521
	TEST2(err, KErrNone);
sl@0
   522
	err = stmt.Next();
sl@0
   523
	TEST2(err, KSqlAtRow);
sl@0
   524
	TEST2(stmt.ColumnInt(0), 1);
sl@0
   525
	err = stmt.Next();
sl@0
   526
	TEST2(err, KSqlAtEnd);
sl@0
   527
	stmt.Close();
sl@0
   528
sl@0
   529
	db.Close();
sl@0
   530
sl@0
   531
	//Run a test thread which will begin a transaction and then simulate a crash within the transaction
sl@0
   532
	TEST2(ThreadCritSect.CreateLocal(), KErrNone);
sl@0
   533
	ThreadCritSect.Wait();
sl@0
   534
	TEST2(MainCritSect.CreateLocal(), KErrNone);
sl@0
   535
	MainCritSect.Wait();
sl@0
   536
	RDebug::Print(_L("+++:MainThread: Create the worker thread\r\n"));
sl@0
   537
	_LIT(KThreadName, "WorkThrd");
sl@0
   538
	RThread thread;
sl@0
   539
	TEST2(thread.Create(KThreadName, &ThreadFunc1, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
sl@0
   540
	TRequestStatus status;
sl@0
   541
	thread.Logon(status);
sl@0
   542
	TEST2(status.Int(), KRequestPending);
sl@0
   543
	thread.Resume();
sl@0
   544
	RDebug::Print(_L("+++:MainThread: Wait SQL statement(s) to be executed...\r\n"));
sl@0
   545
	MainCritSect.Wait();
sl@0
   546
	RDebug::Print(_L("+++:MainThread: Notify the worker thread to simulate a crash...\r\n"));
sl@0
   547
	ThreadCritSect.Signal();
sl@0
   548
	User::WaitForRequest(status);
sl@0
   549
	User::SetJustInTime(ETrue);	// enable debugger panic handling
sl@0
   550
	TEST2(thread.ExitType(), EExitPanic);
sl@0
   551
	TEST2(thread.ExitReason(), KPanicCode);
sl@0
   552
	thread.Close();
sl@0
   553
	MainCritSect.Close();
sl@0
   554
	ThreadCritSect.Close();
sl@0
   555
sl@0
   556
	//Reopen the test database. The failed transaction must be rolled back.
sl@0
   557
	err = db.Open(KTestDb1);
sl@0
   558
	TEST2(err, KErrNone);
sl@0
   559
	//Verify that the database content is the same as before the failed transaction
sl@0
   560
	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
sl@0
   561
	TEST2(err, KErrNone);
sl@0
   562
	err = stmt.Next();
sl@0
   563
	TEST2(err, KSqlAtRow);
sl@0
   564
	TEST2(stmt.ColumnInt(0), 1);
sl@0
   565
	err = stmt.Next();
sl@0
   566
	TEST2(err, KSqlAtEnd);
sl@0
   567
	stmt.Close();
sl@0
   568
	db.Close();
sl@0
   569
sl@0
   570
	err = RSqlDatabase::Delete(KTestDb1);
sl@0
   571
	TEST2(err, KErrNone);
sl@0
   572
	}
sl@0
   573
	
sl@0
   574
void DoTests()
sl@0
   575
	{
sl@0
   576
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1764 Simple private db operations "));
sl@0
   577
	SimpleDbOpTest();
sl@0
   578
sl@0
   579
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1765 Private db operations - attach database "));
sl@0
   580
	AttachDbOpTest();
sl@0
   581
sl@0
   582
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1766 Private db operations - 2 database connections "));
sl@0
   583
	TwoDbOpTest();
sl@0
   584
sl@0
   585
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1767 Private db operations - transaction recovery "));
sl@0
   586
	TransDbOpTest();
sl@0
   587
	}
sl@0
   588
sl@0
   589
TInt E32Main()
sl@0
   590
	{
sl@0
   591
	TheTest.Title();
sl@0
   592
	
sl@0
   593
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   594
	
sl@0
   595
	__UHEAP_MARK;
sl@0
   596
	
sl@0
   597
	CreateTestEnv();
sl@0
   598
	DeleteTestFiles();
sl@0
   599
	DoTests();
sl@0
   600
	DeleteTestFiles();
sl@0
   601
	TheFs.Close();
sl@0
   602
sl@0
   603
	__UHEAP_MARKEND;
sl@0
   604
	
sl@0
   605
	TheTest.End();
sl@0
   606
	TheTest.Close();
sl@0
   607
	
sl@0
   608
	delete tc;
sl@0
   609
sl@0
   610
	User::Heap().Check();
sl@0
   611
	return KErrNone;
sl@0
   612
	}