Update contrib.
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 ///////////////////////////////////////////////////////////////////////////////////////
23 RTest TheTest(_L("t_sqlood test"));
25 #if defined __WINSCW__ || defined __WINS__
27 //The C: drive may be too big and may be used concurently by other applications.
28 //The T: drive is more suitable for the test if running on the emulator
29 const TInt KTestDrive = EDriveT;
30 _LIT(KTestDir, "t:\\test\\");
31 _LIT(KTestDatabase, "t:\\test\\t_sql_ood.db");
33 #elif defined __X86GCC__
35 const TInt KTestDrive = EDriveG;
36 _LIT(KTestDir, "g:\\test\\");
37 _LIT(KTestDatabase, "g:\\test\\t_sql_ood.db");
41 const TInt KTestDrive = EDriveE;
42 _LIT(KTestDir, "e:\\test\\");
43 _LIT(KTestDatabase, "e:\\test\\t_sql_ood.db");
47 //One or more files with KLargeFileName name and ".<n>" extension, where n is
48 //000, 001, 002, 003...
49 //will be created and they will occupy almost all available disk space.
50 //The idea is to perform after that one "delete"
51 //transaction, which must to fail, because there won't be enough available disk space to complete the transaction.
52 #if defined __WINSCW__ || defined __WINS__
54 _LIT(KLargeFileName, "t:\\test\\DeleteMe");
56 #elif defined __X86GCC__
58 _LIT(KLargeFileName, "g:\\test\\DeleteMe");
62 _LIT(KLargeFileName, "e:\\test\\DeleteMe");
66 _LIT8(KDatabasePageSizeConfig, "page_size=1024");
68 const TInt KMaxTestRecordsCount = 350;
69 TInt TestRecordsCount = 0;
71 ///////////////////////////////////////////////////////////////////////////////////////
73 //Assemblesd a file name from "aFileName" and "aFileNumber" parameters and places the resulting string in "aResultPath".
74 void AssembleLargeFileName(const TDesC& aFileName, TInt aFileNumber, TDes& aResultPath)
76 _LIT(KFormatStr, "%S.%03d");
77 aResultPath.Format(KFormatStr, &aFileName, aFileNumber);
80 //Deletes all created large data files.
81 void DeleteLargeDataFiles()
85 while(err == KErrNone)
87 TBuf<KMaxFileName> filePath;
88 ::AssembleLargeFileName(KLargeFileName, ++i, filePath);
89 err = TheFs.Delete(filePath);
93 //Deletes all created test files.
94 void DeleteTestFiles()
96 DeleteLargeDataFiles();
97 (void)RSqlDatabase::Delete(KTestDatabase);
100 ///////////////////////////////////////////////////////////////////////////////////////
101 ///////////////////////////////////////////////////////////////////////////////////////
102 //Test macros and functions
103 void Check1(TInt aValue, TInt aLine)
108 RDebug::Print(_L("*** Line %d\r\n"), aLine);
109 TheTest(EFalse, aLine);
112 void Check2(TInt aValue, TInt aExpected, TInt aLine)
114 if(aValue != aExpected)
117 RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
118 TheTest(EFalse, aLine);
121 #define TEST(arg) ::Check1((arg), __LINE__)
122 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
124 ///////////////////////////////////////////////////////////////////////////////////////
126 //Creates file session instance and the test directory
129 TInt err = TheFs.Connect();
130 TEST2(err, KErrNone);
132 err = TheFs.MkDir(KTestDir);
135 RDebug::Print(_L("*** CreateTestEnv(), RFs::MkDir(), err=%d\r\n"), err);
137 TEST(err == KErrNone || err == KErrAlreadyExists);
140 //Creates one or more large files with the total size near to the size of the available disk space.
141 //The idea is to cause an "out of disk space" condition.
142 void FillLargeDataFile(RFile& aFile, TInt aSize)
144 TInt err = KErrDiskFull;
145 while(err == KErrDiskFull)
147 err = aFile.SetSize(aSize);
154 TEST(err == KErrNone || err == KErrDiskFull);
157 //Gets and returns the available disk space of the tested drive.
158 TInt64 FreeDiskSpace()
160 TVolumeInfo volInfoBefore;
161 TInt err = TheFs.Volume(volInfoBefore, KTestDrive);
162 TEST2(err, KErrNone);
163 return volInfoBefore.iFree;
166 //Creates a large data file with aSize size (in bytes).
167 void DoCreateLargeFile(const TDesC& aPath, TInt aSize)
170 TInt err = file.Replace(TheFs, aPath, EFileRead | EFileWrite);
171 TEST2(err, KErrNone);
172 FillLargeDataFile(file, aSize);
174 TEST2(err, KErrNone);
178 //Creates enough number of large data files to fill the available disk space.
179 void CreateLargeFile()
182 const TInt KLargeFileSize = 1000000000;
183 TInt64 diskSpace = ::FreeDiskSpace();
184 RDebug::Print(_L("CreateLargeFile: free space before = %ld\r\n"), diskSpace);
185 TBuf<KMaxFileName> filePath;
186 while(diskSpace > KLargeFileSize)
188 AssembleLargeFileName(KLargeFileName, fileNo++, filePath);
189 DoCreateLargeFile(filePath, KLargeFileSize);
190 diskSpace = ::FreeDiskSpace();
191 RDebug::Print(_L("----CreateLargeFile, step %d, free space = %ld\r\n"), fileNo, diskSpace);
193 //Reserve almost all disk space, except a small amount - 200 bytes.
196 ::AssembleLargeFileName(KLargeFileName, fileNo++, filePath);
197 const TInt64 KSpaceLeft = 200;
198 TInt64 lastFileSize = diskSpace - KSpaceLeft;
199 TInt lastFileSize32 = I64LOW(lastFileSize);
200 RDebug::Print(_L("----file size32 = %d\r\n"), lastFileSize32);
201 ::DoCreateLargeFile(filePath, lastFileSize32);
202 RDebug::Print(_L("----CreateLargeFile, last step (%d), file size = %ld\r\n"), fileNo, lastFileSize);
204 diskSpace = ::FreeDiskSpace();
205 RDebug::Print(_L("CreateLargeFile: free space after = %ld\r\n"), diskSpace);
209 // Number of bytes in the default journal header size.
210 const TInt KJournalHeaderSize = 0x200;
212 // Number of bytes added to each database page in the journal.
213 const TInt KJournalPageOverhead = 8;
215 // The default amount of reserved space provided by the ReserveDriveSpace API
216 const TInt KReserveDriveSpaceAmount = 64*1024;
218 //Creates and fills with some records a test database
219 void CreateAndFillTestDatabase(RSqlDatabase& aDb)
221 TInt err = aDb.Create(KTestDatabase, &KDatabasePageSizeConfig);
222 TEST2(err, KErrNone);
223 err = aDb.Exec(_L("CREATE TABLE A(Id INTEGER, Data TEXT)"));
227 // Find the page size of the database on this media
230 sql.Copy(_L("PRAGMA page_size"));
231 TSqlScalarFullSelectQuery q(aDb);
233 TRAP(err, pageSize = q.SelectIntL(sql););
234 //RDebug::Print(_L("Error %d Page Size %d"),err,pageSize);
235 TEST2(err, KErrNone);
237 //RDebug::Print(_L("Page Size %d"),pageSize);
240 // Find the sector size of this media
242 TDriveInfo driveInfo;
243 err = TheFs.Drive(driveInfo, KTestDrive);
244 TEST2(err, KErrNone);
245 TVolumeIOParamInfo volumeInfo;
246 err = TheFs.VolumeIOParam(KTestDrive, volumeInfo);
247 TEST2(err, KErrNone);
248 TInt sectorSize = volumeInfo.iBlockSize;
249 //RDebug::Print(_L("Sector Size %d"),sectorSize);
251 TInt journalHeaderSize = Max(sectorSize, KJournalHeaderSize);
252 //RDebug::Print(_L("Journal Header Size %d"),journalHeaderSize);
255 // Keep adding to database until it is a size such that all the data can still be deleted within the reserved disk space size.
256 // Do this piece-wise not in a transaction.
259 for(i=0;i<KMaxTestRecordsCount;++i)
261 sql.Format(_L("INSERT INTO A(Id, Data) VALUES(%d, 'A0123456789B0123456789C0123456789D0123456789E0123456789F0123456789G0123456789H0123456789')"), i + 1);
265 TInt size = aDb.Size();
266 TInt numberOfPages = size/pageSize;
267 TInt predictedJournalSize = journalHeaderSize + numberOfPages * (pageSize + KJournalPageOverhead);
268 //RDebug::Print(_L("Size %d, Pages %d, predictedJournalSize %d"),size, numberOfPages, predictedJournalSize);
270 // Will another page take us over the limit ?
271 if ((predictedJournalSize + (pageSize + KJournalPageOverhead)) >= (KReserveDriveSpaceAmount))
276 TestRecordsCount = i + 1;
278 //RDebug::Print(_L("TestRecordsCount %d"),TestRecordsCount);
282 //Tries to delete test database records
283 TInt DeleteTestRecords(RSqlDatabase& aDb)
285 TInt err = aDb.Exec(_L("BEGIN TRANSACTION"));
287 for(TInt i=0;i<TestRecordsCount;++i)
290 sql.Format(_L("DELETE FROM A WHERE Id = %d"), i + 1);
291 err = aDb.Exec(sql);// May fail with KErrDiskFull
294 (void)aDb.Exec(_L("ROLLBACK TRANSACTION"));
298 err = aDb.Exec(_L("COMMIT TRANSACTION"));// May fail with KErrDiskFull
302 ///////////////////////////////////////////////////////////////////////////////////////
304 //The function simply calls RSqlDatabase::ReserveDriveSpace(), RSqlDatabase::GetReserveAccess(),
305 //RSqlDatabase::ReleaseReserveAccess() methods and checks the return values.
306 //It might be usefull for debugging in case if something gets wrong.
307 void SimpleCallsTest()
309 RSqlDatabase db, db2;
310 TInt err = db.Create(KTestDatabase, &KDatabasePageSizeConfig);
311 TEST2(err, KErrNone);
313 err = db2.Open(KTestDatabase);
314 TEST2(err, KErrNone);
316 //An attempt to get an access to the reserved space (which is not reserved yet).
317 err = db.GetReserveAccess();
318 TEST2(err, KErrNotFound);
321 err = db.ReserveDriveSpace(0);
322 TEST2(err, KErrNone);
324 //An attempt to re-reserve it
325 err = db.ReserveDriveSpace(0);
326 TEST2(err, KErrAlreadyExists);
328 //Get an access to the reserved disk space
329 err = db.GetReserveAccess();
330 TEST2(err, KErrNone);
332 //Reserve disk space from the second connection
333 err = db2.ReserveDriveSpace(0);
334 TEST2(err, KErrNone);
336 //An attempt to get an access to the reserved space twice.
337 err = db.GetReserveAccess();
338 TEST2(err, KErrInUse);
340 //An attempt to get an access to the reserved space from the second connection.
341 err = db2.GetReserveAccess();
342 TEST2(err, KErrNone);
344 db.ReleaseReserveAccess();
346 //An attempt to release the reserved space twice.
347 db.ReleaseReserveAccess();
349 //Free the reserved disk space
350 db.FreeReservedSpace();
352 //Free the reserved disk space twice.
353 db.FreeReservedSpace();
355 //Free the reserved disk space from the second connection.
356 db2.FreeReservedSpace();
360 (void)RSqlDatabase::Delete(KTestDatabase);
364 @SYMTestCaseID SYSLIB-SQL-CT-1649
365 @SYMTestCaseDesc SQL database "out of disk space" tests.
366 The test creates and fills with some records a test database and then reserves a disk space.
367 The second step: the test fills almost all available disk space creting large data files.
368 The third step: the test attempts to delete all records from the test database and fails with
370 The fourth step: the test gets an access to the reserved disk space and attempts to delete
371 records again. This time the test should not fail.
372 @SYMTestPriority High
373 @SYMTestActions SQL database "out of disk space" tests.
374 @SYMTestExpectedResults Test must not fail
378 void DeleteTransactionTest()
380 TVolumeIOParamInfo volIoPrm;
381 TInt err = TheFs.VolumeIOParam(KTestDrive, volIoPrm);
382 TEST2(err, KErrNone);
383 RDebug::Print(_L("--Drive %d. BlockSize=%d, ClusterSize=%d, RecReadBufSize=%d, RecWriteBufSize=%d\r\n"), KTestDrive, volIoPrm.iBlockSize, volIoPrm.iClusterSize, volIoPrm.iRecReadBufSize, volIoPrm.iRecWriteBufSize);
384 /////////////////////////////////////////////////////////
385 RDebug::Print(_L("--Create and fill database \"%S\".\r\n"), &KTestDatabase);
387 CreateAndFillTestDatabase(db);
388 db.Close();//When the database gets closed, the persisted journal file will be deleted.
389 RDebug::Print(_L("--Close and reopen database \"%S\" (in order to get the persisted journal file deleted).\r\n"), &KTestDatabase);
390 err = db.Open(KTestDatabase);
391 TEST2(err, KErrNone);
392 RDebug::Print(_L("--Reserve disk space for database \"%S\".\r\n"), &KTestDatabase);
393 err = db.ReserveDriveSpace(0);
394 TEST2(err, KErrNone);
395 RDebug::Print(_L("--Simulate an \"out of disk space\" situation with creating a very large data file, which occupies almost the all the available disk space.\r\n"));
397 RDebug::Print(_L("--Attempt to delete test data records. The transaction must fail, because of \"out of disk space\".\r\n"));
398 err = DeleteTestRecords(db);
399 TEST2(err, KErrDiskFull);
400 RDebug::Print(_L("--Get an access to the reserved disk space.\r\n"));
401 err = db.GetReserveAccess();
402 TEST2(err, KErrNone);
403 TInt64 diskSpace = ::FreeDiskSpace();
404 RDebug::Print(_L("After GetReserveAccess(), free disk space = %ld. Try again \"Delete records\" transaction. The transaction must not fail.\r\n"), diskSpace);
405 err = DeleteTestRecords(db);
406 RDebug::Print(_L("--DeleteTestRecords() returned %d error.\r\n"), err);
408 //Releases the access to the reserved disk space
409 db.ReleaseReserveAccess();
410 //Frees the reserved disk space
411 db.FreeReservedSpace();
412 //Free the resources, used in the test
413 DeleteLargeDataFiles();
414 //Verify that the records have been deleted
416 err = stmt.Prepare(db, _L("SELECT COUNT(*) FROM A"));
417 TEST2(err, KErrNone);
419 TEST2(err, KSqlAtRow);
420 TInt recCount = stmt.ColumnInt(0);
424 (void)RSqlDatabase::Delete(KTestDatabase);
427 //OOD API tests with more than one connection to the same SQL database.
428 //The test calls ReserveDriveSpace/GetReserveAccess/ReleaseReserveAccess in a different
429 //combinations on four RSqlDatabase objects, connected to the same database .
430 //The test should not fail or panic.
434 CreateAndFillTestDatabase(db1);
437 TInt err = db2.Open(KTestDatabase);
438 TEST2(err, KErrNone);
440 //Play with "ReserveDriveSpace" on both sessions
441 err = db1.ReserveDriveSpace(0);
442 TEST2(err, KErrNone);
443 err = db2.ReserveDriveSpace(0);
444 TEST2(err, KErrNone);
445 db2.FreeReservedSpace();
446 err = db2.ReserveDriveSpace(0);
447 TEST2(err, KErrNone);
449 //Get an access to the reserved space through db2
450 err = db2.GetReserveAccess();
451 TEST2(err, KErrNone);
452 //Free/re-reserve disk space for db1.
453 db1.FreeReservedSpace();
454 err = db1.ReserveDriveSpace(0);
455 TEST2(err, KErrNone);
458 err = db4.Open(KTestDatabase);
459 TEST2(err, KErrNone);
461 //Try to reserve space for db4.
462 err = db4.ReserveDriveSpace(0);
463 TEST2(err, KErrNone);
466 err = db3.Open(KTestDatabase);
467 TEST2(err, KErrNone);
469 //Try to reserve space for session db3.
470 err = db3.ReserveDriveSpace(0);
471 TEST2(err, KErrNone);
473 //Release and free db2 access to the reserved space.
474 db2.ReleaseReserveAccess();
475 db2.FreeReservedSpace();
477 db3.FreeReservedSpace();
480 db4.FreeReservedSpace();
483 //Get an access to the reserved space through db2.
484 //But it was freed, so the call will fail.
485 err = db2.GetReserveAccess();
486 TEST2(err, KErrNotFound);
488 //Free/re-reserve disk space for db1.
489 db1.FreeReservedSpace();
490 err = db1.ReserveDriveSpace(0);
491 TEST2(err, KErrNone);
493 //Get/release the access to the reserved space for db1.
494 err = db1.GetReserveAccess();
495 TEST2(err, KErrNone);
496 db1.ReleaseReserveAccess();
498 //Get an access to the reserved space for db2.
499 //The call will fail because there is no reserved disk space for db2.
500 err = db2.GetReserveAccess();
501 TEST2(err, KErrNotFound);
503 //Free the reserved space - db1
504 db1.FreeReservedSpace();
509 (void)RSqlDatabase::Delete(KTestDatabase);
514 TheTest.Start(_L(" \"Simple calls\" OOD test "));
517 TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1649 \"Delete transaction\" OOD test "));
518 DeleteTransactionTest();
520 TheTest.Next(_L(" Multi db OOD test "));
528 CTrapCleanup* tc = CTrapCleanup::New();
545 User::Heap().Check();