Update contrib.
1 // Copyright (c) 2005-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_cenrep_helper.h"
19 #include "srvrepos_noc.h"
25 //Forward declarations
28 ///////////////////////////////////////////////////////////////////////////////////////
31 static RTest TheTest(_L("T_CenRepTrans"));
32 const TUid KTestRepositoryId = {0x00000010};
33 //CentralRepository server UID
34 static const TUid KCenRepServerUID = {0x10202BE9};
36 static CTestObserver* TheObserver = NULL;
37 static CServerRepository* TheServerRepository = NULL;
39 static const TUint32 KIntSettingID = 843;
40 static const TInt KIntSettingVal = 25;
42 static const TUint32 KRealSettingID = 844;
43 static const TReal KRealSettingVal = 8.76;
44 static const TReal KRealSettingVal2 = 19.234;
46 static const TUint32 KStringSettingID = 845;
47 _LIT8(KStringSettingVal, "a test string");
48 _LIT8(KStringSettingVal2, "another string");
49 _LIT8(KEmptyString, "");
51 _LIT(KTestFile1, "C:\\PRIVATE\\10202BE9\\PERSISTS\\00000010.TXT");
52 _LIT(KCreTestFile1, "C:\\PRIVATE\\10202BE9\\PERSISTS\\00000010.CRE");
53 _LIT(KTmpTestFile1, "C:\\PRIVATE\\10202BE9\\PERSISTS\\00000010.TMP");
56 ///////////////////////////////////////////////////////////////////////////////////////
59 //Delete file function.
60 //Used by the test application to cleanup the C drive at the end of the test.
61 static void DeleteFile(const TDesC& aFullName)
64 TInt err = fsSession.Connect();
68 if(fsSession.Entry(aFullName, entry) == KErrNone)
70 RDebug::Print(_L("Deleting \"%S\" file.\n"), &aFullName);
71 err = fsSession.SetAtt(aFullName, 0, KEntryAttReadOnly);
74 RDebug::Print(_L("Error %d changing \"%S\" file attributes.\n"), err, &aFullName);
76 err = fsSession.Delete(aFullName);
79 RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err, &aFullName);
86 RDebug::Print(_L("Error %d connecting file session. File: %S.\n"), err, &aFullName);
90 static void DeleteFiles()
92 ::DeleteFile(KTestFile1);
93 ::DeleteFile(KCreTestFile1);
94 ::DeleteFile(KTmpTestFile1);
97 static TInt GetFileTimeStamp(const TDesC& aFullName, TTime& aTime)
100 TInt err = fsSession.Connect();
104 if(fsSession.Entry(aFullName, entry) == KErrNone)
106 aTime=entry.iModified;
112 RDebug::Print(_L("Error %d connecting file session. File: %S.\n"), err, &aFullName);
118 ///////////////////////////////////////////////////////////////////////////////////////
119 // Test macros and functions.
121 static void Check(TInt aValue, TInt aLine)
126 TheTest(EFalse, aLine);
129 static void Check(TInt aValue, TInt aExpected, TInt aLine)
131 if(aValue != aExpected)
134 RDebug::Print(_L("*** Expected error: %d, got: %d\r\n"), aExpected, aValue);
135 TheTest(EFalse, aLine);
138 #define TEST(arg) ::Check((arg), __LINE__)
139 #define TEST2(aValue, aExpected) ::Check(aValue, aExpected, __LINE__)
141 ///////////////////////////////////////////////////////////////////////////////////////
143 static void CheckTmpFileDeleted()
146 TInt err = fsSession.Connect();
149 // Allow time for file to be deleted
150 User::After(500*1000);
151 TEST2 (BaflUtils::FileExists (fsSession, KTmpTestFile1), EFalse);
156 RDebug::Print(_L("Error %d connecting file session.\n"), err);
161 ///////////////////////////////////////////////////////////////////////////////////////
162 // CTestObserver class
164 //It is needed for the creation of CServerRepository object.
165 //Used also to track Get/Set/Create/Delete setting notifications, received
166 //from CServerRepository object.
167 class CTestObserver : public CBase, public MObserver
171 void Notify(TUint32 aId);
174 TBool iAssertNotification;
178 CTestObserver::CTestObserver() :
179 iAssertNotification(EFalse)
183 void CTestObserver::Notify(TUint32 aId)
185 RDebug::Print(_L("Notification! Id=%d.\n"), aId);
186 if(iAssertNotification)
192 ///////////////////////////////////////////////////////////////////////////////////////
195 //Deletes the global TheObserver object. Sets it to NULL.
196 static void ReleaseObserver(TAny*)
202 //Deletes the global TheServerRepository object. Sets it to NULL.
203 static void ReleaseRepository(TAny*)
205 TheServerRepository->Close();
206 delete TheServerRepository;
207 TheServerRepository = NULL;
208 TServerResources::iCacheManager->DisableCache(ETrue);
209 TServerResources::iObserver->CloseiOpenRepositories();
210 TServerResources::iCacheManager->EnableCache();
213 //This function is called from CreateSetting_OOMtestL().
214 //It creates an integer, real or string setting in the repository
215 //with ID=KTestRepositoryId. All this is done in an OOM loop.
216 //The output from this test is: the repository file with ID = KTestRepositoryId
217 //will be copied from Z to C drive;
218 // An integer, real or string setting will be created in that repository.
219 //Changes in the repository will be stored on C drive (in the related repository file)
220 static void DoCreateSettingL(TBool aTrapping,TServerSetting::TType aSettingType)
222 TheObserver = new (ELeave) CTestObserver;
223 TCleanupItem clnItem1(&ReleaseObserver, NULL);
224 CleanupStack::PushL(clnItem1);
226 TheServerRepository = new (ELeave) CServerRepository;
227 TCleanupItem clnItem2(&ReleaseRepository, NULL);
228 CleanupStack::PushL(clnItem2);
230 // Turn OOM trapping on Repository Open on/off for testing purposes.
231 TServerResources::iObserver->iTrapOOMOnOpen=aTrapping;
232 // Open the repository
233 TheServerRepository->OpenL(KTestRepositoryId, *TheObserver);
235 // All write operations, including CreateL must be done in a transaction
236 TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
237 TheServerRepository->CleanupCancelTransactionPushL();
239 //Create the appropriate setting based on aValueType
242 case TServerSetting::EInt:
244 // Create setting - fails if value already there (it must not be)
245 User::LeaveIfError(TheServerRepository->TransactionCreateL(KIntSettingID, KIntSettingVal, NULL));
247 // Try to get the value from the transaction cache
249 User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
250 TEST2(val,KIntSettingVal);
254 case TServerSetting::EReal:
256 // Create setting - fails if value already there (it must not be)
257 User::LeaveIfError(TheServerRepository->TransactionCreateL(KRealSettingID, KRealSettingVal, NULL));
259 // Try to get the value from the transaction cache
261 User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
262 TEST(val == KRealSettingVal);
266 case TServerSetting::EString:
268 // Create setting - fails if value already there (it must not be)
269 User::LeaveIfError(TheServerRepository->TransactionCreateL(KStringSettingID, KStringSettingVal, NULL));
271 // Try to get the value from the transaction cache
272 TBuf8<16> bufVal(KEmptyString);
273 User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
274 TEST(bufVal == KStringSettingVal);
281 // Commit the changes
284 User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
286 TServerResources::iObserver->iTrapOOMOnOpen=EFalse;
287 // Close the repository
288 TheServerRepository->Close();
290 CleanupStack::PopAndDestroy(2);//TheObserver & TheServerRepository
293 //This function is called from SetSetting_OOMtest().
294 //It expects that the previous test (CreateSetting_OOMtestL()) had completed successfully
295 //and there is a repository file (ID=0x10) on drive C and there is an integer setting
296 //in that repository (ID=KIntSettingID, Value=KIntSettingVal).
297 //The test will try to change the setting value in a transaction. Because the test is
298 //executed in an OOM loop, CommitChangesL() - will fail when called by SetL, leaving the
299 //repository content inconsistent - the in-memory repository content will be deleted
300 //(no settings), but in the file the setting will exist and its value will still be KIntSettingVal.
301 static void DoSetIntSettingL()
303 TEST(TheObserver != NULL);
304 TEST(TheServerRepository != NULL);
305 // Check if the setting is there - and has (or is restored to) the original value
307 User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
308 TEST2(val,KIntSettingVal);
310 // Change the setting value
311 // All write operations, including CreateL must be done in a transaction
312 TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
313 TheServerRepository->CleanupCancelTransactionPushL();
314 // Change the setting value
315 User::LeaveIfError(TheServerRepository->TransactionSetL(KIntSettingID, KIntSettingVal + 1));
317 // Check if the setting is there - it should be in transaction cache now
318 User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
319 TEST2(val,(KIntSettingVal + 1));
321 // Commit the changes
324 User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
326 // Check if the setting is there. The transaction was successful so the new value
327 // should be returned.
329 User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
330 TEST2(val,(KIntSettingVal + 1));
333 //This function is called from SetSetting_OOMtest().
334 //It expects that the previous test (CreateSetting_OOMtestL()) had completed successfully
335 //and there is a repository file (ID=0x10) on drive C and there is an rea; setting
336 //in that repository (ID=KRealSettingID, Value=KRealSettingVal).
337 //The test will try to change the setting value in a transaction. Because the test is
338 //executed in an OOM loop, CommitChangesL() - will fail when called by SetL, leaving the
339 //repository content inconsistent - the in-memory repository content will be deleted
340 //(no settings), but in the file the setting will exist and its value will still be KRealSettingVal.
341 static void DoSetRealSettingL()
343 TEST(TheObserver != NULL);
344 TEST(TheServerRepository != NULL);
345 // Check if the setting is there - and has (or is restored to) the original value
347 User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
348 TEST(val == KRealSettingVal);
350 // Change the setting value
351 // All write operations, including CreateL must be done in a transaction
352 TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
353 TheServerRepository->CleanupCancelTransactionPushL();
354 // Change the setting value
355 User::LeaveIfError(TheServerRepository->TransactionSetL(KRealSettingID, KRealSettingVal2));
357 // Check if the setting is there - it should be in transaction cache now
358 User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
359 TEST(val == KRealSettingVal2);
361 // Commit the changes
364 User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
366 // Check if the setting is there. The transaction was successful so the new value
367 // should be returned.
369 User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
370 TEST(val == KRealSettingVal2);
373 //This function is called from SetSetting_OOMtest().
374 //It expects that the previous test (CreateSetting_OOMtestL()) had completed successfully
375 //and there is a repository file (ID=0x10) on drive C and there is an string setting
376 //in that repository (ID=KStringSettingID, Value=KStringSettingVal).
377 //The test will try to change the setting value in a transaction. Because the test is
378 //executed in an OOM loop, CommitChangesL() - will fail when called by SetL, leaving the
379 //repository content inconsistent - the in-memory repository content will be deleted
380 //(no settings), but in the file the setting will exist and its value will still
381 //be KStringSettingVal.
382 static void DoSetStringSettingL()
384 TEST(TheObserver != NULL);
385 TEST(TheServerRepository != NULL);
387 // Check if the setting is there - and has (or is restored to) the original value
388 TBuf8<16> bufVal(KEmptyString);
389 User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
390 TEST(bufVal == KStringSettingVal);
392 // Change the setting value
393 // All write operations, including CreateL must be done in a transaction
394 TheServerRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction);
395 TheServerRepository->CleanupCancelTransactionPushL();
397 // Change the setting value
398 User::LeaveIfError(TheServerRepository->TransactionSetL(KStringSettingID, KStringSettingVal2));
400 // Check if the setting is there - it should be in transaction cache now
401 User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
402 TEST(bufVal == KStringSettingVal2);
403 // Commit the changes
406 User::LeaveIfError(TheServerRepository->CommitTransaction(keyInfo));
408 // Check if the setting is there. The transaction was successful so the new value
409 // should be returned.
410 bufVal = KEmptyString;
411 User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
412 TEST(bufVal == KStringSettingVal2);
415 //This function is called from SetSetting_OOMtest()in the case where previous
416 //DoSetSettingL() call failed with KErrNoMemory, leaving the repository object
417 //in inconsisten state - the in-memory presentation of the repository differs from
418 //the repository file content. The function will try to repair the repository,
419 //as it would have to happen in a real situation. Then it will check that
420 //the repository content is consistent, which means the tested setting value should
421 //be the same as it was before DoSetSettingL() call.
422 static void DoRecoverRepositoryL(TServerSetting::TType aSettingType)
424 //Repair the repository as it happens in a real situation
425 //AccessL calls RestoreConsistencyL() indirectly
426 //RestoreConsistencyL() is called in the production code from the session object,
427 //ServiceL() implementation. So, it will be called before any other repository call
428 //and it should restore the repository content - it should repair the repository
430 TServerResources::iObserver->AccessL(KTestRepositoryId);
431 //Check if the setting is there - the old, pre-transactional value should be
435 case TServerSetting::EInt:
438 User::LeaveIfError(TheServerRepository->Get(KIntSettingID, val));
439 TEST2(val,KIntSettingVal);
443 case TServerSetting::EReal:
446 User::LeaveIfError(TheServerRepository->Get(KRealSettingID, val));
447 TEST(val == KRealSettingVal);
451 case TServerSetting::EString:
453 TBuf8<16> bufVal(KEmptyString);
454 User::LeaveIfError(TheServerRepository->Get(KStringSettingID, bufVal));
455 TEST(bufVal == KStringSettingVal);
463 //Inits TServerResources object.
464 //It should be called once, before the creation of CServerRepository object.
465 static void InitEnvL()
467 TServerResources::InitialiseL();
469 delete TServerResources::iDataDirectory; TServerResources::iDataDirectory = NULL;
470 delete TServerResources::iRomDirectory; TServerResources::iRomDirectory = NULL;
471 delete TServerResources::iInstallDirectory; TServerResources::iInstallDirectory = NULL;
473 TServerResources::iDataDirectory = HBufC::NewL(KMaxFileName);
474 TServerResources::iRomDirectory = HBufC::NewL(KMaxFileName);
475 TServerResources::iInstallDirectory = HBufC::NewL(KMaxFileName);
478 uidName.Format(_L("%08X"), KCenRepServerUID.iUid);
480 TPtr ptr1(TServerResources::iDataDirectory->Des());
481 ptr1.Append(_L("C:\\PRIVATE\\"));
482 ptr1.Append(uidName);
483 ptr1.Append(_L("\\PERSISTS\\"));
485 TPtr ptr2(TServerResources::iRomDirectory->Des());
486 ptr2.Append(_L("Z:\\PRIVATE\\"));
487 ptr2.Append(uidName);
488 ptr2.Append(_L("\\"));
490 TPtr ptr3(TServerResources::iInstallDirectory->Des());
491 ptr3.Append(_L("C:\\PRIVATE\\"));
492 ptr3.Append(uidName);
493 ptr3.Append(_L("\\"));
496 //Destroys TServerResources object.
497 //It should be called after the CServerRepository object was destroyed.
498 static void ReleaseEnv()
500 TServerResources::Close();
503 //Creates global TheServerRepository and TheObserver objects
504 //It is used in SetSetting_OOMtest()
505 static void InitEnv2L()
507 TheObserver = new CTestObserver;
508 TEST(TheObserver != NULL);
510 TheServerRepository = new CServerRepository;
511 TEST(TheServerRepository != NULL);
513 TheServerRepository->OpenL(KTestRepositoryId, *TheObserver);
516 //Destroys global TheServerRepository and TheObserver objects
517 //It is used in SetSetting_OOMtest()
518 static void ReleaseEnv2()
520 ::ReleaseRepository(NULL);
521 ::ReleaseObserver(NULL);
524 //Central repository test: it creates a setting in OOM conditions.
525 static void CreateSetting_OOMtestL(TBool aTrapping,TServerSetting::TType aSettingType)
528 TRAP(err, ::InitEnvL());
529 TEST2(err, KErrNone);
530 for(TInt count=1;;++count)
533 __UHEAP_FAILNEXT(count);
536 TRAP(err, ::DoCreateSettingL(aTrapping,aSettingType));
538 TServerResources::iOwnerIdLookUpTable.Reset();
539 TServerResources::iObserver->Reset();
545 RDebug::Print(_L("The test succeeded at heap failure rate=%d.\n"), count);
548 else if(err != KErrNoMemory)
550 TEST2(err, KErrNone);
558 //Central repository test: it sets new setting value in OOM conditions.
559 //Then ComitChangesL() is called to store the new setting's value in the repository
560 //file. Because of the simulated OOM conditions ComitChangesL() can fail, leaving the
561 //repository in inconsistent state.
562 //DoRecoverRepositoryL() is called to repair the repository and restore the consistency.
563 static void SetSetting_OOMtest(TServerSetting::TType aSettingType)
567 // Check that cre file exists and get modification time
568 TEST2 (GetFileTimeStamp(KCreTestFile1, before), KErrNone);
572 TRAP(err, ::InitEnvL());
573 TEST2(err, KErrNone);
574 TRAP(err, ::InitEnv2L());
575 TEST2(err, KErrNone);
576 for(TInt count=1;;++count)
578 __UHEAP_FAILNEXT(count);
583 case TServerSetting::EInt:
585 TRAP(err, ::DoSetIntSettingL());
588 case TServerSetting::EReal:
590 TRAP(err, ::DoSetRealSettingL());
593 case TServerSetting::EString:
595 TRAP(err, ::DoSetStringSettingL());
602 // Get timestamp of cre file. It is now possible for the commit to
603 // fail with KErrNoMemory after a successful write. If filestamp
604 // changes, the cre file has been written and the test is complete
605 TEST2 (GetFileTimeStamp(KCreTestFile1, after), KErrNone);
611 if(err == KErrNoMemory)
614 //DoSetSettingL() call failed (so the transaction). Try to recover.
616 TRAP(err2, ::DoRecoverRepositoryL(aSettingType));
617 TEST2(err2, KErrNone);
619 else if(err == KErrNone)
621 //The heap cannot be checked at this point because memory may have been
622 //allocated which must be freed by a call to ReleaseEnv2.
623 //It is checked outside the loop instead
624 RDebug::Print(_L("The test succeeded at heap failure rate=%d.\n"), count);
630 TEST2(err, KErrNone);
632 // Check that tmp file generated by attempted commit is deleted
633 // by RestoreConsistencyL() in DoRecoverRepositoryL().
634 ::CheckTmpFileDeleted();
636 // Check that no tmp file is left over
639 //This __UHEAP_MARKEND checks the heap after the call to DoSetxSettings
640 //succeeds as calls to CommitTRansactions can allocate memory which must be freed
641 //by calling ReleaseEnv2
646 ///////////////////////////////////////////////////////////////////////////////////////
648 void RunTestsL(TServerSetting::TType aSettingType)
651 TheTest.Next(_L("Create setting - OOM test - OOM Trapping off"));
652 ::CreateSetting_OOMtestL(EFalse,aSettingType);
654 TheTest.Next(_L("Create setting - OOM test - OOM Trapping on"));
655 ::CreateSetting_OOMtestL(ETrue,aSettingType);
657 TheTest.Next(_L("Set setting - transaction - OOM test"));
658 ::SetSetting_OOMtest(aSettingType);
664 // create and install the active scheduler we need for the cache manager in TServerResources::InitialiseL
665 CActiveScheduler* cacheManagerAS=new(ELeave) CActiveScheduler;
666 CleanupStack::PushL(cacheManagerAS);
667 CActiveScheduler::Install(cacheManagerAS);
671 TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-CENTRALREPOSITORY-LEGACY-T_CENREPTRANS-0001 Int Setting Tests "));
672 RunTestsL(TServerSetting::EInt);
674 TheTest.Next(_L("Real Setting Tests"));
675 RunTestsL(TServerSetting::EReal);
677 TheTest.Next(_L("String Setting Tests"));
678 RunTestsL(TServerSetting::EString);
683 CleanupStack::PopAndDestroy(cacheManagerAS);
690 CTrapCleanup* tc = CTrapCleanup::New();
694 TRAPD(err, ::MainL());
695 TEST2(err, KErrNone);
703 User::Heap().Check();