sl@0: // Copyright (c) 2002-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 module contains CScript class sl@0: // sl@0: // sl@0: sl@0: // system includes sl@0: #include sl@0: sl@0: // test system includes sl@0: #include sl@0: #include "Filename.h" sl@0: #include "script.h" sl@0: #include "parseline.h" sl@0: #include "config.h" sl@0: sl@0: #if !defined (__TSU_TESTFRAMEWORK__) sl@0: /** sl@0: * sl@0: * Script files can reference other script files. sl@0: * KMaxDepthRecursion limits the number of references. sl@0: * This is to catch accidental circular references in script files sl@0: * which would otherwise cause the system to continue until all sl@0: * memory had be used making more CScript objects. sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: const TInt KMaxDepthRecursion = 100; sl@0: sl@0: #endif sl@0: sl@0: /** sl@0: * sl@0: * Global data : count of how deep in script files parser is. sl@0: * This is to check against infinite recursion sl@0: * sl@0: * NB : we must patch this out for Unit Testing, where script.cpp sl@0: * is part of a DLL sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: // do not define static if Unit Testing sl@0: #if !defined (__TSU_TESTFRAMEWORK__) sl@0: GLDEF_D TInt CScript::iScriptDepth = 0; sl@0: #endif sl@0: sl@0: /** sl@0: * sl@0: * Console prompts sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: //_LIT(KTxtPressAnyKey,"[press any key to continue]\n"); // EABI warning removal sl@0: //_LIT(KTxtBreakOnError,"The test has failed, press X to terminate this test\n [press any other key to continue]\n"); // EABI warning removal sl@0: sl@0: /** sl@0: * sl@0: * CScript first-phase constructor sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: CScript::CScript() sl@0: { sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * CScript second-phase constructor for a script processor which sl@0: * does not inherit a parser. sl@0: * sl@0: * @param "CTestUtils* aTestUtils" sl@0: * The TestUtils object to use sl@0: * sl@0: * @param "CLog* aLog" sl@0: * The logger to use sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::ConstructL(CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString) sl@0: { sl@0: iLog = aLog; sl@0: iGuardTimer = aGuardTimer; sl@0: sl@0: iMatchString = aMatchString.AllocL(); sl@0: sl@0: iParse = CParseLine::NewL(this, aTestUtils, aLog, aGuardTimer, *iMatchString); sl@0: iParseOwner = ETrue; sl@0: sl@0: iPauseAtEnd = EFalse; sl@0: sl@0: #if !defined (__TSU_TESTFRAMEWORK__) sl@0: iScriptDepth++; sl@0: #endif sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * CScript static constructor for a script processor which sl@0: * does not inherit a parser. sl@0: * sl@0: * @param "CTestUtils* aTestUtils" sl@0: * The TestUtils object to use sl@0: * sl@0: * @param "CLog* aLog" sl@0: * The logger to use sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: CScript* CScript::NewL(CTestUtils* aTestUtils, CLog * aLog, TInt64 aGuardTimer, const TDesC& aMatchString) sl@0: { sl@0: CScript * self = new(ELeave) CScript; sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(aTestUtils, aLog, aGuardTimer, aMatchString); sl@0: CleanupStack::Pop(); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * CScript static constructor for a script processor which sl@0: * does not inherit a parser. sl@0: * sl@0: * @param "CTestUtils* aTestUtils" sl@0: * The TestUtils object to use sl@0: * sl@0: * @param "CLog* aLog" sl@0: * The logger to use sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: CScript* CScript::NewLC(CTestUtils* aTestUtils, CLog * aLog, TInt64 aGuardTimer, const TDesC& aMatchString) sl@0: { sl@0: CScript * self = new(ELeave) CScript; sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(aTestUtils, aLog, aGuardTimer, aMatchString); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * CScript second-phase constructor, for a script processor which sl@0: * inherits a parser. sl@0: * sl@0: * @param "CParseLine* aParse" sl@0: * The parser to use sl@0: * sl@0: * @param "CTestUtils*" sl@0: * Dummy parameter (would be used for constructing a parser); sl@0: * retained to maintain overload distinction sl@0: * sl@0: * @param "CLog * aLog" sl@0: * The logger to use sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::ConstructL(CParseLine* aParse, CTestUtils*, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString) sl@0: { sl@0: iLog = aLog; sl@0: iGuardTimer = aGuardTimer; sl@0: sl@0: iMatchString = aMatchString.AllocL(); // should be the same as that for aParse, for moment don't check sl@0: sl@0: iParse = aParse; sl@0: iParseOwner = EFalse; sl@0: sl@0: iPauseAtEnd = EFalse; sl@0: sl@0: #if !defined (__TSU_TESTFRAMEWORK__) sl@0: iScriptDepth++; sl@0: #endif sl@0: sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * CScript static constructor for a script processor which sl@0: * inherits a parser. sl@0: * sl@0: * @param "CParseLine* aParse" sl@0: * The parser to use sl@0: * sl@0: * @param "CTestUtils* aTestUtils" sl@0: * The TestUtils object to use sl@0: * sl@0: * @param "CLog* aLog" sl@0: * The logger to use sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: CScript* CScript::NewL(CParseLine* aParse, CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString) sl@0: { sl@0: CScript* self = new(ELeave) CScript; sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(aParse, aTestUtils, aLog, aGuardTimer, aMatchString); sl@0: CleanupStack::Pop(); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * CScript static constructor for a script processor which sl@0: * inherits a parser. sl@0: * sl@0: * @param "CParseLine* aParse" sl@0: * The parser to use sl@0: * sl@0: * @param "CTestUtils* aTestUtils" sl@0: * The TestUtils object to use sl@0: * sl@0: * @param "CLog* aLog" sl@0: * The logger to use sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: CScript* CScript::NewLC(CParseLine* aParse, CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString) sl@0: { sl@0: CScript* self = new(ELeave) CScript; sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(aParse, aTestUtils, aLog, aGuardTimer, aMatchString); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * CScript destructor sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: CScript::~CScript() sl@0: { sl@0: // delete parser if we own it sl@0: if(iParseOwner) sl@0: { sl@0: delete iParse; sl@0: iParse = NULL; sl@0: } sl@0: sl@0: // delete scriptbuffer sl@0: delete iScriptBuffer; sl@0: sl@0: delete iMatchString; sl@0: sl@0: #if !defined (__TSU_TESTFRAMEWORK__) sl@0: iScriptDepth--; sl@0: #endif sl@0: } sl@0: sl@0: sl@0: /** sl@0: * sl@0: * Open and read a script file. sl@0: * sl@0: * @param "TFileName aScriptFileName" sl@0: * The script file name sl@0: * sl@0: * @return "TBool" sl@0: * true if script file successfully read sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: #ifdef EXCLUDE_FOR_UNITTEST sl@0: TBool CScript::OpenScriptFile(CFileName* /*aScriptFileName*/) sl@0: { sl@0: // empty function to silence OPT:REF warning under WINS UREL build sl@0: return ETrue; sl@0: } sl@0: #else sl@0: TBool CScript::OpenScriptFile(CFileName* aScriptFileName) sl@0: { sl@0: // get the full pathname default drive name and extension sl@0: _LIT(KRelated,"\\xx.script"); sl@0: TParse parseScriptFileName; sl@0: TInt returnCode = parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, NULL); sl@0: if (returnCode != KErrNone) sl@0: { sl@0: ERR_PRINTF2(_L("Could not set script filename: %S"), &parseScriptFileName.FullName()); sl@0: Pause(); sl@0: return EFalse; sl@0: } sl@0: sl@0: #if !defined (__TSU_TESTFRAMEWORK__) sl@0: if (iScriptDepth > KMaxDepthRecursion) sl@0: { sl@0: // prevent the parser from recursing forever sl@0: ERR_PRINTF2(_L("Script parser aborting: depth:%d"), iScriptDepth); sl@0: return EFalse; sl@0: } sl@0: #if !defined(__WINS__) sl@0: if (iScriptDepth > 3) sl@0: { sl@0: // on target, we are likely to KERN-EXEC 3 if nesting more than 4 levels sl@0: WARN_PRINTF2(_L("Warning : script parser depth = %d"), iScriptDepth); sl@0: } sl@0: #endif sl@0: #endif sl@0: sl@0: // connect to the fileserver sl@0: returnCode = iTheFs.Connect(); sl@0: if (returnCode != KErrNone) sl@0: { sl@0: ERR_PRINTF1(_L("Error trying to connect to the file server") ); sl@0: return EFalse; sl@0: } sl@0: sl@0: sl@0: RFile listfile; sl@0: // have we got a drive letter specified - if not, check all drives sl@0: if (parseScriptFileName.DrivePresent()) sl@0: { sl@0: returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny); sl@0: } sl@0: else sl@0: { sl@0: // checks C, D, E and Z drives - this is ugly, is there a better way of doing this? sl@0: INFO_PRINTF1(_L("Looking for script file on all drives...")); sl@0: _LIT(KDriveC, "C:"); sl@0: parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveC); sl@0: returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny); sl@0: if (returnCode != KErrNone) sl@0: { sl@0: _LIT(KDriveD, "D:"); sl@0: parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveD); sl@0: returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny); sl@0: if (returnCode != KErrNone) sl@0: { sl@0: _LIT(KDriveE, "E:"); sl@0: parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveE); sl@0: returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny); sl@0: if (returnCode != KErrNone) sl@0: { sl@0: _LIT(KDriveZ, "Z:"); sl@0: parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveZ); sl@0: returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: // check if open fails sl@0: if (returnCode != KErrNone) sl@0: { sl@0: parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, NULL); sl@0: ERR_PRINTF2(_L("Failed to open script file : %S"), &parseScriptFileName.FullName()); sl@0: listfile.Close(); sl@0: iTheFs.Close(); sl@0: Pause(); sl@0: return EFalse; sl@0: } sl@0: sl@0: // display the file being processed sl@0: INFO_PRINTF2(_L("Reading script %S"), &parseScriptFileName.FullName()); sl@0: sl@0: // get the script file size sl@0: TInt listfilesize; sl@0: returnCode = listfile.Size(listfilesize); sl@0: if (returnCode != KErrNone) sl@0: { sl@0: ERR_PRINTF2(_L("Failed to read script file: %S size "), &parseScriptFileName.FullName()); sl@0: listfile.Close(); sl@0: iTheFs.Close(); sl@0: return EFalse; sl@0: } sl@0: sl@0: // JW 30-10-02 DEF004555 sl@0: // Buffer was being orphaned if already allocated, where there was more than one sl@0: // script file on the command line sl@0: // Now, we check for this and delete iScriptBuffer if it already exists sl@0: if(iScriptBuffer) sl@0: { sl@0: delete iScriptBuffer; sl@0: iScriptBuffer = NULL; sl@0: } sl@0: sl@0: // get a buffer to read the file into sl@0: TRAPD(err, iScriptBuffer = HBufC8::NewL(listfilesize)); sl@0: if (err != KErrNone || iScriptBuffer == NULL) sl@0: { sl@0: ERR_PRINTF2(_L("Failed to allocate memory for script file %S "), &parseScriptFileName.FullName()); sl@0: listfile.Close(); sl@0: iTheFs.Close(); sl@0: return EFalse; sl@0: } sl@0: sl@0: // get a pointer to the buffer sl@0: TPtr8 ptr = iScriptBuffer->Des(); sl@0: sl@0: // read the file into the buffer sl@0: returnCode = listfile.Read(ptr); sl@0: if (returnCode != KErrNone) sl@0: { sl@0: ERR_PRINTF2(_L("Failed to read script file %S "), &parseScriptFileName.FullName()); sl@0: listfile.Close(); sl@0: iTheFs.Close(); sl@0: return EFalse; sl@0: } sl@0: sl@0: listfile.Close(); sl@0: iTheFs.Close(); sl@0: return ETrue; sl@0: } sl@0: #endif // EXCLUDE_FOR_UNITTEST sl@0: sl@0: /** sl@0: * sl@0: * Parse and execute script file. sl@0: * Assumes script file has been read into iScriptBuffer sl@0: * sl@0: * @return "TVerdict" sl@0: * The script verdict (for logging) sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: TVerdict CScript::ExecuteScriptL() sl@0: { sl@0: // use TLex to decode the script sl@0: TLex8 llex(*iScriptBuffer); sl@0: sl@0: // keep a count of the line number sl@0: TInt8 lineNo = 1; sl@0: sl@0: // loop though processing the rest a line at a time sl@0: while(!llex.Eos()) sl@0: { sl@0: // skip any spaces sl@0: while ( llex.Peek() == ' ' ) sl@0: llex.Inc(); sl@0: sl@0: // mark the start of the line sl@0: llex.Mark(); sl@0: sl@0: // move to the next sl@0: while(!llex.Eos() && llex.Peek() != '\n') sl@0: llex.Inc(); sl@0: sl@0: // step over \n sl@0: if ( llex.Peek() == '\n' ) sl@0: llex.Inc(); sl@0: sl@0: // get the line sl@0: TPtrC8 pline = llex.MarkedToken(); sl@0: if (pline.Length() != 0) sl@0: { sl@0: // and then process sl@0: ProcessLineL(pline, lineNo); sl@0: } sl@0: sl@0: // on to the next line sl@0: lineNo++; sl@0: } sl@0: sl@0: // script processing complete, now return the script verdict sl@0: // Note: the script verdicts are just for the log sl@0: // if no tests failed then return pass for the script sl@0: // this covers scripts which do not test anything sl@0: return (iFail == 0 ? EPass : EFail ); sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * Process a single line from the script file. sl@0: * sl@0: * @param "const TDesC8& aNarrowline" sl@0: * The script line sl@0: * sl@0: * @param "TInt8 lineNo" sl@0: * The script line number sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::ProcessLineL(const TDesC8& aNarrowline, TInt8 aLineNo) sl@0: { sl@0: // call parse to process line sl@0: iParse->ProcessLineL(aNarrowline, aLineNo); sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * Display the accumulated script results. sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::DisplayResults() sl@0: { sl@0: sl@0: INFO_PRINTF1(_L("Test Results Summary ") ); sl@0: INFO_PRINTF1(_L("-------------------- ") ); sl@0: INFO_PRINTF2(_L("Passed :%d"), iPass); sl@0: INFO_PRINTF2(_L("Failed :%d"), iFail); sl@0: INFO_PRINTF2(_L("Inconclusive :%d"), iInconclusive); sl@0: INFO_PRINTF2(_L("Test suite errors :%d"), iTestSuiteError); sl@0: INFO_PRINTF2(_L("Aborted :%d"), iAbort); sl@0: INFO_PRINTF2(_L("KnownFailure :%d"), iKnownFailure); //A new TVerdict sl@0: INFO_PRINTF2(_L("Total :%d"), iTotal); sl@0: sl@0: if(iPauseAtEnd) sl@0: { sl@0: // A pause at the end has been requested sl@0: Pause(); sl@0: } sl@0: sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * Pause testing. sl@0: * NOTE : stubbed pending re-implementation of user input sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::Pause() sl@0: { sl@0: WARN_PRINTF1(_L("Warning : PAUSE not implemented")); sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * Display error on the console and invite abort. sl@0: * NOTE : stubbed pending re-implementation of user input sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: TBool CScript::BreakOnError() sl@0: { sl@0: WARN_PRINTF1(_L("Warning : BREAK_ON_ERROR not implemented")); sl@0: return EFalse; sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * Add a test result to the accumulated totals. sl@0: * sl@0: * @param "TVerdict aTestVerdict" sl@0: * The test verdict sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::AddResult(TVerdict aTestVerdict) sl@0: { sl@0: // another test complete, so increment total sl@0: iTotal++; sl@0: sl@0: // add in the current result sl@0: switch (aTestVerdict) sl@0: { sl@0: case EPass: sl@0: iPass++; sl@0: break; sl@0: case EFail: sl@0: iFail++; sl@0: break; sl@0: case EInconclusive: sl@0: iInconclusive++; sl@0: break; sl@0: case ETestSuiteError: sl@0: iTestSuiteError++; sl@0: break; sl@0: case EAbort: sl@0: iAbort++; sl@0: break; sl@0: case EKnownFailure: //A new TVerdict for a known failed test sl@0: iKnownFailure++; sl@0: break; sl@0: } sl@0: sl@0: // display the result sl@0: TPtrC verdictText = CLog::TestResultText(aTestVerdict); sl@0: TPtrC currentSuiteName = iParse->CurrentSuiteName(); sl@0: TPtrC currentStepName = iParse->CurrentStepName(); sl@0: sl@0: iLog->LogResult(aTestVerdict, _L("Test Result for %S:%S is %S "), sl@0: ¤tSuiteName, ¤tStepName, &verdictText); sl@0: sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * Add a test result from a subscript to the accumulated totals. sl@0: * sl@0: * @param "CScript* aSubScript" sl@0: * The subscript sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::AddResult(CScript* aSubScript) sl@0: { sl@0: sl@0: iPass += aSubScript->iPass; sl@0: iFail += aSubScript->iFail; sl@0: iInconclusive += aSubScript->iInconclusive; sl@0: iTestSuiteError += aSubScript->iTestSuiteError; sl@0: iAbort += aSubScript->iAbort; sl@0: iKnownFailure += aSubScript->iKnownFailure; sl@0: iTotal +=aSubScript->iTotal; sl@0: } sl@0: sl@0: /** sl@0: * sl@0: * Traceable logging function for parseline. sl@0: * sl@0: * @param "const TText8* aFile" sl@0: * Source code file name sl@0: * sl@0: * @param "TInt aLine" sl@0: * Source code line sl@0: * sl@0: * @param "TInt aSeverity" sl@0: * Severity level required to log sl@0: * sl@0: * @param "TRefByValue aFmt" sl@0: * Printf-style format. sl@0: * sl@0: * @param "..." sl@0: * Variable print parameters sl@0: * sl@0: * @xxxx sl@0: * sl@0: */ sl@0: void CScript::LogExtra(const TText8* aFile, TInt aLine, TInt aSeverity, sl@0: TRefByValue aFmt,...) sl@0: { sl@0: VA_LIST aList; sl@0: VA_START(aList, aFmt); sl@0: sl@0: if(aSeverity) sl@0: { sl@0: if(iLog) sl@0: { sl@0: iLog->LogExtra(aFile, aLine, aSeverity, aFmt, aList); sl@0: } sl@0: } sl@0: sl@0: VA_END(aList); sl@0: }