sl@0: // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // This file contains the definition of the class CTransition sl@0: // This file comment is for DOxygen only and is ignored by EDoc. sl@0: // sl@0: // sl@0: sl@0: /** sl@0: @test sl@0: */ sl@0: sl@0: #if defined (_MSC_VER) && (_MSC_VER >= 1000) sl@0: #pragma once sl@0: #endif sl@0: #ifndef _INC_TRANSITION_3A23BFC30021_INCLUDED sl@0: #define _INC_TRANSITION_3A23BFC30021_INCLUDED sl@0: sl@0: #include sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: sl@0: class CDataLogger; sl@0: sl@0: /** sl@0: @internalAll sl@0: Comments : Provide the base class definition for a Unit Test's specific data. This class sl@0: is CBase derived so that test developers can derive from this class and add owned member sl@0: data if they wish. sl@0: */ sl@0: sl@0: class CUnitTestContext : public CBase sl@0: { sl@0: public: sl@0: /** sl@0: @fn CUnitTestContext(CDataLogger& aDataLogger, sl@0: MStateAccessor& aAccessor, sl@0: MTransitionObserver& aObserver) sl@0: Intended Usage : Default constructor. sl@0: @since 7.0 sl@0: @param aDataLogger The output logging object. sl@0: @param aObserver The observer of this UnitTest's Transitions. sl@0: @param aAccessor WhiteBox state access to the class under test. sl@0: @pre None. sl@0: @post CUnitTestContext is fully constructed, and initialized. sl@0: */ sl@0: sl@0: CUnitTestContext(CDataLogger& aDataLogger, sl@0: MStateAccessor& aAccessor, sl@0: MTransitionObserver& aObserver); sl@0: sl@0: /** sl@0: @fn CDataLogger& DataLogger() const sl@0: Intended Usage : Provides access to the CDataLogger sl@0: @since 7.0 sl@0: @return A reference to a CDataLogger sl@0: @pre Object is fully constructed sl@0: */ sl@0: sl@0: CDataLogger& DataLogger() const; sl@0: sl@0: /** sl@0: @fn MTransitionObserver& TransitionObserver() const sl@0: Intended Usage : Provides access to the MTransitionObserver sl@0: @since 7.0 sl@0: @return A reference to an MTransitionObserver sl@0: @pre Object is fully constructed sl@0: */ sl@0: sl@0: MTransitionObserver& TransitionObserver() const; sl@0: sl@0: /** sl@0: @fn MStateAccessor& StateAccessor() const sl@0: Intended Usage : Provides access to the MStateAccessor sl@0: @since 7.0 sl@0: @return A reference to an MStateAccessor sl@0: @pre Object is fully constructed sl@0: */ sl@0: sl@0: MStateAccessor& StateAccessor() const; sl@0: sl@0: protected: sl@0: /** The output logging object. */ sl@0: sl@0: CDataLogger& iDataLogger; sl@0: /** The observer of this UnitTest's Transitions. */ sl@0: sl@0: MTransitionObserver& iObserver; sl@0: /** WhiteBox state access to the class under test. */ sl@0: sl@0: MStateAccessor& iStateAccessor; sl@0: }; sl@0: sl@0: /** sl@0: @internalAll sl@0: @enum TTestBedAsyncState sl@0: Dependencies : None sl@0: Comments : Represents the possible states of an asynchronous transition when sl@0: ValidatePostConditions is called. EAsyncCalled means the async request has been made but sl@0: not satisfied, EAsyncCompleted is when the request has been satisfied. sl@0: */ sl@0: enum TTestBedAsyncState {EAsyncCalled, EAsyncCompleted}; sl@0: sl@0: /** sl@0: @internalAll sl@0: Comments : Placeholder for the functions which validate the state of a transition sl@0: before and after its execution. sl@0: */ sl@0: sl@0: class TTransitionValidator sl@0: { sl@0: public: sl@0: /** sl@0: @fn TTransitionValidator(CUnitTestContext& aUTContext) sl@0: Intended Usage : Standard constructor sl@0: @since 7.0 sl@0: @param aUTContext The context within which this transition is executing sl@0: */ sl@0: sl@0: TTransitionValidator(CUnitTestContext& aUTContext); sl@0: sl@0: /** sl@0: @fn ~TTransitionValidator() sl@0: Intended Usage : Standard Destructor. sl@0: Error Condition : None. sl@0: @since 7.0 sl@0: */ sl@0: sl@0: virtual ~TTransitionValidator(); sl@0: sl@0: /** sl@0: @fn virtual TBool ValidatePreConditions() sl@0: Intended Usage : Overridden in the derived transition to check the transition pre sl@0: conditions. sl@0: @since 7.0 sl@0: @return TBool ETrue if the Preconditions were valid, EFalse if not. sl@0: @pre This CTransition is fully constructed sl@0: @post Returns the validity of the preconditions sl@0: */ sl@0: sl@0: virtual TBool ValidatePreConditions(); sl@0: sl@0: /** sl@0: @fn virtual TBool ValidatePostConditions(TTestBedAsyncState aAsyncState) sl@0: Intended Usage : Overridden in the derived transition to check the transition post sl@0: conditions. sl@0: When overriding, if the transition calls an asynchronous function ValidatePostConditions sl@0: will be called twice. Firstly, after the asynchronous function has been called and, sl@0: secondly, after the asynchronous request has completed. The parameter aAsyncState can sl@0: be used to distinguish between these two cases. sl@0: @since 7.0 sl@0: @param aAsyncState EAsyncCalled if the async function has been just been called, sl@0: EAsyncCompleted if the function has completed. sl@0: @return TBool ETrue if the Postconditions were valid, EFalse if not. sl@0: @pre This CTransition is fully constructed sl@0: @post Unspecified sl@0: */ sl@0: sl@0: virtual TBool ValidatePostConditions(TTestBedAsyncState aAsyncState); sl@0: sl@0: protected: sl@0: /** The context that this transition will be executing in */ sl@0: sl@0: CUnitTestContext& iUTContext; sl@0: }; sl@0: sl@0: /** sl@0: @internalAll sl@0: @struct TTransitionInfo sl@0: Dependencies : None sl@0: Comments : Structure for storing the current transition id and iteration number sl@0: */ sl@0: sl@0: struct TTransitionInfo sl@0: { sl@0: sl@0: TTransitionInfo(const TDesC& aTransitionId, sl@0: CDataLogger& aDataLogger); sl@0: /** Descriptor containing the transition identifier */ sl@0: sl@0: const TDesC& iTransitionId; sl@0: /** The iteration that this transition is currently on */ sl@0: sl@0: TInt iIteration; sl@0: /** The Data Logger that this transition is currently using */ sl@0: sl@0: const CDataLogger& iDataLogger; sl@0: }; sl@0: sl@0: const TInt KFirstTransitionIteration = 1; sl@0: sl@0: /** sl@0: @internalAll sl@0: Comments : Base class from which test developers sl@0: can derive their own transitions for both sl@0: synchronous and asynchronous methods on sl@0: the test class. sl@0: sl@0: The default behaviour is to log its activity, sl@0: and claim that the pre and post conditions are true. sl@0: The TransitMethodL implementation does nothing. sl@0: To write a complete test class method transition sl@0: override: sl@0: sl@0: 1. The c'tor, (To obtain the test class reference, sl@0: (or pointer reference for a NewL transit), sl@0: and any parameters for the TransitMethodL). sl@0: sl@0: 2. SetStartStateL() to set the test class state, sl@0: Using the TStateAccessor provided in the c'tor. sl@0: sl@0: 3. ValidatePreConditions() to test the test class state sl@0: is valid. sl@0: sl@0: 4. TransitMethodL() to define the transition behaviour, sl@0: calling the method to test with the appropriate parameters. sl@0: sl@0: 5. ValidatePostConditions(), to check the end state is valid. sl@0: sl@0: The simplest case of implementation is to derive an empty sl@0: class, that relies entirely upon the default behaviour, sl@0: and implement the remaining methods as appropriate. sl@0: (See the the EXAMPLE implementation.) sl@0: */ sl@0: sl@0: class CTransition : public CActive sl@0: { sl@0: public: sl@0: /** sl@0: @fn IMPORT_C ~CTransition() sl@0: Intended Usage : Standardized virtual destruction method sl@0: @since 7.0 sl@0: @pre The CTransition exists sl@0: @post The object has been destroyed sl@0: */ sl@0: sl@0: IMPORT_C ~CTransition(); sl@0: sl@0: /** sl@0: @fn virtual void SetStartStateL() sl@0: Intended Usage : Sets the state of the test object to that specified sl@0: Error Condition : sl@0: @since 7.0 sl@0: @pre The CTransition is instantiated sl@0: @post The CTransition is in the state specified sl@0: */ sl@0: sl@0: IMPORT_C virtual void SetStartStateL(); sl@0: sl@0: /** sl@0: @fn IMPORT_C const TDesC& TransitionId() const sl@0: Intended Usage : Returns the transition identifier sl@0: Error Condition : sl@0: @since 7.0 sl@0: @return const TDesC& The identifier of this transition sl@0: @pre The CTransition has been instantiated sl@0: */ sl@0: sl@0: IMPORT_C const TDesC& TransitionId() const; sl@0: sl@0: /** sl@0: @fn IMPORT_C const TTransitionInfo& TransitionInfo() const sl@0: Intended Usage : Get information on this transition. Return struct contains the sl@0: transition ID, the current iteration and the logging mechanism used by this transition sl@0: Error Condition : sl@0: @since 7.0 sl@0: @return const TTransitionInfo Information on the current transition sl@0: @pre This CTransition is initialized sl@0: */ sl@0: sl@0: IMPORT_C const TTransitionInfo& TransitionInfo() const; sl@0: sl@0: /** sl@0: @fn virtual void RunTransition(TRequestStatus* aUnitTestStatus) sl@0: Intended Usage : Indicates that this transition should be run as soon as possible. sl@0: This function calls ValidatePreConditions to ensure the object is in the appropriate start state sl@0: and then sets the transition active causing its RunL function to be called. sl@0: Error Condition : sl@0: @since 7.0 sl@0: @param aUnitTestStatus Status word of the calling CUnitTest sl@0: @pre The test object is in an appropriate state for this transition sl@0: @post The test object is in the end state of this transition sl@0: */ sl@0: sl@0: IMPORT_C virtual void RunTransition(TRequestStatus* aUnitTestStatus); sl@0: sl@0: /** sl@0: @fn IMPORT_C void RepeatOnce() sl@0: Intended Usage : Flag this transition to be repeated sl@0: @since 7.0 sl@0: @pre None sl@0: @post The iRepeatThis flag is set to true. sl@0: */ sl@0: sl@0: IMPORT_C void RepeatOnce(); sl@0: sl@0: /** sl@0: @fn IMPORT_C TBool IsBlockingTransition() const sl@0: Intended Usage : A blocking transition is one which does not run until all sl@0: outstanding asynchronous transitions have completed. This function returns sl@0: whether this is a blocking transition. sl@0: This function does not need to be used by test developers as it is taken sl@0: care of in the transition and unittest base classes. Blocking transitions sl@0: can be added using AddBlockingTransitionL() in CUnitTest::ConstructL(). sl@0: @since 7.0 sl@0: @return TBool Flag indicating whether this is a blocking transition or not. sl@0: @pre None sl@0: */ sl@0: sl@0: IMPORT_C TBool IsBlockingTransition() const; sl@0: sl@0: /** sl@0: @fn IMPORT_C void SetBlockingTransition(TBool aBlocking) sl@0: Intended Usage : A blocking transition is one which does not run until all sl@0: outstanding asynchronous transitions have completed. sl@0: This function does not need to be used by test developers as it is taken sl@0: care of in the transition and unittest base classes. Blocking transitions sl@0: can be added using AddBlockingTransitionL() in CUnitTest::ConstructL(). sl@0: @since 7.0 sl@0: @param aBlocking Pass ETrue to set this as a blocking transition sl@0: @pre This object is fully constructed sl@0: @post The blocking status of this CTransition is set sl@0: */ sl@0: sl@0: IMPORT_C void SetBlockingTransition(TBool aBlocking); sl@0: sl@0: /** sl@0: @fn GetErrorCode() const sl@0: Intended Usage : Returns the error code last recorded by the transition. sl@0: @since 7.0 sl@0: @return The error code that was last recorded during the transition execution. sl@0: @pre This object is fully constructed. sl@0: */ sl@0: sl@0: IMPORT_C TInt GetErrorCode() const; sl@0: sl@0: protected: sl@0: /** sl@0: @fn CTransition(const TDesC& aTransactionId, sl@0: CUnitTestContext& aUTContext, sl@0: TTransitionValidator& aValidator) sl@0: Intended Usage : Default Constructor sl@0: @param aTransitionId The identifier of this transition sl@0: @param aUTContext The context within which this transition should run sl@0: @param aValidator Provides Pre & Post condition validation sl@0: @since 7.0 sl@0: @pre None sl@0: @post First phase of two phase construction is complete sl@0: */ sl@0: IMPORT_C CTransition(const TDesC& aTransitionId, sl@0: CUnitTestContext& aUTContext, sl@0: TTransitionValidator& aValidator); sl@0: sl@0: /** sl@0: @fn virtual void DoCancel() sl@0: Intended Usage : Standard Active Object method for cancelling the current request sl@0: Error Condition : sl@0: @since 7.0 sl@0: @pre None sl@0: @post Any outstanding requests are cancelled sl@0: */ sl@0: sl@0: IMPORT_C virtual void DoCancel(); sl@0: sl@0: /** sl@0: @fn RunL() sl@0: Intended Usage : Implementation of CActive pure virtual method. For each transition execution sl@0: RunL is performed twice. sl@0: The first time it calls TransitMethodL() which is the developer defined function which executes the sl@0: test method. It then immediately sets itself active and attempts to complete. If TransitMethodL() sl@0: called a synchronous function then iStatus will not be KRequestPending and the RunL will be called as sl@0: soon as it is scheduled. If TransitMethodL() calls an asyncronous function then it will pass iStatus to the function and RunL sl@0: will be called when the asynchronous function completes. sl@0: Whichever method is used the second call of RunL() will complete the request of the CUnitTest which sl@0: called this transition by setting its status word. If it has been requested that this transition is sl@0: repeated then the status word will be set to KTestBedRepeatTest. sl@0: Error Condition : sl@0: @since 7.0 sl@0: @pre RunTransition() ensures the preconditions for this function sl@0: @post After 1st run - test method has been called sl@0: After 2nd run - Unittest has been completed. sl@0: */ sl@0: sl@0: IMPORT_C virtual void RunL(); sl@0: sl@0: /** sl@0: @fn TransitMethodL() sl@0: Intended Usage : Calls the method on the class being tested. If the call is to sl@0: an asynchronous service provider you must set iStatus to KRequestPending as usual sl@0: Error Condition : sl@0: @since 7.0 sl@0: @pre Preconditions for this transition have been validated sl@0: @post The test method has been executed sl@0: */ sl@0: sl@0: IMPORT_C virtual void TransitMethodL() = 0; sl@0: sl@0: /** sl@0: @fn RunError(TInt anErrorCode) sl@0: Intended Usage : Called by the active scheduler if RunL leaves sl@0: @since 7.0 sl@0: @param anErrorCode The error code which RunL left with sl@0: @return KErrNone because if a CTransition::RunL leaves then we simply log sl@0: the event and inform the CUnitTest. sl@0: @pre This function is only called by the active scheduler if RunL leaves. sl@0: @post The owning CUnitTest has been informed that this CTransition left sl@0: */ sl@0: sl@0: IMPORT_C virtual TInt RunError(TInt aErrorCode); sl@0: sl@0: /** sl@0: @fn PostTransitionCleanup() sl@0: Intended Usage : Called from the RunL immediately after TransitMethodL() executes sl@0: @since 7.0 sl@0: @pre Must be called from RunL only after TransitMethodL() has completed. sl@0: @post Default behaviour is to do nothing. sl@0: See the derived classes for additional functionality. sl@0: */ sl@0: sl@0: IMPORT_C virtual void PostTransitionCleanup(); sl@0: sl@0: protected: sl@0: /** The transition identifier*/ sl@0: sl@0: const TDesC& iTransitionId; sl@0: /** The context in which this transition is running (logging, state accessor & observer) */ sl@0: sl@0: CUnitTestContext& iUTContext; sl@0: /** Used for checking the state of the test object - provides pre and post condition sl@0: validation functions */ sl@0: sl@0: TTransitionValidator& iValidator; sl@0: /** Indicates that this transition should be re-run after the current run */ sl@0: sl@0: TBool iRepeatThis; sl@0: /** The iStatus of the UnitTest which owns me so that I can complete it */ sl@0: sl@0: TRequestStatus* iUnitTestStatus; sl@0: /** Indicates that the transition has executed and can be completed */ sl@0: sl@0: TBool iTransitionFinished; sl@0: /** indicates the code that the RunL left with when completing with a KErrTestBedLeft */ sl@0: sl@0: TInt iLeaveError; sl@0: /** Indicates that this transition should not execute until all previous async transitions sl@0: have finished */ sl@0: sl@0: TBool iBlockingTransition; sl@0: /** Indicates that this transition is an asynchronous transition*/ sl@0: TBool iAsyncTransition; sl@0: /** Information on this transition including its ID, its iteration number and the data logger sl@0: it is currently using */ sl@0: sl@0: TTransitionInfo iTransitionInfo; sl@0: sl@0: /** A friend for self-test operation */ sl@0: friend class TTransition_StateAccessor; sl@0: }; sl@0: sl@0: #include sl@0: sl@0: #endif