Update contrib.
1 // Copyright (c) 2002-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.
14 // This module contains CScript class
21 // test system includes
22 #include <testframework.h>
25 #include "parseline.h"
28 #if !defined (__TSU_TESTFRAMEWORK__)
31 * Script files can reference other script files.
32 * KMaxDepthRecursion limits the number of references.
33 * This is to catch accidental circular references in script files
34 * which would otherwise cause the system to continue until all
35 * memory had be used making more CScript objects.
40 const TInt KMaxDepthRecursion = 100;
46 * Global data : count of how deep in script files parser is.
47 * This is to check against infinite recursion
49 * NB : we must patch this out for Unit Testing, where script.cpp
55 // do not define static if Unit Testing
56 #if !defined (__TSU_TESTFRAMEWORK__)
57 GLDEF_D TInt CScript::iScriptDepth = 0;
67 //_LIT(KTxtPressAnyKey,"[press any key to continue]\n"); // EABI warning removal
68 //_LIT(KTxtBreakOnError,"The test has failed, press X to terminate this test\n [press any other key to continue]\n"); // EABI warning removal
72 * CScript first-phase constructor
83 * CScript second-phase constructor for a script processor which
84 * does not inherit a parser.
86 * @param "CTestUtils* aTestUtils"
87 * The TestUtils object to use
95 void CScript::ConstructL(CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString)
98 iGuardTimer = aGuardTimer;
100 iMatchString = aMatchString.AllocL();
102 iParse = CParseLine::NewL(this, aTestUtils, aLog, aGuardTimer, *iMatchString);
105 iPauseAtEnd = EFalse;
107 #if !defined (__TSU_TESTFRAMEWORK__)
114 * CScript static constructor for a script processor which
115 * does not inherit a parser.
117 * @param "CTestUtils* aTestUtils"
118 * The TestUtils object to use
120 * @param "CLog* aLog"
126 CScript* CScript::NewL(CTestUtils* aTestUtils, CLog * aLog, TInt64 aGuardTimer, const TDesC& aMatchString)
128 CScript * self = new(ELeave) CScript;
129 CleanupStack::PushL(self);
130 self->ConstructL(aTestUtils, aLog, aGuardTimer, aMatchString);
137 * CScript static constructor for a script processor which
138 * does not inherit a parser.
140 * @param "CTestUtils* aTestUtils"
141 * The TestUtils object to use
143 * @param "CLog* aLog"
149 CScript* CScript::NewLC(CTestUtils* aTestUtils, CLog * aLog, TInt64 aGuardTimer, const TDesC& aMatchString)
151 CScript * self = new(ELeave) CScript;
152 CleanupStack::PushL(self);
153 self->ConstructL(aTestUtils, aLog, aGuardTimer, aMatchString);
159 * CScript second-phase constructor, for a script processor which
162 * @param "CParseLine* aParse"
165 * @param "CTestUtils*"
166 * Dummy parameter (would be used for constructing a parser);
167 * retained to maintain overload distinction
169 * @param "CLog * aLog"
175 void CScript::ConstructL(CParseLine* aParse, CTestUtils*, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString)
178 iGuardTimer = aGuardTimer;
180 iMatchString = aMatchString.AllocL(); // should be the same as that for aParse, for moment don't check
183 iParseOwner = EFalse;
185 iPauseAtEnd = EFalse;
187 #if !defined (__TSU_TESTFRAMEWORK__)
195 * CScript static constructor for a script processor which
198 * @param "CParseLine* aParse"
201 * @param "CTestUtils* aTestUtils"
202 * The TestUtils object to use
204 * @param "CLog* aLog"
210 CScript* CScript::NewL(CParseLine* aParse, CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString)
212 CScript* self = new(ELeave) CScript;
213 CleanupStack::PushL(self);
214 self->ConstructL(aParse, aTestUtils, aLog, aGuardTimer, aMatchString);
221 * CScript static constructor for a script processor which
224 * @param "CParseLine* aParse"
227 * @param "CTestUtils* aTestUtils"
228 * The TestUtils object to use
230 * @param "CLog* aLog"
236 CScript* CScript::NewLC(CParseLine* aParse, CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString)
238 CScript* self = new(ELeave) CScript;
239 CleanupStack::PushL(self);
240 self->ConstructL(aParse, aTestUtils, aLog, aGuardTimer, aMatchString);
253 // delete parser if we own it
260 // delete scriptbuffer
261 delete iScriptBuffer;
265 #if !defined (__TSU_TESTFRAMEWORK__)
273 * Open and read a script file.
275 * @param "TFileName aScriptFileName"
276 * The script file name
279 * true if script file successfully read
284 #ifdef EXCLUDE_FOR_UNITTEST
285 TBool CScript::OpenScriptFile(CFileName* /*aScriptFileName*/)
287 // empty function to silence OPT:REF warning under WINS UREL build
291 TBool CScript::OpenScriptFile(CFileName* aScriptFileName)
293 // get the full pathname default drive name and extension
294 _LIT(KRelated,"\\xx.script");
295 TParse parseScriptFileName;
296 TInt returnCode = parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, NULL);
297 if (returnCode != KErrNone)
299 ERR_PRINTF2(_L("Could not set script filename: %S"), &parseScriptFileName.FullName());
304 #if !defined (__TSU_TESTFRAMEWORK__)
305 if (iScriptDepth > KMaxDepthRecursion)
307 // prevent the parser from recursing forever
308 ERR_PRINTF2(_L("Script parser aborting: depth:%d"), iScriptDepth);
311 #if !defined(__WINS__)
312 if (iScriptDepth > 3)
314 // on target, we are likely to KERN-EXEC 3 if nesting more than 4 levels
315 WARN_PRINTF2(_L("Warning : script parser depth = %d"), iScriptDepth);
320 // connect to the fileserver
321 returnCode = iTheFs.Connect();
322 if (returnCode != KErrNone)
324 ERR_PRINTF1(_L("Error trying to connect to the file server") );
330 // have we got a drive letter specified - if not, check all drives
331 if (parseScriptFileName.DrivePresent())
333 returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny);
337 // checks C, D, E and Z drives - this is ugly, is there a better way of doing this?
338 INFO_PRINTF1(_L("Looking for script file on all drives..."));
340 parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveC);
341 returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny);
342 if (returnCode != KErrNone)
345 parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveD);
346 returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny);
347 if (returnCode != KErrNone)
350 parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveE);
351 returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny);
352 if (returnCode != KErrNone)
355 parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, &KDriveZ);
356 returnCode = listfile.Open(iTheFs, parseScriptFileName.FullName(), EFileRead | EFileShareAny);
362 // check if open fails
363 if (returnCode != KErrNone)
365 parseScriptFileName.Set(aScriptFileName->FileName(), &KRelated, NULL);
366 ERR_PRINTF2(_L("Failed to open script file : %S"), &parseScriptFileName.FullName());
373 // display the file being processed
374 INFO_PRINTF2(_L("Reading script %S"), &parseScriptFileName.FullName());
376 // get the script file size
378 returnCode = listfile.Size(listfilesize);
379 if (returnCode != KErrNone)
381 ERR_PRINTF2(_L("Failed to read script file: %S size "), &parseScriptFileName.FullName());
387 // JW 30-10-02 DEF004555
388 // Buffer was being orphaned if already allocated, where there was more than one
389 // script file on the command line
390 // Now, we check for this and delete iScriptBuffer if it already exists
393 delete iScriptBuffer;
394 iScriptBuffer = NULL;
397 // get a buffer to read the file into
398 TRAPD(err, iScriptBuffer = HBufC8::NewL(listfilesize));
399 if (err != KErrNone || iScriptBuffer == NULL)
401 ERR_PRINTF2(_L("Failed to allocate memory for script file %S "), &parseScriptFileName.FullName());
407 // get a pointer to the buffer
408 TPtr8 ptr = iScriptBuffer->Des();
410 // read the file into the buffer
411 returnCode = listfile.Read(ptr);
412 if (returnCode != KErrNone)
414 ERR_PRINTF2(_L("Failed to read script file %S "), &parseScriptFileName.FullName());
424 #endif // EXCLUDE_FOR_UNITTEST
428 * Parse and execute script file.
429 * Assumes script file has been read into iScriptBuffer
432 * The script verdict (for logging)
437 TVerdict CScript::ExecuteScriptL()
439 // use TLex to decode the script
440 TLex8 llex(*iScriptBuffer);
442 // keep a count of the line number
445 // loop though processing the rest a line at a time
449 while ( llex.Peek() == ' ' )
452 // mark the start of the line
456 while(!llex.Eos() && llex.Peek() != '\n')
460 if ( llex.Peek() == '\n' )
464 TPtrC8 pline = llex.MarkedToken();
465 if (pline.Length() != 0)
468 ProcessLineL(pline, lineNo);
471 // on to the next line
475 // script processing complete, now return the script verdict
476 // Note: the script verdicts are just for the log
477 // if no tests failed then return pass for the script
478 // this covers scripts which do not test anything
479 return (iFail == 0 ? EPass : EFail );
484 * Process a single line from the script file.
486 * @param "const TDesC8& aNarrowline"
489 * @param "TInt8 lineNo"
490 * The script line number
495 void CScript::ProcessLineL(const TDesC8& aNarrowline, TInt8 aLineNo)
497 // call parse to process line
498 iParse->ProcessLineL(aNarrowline, aLineNo);
503 * Display the accumulated script results.
508 void CScript::DisplayResults()
511 INFO_PRINTF1(_L("Test Results Summary ") );
512 INFO_PRINTF1(_L("-------------------- ") );
513 INFO_PRINTF2(_L("Passed :%d"), iPass);
514 INFO_PRINTF2(_L("Failed :%d"), iFail);
515 INFO_PRINTF2(_L("Inconclusive :%d"), iInconclusive);
516 INFO_PRINTF2(_L("Test suite errors :%d"), iTestSuiteError);
517 INFO_PRINTF2(_L("Aborted :%d"), iAbort);
518 INFO_PRINTF2(_L("KnownFailure :%d"), iKnownFailure); //A new TVerdict
519 INFO_PRINTF2(_L("Total :%d"), iTotal);
523 // A pause at the end has been requested
532 * NOTE : stubbed pending re-implementation of user input
537 void CScript::Pause()
539 WARN_PRINTF1(_L("Warning : PAUSE not implemented"));
544 * Display error on the console and invite abort.
545 * NOTE : stubbed pending re-implementation of user input
550 TBool CScript::BreakOnError()
552 WARN_PRINTF1(_L("Warning : BREAK_ON_ERROR not implemented"));
558 * Add a test result to the accumulated totals.
560 * @param "TVerdict aTestVerdict"
566 void CScript::AddResult(TVerdict aTestVerdict)
568 // another test complete, so increment total
571 // add in the current result
572 switch (aTestVerdict)
583 case ETestSuiteError:
589 case EKnownFailure: //A new TVerdict for a known failed test
594 // display the result
595 TPtrC verdictText = CLog::TestResultText(aTestVerdict);
596 TPtrC currentSuiteName = iParse->CurrentSuiteName();
597 TPtrC currentStepName = iParse->CurrentStepName();
599 iLog->LogResult(aTestVerdict, _L("Test Result for %S:%S is %S "),
600 ¤tSuiteName, ¤tStepName, &verdictText);
606 * Add a test result from a subscript to the accumulated totals.
608 * @param "CScript* aSubScript"
614 void CScript::AddResult(CScript* aSubScript)
617 iPass += aSubScript->iPass;
618 iFail += aSubScript->iFail;
619 iInconclusive += aSubScript->iInconclusive;
620 iTestSuiteError += aSubScript->iTestSuiteError;
621 iAbort += aSubScript->iAbort;
622 iKnownFailure += aSubScript->iKnownFailure;
623 iTotal +=aSubScript->iTotal;
628 * Traceable logging function for parseline.
630 * @param "const TText8* aFile"
631 * Source code file name
633 * @param "TInt aLine"
636 * @param "TInt aSeverity"
637 * Severity level required to log
639 * @param "TRefByValue<const TDesC16> aFmt"
640 * Printf-style format.
643 * Variable print parameters
648 void CScript::LogExtra(const TText8* aFile, TInt aLine, TInt aSeverity,
649 TRefByValue<const TDesC16> aFmt,...)
652 VA_START(aList, aFmt);
658 iLog->LogExtra(aFile, aLine, aSeverity, aFmt, aList);