os/persistentdata/persistentstorage/sql/TEST/t_sqltrans.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-2009 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
#define UNUSED_VAR(a) (a) = (a)
sl@0
    23
sl@0
    24
RTest TheTest(_L("t_sqltrans test"));
sl@0
    25
sl@0
    26
_LIT(KTestDir, "c:\\test\\");
sl@0
    27
_LIT(KTestDbName, "c:\\test\\t_sqltrans.db");
sl@0
    28
sl@0
    29
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    30
sl@0
    31
void DeleteTestFiles()
sl@0
    32
	{
sl@0
    33
	RSqlDatabase::Delete(KTestDbName);
sl@0
    34
	}
sl@0
    35
sl@0
    36
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    37
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    38
//Test macros and functions
sl@0
    39
void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
sl@0
    40
	{
sl@0
    41
	if(!aValue)
sl@0
    42
		{
sl@0
    43
		DeleteTestFiles();
sl@0
    44
		if(aPrintThreadName)
sl@0
    45
			{
sl@0
    46
			RThread th;
sl@0
    47
			TName name = th.Name();
sl@0
    48
			RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
sl@0
    49
			}
sl@0
    50
		else
sl@0
    51
			{
sl@0
    52
			RDebug::Print(_L("*** Line %d\r\n"), aLine);
sl@0
    53
			}
sl@0
    54
		TheTest(EFalse, aLine);
sl@0
    55
		}
sl@0
    56
	}
sl@0
    57
void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
sl@0
    58
	{
sl@0
    59
	if(aValue != aExpected)
sl@0
    60
		{
sl@0
    61
		DeleteTestFiles();
sl@0
    62
		if(aPrintThreadName)
sl@0
    63
			{
sl@0
    64
			RThread th;
sl@0
    65
			TName name = th.Name();
sl@0
    66
			RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
sl@0
    67
			}
sl@0
    68
		else
sl@0
    69
			{
sl@0
    70
			RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
sl@0
    71
			}
sl@0
    72
		TheTest(EFalse, aLine);
sl@0
    73
		}
sl@0
    74
	}
sl@0
    75
#define TEST(arg) ::Check1((arg), __LINE__)
sl@0
    76
#define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
sl@0
    77
#define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
sl@0
    78
#define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
sl@0
    79
sl@0
    80
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    81
sl@0
    82
void CreateTestDir()
sl@0
    83
    {
sl@0
    84
    RFs fs;
sl@0
    85
	TInt err = fs.Connect();
sl@0
    86
	TEST2(err, KErrNone);
sl@0
    87
sl@0
    88
	err = fs.MkDir(KTestDir);
sl@0
    89
	TEST(err == KErrNone || err == KErrAlreadyExists);
sl@0
    90
	
sl@0
    91
	fs.Close();
sl@0
    92
	}
sl@0
    93
sl@0
    94
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    95
sl@0
    96
_LIT8(KTestSql1, "INSERT INTO A(Id) VALUES(1); INSERT INTO A(Id) VALUES(2);");
sl@0
    97
sl@0
    98
const TPtrC8 KSqls[] = {KTestSql1()};
sl@0
    99
sl@0
   100
static RCriticalSection ThreadCritSect;
sl@0
   101
static RCriticalSection MainCritSect;
sl@0
   102
sl@0
   103
static TInt TheSqlIdx = 0;
sl@0
   104
sl@0
   105
_LIT(KPanicCategory, "TransFail");
sl@0
   106
const TInt KPanicCode = 0x1234;
sl@0
   107
sl@0
   108
//Test thread function
sl@0
   109
TInt ThreadFunc1(void*)
sl@0
   110
	{
sl@0
   111
	__UHEAP_MARK;
sl@0
   112
	
sl@0
   113
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   114
	TTEST(tc != NULL);
sl@0
   115
sl@0
   116
	__ASSERT_ALWAYS(TheSqlIdx >= 0 && TheSqlIdx < (TInt)(sizeof(KSqls) / sizeof(KSqls[0])), User::Invariant());
sl@0
   117
	const TPtrC8 sql = KSqls[TheSqlIdx];
sl@0
   118
sl@0
   119
	//Open test database
sl@0
   120
	RSqlDatabase db;
sl@0
   121
	TInt err = db.Open(KTestDbName);
sl@0
   122
	TTEST2(err, KErrNone);
sl@0
   123
sl@0
   124
	RDebug::Print(_L("---:WorkThread: Begin transaction. Exec SQL...\r\n"));
sl@0
   125
	
sl@0
   126
	//Begin a transaction
sl@0
   127
	_LIT8(KBeginTrans, "BEGIN");
sl@0
   128
	err = db.Exec(KBeginTrans);
sl@0
   129
	TTEST(err >= 0);	
sl@0
   130
sl@0
   131
	//Execute the SQL statement(s)
sl@0
   132
	err = db.Exec(sql);
sl@0
   133
	TTEST(err >= 0);	
sl@0
   134
sl@0
   135
	RDebug::Print(_L("---:WorkThread: Notify the main thread about the SQL statement execution\r\n"));
sl@0
   136
	MainCritSect.Signal();
sl@0
   137
sl@0
   138
	RDebug::Print(_L("---:WorkThread: Wait for permisson to continue...\r\n"));
sl@0
   139
	ThreadCritSect.Wait();
sl@0
   140
sl@0
   141
	User::SetJustInTime(EFalse);	// disable debugger panic handling
sl@0
   142
sl@0
   143
	//Panic current thread without commiting the transaction
sl@0
   144
	RDebug::Print(_L("---:WorkThread: Panic!\r\n"));
sl@0
   145
	User::Panic(KPanicCategory, KPanicCode);
sl@0
   146
sl@0
   147
	delete tc;	
sl@0
   148
	
sl@0
   149
	__UHEAP_MARKEND;
sl@0
   150
	
sl@0
   151
	return KErrNone;		
sl@0
   152
	}
sl@0
   153
sl@0
   154
/**
sl@0
   155
@SYMTestCaseID			SYSLIB-SQL-CT-1623
sl@0
   156
@SYMTestCaseDesc		Transaction atomicity test.
sl@0
   157
						Create a test database with a table. 
sl@0
   158
						Create a worker thread and make some "insert record" operations in a transaction from 
sl@0
   159
						that thread. Before commiting the transaction notify the main thread that the
sl@0
   160
						insert operation completed and wait for a notification from the main thread.
sl@0
   161
						The main thread notifies the worker thread to panic and checks the test table 
sl@0
   162
						content. No records should be found there.
sl@0
   163
@SYMTestPriority		High
sl@0
   164
@SYMTestActions			Transaction atomicity test.
sl@0
   165
@SYMTestExpectedResults Test must not fail
sl@0
   166
@SYMREQ					REQ5792
sl@0
   167
                        REQ5793
sl@0
   168
*/	
sl@0
   169
void TransactionTest1()
sl@0
   170
	{
sl@0
   171
	RDebug::Print(_L("+++:MainThread: Create critical sections\r\n"));
sl@0
   172
	TEST2(ThreadCritSect.CreateLocal(), KErrNone);
sl@0
   173
	ThreadCritSect.Wait();
sl@0
   174
	TEST2(MainCritSect.CreateLocal(), KErrNone);
sl@0
   175
	MainCritSect.Wait();
sl@0
   176
	
sl@0
   177
	RDebug::Print(_L("+++:MainThread: Create test database\r\n"));
sl@0
   178
	(void)RSqlDatabase::Delete(KTestDbName);
sl@0
   179
	RSqlDatabase db;
sl@0
   180
	TInt err = db.Create(KTestDbName);
sl@0
   181
	TEST2(err, KErrNone);
sl@0
   182
sl@0
   183
	RDebug::Print(_L("+++:MainThread: Create a table in the test database\r\n"));
sl@0
   184
	_LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER)");
sl@0
   185
	err = db.Exec(KCreateSql);
sl@0
   186
	TEST(err >= 0);	
sl@0
   187
	
sl@0
   188
	db.Close();
sl@0
   189
sl@0
   190
	RDebug::Print(_L("+++:MainThread: Create the worker thread\r\n"));
sl@0
   191
	_LIT(KThreadName, "WorkThrd");
sl@0
   192
	RThread thread;
sl@0
   193
	TheSqlIdx = 0;
sl@0
   194
	TEST2(thread.Create(KThreadName, &ThreadFunc1, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
sl@0
   195
	TRequestStatus status;
sl@0
   196
	thread.Logon(status);
sl@0
   197
	TEST2(status.Int(), KRequestPending);
sl@0
   198
	thread.Resume();
sl@0
   199
sl@0
   200
	RDebug::Print(_L("+++:MainThread: Wait SQL statement(s) to be executed...\r\n"));
sl@0
   201
	MainCritSect.Wait();
sl@0
   202
sl@0
   203
	RDebug::Print(_L("+++:MainThread: Notify the worker thread to panic...\r\n"));
sl@0
   204
	ThreadCritSect.Signal();
sl@0
   205
	
sl@0
   206
	User::WaitForRequest(status);
sl@0
   207
	User::SetJustInTime(ETrue);	// enable debugger panic handling
sl@0
   208
sl@0
   209
	TEST2(thread.ExitType(), EExitPanic);
sl@0
   210
	TEST2(thread.ExitReason(), KPanicCode);
sl@0
   211
	
sl@0
   212
	thread.Close();
sl@0
   213
sl@0
   214
	RDebug::Print(_L("+++:MainThread: Check the database content...\r\n"));
sl@0
   215
	err = db.Open(KTestDbName);
sl@0
   216
	TEST2(err, KErrNone);
sl@0
   217
	_LIT8(KSelectSql, "SELECT COUNT(*) AS CNT FROM A");
sl@0
   218
	RSqlStatement stmt;
sl@0
   219
	err = stmt.Prepare(db, KSelectSql);
sl@0
   220
	TEST2(err, KErrNone);
sl@0
   221
	err = stmt.Next();
sl@0
   222
	TEST2(err, KSqlAtRow);
sl@0
   223
	TInt val = stmt.ColumnInt(0);
sl@0
   224
	TEST(val == 0);
sl@0
   225
	stmt.Close();
sl@0
   226
	db.Close();
sl@0
   227
sl@0
   228
	err = RSqlDatabase::Delete(KTestDbName);
sl@0
   229
	TEST2(err, KErrNone);
sl@0
   230
	}
sl@0
   231
sl@0
   232
/**
sl@0
   233
@SYMTestCaseID			SYSLIB-SQL-CT-1624
sl@0
   234
@SYMTestCaseDesc		Transaction consistency test.
sl@0
   235
						Create a test database with a table with a field with a CHECK constraint. 
sl@0
   236
						Try to insert some records in a transaction violating the CHECK constraint.
sl@0
   237
						The transaction should fail.
sl@0
   238
						No records should be found in the test table.
sl@0
   239
@SYMTestPriority		High
sl@0
   240
@SYMTestActions			Transaction atomicity test.
sl@0
   241
@SYMTestExpectedResults Test must not fail
sl@0
   242
@SYMREQ					REQ5792
sl@0
   243
                        REQ5793
sl@0
   244
*/	
sl@0
   245
void TransactionTest2()
sl@0
   246
	{
sl@0
   247
	//Create a test database
sl@0
   248
	(void)RSqlDatabase::Delete(KTestDbName);
sl@0
   249
	RSqlDatabase db;
sl@0
   250
	TInt err = db.Create(KTestDbName);
sl@0
   251
	TEST2(err, KErrNone);
sl@0
   252
sl@0
   253
	//Create a test table
sl@0
   254
	_LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER, CHECK(Id > 10 AND Id <= 20))");
sl@0
   255
	err = db.Exec(KCreateSql);
sl@0
   256
	TEST(err >= 0);	
sl@0
   257
sl@0
   258
	//Begin a transaction
sl@0
   259
	_LIT8(KBeginTrans, "BEGIN");
sl@0
   260
	err = db.Exec(KBeginTrans);
sl@0
   261
	TEST(err >= 0);	
sl@0
   262
	
sl@0
   263
	//Exec SQL, viloate constraint.
sl@0
   264
	_LIT8(KInsertSql, "INSERT INTO A(Id) VALUES(15); INSERT INTO A(Id) VALUES(38);");
sl@0
   265
	err = db.Exec(KInsertSql);
sl@0
   266
	TEST2(err, KSqlErrConstraint);
sl@0
   267
sl@0
   268
	//Rollback transaction
sl@0
   269
	_LIT8(KRollbackTrans, "ROLLBACK");
sl@0
   270
	err = db.Exec(KRollbackTrans);
sl@0
   271
	TEST(err >= 0);	
sl@0
   272
sl@0
   273
	//Check the database content
sl@0
   274
	_LIT8(KSelectSql, "SELECT COUNT(*) AS CNT FROM A");
sl@0
   275
	RSqlStatement stmt;
sl@0
   276
	err = stmt.Prepare(db, KSelectSql);
sl@0
   277
	TEST2(err, KErrNone);
sl@0
   278
	err = stmt.Next();
sl@0
   279
	TEST2(err, KSqlAtRow);
sl@0
   280
	TInt val = stmt.ColumnInt(0);
sl@0
   281
	TEST2(val, 0);
sl@0
   282
	stmt.Close();
sl@0
   283
	
sl@0
   284
	db.Close();
sl@0
   285
sl@0
   286
	err = RSqlDatabase::Delete(KTestDbName);
sl@0
   287
	TEST2(err, KErrNone);
sl@0
   288
	}
sl@0
   289
sl@0
   290
void DoTests()
sl@0
   291
	{
sl@0
   292
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1623 Transaction test 1 "));
sl@0
   293
	TransactionTest1();
sl@0
   294
sl@0
   295
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1624 Transaction test 2 "));
sl@0
   296
	TransactionTest2();
sl@0
   297
	}
sl@0
   298
sl@0
   299
TInt E32Main()
sl@0
   300
	{
sl@0
   301
	TheTest.Title();
sl@0
   302
	
sl@0
   303
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   304
	
sl@0
   305
	__UHEAP_MARK;
sl@0
   306
	
sl@0
   307
	CreateTestDir();
sl@0
   308
	DeleteTestFiles();
sl@0
   309
	DoTests();
sl@0
   310
	DeleteTestFiles();
sl@0
   311
sl@0
   312
	__UHEAP_MARKEND;
sl@0
   313
	
sl@0
   314
	TheTest.End();
sl@0
   315
	TheTest.Close();
sl@0
   316
	
sl@0
   317
	delete tc;
sl@0
   318
sl@0
   319
	User::Heap().Check();
sl@0
   320
	return KErrNone;
sl@0
   321
	}