First public contribution.
1 // Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
20 ///////////////////////////////////////////////////////////////////////////////////////
22 #define UNUSED_VAR(a) (a) = (a)
24 RTest TheTest(_L("t_sqltrans test"));
26 _LIT(KTestDir, "c:\\test\\");
27 _LIT(KTestDbName, "c:\\test\\t_sqltrans.db");
29 ///////////////////////////////////////////////////////////////////////////////////////
31 void DeleteTestFiles()
33 RSqlDatabase::Delete(KTestDbName);
36 ///////////////////////////////////////////////////////////////////////////////////////
37 ///////////////////////////////////////////////////////////////////////////////////////
38 //Test macros and functions
39 void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
47 TName name = th.Name();
48 RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
52 RDebug::Print(_L("*** Line %d\r\n"), aLine);
54 TheTest(EFalse, aLine);
57 void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
59 if(aValue != aExpected)
65 TName name = th.Name();
66 RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
70 RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
72 TheTest(EFalse, aLine);
75 #define TEST(arg) ::Check1((arg), __LINE__)
76 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
77 #define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
78 #define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
80 ///////////////////////////////////////////////////////////////////////////////////////
85 TInt err = fs.Connect();
88 err = fs.MkDir(KTestDir);
89 TEST(err == KErrNone || err == KErrAlreadyExists);
94 ///////////////////////////////////////////////////////////////////////////////////////
96 _LIT8(KTestSql1, "INSERT INTO A(Id) VALUES(1); INSERT INTO A(Id) VALUES(2);");
98 const TPtrC8 KSqls[] = {KTestSql1()};
100 static RCriticalSection ThreadCritSect;
101 static RCriticalSection MainCritSect;
103 static TInt TheSqlIdx = 0;
105 _LIT(KPanicCategory, "TransFail");
106 const TInt KPanicCode = 0x1234;
108 //Test thread function
109 TInt ThreadFunc1(void*)
113 CTrapCleanup* tc = CTrapCleanup::New();
116 __ASSERT_ALWAYS(TheSqlIdx >= 0 && TheSqlIdx < (TInt)(sizeof(KSqls) / sizeof(KSqls[0])), User::Invariant());
117 const TPtrC8 sql = KSqls[TheSqlIdx];
121 TInt err = db.Open(KTestDbName);
122 TTEST2(err, KErrNone);
124 RDebug::Print(_L("---:WorkThread: Begin transaction. Exec SQL...\r\n"));
126 //Begin a transaction
127 _LIT8(KBeginTrans, "BEGIN");
128 err = db.Exec(KBeginTrans);
131 //Execute the SQL statement(s)
135 RDebug::Print(_L("---:WorkThread: Notify the main thread about the SQL statement execution\r\n"));
136 MainCritSect.Signal();
138 RDebug::Print(_L("---:WorkThread: Wait for permisson to continue...\r\n"));
139 ThreadCritSect.Wait();
141 User::SetJustInTime(EFalse); // disable debugger panic handling
143 //Panic current thread without commiting the transaction
144 RDebug::Print(_L("---:WorkThread: Panic!\r\n"));
145 User::Panic(KPanicCategory, KPanicCode);
155 @SYMTestCaseID SYSLIB-SQL-CT-1623
156 @SYMTestCaseDesc Transaction atomicity test.
157 Create a test database with a table.
158 Create a worker thread and make some "insert record" operations in a transaction from
159 that thread. Before commiting the transaction notify the main thread that the
160 insert operation completed and wait for a notification from the main thread.
161 The main thread notifies the worker thread to panic and checks the test table
162 content. No records should be found there.
163 @SYMTestPriority High
164 @SYMTestActions Transaction atomicity test.
165 @SYMTestExpectedResults Test must not fail
169 void TransactionTest1()
171 RDebug::Print(_L("+++:MainThread: Create critical sections\r\n"));
172 TEST2(ThreadCritSect.CreateLocal(), KErrNone);
173 ThreadCritSect.Wait();
174 TEST2(MainCritSect.CreateLocal(), KErrNone);
177 RDebug::Print(_L("+++:MainThread: Create test database\r\n"));
178 (void)RSqlDatabase::Delete(KTestDbName);
180 TInt err = db.Create(KTestDbName);
181 TEST2(err, KErrNone);
183 RDebug::Print(_L("+++:MainThread: Create a table in the test database\r\n"));
184 _LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER)");
185 err = db.Exec(KCreateSql);
190 RDebug::Print(_L("+++:MainThread: Create the worker thread\r\n"));
191 _LIT(KThreadName, "WorkThrd");
194 TEST2(thread.Create(KThreadName, &ThreadFunc1, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
195 TRequestStatus status;
196 thread.Logon(status);
197 TEST2(status.Int(), KRequestPending);
200 RDebug::Print(_L("+++:MainThread: Wait SQL statement(s) to be executed...\r\n"));
203 RDebug::Print(_L("+++:MainThread: Notify the worker thread to panic...\r\n"));
204 ThreadCritSect.Signal();
206 User::WaitForRequest(status);
207 User::SetJustInTime(ETrue); // enable debugger panic handling
209 TEST2(thread.ExitType(), EExitPanic);
210 TEST2(thread.ExitReason(), KPanicCode);
214 RDebug::Print(_L("+++:MainThread: Check the database content...\r\n"));
215 err = db.Open(KTestDbName);
216 TEST2(err, KErrNone);
217 _LIT8(KSelectSql, "SELECT COUNT(*) AS CNT FROM A");
219 err = stmt.Prepare(db, KSelectSql);
220 TEST2(err, KErrNone);
222 TEST2(err, KSqlAtRow);
223 TInt val = stmt.ColumnInt(0);
228 err = RSqlDatabase::Delete(KTestDbName);
229 TEST2(err, KErrNone);
233 @SYMTestCaseID SYSLIB-SQL-CT-1624
234 @SYMTestCaseDesc Transaction consistency test.
235 Create a test database with a table with a field with a CHECK constraint.
236 Try to insert some records in a transaction violating the CHECK constraint.
237 The transaction should fail.
238 No records should be found in the test table.
239 @SYMTestPriority High
240 @SYMTestActions Transaction atomicity test.
241 @SYMTestExpectedResults Test must not fail
245 void TransactionTest2()
247 //Create a test database
248 (void)RSqlDatabase::Delete(KTestDbName);
250 TInt err = db.Create(KTestDbName);
251 TEST2(err, KErrNone);
253 //Create a test table
254 _LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER, CHECK(Id > 10 AND Id <= 20))");
255 err = db.Exec(KCreateSql);
258 //Begin a transaction
259 _LIT8(KBeginTrans, "BEGIN");
260 err = db.Exec(KBeginTrans);
263 //Exec SQL, viloate constraint.
264 _LIT8(KInsertSql, "INSERT INTO A(Id) VALUES(15); INSERT INTO A(Id) VALUES(38);");
265 err = db.Exec(KInsertSql);
266 TEST2(err, KSqlErrConstraint);
268 //Rollback transaction
269 _LIT8(KRollbackTrans, "ROLLBACK");
270 err = db.Exec(KRollbackTrans);
273 //Check the database content
274 _LIT8(KSelectSql, "SELECT COUNT(*) AS CNT FROM A");
276 err = stmt.Prepare(db, KSelectSql);
277 TEST2(err, KErrNone);
279 TEST2(err, KSqlAtRow);
280 TInt val = stmt.ColumnInt(0);
286 err = RSqlDatabase::Delete(KTestDbName);
287 TEST2(err, KErrNone);
292 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1623 Transaction test 1 "));
295 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1624 Transaction test 2 "));
303 CTrapCleanup* tc = CTrapCleanup::New();
319 User::Heap().Check();