Update contrib.
1 // Copyright (c) 2008-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.
22 ///////////////////////////////////////////////////////////////////////////////////////
24 RTest TheTest(_L("t_sqlcompact1 test"));
28 const TInt KTextLen = 400;
29 TBuf<KTextLen> TheText;
30 TBuf<KTextLen + 100> TheSqlBuf;
32 _LIT(KTestDir, "c:\\test\\");
33 _LIT(KDbName1, "c:\\test\\t_sqlcompact1_1.db");
34 _LIT(KDbName2, "c:\\test\\t_sqlcompact1_2.db");
35 _LIT(KDbName3, "c:\\test\\t_sqlcompact1_3.db");
36 _LIT(KDbName4, "c:\\test\\t_sqlcompact1_4.db");
38 _LIT(KAttachName1, "UOOO");
39 _LIT(KAttachName2, "FOOO");
40 _LIT(KAttachName3, "AOOO");
41 _LIT(KAttachName4, "EOOO");
43 ///////////////////////////////////////////////////////////////////////////////////////
48 (void)RSqlDatabase::Delete(KDbName4);
49 (void)RSqlDatabase::Delete(KDbName3);
50 (void)RSqlDatabase::Delete(KDbName2);
51 (void)RSqlDatabase::Delete(KDbName1);
54 ///////////////////////////////////////////////////////////////////////////////////////
55 ///////////////////////////////////////////////////////////////////////////////////////
56 //Test macros and functions
57 void Check(TInt aValue, TInt aLine)
62 RDebug::Print(_L("*** Test failure. Boolean expression evaluates to false.\r\n"));
63 TheTest(EFalse, aLine);
66 void Check2(TInt aValue, TInt aExpected, TInt aLine)
68 if(aValue != aExpected)
71 RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
72 TheTest(EFalse, aLine);
75 #define TEST(arg) ::Check((arg), __LINE__)
76 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
78 ///////////////////////////////////////////////////////////////////////////////////////
80 //t_sqlcompact1 timeouts in WDP builds.
81 //This function is used to check whether the time limit is reaqched or not.
82 TBool IsTimeLimitReached()
93 static TStartTime startTime;
94 const TInt KTestTimeLimit = 100;//seconds
99 TTimeIntervalSeconds s;
100 TInt err = currTime.SecondsFrom(startTime.iTime, s);
101 TEST2(err, KErrNone);
102 return s.Int() > KTestTimeLimit;
105 void GetHomeTimeAsString(TDes& aStr)
109 TDateTime dt = time.DateTime();
110 aStr.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
116 TInt err = fs.Connect();
117 TEST2(err, KErrNone);
119 err = fs.MkDir(KTestDir);
120 TEST(err == KErrNone || err == KErrAlreadyExists);
125 ///////////////////////////////////////////////////////////////////////////////////////
127 void ReplaceDb(const TDesC& aDbName, TInt aPageSize)
129 (void)RSqlDatabase::Delete(aDbName);
130 _LIT8(KConfigStr, "compaction=manual;page_size=");
132 config.Copy(KConfigStr);
133 config.AppendNum(aPageSize);
134 TInt err = TheDb.Create(aDbName, &config);
135 TEST2(err, KErrNone);
139 void CreateTable(const TDesC& aDbName)
141 TInt err = TheDb.Open(aDbName);
142 TEST2(err, KErrNone);
143 err = TheDb.Exec(_L("CREATE TABLE A(I INTEGER, T TEXT)"));
148 void InsertRecords(const TDesC& aDbName)
150 TInt err = TheDb.Open(aDbName);
151 TEST2(err, KErrNone);
152 err = TheDb.Exec(_L("BEGIN TRANSACTION"));
154 TheText.SetLength(TheText.MaxLength());
155 TheText.Fill(TChar('A'));
156 for(TInt i=0;i<100;++i)
158 TheSqlBuf.Format(_L("INSERT INTO A VALUES(%d, '%S')"), i + 1, &TheText);
159 err = TheDb.Exec(TheSqlBuf);
162 err = TheDb.Exec(_L("COMMIT TRANSACTION"));
167 TInt DeleteRecords(const TDesC& aDbName, TInt aPageCount, TInt aPageSize)
169 TInt freePageCount = -1;
170 TInt err = TheDb.Open(aDbName);
171 TEST2(err, KErrNone);
172 err = TheDb.Exec(_L("BEGIN TRANSACTION"));
176 TheSqlBuf.Format(_L("DELETE FROM A WHERE I=%d"), i + 1);
177 err = TheDb.Exec(TheSqlBuf);
179 RSqlDatabase::TSize s;
181 TEST2(err, KErrNone);
182 freePageCount = s.iFree / aPageSize;
183 if(freePageCount >= aPageCount)
188 err = TheDb.Exec(_L("COMMIT TRANSACTION"));
191 return freePageCount;
195 @SYMTestCaseID SYSLIB-SQL-UT-4072
196 @SYMTestCaseDesc Manual compaction on attached databases with different page size.
197 The test creates couple of databases with manual compaction and
198 different page sizes, then inserts some records and deletes part of
199 the just inserted records thus making some free pages.
200 The test opens the first database and attaches all other databases the the first one.
201 Then the test checks that RSqlDatabase::Size() returns correct information
202 about the free database space. The test runs the manual compaction on the
203 databases and checks again that the free database space is reported correctly.
204 @SYMTestPriority Medium
205 @SYMTestActions Manual compaction on attached databases with different page size.
206 @SYMTestExpectedResults Test must not fail
210 void CompactDbTest1()
212 const TPtrC KDbName[] = {KDbName1(), KDbName2(), KDbName3(), KDbName4()};
213 const TPtrC KDbAttachName[]={KAttachName1(),KAttachName2(), KAttachName3(), KAttachName4()};
214 const TInt KDbPageSize[] = {8192, 1024, 4096, 2048};
215 const TInt KFreePageCount[]={9, 30, 17, 7};
216 TInt freePageCount[] ={0, 0, 0, 0};
217 const TInt KSize = sizeof(KDbName) / sizeof(KDbName[0]);
222 GetHomeTimeAsString(timeBuf);
223 TheTest.Printf(_L("===Time1: %S\r\n"), &timeBuf);
225 //Create databases, tables, insert records, delete part of the just inserted records.
228 ReplaceDb(KDbName[i], KDbPageSize[i]);
229 CreateTable(KDbName[i]);
230 InsertRecords(KDbName[i]);
231 freePageCount[i] = DeleteRecords(KDbName[i], KFreePageCount[i], KDbPageSize[i]);
234 GetHomeTimeAsString(timeBuf);
235 TheTest.Printf(_L("===Time2: %S\r\n"), &timeBuf);
237 //Open the first database, attach all others.
238 TInt err = TheDb.Open(KDbName1());
239 TEST2(err, KErrNone);
242 err = TheDb.Attach(KDbName[i], KDbAttachName[i]);
243 TEST2(err, KErrNone);
246 GetHomeTimeAsString(timeBuf);
247 TheTest.Printf(_L("===Time3: %S\r\n"), &timeBuf);
249 //Check the size of the main database.
250 RSqlDatabase::TSize size;
251 err = TheDb.Size(size);
252 TEST2(err, KErrNone);
253 TEST2((size.iFree / KDbPageSize[0]), freePageCount[0]);
255 //For all attached database: check the size of the database, compact, check the size again.
258 err = TheDb.Size(size, KDbAttachName[i]);
259 TEST2(err, KErrNone);
260 TEST2((size.iFree / KDbPageSize[i]), freePageCount[i]);
262 const TInt KCompactPageCount = 3;
263 TInt rc = TheDb.Compact(KCompactPageCount * KDbPageSize[i], KDbAttachName[i]);
264 TInt expected = KCompactPageCount * KDbPageSize[i];
266 err = TheDb.Size(size, KDbAttachName[i]);
267 TEST2(err, KErrNone);
268 TInt count = size.iFree / KDbPageSize[i];
269 expected = freePageCount[i] - KCompactPageCount;
270 TEST2(count, expected);
273 GetHomeTimeAsString(timeBuf);
274 TheTest.Printf(_L("===Time4: %S\r\n"), &timeBuf);
276 //Detach databases and close the main database.
279 err = TheDb.Detach(KDbAttachName[i]);
280 TEST2(err, KErrNone);
284 GetHomeTimeAsString(timeBuf);
285 TheTest.Printf(_L("===Time5: %S\r\n"), &timeBuf);
290 (void)RSqlDatabase::Delete(KDbName[i]);
294 //Creates a test database (with KDbName1 name). Inserts aRecordCount records.
295 //The page size is specified in aPageSize parameter. But practically the page size is always 1024 bytes.
296 //The record size is such that there is only one record per page.
297 void PrepareDb(TInt aPageSize, TInt aRecordCount, TBool aManualCompaction = EFalse)
299 //Create the database
300 (void)RSqlDatabase::Delete(KDbName1);
301 _LIT8(KConfigStr, "page_size=");
303 config.Copy(KConfigStr);
304 config.AppendNum(aPageSize);
305 if(aManualCompaction)
307 config.Append(_L(";compaction=manual;"));
309 TInt err = TheDb.Create(KDbName1, &config);
310 TEST2(err, KErrNone);
312 err = TheDb.Exec(_L("BEGIN TRANSACTION"));
314 err = TheDb.Exec(_L("CREATE TABLE A(I INTEGER, T TEXT)"));
317 TheText.SetLength(TheText.MaxLength());
318 TheText.Fill(TChar('A'));
319 for(TInt i=0;i<aRecordCount;++i)
321 TheSqlBuf.Format(_L("INSERT INTO A VALUES(%d, '%S')"), i + 1, &TheText);
322 err = TheDb.Exec(TheSqlBuf);
325 err = TheDb.Exec(_L("COMMIT TRANSACTION"));
327 //Delete all records making a lot of free pages. This operation should kick-off the background compaction
328 err = TheDb.Exec(_L("DELETE FROM A WHERE 1"));
329 TEST2(err, aRecordCount);
333 @SYMTestCaseID SYSLIB-SQL-UT-4073
334 @SYMTestCaseDesc Background compaction steps test.
335 The test creates a database with background compaction mode,
336 then inserts records and deletes all of them. The count of records is such that when
337 the records get deleted, the number of the free pages is very big and all free pages cannot
338 be removed for just one compaction step.
339 The test waits for ("compaction interval"/10 ms) time and checks that no compaction
340 step has been run by the server during the pause and the free space size is the same as before the
341 pause. Then the test waits for ("compaction interval" + "compaction step") time and checks that
342 the background compaction step really happened and removed only part of the free pages.
343 The same test is repeated again and the same check is performed again.
344 @SYMTestPriority Medium
345 @SYMTestActions Background compaction steps test.
346 @SYMTestExpectedResults Test must not fail
350 void CompactDbTest2()
353 GetHomeTimeAsString(timeBuf);
354 TheTest.Printf(_L("===Time1: %S\r\n"), &timeBuf);
356 const TInt KPageSize = 1024;
357 //Number of records to be added and removed from database. Need to be increased when testing on a faster
358 // hardware, otherwise at fastest case the background compaction could be finished in just 1 step.
359 const TInt KRecordCount = 2000;
360 PrepareDb(KPageSize, KRecordCount);
362 GetHomeTimeAsString(timeBuf);
363 TheTest.Printf(_L("===Time2: %S\r\n"), &timeBuf);
365 //Check the free space-1
366 RSqlDatabase::TSize size1;
367 TInt err = TheDb.Size(size1);
368 TEST2(err, KErrNone);
369 TheTest.Printf(_L("===Free space before compaction, pages=%d\r\n"), size1.iFree / KPageSize);
370 TEST(size1.iSize >= (KRecordCount * KPageSize));
372 //Wait KSqlCompactStepIntervalMs/10 ms. The background compaction should not be kicked-off.
375 User::After((KSqlCompactStepIntervalMs / 10) * 1000);
378 TTimeIntervalMicroSeconds intervalUs = time2.MicroSecondsFrom(time1);
379 //Check the free space-2
380 RSqlDatabase::TSize size2;
381 err = TheDb.Size(size2);
382 TEST2(err, KErrNone);
383 TheTest.Printf(_L("=== Wait time: %ld ms. Free space after compaction-1, pages=%d\r\n"), intervalUs.Int64() / 1000 ,size2.iFree / KPageSize);
384 if(intervalUs > KSqlCompactStepIntervalMs * 1000)
386 TEST(size2.iFree <= size1.iFree);
390 TEST(size2.iFree == size1.iFree);
393 GetHomeTimeAsString(timeBuf);
394 TheTest.Printf(_L("===Time3: %S\r\n"), &timeBuf);
396 //Wait (KSqlCompactStepIntervalMs + KSqlCompactStepLengthMs) ms. During the pause only part of the free pages
397 //should be removed (whatever can be completed for KSqlCompactStepLengthMs ms).
398 User::After((KSqlCompactStepIntervalMs + KSqlCompactStepLengthMs) * 1000);
399 //Check the free space-3
400 RSqlDatabase::TSize size3;
401 err = TheDb.Size(size3);
402 TEST2(err, KErrNone);
403 TheTest.Printf(_L("===Free space after compaction-2, pages=%d\r\n"), size3.iFree / KPageSize);
406 TheTest.Printf(_L("WARNING: Background compaction finished in 1 step. Initial number of records need to be increased.\r\n"));
408 TEST(size3.iFree > 0 && size3.iFree < size2.iFree);
410 GetHomeTimeAsString(timeBuf);
411 TheTest.Printf(_L("===Time4: %S\r\n"), &timeBuf);
413 //Wait another (KSqlCompactStepIntervalMs + KSqlCompactStepLengthMs) ms. During the pause only part of the free pages
414 //should be removed (whatever can be completed for KSqlCompactStepLengthMs ms).
415 User::After((KSqlCompactStepIntervalMs + KSqlCompactStepLengthMs) * 1000);
416 //Check the free space-4
417 RSqlDatabase::TSize size4;
418 err = TheDb.Size(size4);
419 TEST2(err, KErrNone);
420 TheTest.Printf(_L("===Free space after compaction-3, pages=%d\r\n"), size4.iFree / KPageSize);
421 TEST((size4.iFree > 0 && size4.iFree < size3.iFree) || (size4.iFree == 0));
423 GetHomeTimeAsString(timeBuf);
424 TheTest.Printf(_L("===Time5: %S\r\n"), &timeBuf);
428 (void)RSqlDatabase::Delete(KDbName1);
432 @SYMTestCaseID SYSLIB-SQL-UT-4074
433 @SYMTestCaseDesc Background compaction timer test.
434 The test creates a database with background compaction mode,
435 then inserts records and deletes all of them. The count of records is such that when
436 the records get deleted, the number of the free pages is very big and all free pages cannot
437 be removed for just one compaction step.
438 Then the test executes a set of operations with the server. The amount of time needed for the
439 operations to be executed is bigger than the ("compaction interval" + "compaction step") time.
440 No compaction step should be executed during that time, because every operation resets the background
442 @SYMTestPriority Medium
443 @SYMTestActions Background compaction timer test.
444 @SYMTestExpectedResults Test must not fail
448 void CompactDbTest3()
451 GetHomeTimeAsString(timeBuf);
452 TheTest.Printf(_L("===Time1: %S\r\n"), &timeBuf);
454 const TInt KPageSize = 1024;
455 const TInt KRecordCount = 1000;
456 PrepareDb(KPageSize, KRecordCount);
458 GetHomeTimeAsString(timeBuf);
459 TheTest.Printf(_L("===Time2: %S\r\n"), &timeBuf);
461 //Check the free space-1
462 RSqlDatabase::TSize size1;
463 TInt err = TheDb.Size(size1);
464 TEST2(err, KErrNone);
465 TheTest.Printf(_L("===Free space before operations, pages=%d. Db.Size=%d, Db.Free=%d\r\n"), size1.iFree / KPageSize, size1.iSize, size1.iFree);
466 TEST(size1.iSize >= (KRecordCount * KPageSize));
468 //Execute a set of operations. The time needed for the operations to complete is bigger than
469 //(KSqlCompactStepIntervalMs + KSqlCompactStepLengthMs) ms
471 TEST2(HAL::Get(HAL::EFastCounterFrequency, freq), KErrNone);
472 TUint32 begin = User::FastCounter();
477 err = TheDb.Exec(_L("SELECT COUNT(*) FROM A"));
479 TUint32 current = User::FastCounter();
480 TInt64 diffTicks = (TInt64)current - (TInt64)begin;
483 diffTicks = KMaxTUint32 + diffTicks + 1;
485 const TInt KMicroSecIn1Sec = 1000000;
486 TInt32 us = (diffTicks * KMicroSecIn1Sec) / freq;
488 if(time > ((KSqlCompactStepIntervalMs + KSqlCompactStepLengthMs)))
494 GetHomeTimeAsString(timeBuf);
495 TheTest.Printf(_L("===Time3: %S\r\n"), &timeBuf);
497 //Check the free space-2
498 RSqlDatabase::TSize size2;
499 err = TheDb.Size(size2);
500 TEST2(err, KErrNone);
501 TheTest.Printf(_L("===%d operations completed for %d ms\r\n"), count, time);
502 TheTest.Printf(_L("===Free space after operations, pages=%d\r\n"), size2.iFree / KPageSize);
503 TEST(size1.iFree == size2.iFree);
507 (void)RSqlDatabase::Delete(KDbName1);
511 @SYMTestCaseID SYSLIB-SQL-UT-4103
512 @SYMTestCaseDesc Big manual compaction test.
513 The test creates a database with 1000 free pages, then calls
514 RSqlDatabase::Compact(RSqlDatabase::EMaxCompaction).
515 @SYMTestPriority Medium
516 @SYMTestActions Big manual compaction test.
517 @SYMTestExpectedResults Test must not fail
521 void ManualCompactTest()
524 GetHomeTimeAsString(timeBuf);
525 TheTest.Printf(_L("===Time1: %S\r\n"), &timeBuf);
527 //Create a database with 1000 free pages
528 const TInt KPageSize = 1024;
529 const TInt KRecordCount = 1000;
530 PrepareDb(KPageSize, KRecordCount, ETrue);//create the database with manual compaction mode
532 GetHomeTimeAsString(timeBuf);
533 TheTest.Printf(_L("===Time2: %S\r\n"), &timeBuf);
535 //Check the free space-1
536 RSqlDatabase::TSize size1;
537 TInt err = TheDb.Size(size1);
538 TEST2(err, KErrNone);
539 const TInt KFreePageCount = size1.iFree / KPageSize;
540 TheTest.Printf(_L("===Free space before operations, pages=%d\r\n"), KFreePageCount);
541 TEST(size1.iSize >= (KRecordCount * KPageSize));
543 err = TheDb.Compact(RSqlDatabase::EMaxCompaction);
544 TEST2(err, size1.iFree);
546 GetHomeTimeAsString(timeBuf);
547 TheTest.Printf(_L("===Time3: %S\r\n"), &timeBuf);
551 (void)RSqlDatabase::Delete(KDbName1);
556 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-UT-4072 Manual Compact() - attached databases, different page sizes"));
559 if(IsTimeLimitReached())
561 TheTest.Printf(_L("===Test timeout!\r\n"));
565 TheTest.Next( _L(" @SYMTestCaseID:SYSLIB-SQL-UT-4073 Background compaction steps test"));
568 if(IsTimeLimitReached())
570 TheTest.Printf(_L("===Test timeout!\r\n"));
574 TheTest.Next( _L(" @SYMTestCaseID:SYSLIB-SQL-UT-4074 Background compaction timer test"));
577 if(IsTimeLimitReached())
579 TheTest.Printf(_L("===Test timeout!\r\n"));
583 TheTest.Next( _L(" @SYMTestCaseID:SYSLIB-SQL-UT-4103 Big manual compaction test"));
591 CTrapCleanup* tc = CTrapCleanup::New();
597 TRAPD(err, DoTestsL());
599 TEST2(err, KErrNone);
608 User::Heap().Check();