First public contribution.
1 // Copyright (c) 1998-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.
16 #include "t_dbstress.h"
18 //#define __DUMP_STATE
20 //#define _INSTALL_FILE_SYSTEM
23 GLDEF_D RTest TheTest(_L("t_dbstress: Stress testing DBMS"));
25 GLDEF_D TPtrC KTestDir=_S("\\DBMS-TST\\");
26 GLDEF_D TPtrC KLogFile=_L("T_STRESS.LOG");
27 GLDEF_D TPtrC KTestDatabase=_S("T_STRESS.DB");
28 GLDEF_D TInt NewCount,OldCount;
30 GLDEF_D Timer RunTimer;
34 LOCAL_D RThread TheThread;
35 LOCAL_D TRequestStatus TheStatus;
36 LOCAL_D RDbStoreDatabase TheDatabase;
38 LOCAL_D RDbView Trans;
39 LOCAL_D RDbTable TheTable;
40 LOCAL_D TInt Shot,ShotDuringCommit;
41 LOCAL_D TInt64 RunningTime(1);
42 LOCAL_D Timer Stopwatch;
45 const TPtrC KDumpFile=_S("T_STRESS.DMP");
47 const TInt KTestCleanupStack=0x20;
51 iTime.UniversalTime();
58 return ((t.MicroSecondsFrom(iTime).Int64()) + 500)/1000;
64 TheTest.Printf(_L(" %u milliseconds\n"), I64LOW(milli) );
77 static CDbColSet* CreateL(const SColDef* aDef);
79 CDbColSet* Set::CreateL(const SColDef* aDef)
81 CDbColSet *set=CDbColSet::NewLC();
82 for (;aDef->iName!=NULL;++aDef)
84 TDbCol col(*aDef->iName,aDef->iType);
85 col.iAttributes=aDef->iAttributes;
93 const TPtrC KAccounts=_S("ACCOUNTS");
94 const TPtrC KAccountsID=_S("ID");
95 const TPtrC KAccountsBalance=_S("BALANCE");
96 const TPtrC KAccountsStatement=_S("STATEMENT_BALANCE");
97 Set::SColDef const AccountsDef[]=
99 {&KAccountsID,EDbColInt32,TDbCol::ENotNull},
100 {&KAccountsBalance,EDbColInt32,TDbCol::ENotNull},
101 {&KAccountsStatement,EDbColInt32,TDbCol::ENotNull},
104 const TInt KInitialCash=100000;
105 const TInt KInitialBalance=1000;
108 const TPtrC KTransactions=_S("TRANSACTIONS");
109 const TPtrC KTransactionDate=_S("T_DATE");
110 const TPtrC KTransactionFrom=_S("FROM_ID");
111 const TPtrC KTransactionTo=_S("TO_ID");
112 const TPtrC KTransactionAmount=_S("AMOUNT");
113 Set::SColDef const TransactionsDef[]=
115 // {&KTransactionDate,EDbColDateTime,TDbCol::ENotNull},
116 {&KTransactionDate,EDbColInt32,TDbCol::ENotNull},
117 {&KTransactionFrom,EDbColInt32,TDbCol::ENotNull},
118 {&KTransactionTo,EDbColInt32,TDbCol::ENotNull},
119 {&KTransactionAmount,EDbColInt32,TDbCol::ENotNull},
123 LOCAL_D TInt32 TotalMonies;
124 LOCAL_D TBuf<100> Buf;
126 GLDEF_C TInt Random(TInt aRange)
128 return (Math::Random()>>11)%aRange;
131 ///////////////////////////////////////////////////////////////////////////////////////
132 ///////////////////////////////////////////////////////////////////////////////////////
134 TPtrC FileName(const TText* aFile)
137 TInt ix=p.LocateReverse('\\');
139 ix=p.LocateReverse('/');
145 //Test macros and functions
146 void Check1(TInt aValue, const TText* aFile, TInt aLine)
150 TPtrC fname(FileName(aFile));
151 TheTest.Printf(_L("*** Expression evaluated to false. %S-%d\r\n"), &fname, aLine);
152 TheTest(EFalse, aLine);
155 void Check2(TInt aValue, TInt aExpected, const TText* aFile, TInt aLine)
157 if(aValue != aExpected)
159 TPtrC fname(FileName(aFile));
160 TheTest.Printf(_L("*** Expected %d, got %d. %S-%d\r\n"), aExpected, aValue, &fname, aLine);
161 TheTest(EFalse, aLine);
165 ///////////////////////////////////////////////////////////////////////////////////////
167 LOCAL_C void CreateIndexL(RDbDatabase& aDatabase,const TDesC& aTable,const TDesC& aColumn,TBool aUnique)
169 CDbKey* key=CDbKey::NewLC();
173 TEST2(aDatabase.CreateIndex(aColumn,aTable,*key),KErrNone);
174 CleanupStack::PopAndDestroy();
178 // Create the database
180 LOCAL_C void CreateDatabaseL()
182 CFileStore* store=CPermanentFileStore::ReplaceLC(TheFs,KTestDatabase,EFileRead|EFileWrite);
183 store->SetTypeL(KPermanentFileStoreLayoutUid);
184 store->SetRootL(TheDatabase.CreateL(store));
187 CDbColSet* set=Set::CreateL(AccountsDef);
188 TEST2(TheDatabase.CreateTable(KAccounts,*set),KErrNone);
190 CreateIndexL(TheDatabase,KAccounts,KAccountsID,ETrue);
191 CreateIndexL(TheDatabase,KAccounts,KAccountsBalance,EFalse);
192 set=Set::CreateL(TransactionsDef);
193 TEST2(TheDatabase.CreateTable(KTransactions,*set),KErrNone);
195 CreateIndexL(TheDatabase,KTransactions,KTransactionDate,EFalse);
196 TEST2(TheDatabase.Commit(),KErrNone);
198 // prepare Accs table
200 TEST2(Accs.Prepare(TheDatabase,_L("select * from accounts"),Accs.EInsertOnly),KErrNone);
202 Accs.SetColL(1,TInt32(ECash));
203 Accs.SetColL(2,KInitialCash);
204 Accs.SetColL(3,KInitialCash);
206 TotalMonies=KInitialCash;
207 for (TInt ii=EJohn;ii<=EPenny;++ii)
211 Accs.SetColL(2,KInitialBalance);
212 Accs.SetColL(3,KInitialBalance);
214 TotalMonies+=KInitialBalance;
216 TEST2(TheDatabase.Commit(),KErrNone);
219 CleanupStack::PopAndDestroy(); // store
224 LOCAL_C void DumpStateL()
227 CleanupClosePushL(file);
228 User::LeaveIfError(file.Replace(TheFs,KLogFile,EFileWrite|EFileStreamText));
229 RDbRowSet::RConstraint match;
230 CleanupClosePushL(match);
231 for (TInt id=ECash;id<=EPenny;++id)
233 Buf.Format(_L("id=%d"),id);
235 TEST(Accs.FindL(Accs.EForwards,Buf)>=0);
237 TInt balance=Accs.ColInt(2);
238 Buf.Format(_L("\nStatement for account %d: Previous balance %d\n"),id,balance);
239 User::LeaveIfError(file.Write(Buf));
240 Buf.Format(_L("from_id=%d or to_id=%d"),id,id);
241 User::LeaveIfError(match.Open(Trans,Buf));
242 for (Trans.BeginningL();Trans.NextL();)
244 if (Trans.MatchL(match))
247 TInt from=Trans.ColInt(2);
248 TInt amount=Trans.ColInt(4);
249 Buf.Format(_L("%04d: %6s %5d\n"),Trans.ColInt(1),from==id?_S("debit"):_S("credit"),amount);
250 User::LeaveIfError(file.Write(Buf));
257 Buf.Format(_L("Closing balance %d\n"),balance);
258 User::LeaveIfError(file.Write(Buf));
259 Buf.Format(_L("Account balance %d\n"),Accs.ColInt(3));
260 User::LeaveIfError(file.Write(Buf));
262 CleanupStack::PopAndDestroy(2);
268 // Check that the database structure is fully intact
270 LOCAL_C void VerifyDatabaseL(CPersistentStore& aStore)
272 TheDatabase.OpenL(&aStore,aStore.Root());
274 TEST2(TheTable.Open(TheDatabase,KAccounts,TheTable.EReadOnly),KErrNone);
275 TEST2(TheTable.CountL(),KAccountIDs);
276 TInt r=TheTable.SetIndex(KAccountsID);
280 TEST2(TheTable.CountL(),KAccountIDs);
281 for (TInt id=ECash;id<=EPenny;++id)
283 TEST(TheTable.NextL());
285 TEST2(TheTable.ColInt(1),id);
287 TEST(!TheTable.NextL());
289 r=TheTable.SetIndex(KAccountsBalance);
293 TEST2(TheTable.CountL(),KAccountIDs);
294 TEST(TheTable.FirstL());
296 TInt last=TheTable.ColInt(2);
297 for (TInt ii=1;ii<KAccountIDs;++ii)
299 TEST(TheTable.NextL());
301 TInt bal=TheTable.ColInt(2);
305 TEST(!TheTable.NextL());
308 TEST2(TheTable.Open(TheDatabase,KTransactions,TheTable.EReadOnly),KErrNone);
309 TInt count=TheTable.CountL();
310 r=TheTable.SetIndex(KTransactionDate);
314 TEST2(TheTable.CountL(),count);
317 TEST(TheTable.FirstL());
319 TInt last=TheTable.ColInt(1);
322 TEST(TheTable.NextL());
324 TInt date=TheTable.ColInt(1);
328 TEST(!TheTable.NextL());
331 TEST(!TheTable.FirstL());
334 // verify database integrity
335 TInt balance[KAccountIDs];
336 TEST2(Accs.Prepare(TheDatabase,_L("select id,statement_balance,balance from accounts"),Accs.EReadOnly),KErrNone);
337 TEST2(Accs.CountL(),KAccountIDs);
341 TInt id=Accs.ColInt(1);
342 balance[id]=Accs.ColInt(2);
344 TEST2(Trans.Prepare(TheDatabase,_L("select t_date,from_id,to_id,amount from Transactions"),Trans.EReadOnly),KErrNone);
346 while (Trans.NextL())
350 TInt from=Trans.ColInt(2);
351 TInt to=Trans.ColInt(3);
352 TInt amount=Trans.ColInt(4);
353 balance[from]-=amount;
356 TEST2(transact,Trans.CountL());
357 if (NewCount!=-1 && transact!=NewCount)
359 TEST2(transact,OldCount);
362 OldCount=NewCount=transact;
364 for (Accs.BeginningL();Accs.NextL();)
367 TInt id=Accs.ColInt(1);
369 if (balance[id]!=Accs.ColInt(3))
372 TEST(balance[id]==Accs.ColInt(3));
376 TEST2(total,TotalMonies);
382 LOCAL_C TInt Verify(CPersistentStore& aStore)
384 TRAPD(r,VerifyDatabaseL(aStore));
388 LOCAL_C TInt Recover(CPersistentStore& aStore)
390 TRAPD(r,TheDatabase.OpenL(&aStore,aStore.Root()));
393 r=TheDatabase.Recover();
399 LOCAL_C void CompactL(CStreamStore& aStore)
401 TInt t=aStore.ReclaimL();
403 t-=aStore.CompactL();
404 TheTest.Printf(_L(" compacted %d byte(s) in"),t);
409 LOCAL_C TInt Compact(CStreamStore& aStore)
411 TRAPD(r,CompactL(aStore));
415 LOCAL_C TInt EndThread()
417 RunningTime+=RunTimer.Stop();
418 if (TheStatus==KRequestPending)
420 User::WaitForRequest(TheStatus);
422 if (TheThread.ExitType()==EExitKill)
423 r=TheThread.ExitReason();
430 //aTestExecutionTime - desired test execution time in minutes
431 LOCAL_C void RunTestL(TInt aTestExecutionTime = 0)
433 __ASSERT_ALWAYS(aTestExecutionTime >= 0, User::Invariant());
435 RThread().SetPriority(EPriorityMore);
436 TheTest.Start(_L("Create the database"));
439 TTimeIntervalMinutes timeInterval(aTestExecutionTime);
442 timeCurrent.UniversalTime();
443 TTime timeEnd(timeCurrent);
444 timeEnd += timeInterval;
446 for (TBool condition=ETrue; condition; condition = aTestExecutionTime > 0 ? (timeCurrent < timeEnd) : ETrue)
448 TheTest.Next(_L("Main loop"));
449 TheTest.Start(_L("Kick off the thread"));
450 TEST2 (StartThread(TheThread,TheStatus),KErrNone);
455 if (TheStatus!=KRequestPending)
460 TheTest.Next(_L("End the thread"));
461 TInt exit=EndThread();
463 TheTest.Printf(_L(" thread failed with error %d\n"),exit);
466 CFileStore* store=NULL;
467 for (TInt ii=0;;++ii)
469 TheTest.Printf(_L("Opening %d\r"),ii);
470 TRAPD(r,store=CFileStore::OpenL(TheFs,KTestDatabase,EFileRead|EFileWrite|EFileWriteDirectIO));
476 TheTest.Next(_L("Verify & Recover"));
477 TEST2 (Verify(*store),KErrNone);
481 TheTest.Printf(_L(" Iteration %d, TPS %d, during commit %d%%\n"),Shot,I64LOW(tps),(100*ShotDuringCommit)/Shot);
482 TInt r=Recover(*store);
483 if (r==KErrNoMemory || r==KErrDiskFull)
484 { // need to compact before completing recovery
485 TheTest.Next(_L("No space, compacting"));
486 TEST2 (Compact(*store),KErrNone);
487 TheTest.Next(_L("Verify & Recover again"));
488 TEST2 (Verify(*store),KErrNone);
492 TheTest.Next(_L("Verify & Compact"));
493 // TEST2 (Verify(*store),KErrNone);
494 TEST2 (Compact(*store),KErrNone);
495 TheTest.Next(_L("Verify"));
496 TEST2 (Verify(*store),KErrNone);
501 timeCurrent.UniversalTime();
507 @SYMTestCaseID SYSLIB-DBMS-CT-0636
508 @SYMTestCaseDesc DBMS stess testing.
509 @SYMTestPriority Medium
510 @SYMTestActions Tests for verifying the database integrity.
511 @SYMTestExpectedResults Test must not fail
514 static void RunVerify()
516 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0636 Open store "));
517 CFileStore* store=NULL;
518 TRAPD(r,store=CFileStore::OpenL(TheFs,KTestDatabase,EFileRead|EFileWrite|EFileWriteDirectIO));
520 TheTest.Next(_L("Verify"));
522 TotalMonies=KInitialCash + (EPenny-EJohn+1)*KInitialBalance;
523 TEST2 (Verify(*store),KErrNone);
524 TheTest.Next(_L("Recover"));
525 TEST2 (Recover(*store),KErrNone);
526 TheTest.Next(_L("Verify"));
527 TEST2 (Verify(*store),KErrNone);
533 // Prepare the test directory.
535 LOCAL_C void setupTestDirectory()
537 TInt r=TheFs.Connect();
540 r=TheFs.MkDir(KTestDir);
541 TEST(r==KErrNone || r==KErrAlreadyExists);
542 r=TheFs.SetSessionPath(KTestDir);
547 // Initialise the cleanup stack.
549 LOCAL_C CTrapCleanup* setupCleanup()
551 CTrapCleanup* cleanup=CTrapCleanup::New();
555 for (TInt i=KTestCleanupStack;i>0;i--)\
556 CleanupStack::PushL((TAny*)0);\
557 CleanupStack::Pop(KTestCleanupStack);\
567 // t_stress [-v]|[0]|[<positive number>]
569 // -v - a verification test will be run;
570 // 0 - a stress test will be run for indefinite time;
571 // <positive number> - a stress test will be run for <positive number> minutes;
572 // If the test is run without arguments, the test execution time will be 10 minutes
573 // (KDefaultTestExecutionTime constant bellow).
574 GLDEF_C TInt E32Main()
577 setupTestDirectory();
578 CTrapCleanup* cleanup=setupCleanup();
582 User::CommandLine(cmd);
587 TPtrC arg(lex.NextToken());
588 if(arg.Length() == 0)
590 const TInt KDefaultTestExecutionTime = 10;//default test execution time - minutes
591 TRAP(err, RunTestL(KDefaultTestExecutionTime));
594 else if(arg.CompareF(_L("-v")) == 0)
601 TInt32 testExecutionTime = 0;
603 (void)lex.Val(testExecutionTime);
604 TRAP(err, RunTestL(testExecutionTime));
608 TInt err2 = TheFs.Delete(KTestDatabase);
611 RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err2, &KTestDatabase);
613 err2 = TheFs.Delete(KLogFile);
616 RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err2, &KLogFile);
618 TEST2(err, KErrNone);