os/ossrv/lowlevellibsandfws/pluginfw/Test_Bed/test_bed/UnitTest.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 1997-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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Implementation of the CUnitTest base class.
    15 // 
    16 //
    17 
    18 #include <ecom/test_bed/unittest.h>
    19 #include <ecom/test_bed/transition.h>
    20 #include <ecom/test_bed/datalogger.h>
    21 #include <ecom/test_bed/testbeddefinitions.h>
    22 
    23 
    24 EXPORT_C CUnitTest::~CUnitTest()
    25 	{
    26 	Cancel();
    27 
    28 	// Delete any outstanding asynchronous transitions
    29 	if(iOutstandingTransitions)
    30 		{
    31 		iOutstandingTransitions->Reset();
    32 		delete iOutstandingTransitions;
    33 		}
    34 
    35 	if(iTransitions)
    36 		{
    37 		iTransitions->ResetAndDestroy();
    38 		delete iTransitions;
    39 		}
    40 
    41 	iLeaveErrorArray.Reset();
    42 
    43 	delete iFileMan;
    44 	iFs.Close();
    45 	}
    46 
    47 EXPORT_C void CUnitTest::UnitTestConstructL()
    48 	{
    49 	User::LeaveIfError(iFs.Connect());
    50 	iFileMan = CFileMan::NewL(iFs);
    51 	CTimer::ConstructL();
    52 	iTransitions = new(ELeave) RPointerArray<CTransition>;
    53 	iOutstandingTransitions = new(ELeave) RPointerArray<CTransition>;
    54 	_LIT(KConstructingUnitTestMsg, "Constructed Unit Test named %S");
    55 	iDataLogger.LogInformationWithParameters(KConstructingUnitTestMsg, &iUnitTestName);
    56 	}
    57 
    58 
    59 CUnitTestInfo* CUnitTest::TransitionSetL() const
    60 	{
    61 	CUnitTestInfo* transitionSet = CUnitTestInfo::NewL(iUnitTestName);
    62 	return transitionSet;
    63 	}
    64 
    65 
    66 EXPORT_C void CUnitTest::RunTest(TTimeIntervalMicroSeconds32 aTimeAfter /*= 0*/)
    67 	{
    68 	After(aTimeAfter);
    69 	_LIT(KTxtSeparator, "-----------------------------------------------------------------------------------");
    70 	_LIT(KStartingUnitTest, "Beginning Unit Test named %S");
    71 	iDataLogger.LogInformation(KTxtSeparator);
    72 	iDataLogger.LogInformationWithParameters(KStartingUnitTest, &iUnitTestName);
    73 	iDataLogger.ReportInformationWithParameters(KStartingUnitTest, &iUnitTestName);
    74 	}
    75 
    76 EXPORT_C void CUnitTest::RunL()
    77 	{
    78 	_LIT(KUnitTestRunLPanic, "CUnitTest::RunL");
    79 
    80 	TInt status = iStatus.Int();
    81 	switch(status)
    82 		{
    83 		case (KTestBedRepeatTest):			/* A stub has requested repeat of the last test */
    84 			// Go back one so that we repeat the last test
    85 			--iNextTransitionIndex;
    86 			break;
    87 
    88 		case (KTestBedTestLeft):				/* The last transition's RunL left */
    89 		case (KTestBedTestCancel):				/* The last transition was cancelled */
    90 		case (KTestBedLeakTestLoopDetected):	/* A leak test detected an infinite loop */
    91 		case (KTestBedFailedPreConditions):		/* The last transition failed it's pre conditions */
    92 		case (KTestBedFailedPostConditions):	/* The last transition failed it's post conditions */
    93 			// Go to the end of the test so that it finishes
    94 			iNextTransitionIndex = iTransitions->Count();
    95 			break;
    96 
    97 		case (KTestBedAsynchronousTransition):	/* The last transition started an async request */
    98 			// Remember that we have an outstanding request and then carry on
    99 			iOutstandingTransitions->Append((*iTransitions)[iNextTransitionIndex - 1]);
   100 			break;
   101 
   102 		case (KErrNone):
   103 			break;
   104 
   105 		default:
   106 			User::Panic(KUnitTestRunLPanic, KTestBedInvalidStatus);
   107 		}
   108 
   109 	// If we still have more transitions to run
   110 	if(iNextTransitionIndex < iTransitions->Count())
   111 		{
   112 		iStatus = KRequestPending;
   113 		SetActive();
   114 
   115 		// If the next transition is a blocking one then wait for all outstanding async
   116 		// requests to complete.  Otherwise just run the next transition
   117 		if(((*iTransitions)[iNextTransitionIndex]->IsBlockingTransition()) && 
   118 			(iOutstandingTransitions->Count() > 0))
   119 			{
   120 			iWaitingForCompletion = ETrue;
   121 			}
   122 		else
   123 			{
   124 			(*iTransitions)[iNextTransitionIndex]->RunTransition(&iStatus);
   125 			++iNextTransitionIndex;
   126 			}
   127 		}
   128 	else
   129 		{
   130 		// If we still have outstanding async requests then wait for these to complete
   131 		// otherwise we have finished this test
   132 		if(iOutstandingTransitions->Count() > 0)
   133 			{
   134 			iWaitingForCompletion = ETrue;
   135 			iStatus = KRequestPending;
   136 			SetActive();
   137 			}
   138 		else
   139 			{
   140 			iUnitTestObserver.Complete(this);
   141 
   142 			_LIT(KInfoPrintFailed, "Failed: Unit Test");
   143 			_LIT(KTestLeft, "Failed: Unit Test %S left");
   144 			_LIT(KTestLeftWithUnexpectedError, "Failed: Test %S left with unexpected error");
   145 			_LIT(KTestFailed, "Failed: Unit Test %S failed a pre/post condition validation check");
   146 			_LIT(KTestLeftWithExpectedError, "Test %S left with an anticipated error");
   147 			_LIT(KTestCancelled, "Cancelled: Unit Test Transition %S was cancelled");
   148 			_LIT(KTestEnteredInfiniteLoop, "Unit Test Transition %S aborted (infinitely looping)");
   149 			_LIT(KEndingUnitTest, "Successfully completed Unit Test %S");
   150 			// We use RTest if it is present in the framework to validate
   151 			// status codes for errors. Note: not all non KErrNone code mean 
   152 			// there was an error and so we need to selective over which 
   153 			// cases below we use RTest().
   154 			switch(status)
   155 				{
   156 				case (KTestBedTestLeft):
   157 					{
   158 					TInt leaveCode = iCurrentlyExecutingTransition->GetErrorCode();
   159 					//Check to see if the leave code is NOT on the list of known leaving codes
   160 					if(iLeaveErrorArray.Find(leaveCode) == KErrNotFound)
   161 						{
   162 						iDataLogger.LogInformationWithParameters(KTestLeft, &iUnitTestName);
   163 						iDataLogger.ReportInformationWithParameters(KTestLeft, &iUnitTestName);
   164 						if(iRTest) 
   165 							{
   166 							(*iRTest)(status==KErrNone);
   167 							}
   168 						}
   169 					else	//Leave code is on the list 
   170 						{
   171 						TInt count = iTransitions->Count();
   172 						//Check transition number and if it is the last transition then this is an expected error
   173 						CTransition* lastTransition = (*iTransitions)[count-1];
   174 						if(iCurrentlyExecutingTransition == lastTransition)
   175 							{
   176 							iDataLogger.LogInformationWithParameters(KTestLeftWithExpectedError, &iUnitTestName);
   177 							iDataLogger.ReportInformationWithParameters(KEndingUnitTest, &iUnitTestName);
   178 							}
   179 						else	//Otherwise, if not the last transition, the test failed with an unexpected error
   180 							{
   181 							User::InfoPrint(KInfoPrintFailed);
   182 							User::InfoPrint(iUnitTestName);
   183 							iDataLogger.LogInformationWithParameters(KTestLeftWithUnexpectedError, &iUnitTestName);
   184 							iDataLogger.ReportInformationWithParameters(KTestLeftWithUnexpectedError, &iUnitTestName);
   185 							if(iRTest) 
   186 								{
   187 								(*iRTest)(status==KErrNone);
   188 								}
   189 							}
   190 						}
   191 					}
   192 					break;
   193 
   194 				case (KTestBedFailedPreConditions):
   195 				case (KTestBedFailedPostConditions):
   196 					{
   197 					User::InfoPrint(KInfoPrintFailed);
   198 					User::InfoPrint(iUnitTestName);
   199 					iDataLogger.LogInformationWithParameters(KTestFailed, &iUnitTestName);
   200 					iDataLogger.ReportInformationWithParameters(KTestFailed, &iUnitTestName);
   201 					if(iRTest) 
   202 						{
   203 						(*iRTest)(status==KErrNone);
   204 						}
   205 					}
   206 					break;
   207 
   208 				case (KTestBedTestCancel):
   209 					iDataLogger.LogInformationWithParameters(KTestCancelled, &iUnitTestName);
   210 					iDataLogger.ReportInformationWithParameters(KTestCancelled, &iUnitTestName);
   211 					if(iRTest) 
   212 						{
   213 						(*iRTest)(status==KErrNone);
   214 						}
   215 					break;
   216 
   217 				case (KTestBedLeakTestLoopDetected):
   218 					iDataLogger.LogInformationWithParameters(KTestEnteredInfiniteLoop, &iUnitTestName);
   219 					iDataLogger.ReportInformationWithParameters(KTestEnteredInfiniteLoop, &iUnitTestName);
   220 					if(iRTest) 
   221 						{
   222 						(*iRTest)(status==KErrNone);
   223 						}
   224 					break;
   225 
   226 				case (KErrNone):
   227 					iDataLogger.LogInformationWithParameters(KEndingUnitTest, &iUnitTestName);
   228 					iDataLogger.ReportInformationWithParameters(KEndingUnitTest, &iUnitTestName);
   229 					break;
   230 
   231 				default:
   232 					User::Panic(KUnitTestRunLPanic, KTestBedInvalidStatus);
   233 				}
   234 			}
   235 		}
   236 	}
   237 
   238 EXPORT_C void CUnitTest::AddTransitionL(CTransition* aTransition)
   239 	{
   240 	__ASSERT_DEBUG(aTransition, User::Panic(_L("CUnitTest"), KErrTestBedInvalidTransition));
   241 	CleanupStack::PushL(aTransition);
   242 	User::LeaveIfError(iTransitions->Append(aTransition));
   243 	CleanupStack::Pop(aTransition);
   244 	}
   245 
   246 EXPORT_C void CUnitTest::AddBlockingTransitionL(CTransition* aTransition)
   247 	{
   248 	__ASSERT_DEBUG(aTransition, User::Panic(_L("CUnitTest"), KErrTestBedInvalidTransition));
   249 	aTransition->SetBlockingTransition(ETrue);
   250 	CleanupStack::PushL(aTransition);
   251 	User::LeaveIfError(iTransitions->Append(aTransition));
   252 	CleanupStack::Pop(aTransition);
   253 	}
   254 
   255 EXPORT_C void CUnitTest::AddLeaveErrorCodeL(TInt aLeaveErrorCode)
   256 	{
   257 	User::LeaveIfError(iLeaveErrorArray.Append(aLeaveErrorCode));
   258 	}
   259 
   260 EXPORT_C CTransition& CUnitTest::GetCurrentTransition() const
   261 	{
   262 	// Check fror a stray stub call
   263 	// We will always have a valid pointer here IF called
   264 	// from a stub in response to that transition's call 
   265 	// on the stub's methods.
   266 	__ASSERT_DEBUG(iCurrentlyExecutingTransition, User::Invariant());
   267 	return *iCurrentlyExecutingTransition;
   268 	}
   269 
   270 EXPORT_C void CUnitTest::SetCurrentTransition(CTransition& aTransition)
   271 	{
   272 	iCurrentlyExecutingTransition = &aTransition;
   273 	}
   274 
   275 EXPORT_C void CUnitTest::Complete(CTransition& aTransition, TInt aAsyncPostCheckError)
   276 	{
   277 	// Should never be NULL at this point
   278 	__ASSERT_DEBUG(iCurrentlyExecutingTransition, User::Invariant());
   279 	if(iCurrentlyExecutingTransition == &aTransition)
   280 		iCurrentlyExecutingTransition = NULL;	// Clear the current transition
   281 
   282 	// Oops the code will crash if this is ever false
   283 	__ASSERT_DEBUG(iOutstandingTransitions, User::Invariant());
   284 
   285 	// Look-up the transition passed in...
   286 	TInt index = iOutstandingTransitions->Find(&aTransition);
   287 	if(index != KErrNotFound)
   288 		{
   289 		// ... and remove from the set of outstanding ones.
   290 		iOutstandingTransitions->Remove(index);
   291 
   292 		// Did we fail a second-phase post-condition validation on an asynchronous transition?
   293 		// Or was it a normal transition completion?  Regardless, we go for another iteration 
   294 		// of the AO, passing through the error code.
   295 		TBool completeIt = (aAsyncPostCheckError != KErrNone);
   296 		if(iWaitingForCompletion && (iOutstandingTransitions->Count() == 0))
   297 			completeIt = ETrue;
   298 
   299 		if (completeIt)
   300 			{
   301 			TRequestStatus* status = &iStatus;
   302 			User::RequestComplete(status, aAsyncPostCheckError);
   303 			}
   304 		}
   305 	else
   306 		{
   307 		__ASSERT_DEBUG(ETrue, 
   308 			User::Panic(_L("CUnitTest"), KErrTestBedInvalidTransition));
   309 		}
   310 	}
   311 
   312 EXPORT_C void CUnitTest::SetParametersL(TAny* /*aParams*/)
   313 	{
   314 	// Do nothing
   315 	}
   316 
   317 EXPORT_C void CUnitTest::DoCancel()
   318 	{
   319 	CTimer::DoCancel();
   320 
   321 	if(iCurrentlyExecutingTransition)
   322 		iCurrentlyExecutingTransition->Cancel();
   323 
   324 	// Cancel any outstanding asynchronous transitions
   325 	if(iOutstandingTransitions)
   326 		{
   327 		TInt count = iOutstandingTransitions->Count();
   328 		for(TInt index = 0; index < count; ++index)
   329 			{
   330 			(*iOutstandingTransitions)[index]->Cancel();
   331 			}
   332 		}
   333 
   334 	iUnitTestObserver.Complete(this);
   335 	}
   336