First public contribution.
1 // Copyright (c) 2007-2010 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
23 #include "SqliteSymbian.h"
25 ///////////////////////////////////////////////////////////////////////////////////////
27 RTest TheTest(_L("t_sqlfserr test"));
28 _LIT(KTestDir, "c:\\test\\");
29 _LIT(KTestDbName, "c:\\test\\t_fserr.db");
30 _LIT(KPrivateTestDbName, "c:\\private\\212A2C27\\t_fserr2.db");
31 _LIT(KSecureTestDbName, "c:[212A2C27]t_fserr3.db");
33 TFileName TheRmvMediaDbFileName;//The name of the file used for tests on a removable media
37 //The next constants are used in the "blob write" test
38 const TInt KWriteCnt = 9;
39 const TInt KBlobSize = 397 * KWriteCnt;
40 _LIT(KAttachDb, "AttachDb");
42 //In order to be able to compile the test, the following variables are defined (used inside the OS porting layer, when _SQLPROFILER macro is defined)
44 TInt TheSqlSrvProfilerFileRead = 0;
45 TInt TheSqlSrvProfilerFileWrite = 0;
46 TInt TheSqlSrvProfilerFileSync = 0;
47 TInt TheSqlSrvProfilerFileSetSize = 0;
50 ///////////////////////////////////////////////////////////////////////////////////////
52 TBool FileExists(const TDesC& aFileName)
55 return TheFs.Entry(aFileName, entry) == KErrNone;
61 (void)RSqlDatabase::Delete(KSecureTestDbName);
62 (void)RSqlDatabase::Delete(KPrivateTestDbName);
63 (void)RSqlDatabase::Delete(KTestDbName);
64 (void)RSqlDatabase::Delete(TheRmvMediaDbFileName);
66 sqlite3SymbianLibFinalize();
70 ///////////////////////////////////////////////////////////////////////////////////////
71 ///////////////////////////////////////////////////////////////////////////////////////
72 //Test macros and functions
73 void Check(TInt aValue, TInt aLine)
78 TheTest(EFalse, aLine);
81 void Check(TInt aValue, TInt aExpected, TInt aLine)
83 if(aValue != aExpected)
86 RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
87 TheTest(EFalse, aLine);
90 #define TEST(arg) ::Check((arg), __LINE__)
91 #define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
93 void SqliteCheck(sqlite3* aDbHandle, TInt aValue, TInt aExpected, TInt aLine)
95 if(aValue != aExpected)
97 RDebug::Print(_L("*** SQLITE: Expected error: %d, got: %d\r\n"), aExpected, aValue);
100 const char* errMsg = sqlite3_errmsg(aDbHandle);
101 TPtrC8 ptr8((const TUint8*)errMsg, strlen(errMsg));
104 RDebug::Print(_L("*** SQLITE error message: \"%S\"\r\n"), &buf);
107 TheTest(EFalse, aLine);
110 #define SQLITE_TEST(aDbHandle, aValue, aExpected) ::SqliteCheck(aDbHandle, aValue, aExpected, __LINE__)
112 ///////////////////////////////////////////////////////////////////////////////////////
116 TInt err = TheFs.Connect();
117 TEST2(err, KErrNone);
119 err = TheFs.MkDir(KTestDir);
120 TEST(err == KErrNone || err == KErrAlreadyExists);
122 sqlite3SymbianLibInit();
125 TBool CheckRecord(const TDesC& aDbName, TInt aId, const TDesC& aExpectedName, TBool aOpenDb = ETrue)
129 TEST2(TheDb.Open(aDbName), KErrNone);
132 sql.Copy(_L("SELECT Name FROM A WHERE Id="));
134 TSqlScalarFullSelectQuery q(TheDb);
136 TRAPD(err, (void)q.SelectTextL(sql, name));
137 TEST2(err, KErrNone);
142 return name == aExpectedName;
145 ///////////////////////////////////////////////////////////////////////////////////////
148 @SYMTestCaseID SYSLIB-SQL-UT-3419
149 @SYMTestCaseDesc Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
150 The test creates a test database with one table, inserts one record.
151 Then the test attempts to update the existing record while simulating file I/O failures.
152 After each test iteration the database content is tested and is expected to be the same
153 as it was before the test. RSqlDatabase::Exec() is used for the update operation.
154 @SYMTestPriority High
155 @SYMTestActions Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
156 @SYMTestExpectedResults The test must not fail
159 void AlterDatabaseTest()
161 (void)RSqlDatabase::Delete(KTestDbName);
162 TInt err = TheDb.Create(KTestDbName);
163 TEST2(err, KErrNone);
164 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
168 for(TInt cnt=0;err<KErrNone;++cnt)
170 TheTest.Printf(_L("%d \r"), cnt);
171 for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
174 TEST2(TheDb.Open(KTestDbName), KErrNone);
175 (void)TheDb.Exec(_L("DELETE FROM A WHERE Id=1"));
176 err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
179 (void)TheFs.SetErrorCondition(fsError, cnt);
180 err = TheDb.Exec(_L("UPDATE A SET Name='Name2' WHERE Id=1"));
181 (void)TheFs.SetErrorCondition(KErrNone);
184 TheDb.Close();//close the database to recover from the last error
185 //check the database content - all bets are off in a case of an I/O error.
186 //The existing record might have been updated.
187 TEST(CheckRecord(KTestDbName, 1, _L("Name")) || CheckRecord(KTestDbName, 1, _L("Name2")));
192 //check the database content has been modified by the operation.
193 TEST(CheckRecord(KTestDbName, 1, _L("Name2"), EFalse));
198 (void)TheFs.SetErrorCondition(KErrNone);
200 //check the database content (transaction durability).
201 TEST(CheckRecord(KTestDbName, 1, _L("Name2")));
202 err = RSqlDatabase::Delete(KTestDbName);
203 TEST2(err, KErrNone);
204 TheTest.Printf(_L("\r\n"));
208 @SYMTestCaseID SYSLIB-SQL-UT-3420
209 @SYMTestCaseDesc Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
210 The test creates a test database with one table, inserts one record.
211 Then the test attempts to update the existing record while simulating file I/O failures.
212 After each test iteration the database content is tested and is expected to be the same
213 as it was before the test. RSqlStatement::Exec() is used for the update operation.
214 @SYMTestPriority High
215 @SYMTestActions Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
216 @SYMTestExpectedResults The test must not fail
219 void AlterDatabaseTest2()
221 (void)RSqlDatabase::Delete(KTestDbName);
222 TInt err = TheDb.Create(KTestDbName);
223 TEST2(err, KErrNone);
224 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
228 for(TInt cnt=0;err<KErrNone;++cnt)
230 TheTest.Printf(_L("%d \r"), cnt);
231 for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
234 TEST2(TheDb.Open(KTestDbName), KErrNone);
235 (void)TheDb.Exec(_L("DELETE FROM A WHERE Id=1"));
236 err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
239 (void)TheFs.SetErrorCondition(fsError, cnt);
241 err = stmt.Prepare(TheDb, _L("UPDATE A SET Name='Name2' WHERE Id=1"));
246 (void)TheFs.SetErrorCondition(KErrNone);
250 TheDb.Close();//close the database to recover from the last error
251 //check the database content - all bets are off in a case of an I/O error.
252 //The existing record might have been updated.
253 TEST(CheckRecord(KTestDbName, 1, _L("Name")) || CheckRecord(KTestDbName, 1, _L("Name2")));
258 //check the database content has been modified by the operation.
259 TEST(CheckRecord(KTestDbName, 1, _L("Name2"), EFalse));
264 (void)TheFs.SetErrorCondition(KErrNone);
266 //check the database content has been modified by the operation.
267 TEST(CheckRecord(KTestDbName, 1, _L("Name2")));
268 err = RSqlDatabase::Delete(KTestDbName);
269 TEST2(err, KErrNone);
270 TheTest.Printf(_L("\r\n"));
273 void CreateTestSecurityPolicy(RSqlSecurityPolicy& aSecurityPolicy)
275 TSecurityPolicy alwaysPassPolicy(TSecurityPolicy::EAlwaysPass);
276 TInt err = aSecurityPolicy.Create(alwaysPassPolicy);
277 TEST2(err, KErrNone);
279 err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::ESchemaPolicy, alwaysPassPolicy);
280 TEST2(err, KErrNone);
281 err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::EWritePolicy, alwaysPassPolicy);
282 TEST2(err, KErrNone);
283 err = aSecurityPolicy.SetDbPolicy(RSqlSecurityPolicy::EReadPolicy, alwaysPassPolicy);
284 TEST2(err, KErrNone);
287 //Creates public shared, private secure and public secure databases.
288 void DoCreateTestDatabases(const TPtrC aDbName[], TInt aCount)
291 for(TInt i=0;i<aCount;++i)
293 TheTest.Printf(_L("Database: \"%S\"\r\n"), &aDbName[i]);
294 (void)RSqlDatabase::Delete(aDbName[i]);
295 TInt err = KErrGeneral;
296 if(i == (aCount - 1))
298 RSqlSecurityPolicy policy;
299 CreateTestSecurityPolicy(policy);
300 err = TheDb.Create(aDbName[i], policy);
305 err = TheDb.Create(aDbName[i]);
307 TEST2(err, KErrNone);
308 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
310 err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
317 @SYMTestCaseID SYSLIB-SQL-UT-3421
318 @SYMTestCaseDesc Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
319 The test creates a test database with one table, inserts one record.
320 Then the test attempts to open the database while simulating file I/O failures.
321 At the end of the test the database content is tested and is expected to be the same
322 as it was before the test. RSqlStatement::Open() is used in the test.
323 @SYMTestPriority High
324 @SYMTestActions Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
325 @SYMTestExpectedResults The test must not fail
328 void OpenDatabaseTest()
330 TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
331 const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
332 DoCreateTestDatabases(dbName, KDbNameCnt);
333 for(TInt k=0;k<KDbNameCnt;++k)
335 TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);
336 TInt err = KErrNotFound;
337 for(TInt cnt=0;err<KErrNone;++cnt)
339 TheTest.Printf(_L("%d \r"), cnt);
340 for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
342 (void)TheFs.SetErrorCondition(fsError, cnt);
343 err = TheDb.Open(dbName[k]);
344 (void)TheFs.SetErrorCondition(KErrNone);
347 TheDb.Close();//close the database to recover from the last error
348 //check the database content is still the same as before the "open" call
349 TEST(CheckRecord(dbName[k], 1, _L("Name")));
353 TEST2(err, KErrNone);
354 //check the database content is still the same as before the operation, without closing the database
355 TEST(CheckRecord(dbName[k], 1, _L("Name"), EFalse));
360 (void)TheFs.SetErrorCondition(KErrNone);
361 TEST2(err, KErrNone);
362 //check the database content is the same as before the operation, after reopening the database.
363 TEST(CheckRecord(dbName[k], 1, _L("Name")));
364 err = RSqlDatabase::Delete(dbName[k]);
365 TEST2(err, KErrNone);
366 TheTest.Printf(_L("\r\n"));
367 }//end of: for(TInt k=0;k<KDbNameCnt;++k)
371 @SYMTestCaseID SYSLIB-SQL-UT-3434
372 @SYMTestCaseDesc Test for DEF104820 "SQL, RSqlDatabase::Create() does not delete the file if fails".
373 Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
374 Then the test attempts to create a database while simulating file I/O failures.
375 When the test succeeds, the test verifies that the database file does exist.
376 @SYMTestPriority High
377 @SYMTestActions Test for DEF104820 "SQL, RSqlDatabase::Create() does not delete the file if fails".
378 Test for DEF103859 "SQLITE panic, _DEBUG mode, persistent file I/O error simulation".
379 @SYMTestExpectedResults The test must not fail
382 void CreateDatabaseTest()
384 RSqlSecurityPolicy policy;
385 CreateTestSecurityPolicy(policy);
387 TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
388 const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
389 for(TInt k=0;k<KDbNameCnt;++k)
391 TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);
393 for(TInt cnt=0;err<KErrNone;++cnt)
395 TheTest.Printf(_L("%d \r"), cnt);
396 for (TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
398 //Ideally, the database should be deleted by the SQL server, if RSqlDatabase::Create() fails.
399 //But SetErrorCondition() makes the error persistent, so the SQL server will fail to delete the file.
400 //This is the reason, RSqlDatabase::Delete()to be used, before simulating file I/O error.
401 (void)RSqlDatabase::Delete(dbName[k]);
402 (void)TheFs.SetErrorCondition(fsError, cnt);
403 err = (k == (KDbNameCnt - 1)) ? TheDb.Create(dbName[k], policy) : TheDb.Create(dbName[k]);
404 (void)TheFs.SetErrorCondition(KErrNone);
406 //If err != KErrNone, the database file should have been already deleted by the server and here is
407 //the place to check that. But since the file I/O failure simulation makes the file I/O error
408 //persistent, the file cannot be deleted by the server, because the "file delete" operation also fails.
411 (void)TheFs.SetErrorCondition(KErrNone);
413 TEST2(err, KErrNone);
414 if( k != (KDbNameCnt - 1))
416 TEST(FileExists(dbName[k]));
418 err = RSqlDatabase::Delete(dbName[k]);
419 TEST2(err, KErrNone);
420 TheTest.Printf(_L("\r\n"));
426 @SYMTestCaseID PDS-SQL-UT-4189
427 @SYMTestCaseDesc Test for DEF145125 "SQL, low code coverage".
428 The test creates public shared, private secure and public secure test databases.
429 Then the test opens the publich shared database and attempts to attach one of the other two
430 in a file I/O error simulation loop.
431 @SYMTestPriority High
432 @SYMTestActions Test for DEF145125 - "SQL, low code coverage".
433 @SYMTestExpectedResults The test must not fail
436 void AttachDatabaseTest()
438 TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
439 const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
440 DoCreateTestDatabases(dbName, KDbNameCnt);
441 for(TInt k=1;k<KDbNameCnt;++k)
443 TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);
444 TInt err = KErrGeneral;
445 for(TInt cnt=0;err<KErrNone;++cnt)
447 TheTest.Printf(_L("%d \r"), cnt);
448 for(TInt fsError=KErrNotFound;fsError>=KErrDied;--fsError)
450 err = TheDb.Open(KTestDbName);
451 TEST2(err, KErrNone);
452 (void)TheFs.SetErrorCondition(fsError, cnt);
453 err = TheDb.Attach(dbName[k], _L("DB2"));
454 (void)TheFs.SetErrorCondition(KErrNone);
455 (void)TheDb.Detach(_L("DB2"));
456 TheDb.Close();//close the database to recover from the last error
459 TEST2(err, KErrNone);
460 err = RSqlDatabase::Delete(dbName[k]);
461 TEST2(err, KErrNone);
462 TheTest.Printf(_L("\r\n"));
467 @SYMTestCaseID PDS-SQL-UT-4190
468 @SYMTestCaseDesc Test for DEF145125 "SQL, low code coverage".
469 The tests attempts to delete a database in a file I/O error simulation loop.
470 @SYMTestPriority High
471 @SYMTestActions Test for DEF145125 - "SQL, low code coverage".
472 @SYMTestExpectedResults The test must not fail
475 void DeleteDatabaseTest()
477 TPtrC dbName[] = {KTestDbName(), KPrivateTestDbName(), KSecureTestDbName()};
478 const TInt KDbNameCnt = sizeof(dbName) / sizeof(dbName[0]);
479 DoCreateTestDatabases(dbName, KDbNameCnt);
480 for(TInt k=0;k<KDbNameCnt;++k)
482 TheTest.Printf(_L("Database: \"%S\"\r\n"), &dbName[k]);
483 TInt err = KErrGeneral;
484 for(TInt cnt=1;err<KErrNone;++cnt)
486 TheTest.Printf(_L("%d \r"), cnt);
487 (void)TheFs.SetErrorCondition(KErrGeneral, cnt);
488 err = RSqlDatabase::Delete(dbName[k]);
489 (void)TheFs.SetErrorCondition(KErrNone);
491 TEST2(err, KErrNone);
492 err = RSqlDatabase::Delete(KTestDbName);
493 TEST2(err, KErrNotFound);
498 @SYMTestCaseID SYSLIB-SQL-UT-3462
499 @SYMTestCaseDesc Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
500 The test creates a test database with one table, inserts one record.
501 Then the test attempts to retrieve the existing record while simulating file I/O failures.
502 After each iteration, the database content is tested, that it has not been modified by the operation.
503 @SYMTestPriority High
504 @SYMTestActions Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
505 @SYMTestExpectedResults The test must not fail
508 void SelectRecordTest()
510 (void)RSqlDatabase::Delete(KTestDbName);
511 TInt err = TheDb.Create(KTestDbName);
512 TEST2(err, KErrNone);
513 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER, Name TEXT)"));
515 err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
519 for(TInt cnt=0;err<KErrNone;++cnt)
521 TheTest.Printf(_L("%d \r"), cnt);
522 err = TheDb.Open(KTestDbName);
523 TEST2(err, KErrNone);
525 (void)TheFs.SetErrorCondition(KErrGeneral, cnt);
526 err = stmt.Prepare(TheDb, _L("SELECT * FROM A WHERE Id=?"));
529 err = stmt.BindInt(0, 1);
533 TEST(err == KSqlAtRow || err < 0);
536 TInt id = stmt.ColumnInt(0);
539 err = stmt.ColumnText(1, name);
540 TEST2(err, KErrNone);
541 TEST(name == _L("Name"));
545 (void)TheFs.SetErrorCondition(KErrNone);
548 //check the database content is the same as before the operation
549 TEST(CheckRecord(KTestDbName, 1, _L("Name")));
551 (void)TheFs.SetErrorCondition(KErrNone);
554 //check the database content is the same as before the operation, after reopening the database.
555 TEST(CheckRecord(KTestDbName, 1, _L("Name")));
556 err = RSqlDatabase::Delete(KTestDbName);
557 TEST2(err, KErrNone);
558 TheTest.Printf(_L("\r\n"));
562 @SYMTestCaseID SYSLIB-SQL-UT-3463
563 @SYMTestCaseDesc Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
564 The test creates a test database with one table, inserts one record.
565 Then the test attempts to insert another while simulating file I/O failures.
566 After each iteration, the database content is tested, that it has not been modified by the operation.
567 If the operation succeeds, the database content is tested again to check that the inserted record is there.
568 @SYMTestPriority High
569 @SYMTestActions Test for DEF105434 "SQL, persistent file I/O simulation, COMMIT problem".
570 @SYMTestExpectedResults The test must not fail
573 void InsertRecordTest()
575 (void)RSqlDatabase::Delete(KTestDbName);
576 TInt err = TheDb.Create(KTestDbName);
577 TEST2(err, KErrNone);
578 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Name TEXT)"));
580 err = TheDb.Exec(_L("INSERT INTO A(Id,Name) VALUES(1,'Name')"));
584 for(TInt cnt=0;err<KErrNone;++cnt)
586 TheTest.Printf(_L("%d \r"), cnt);
587 err = TheDb.Open(KTestDbName);
588 TEST2(err, KErrNone);
590 (void)TheFs.SetErrorCondition(KErrGeneral, cnt);
591 err = stmt.Prepare(TheDb, _L("INSERT INTO A(Id,Name) VALUES(2, 'Name2')"));
594 err = TheDb.Exec(_L("BEGIN TRANSACTION"));
598 TEST(err == 1 || err < 0);
601 err = TheDb.Exec(_L("COMMIT TRANSACTION"));
605 (void)TheFs.SetErrorCondition(KErrNone);
609 TheDb.Close();//close the database to recover from the last error
610 //check that the database contains the "name" record that has been inserted before the file I/O failure test.
611 TEST(CheckRecord(KTestDbName, 1, _L("Name")));
616 //check the database content has been modified by the operation, without closing the database.
617 TEST(CheckRecord(KTestDbName, 1, _L("Name"), EFalse));
618 TEST(CheckRecord(KTestDbName, 2, _L("Name2"), EFalse));
622 (void)TheFs.SetErrorCondition(KErrNone);
624 //check the database content (transaction durability).
625 TEST(CheckRecord(KTestDbName, 1, _L("Name")));
626 TEST(CheckRecord(KTestDbName, 2, _L("Name2")));
627 (void)RSqlDatabase::Delete(KTestDbName);
628 TheTest.Printf(_L("\r\n"));
631 /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
632 ////////////////////////////////// Removable media robustness test /////////////////////////////////////////
633 /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
635 _LIT8(KNameColData, "A123456789012345678901234567890");
636 _LIT8(KUpdatedNameColData, "1234");
638 //TRemovableMediaTest class is used for testing the SQLITE behaviour when the database file is created on a removable media where
639 //the cluster size is bigger than the page size and in case of a power failure is not guaranteed that the content
640 //of the last updated cluster will be preserved.
641 class TRemovableMediaTest
643 enum {KTestRecCnt = 200};
644 enum {KMinCachePageSize = 512};
650 TInt GetRemovableMediaDriveNo();
651 TInt ClusterSize(TInt aDriveNo);
652 void CreateDatabase(TInt aDriveNo, TInt aCachePageSize);
653 void CheckRecord(sqlite3_stmt* aStmt, TInt aRecId);
654 void VerifyDatabase();
659 //Returns the number of the removable media drive, KErrNotFound otherwise.
660 TInt TRemovableMediaTest::GetRemovableMediaDriveNo()
662 for(TInt driveNo=EDriveA;driveNo<=EDriveZ;++driveNo)
664 TDriveInfo driveInfo;
665 TInt err = TheFs.Drive(driveInfo, driveNo);
668 _LIT(KType1, "Not present");
669 _LIT(KType2, "Unknown");
670 _LIT(KType3, "Floppy");
671 _LIT(KType4, "Hard disk");
672 _LIT(KType5, "CD ROM");
673 _LIT(KType6, "RAM disk");
674 _LIT(KType7, "Flash");
675 _LIT(KType8, "ROM drive");
676 _LIT(KType9, "Remote drive");
677 _LIT(KType10,"NAND flash");
678 _LIT(KType11,"Rotating media");
681 TPtrC KMediaTypeNames[] = {KType1(), KType2(), KType3(), KType4(), KType5(), KType6(), KType7(), KType8(), KType9(), KType10(), KType11()};
682 TheTest.Printf(_L("Drive: %C:, %S, Removable: %S\r\n"), 'A' + driveNo, &KMediaTypeNames[driveInfo.iType],
683 driveInfo.iDriveAtt & KDriveAttRemovable ? &KYes : &KNo);
684 if(driveInfo.iDriveAtt & KDriveAttRemovable)
686 TheTest.Printf(_L("Removable drive to test: %C:\r\n"), 'A' + driveNo);
694 //Get the cluster size of aDriveNo drive
695 TInt TRemovableMediaTest::ClusterSize(TInt aDriveNo)
697 __ASSERT_DEBUG((TUint)aDriveNo <= EDriveZ, User::Invariant());
698 TVolumeIOParamInfo volIoParams;
699 TInt err = TheFs.VolumeIOParam(aDriveNo, volIoParams);
700 return (err == KErrNone) ? volIoParams.iClusterSize : err;
703 //Create a test database on aDriveNo with aCachePageSize page size.
704 //Insert KTestRecCnt records.
705 void TRemovableMediaTest::CreateDatabase(TInt aDriveNo, TInt aCachePageSize)
707 __ASSERT_DEBUG((TUint)aDriveNo <= EDriveZ, User::Invariant());
708 __ASSERT_DEBUG(aCachePageSize > 0, User::Invariant());
709 TDriveUnit drvUnit(aDriveNo);
710 _LIT(KDbName, "\\flashmedia.db");
712 parse.Set(drvUnit.Name(), &KDbName, 0);
713 TheRmvMediaDbFileName.Copy(parse.FullName());
714 TBuf8<KMaxFileName + 1> dbFileName8;
715 dbFileName8.Copy(TheRmvMediaDbFileName);
716 (void)TheFs.Delete(TheRmvMediaDbFileName);
718 sqlite3* dbHandle = NULL;
719 TInt rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
720 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
721 __ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
724 config.Copy(_L8("PRAGMA PAGE_SIZE="));
725 config.AppendNum(aCachePageSize);
726 rc = sqlite3_exec(dbHandle, (const char*)config.PtrZ(), 0, 0, 0);
727 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
729 rc = sqlite3_exec(dbHandle, "CREATE TABLE A(Id INTEGER,Name TEXT)", 0, 0, 0);
730 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
731 rc = sqlite3_exec(dbHandle, "BEGIN", 0, 0, 0);
732 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
733 for(TInt recid=1;recid<=KTestRecCnt;++recid)
737 sql.Copy(_L8("INSERT INTO A VALUES("));
738 sql.AppendNum(recid);
739 sql.Append(_L8(",'"));
740 sql.Append(KNameColData);
741 sql.Append(_L8("')"));
742 rc = sqlite3_exec(dbHandle, (const char*)sql.PtrZ(), 0, 0, 0);
743 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
745 rc = sqlite3_exec(dbHandle, "COMMIT", 0, 0, 0);
746 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
747 sqlite3_close(dbHandle);
750 //Checks the content of a single record
751 void TRemovableMediaTest::CheckRecord(sqlite3_stmt* aStmt, TInt aRecId)
753 __ASSERT_DEBUG(aStmt != NULL, User::Invariant());
754 TInt id = sqlite3_column_int(aStmt, 0);
756 const TUint8* text = (const TUint8*)sqlite3_column_text(aStmt, 1);
757 TPtrC8 name(text, User::StringLength(text));
758 TEST(KNameColData() == name || KUpdatedNameColData() == name);
761 //Verifies that the database content is either the same as it was before the UPDATE operation or
762 //it has been updated with the new data.
763 void TRemovableMediaTest::VerifyDatabase()
765 TBuf8<KMaxFileName + 1> dbFileName8;
766 dbFileName8.Copy(TheRmvMediaDbFileName);
768 sqlite3* dbHandle = NULL;
769 TInt rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
770 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
771 __ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
773 sqlite3_stmt* stmtHandle = NULL;
774 rc = sqlite3_prepare(dbHandle, "SELECT Id,Name FROM A", -1, &stmtHandle, 0);
775 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
776 __ASSERT_DEBUG(stmtHandle != NULL, User::Invariant());
778 for(TInt recid=1;recid<=KTestRecCnt;++recid)
780 rc = sqlite3_step(stmtHandle);
781 SQLITE_TEST(dbHandle, rc, SQLITE_ROW);
782 CheckRecord(stmtHandle, recid);
784 rc = sqlite3_step(stmtHandle);
785 SQLITE_TEST(dbHandle, rc, SQLITE_DONE);
787 sqlite3_finalize(stmtHandle);
788 sqlite3_close(dbHandle);
791 //Simulates a file system error in a loop.
792 //Attempts to update single record in a transaction.
793 //If the UPDATE operation fails - verifies the database content on each iteration.
794 //Note: pages are stored at the moment, not clusters. The database operations are not more robust if
795 // clusters are stored in a case of a removable media.
796 void TRemovableMediaTest::DoTest()
798 TheTest.Printf(_L("Update 1 record in a file I/o simulation loop\r\n"));
800 TBuf8<KMaxFileName + 1> dbFileName8;
801 dbFileName8.Copy(TheRmvMediaDbFileName);
802 for(TInt cnt=0;rc!=SQLITE_OK;++cnt)
804 TheTest.Printf(_L("%d \r"), cnt);
805 sqlite3* dbHandle = NULL;
806 rc = sqlite3_open((const char*)dbFileName8.PtrZ(), &dbHandle);
807 SQLITE_TEST(dbHandle, rc, SQLITE_OK);
808 __ASSERT_DEBUG(dbHandle != NULL, User::Invariant());
809 (void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
810 rc = sqlite3_exec(dbHandle, "BEGIN IMMEDIATE", 0, 0, 0);
813 rc = sqlite3_exec(dbHandle, "UPDATE A SET Name='1234' WHERE Id=1", 0, 0, 0);
816 TInt cnt = sqlite3_changes(dbHandle);
818 rc = sqlite3_exec(dbHandle, "COMMIT", 0, 0, 0);
821 (void)TheFs.SetErrorCondition(KErrNone);
822 sqlite3_close(dbHandle);
828 TEST2(rc, SQLITE_OK);
832 void TRemovableMediaTest::Run()
834 TInt driveNo = GetRemovableMediaDriveNo();
835 if(driveNo == KErrNotFound)
837 TheTest.Printf(_L("No removable media discovered. Test case not executed.\r\n"));
840 TInt clusterSize = ClusterSize(driveNo);
843 TheTest.Printf(_L("Error %d retrieving the cluster size of drive %C. Test case not executed.\r\n"), clusterSize, 'A' + driveNo);
846 if(clusterSize <= KMinCachePageSize)
848 TheTest.Printf(_L("Cluster size: %d. No appropriate cache page size found. Test case not executed.\r\n"), clusterSize);
852 TheTest.Printf(_L("Cluster size: %d. Cache page size %d.\r\nBegin test.\r\n"), clusterSize, KMinCachePageSize);
853 CreateDatabase(driveNo, KMinCachePageSize);
855 (void)TheFs.Delete(TheRmvMediaDbFileName);
856 TheTest.Printf(_L("End test.\r\n"));
860 @SYMTestCaseID SYSLIB-SQL-UT-3516
861 @SYMTestCaseDesc Removable media robustness test
862 The test creates a test database with a table with some records. Then the test verifies
863 that the database content cannot be corrupted by file I/O failures during database updates,
864 when the database file is on a removable media and the media cluster size is bigger than the
866 @SYMTestPriority High
867 @SYMTestActions Removable media robustness test
868 @SYMTestExpectedResults The test must not fail
871 void RemovableMediaRobustnessTest()
873 TRemovableMediaTest removableMediaTest;
874 removableMediaTest.Run();
878 @SYMTestCaseID SYSLIB-SQL-UT-4044
879 @SYMTestCaseDesc RSqlDatabase::Size(TSize&), file I/O error simulation test.
880 The test creates a database and executes RSqldatabase::Size(TSize&)
881 during a file I/O error simulation. The database should not be corrupted
883 @SYMTestPriority High
884 @SYMTestActions RSqlDatabase::Size(TSize&), file I/O error simulation test.
885 @SYMTestExpectedResults Test must not fail
890 (void)RSqlDatabase::Delete(KTestDbName);
891 TInt err = TheDb.Create(KTestDbName);
892 TEST2(err, KErrNone);
893 err = TheDb.Exec(_L("BEGIN;CREATE TABLE A(Id INTEGER,Data BLOB);INSERT INTO A VALUES(1, x'11223344');COMMIT;"));
895 RSqlDatabase::TSize size1 = {-1, -1};
896 err = TheDb.Size(size1);
897 TEST2(err, KErrNone);
898 TEST(size1.iSize > 0);
899 TEST2(size1.iFree, 0);
901 //"File I/O" error simulation loop
903 for(TInt cnt=0;err<KErrNone;++cnt)
905 TheTest.Printf(_L("%d \r"), cnt);
906 TEST2(TheDb.Open(KTestDbName), KErrNone);
907 (void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
908 RSqlDatabase::TSize size2 = {-1, -1};
909 err = TheDb.Size(size2);
910 (void)TheFs.SetErrorCondition(KErrNone);
914 TEST(size2.iSize == size1.iSize);
915 TEST(size2.iFree == size1.iFree);
920 //check the database content - all bets are off in a case of an I/O error.
921 TEST2(TheDb.Open(KTestDbName), KErrNone);
922 TSqlScalarFullSelectQuery q(TheDb);
924 TRAPD(err2, recCnt = q.SelectIntL(_L8("SELECT COUNT(*) FROM A")));
926 TEST2(err2, KErrNone);
930 (void)RSqlDatabase::Delete(KTestDbName);
934 @SYMTestCaseID SYSLIB-SQL-UT-4045
935 @SYMTestCaseDesc RSqlDatabase::Compact(), file I/O error simulation test.
936 The test creates a database and executes RSqlDatabase::Compact()
937 during a file I/O error simulation. The database should not be corrupted
939 @SYMTestPriority High
940 @SYMTestActions RSqlDatabase::Compact(), file I/O error simulation test.
941 @SYMTestExpectedResults Test must not fail
946 (void)RSqlDatabase::Delete(KTestDbName);
947 _LIT8(KConfig, "compaction=manual");
948 TInt err = TheDb.Create(KTestDbName, &KConfig);
949 TEST2(err, KErrNone);
950 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
953 err = TheDb.Exec(_L8("BEGIN"));
955 const TInt KRecLen = 1000;
956 TBuf8<KRecLen> sqlfmt;
957 sqlfmt.Copy(_L8("INSERT INTO A VALUES(%d,x'"));
958 for(TInt j=0;j<(KRecLen-50);++j)
960 sqlfmt.Append(_L8("A"));
962 sqlfmt.Append(_L8("')"));
963 const TInt KRecCount = 100;
964 for(TInt i=0;i<KRecCount;++i)
967 sql.Format(sqlfmt, i + 1);
968 err = TheDb.Exec(sql);
971 err = TheDb.Exec(_L8("COMMIT"));
974 const TInt KDeletedRecCnt = KRecCount - 10;
975 err = TheDb.Exec(_L8("DELETE FROM A WHERE Id > 10"));
977 //Get the database size
978 RSqlDatabase::TSize size;
979 err = TheDb.Size(size);
980 TEST2(err, KErrNone);
982 TEST(size.iSize > 0);
983 TEST(size.iFree > 0);
984 //"File I/O" error simulation loop
986 for(TInt cnt=0;err<KErrNone;++cnt)
988 TheTest.Printf(_L("%d \r"), cnt);
989 TEST2(TheDb.Open(KTestDbName), KErrNone);
990 (void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
991 err = TheDb.Compact(RSqlDatabase::EMaxCompaction);
992 (void)TheFs.SetErrorCondition(KErrNone);
1000 //check the database content - all bets are off in a case of an I/O error.
1001 //The database maight have been compacted, so - no check for that.
1002 TEST2(TheDb.Open(KTestDbName), KErrNone);
1003 TSqlScalarFullSelectQuery q(TheDb);
1005 TRAPD(err2, recCnt = q.SelectIntL(_L8("SELECT COUNT(*) FROM A")));
1007 TEST2(err2, KErrNone);
1008 TEST2(recCnt, (KRecCount - KDeletedRecCnt));
1011 TheTest.Printf(_L("\r\n"));
1012 //Check that the database has been really compacted
1013 TEST2(TheDb.Open(KTestDbName), KErrNone);
1014 RSqlDatabase::TSize size2;
1015 err = TheDb.Size(size2);
1016 TEST2(err, KErrNone);
1018 (void)RSqlDatabase::Delete(KTestDbName);
1019 TEST(size.iSize > size2.iSize);
1020 TEST2(size2.iFree, 0);
1023 void DoBlobWriteStreamTestL(TBool aAttachDb)
1025 RSqlBlobWriteStream strm;
1026 CleanupClosePushL(strm);
1029 strm.OpenL(TheDb, _L("A"), _L("Data"), 1, KAttachDb);
1033 strm.OpenL(TheDb, _L("A"), _L("Data"), 1);
1036 TBuf8<KBlobSize / KWriteCnt> data;
1037 data.SetLength(KBlobSize / KWriteCnt);
1040 for(TInt i=0;i<KWriteCnt;++i)
1047 CleanupStack::PopAndDestroy(&strm);
1051 @SYMTestCaseID SYSLIB-SQL-UT-4089
1052 @SYMTestCaseDesc RSqlBlobWriteStream::WriteL(), file I/O error simulation test.
1053 The test creates a database and executes RSqlBlobWriteStream::WriteL()
1054 during a file I/O error simulation. The database should not be corrupted
1056 @SYMTestPriority High
1057 @SYMTestActions RSqlBlobWriteStream::WriteL(), file I/O error simulation test.
1058 @SYMTestExpectedResults Test must not fail
1062 void BlobWriteStreamTest(TBool aAttachDb)
1064 (void)RSqlDatabase::Delete(KTestDbName);
1065 TInt err = TheDb.Create(KTestDbName);
1066 TEST2(err, KErrNone);
1067 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
1070 sql.Format(_L8("INSERT INTO A VALUES(1, zeroblob(%d))"), KBlobSize);
1071 err = TheDb.Exec(sql);
1076 for(TInt cnt=0;err<KErrNone;++cnt)
1078 TheTest.Printf(_L("%d \r"), cnt);
1079 TEST2(TheDb.Open(KTestDbName), KErrNone);
1082 TEST2(TheDb.Attach(KTestDbName, KAttachDb), KErrNone);
1084 (void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
1085 TRAP(err, DoBlobWriteStreamTestL(aAttachDb));
1086 (void)TheFs.SetErrorCondition(KErrNone);
1089 TEST2(TheDb.Detach(KAttachDb), KErrNone);
1093 TheTest.Printf(_L("\r\n"));
1095 TEST2(TheDb.Open(KTestDbName), KErrNone);
1098 err = stmt.Prepare(TheDb, _L8("SELECT * FROM A"));
1099 TEST2(err, KErrNone);
1101 TEST2(err, KSqlAtRow);
1103 err = stmt.ColumnBinary(1, data);
1104 TEST2(err, KErrNone);
1105 TEST2(data.Length(), KBlobSize);
1106 for(TInt j=0;j<KBlobSize;++j)
1114 (void)RSqlDatabase::Delete(KTestDbName);
1117 void DoBlobReadStreamTestL(TBool aAttachDb, TDes8& aDes)
1119 RSqlBlobReadStream strm;
1120 CleanupClosePushL(strm);
1123 strm.OpenL(TheDb, _L("A"), _L("Data"), 1, KAttachDb);
1127 strm.OpenL(TheDb, _L("A"), _L("Data"), 1);
1130 TBuf8<KBlobSize / KWriteCnt> data;
1133 for(TInt i=0;i<KWriteCnt;++i)
1139 CleanupStack::PopAndDestroy(&strm);
1143 @SYMTestCaseID SYSLIB-SQL-UT-4090
1144 @SYMTestCaseDesc RSqlBlobReadStream::ReadL(), file I/O error simulation test.
1145 The test creates a database and executes RSqlBlobReadStream::ReadL()
1146 during a file I/O error simulation. The database should not be corrupted
1148 @SYMTestPriority High
1149 @SYMTestActions RSqlBlobReadStream::ReadL(), file I/O error simulation test.
1150 @SYMTestExpectedResults Test must not fail
1155 void BlobReadStreamTest(TBool aAttachDb)
1157 (void)RSqlDatabase::Delete(KTestDbName);
1158 TInt err = TheDb.Create(KTestDbName);
1159 TEST2(err, KErrNone);
1160 err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER,Data BLOB)"));
1163 sql.Format(_L8("INSERT INTO A VALUES(1, zeroblob(%d))"), KBlobSize);
1164 err = TheDb.Exec(sql);
1166 TRAP(err, DoBlobWriteStreamTestL(EFalse));
1167 TEST2(err, KErrNone);
1170 HBufC8* buf = HBufC8::New(KBlobSize);
1172 TPtr8 bufptr = buf->Des();
1175 for(TInt cnt=0;err<KErrNone;++cnt)
1177 TheTest.Printf(_L("%d \r"), cnt);
1178 TEST2(TheDb.Open(KTestDbName), KErrNone);
1181 TEST2(TheDb.Attach(KTestDbName, KAttachDb), KErrNone);
1183 (void)TheFs.SetErrorCondition(KErrCorrupt, cnt);
1184 TRAP(err, DoBlobReadStreamTestL(aAttachDb, bufptr));
1185 (void)TheFs.SetErrorCondition(KErrNone);
1188 TEST2(TheDb.Detach(KAttachDb), KErrNone);
1192 TheTest.Printf(_L("\r\n"));
1194 TEST2(bufptr.Length(), KBlobSize);
1195 for(TInt j=0;j<KBlobSize;++j)
1197 TUint8 d = bufptr[j];
1203 (void)RSqlDatabase::Delete(KTestDbName);
1206 /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
1210 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3419 Alter database during file I/O error "));
1211 AlterDatabaseTest();
1212 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3420 Alter database during file I/O error (using statement object) "));
1213 AlterDatabaseTest2();
1214 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3421 Open database during file I/O error "));
1216 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3434 Create database during file I/O error "));
1217 CreateDatabaseTest();
1218 TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-UT-4189 Attach database during file I/O error "));
1219 AttachDatabaseTest();
1220 TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-UT-4190 Delete database during file I/O error "));
1221 DeleteDatabaseTest();
1222 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3462 Select record test during file I/O error "));
1224 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3463 Insert record test during file I/O error "));
1226 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-3516 Removable Media robustness test "));
1227 RemovableMediaRobustnessTest();
1228 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4044 Database size test during file I/O error"));
1230 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4045 Compact database test during file I/O error"));
1232 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4089 RSqlBlobWriteStream::WriteL() test during file I/O error"));
1233 BlobWriteStreamTest(EFalse);
1234 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4089 RSqlBlobWriteStream::WriteL()+attached database test during file I/O error"));
1235 BlobWriteStreamTest(ETrue);
1236 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4090 RSqlBlobReadStream::ReadL() test during file I/O error"));
1237 BlobReadStreamTest(EFalse);
1238 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4090 RSqlBlobReadStream::ReadL()+attached database test during file I/O error"));
1239 BlobReadStreamTest(ETrue);
1246 CTrapCleanup* tc = CTrapCleanup::New();
1261 User::Heap().Check();