VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testdriver/reporter.py@ 75713

Last change on this file since 75713 was 72694, checked in by vboxsync, 7 years ago

ValKit/reporter.py: python 3.x fix for local logger.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 60.4 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: reporter.py 72694 2018-06-26 13:58:57Z vboxsync $
3# pylint: disable=C0302
4
5"""
6Testdriver reporter module.
7"""
8
9from __future__ import print_function;
10
11__copyright__ = \
12"""
13Copyright (C) 2010-2017 Oracle Corporation
14
15This file is part of VirtualBox Open Source Edition (OSE), as
16available from http://www.virtualbox.org. This file is free software;
17you can redistribute it and/or modify it under the terms of the GNU
18General Public License (GPL) as published by the Free Software
19Foundation, in version 2 as it comes in the "COPYING" file of the
20VirtualBox OSE distribution. VirtualBox OSE is distributed in the
21hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
22
23The contents of this file may alternatively be used under the terms
24of the Common Development and Distribution License Version 1.0
25(CDDL) only, as it comes in the "COPYING.CDDL" file of the
26VirtualBox OSE distribution, in which case the provisions of the
27CDDL are applicable instead of those of the GPL.
28
29You may elect to license modified versions of this file under the
30terms and conditions of either the GPL or the CDDL or both.
31"""
32__version__ = "$Revision: 72694 $"
33
34
35# Standard Python imports.
36import array
37import datetime
38import errno
39import gc
40import os
41import os.path
42import sys
43import time
44import threading
45import traceback
46
47# Validation Kit imports.
48from common import utils;
49
50## test reporter instance
51g_oReporter = None; # type: ReporterBase
52g_sReporterName = None;
53
54
55class ReporterLock(object):
56 """
57 Work around problem with garbage collection triggering __del__ method with
58 logging while inside the logger lock and causing a deadlock.
59 """
60
61 def __init__(self, sName):
62 self.sName = sName;
63 self.oLock = threading.RLock();
64 self.oOwner = None;
65 self.cRecursion = 0;
66 self.fRestoreGC = False;
67
68 def acquire(self):
69 """ Acquire the lock. """
70 oSelf = threading.current_thread();
71
72 # Take the lock.
73 if not self.oLock.acquire():
74 return False;
75
76 self.oOwner = oSelf;
77 self.cRecursion += 1;
78
79 # Disable GC to avoid __del__ w/ log statement randomly reenter the logger.
80 if self.cRecursion == 1:
81 self.fRestoreGC = gc.isenabled();
82 if self.fRestoreGC:
83 gc.disable();
84
85 return True;
86
87 def release(self):
88 """ Release the lock. """
89 oSelf = threading.current_thread();
90
91 # Check the ownership.
92 if oSelf != self.oOwner:
93 raise threading.ThreadError();
94
95 # Drop one recursion.
96 self.cRecursion -= 1;
97 if self.cRecursion <= 0:
98
99 # Final recursion. Clear owner and re-enable GC.
100 self.oOwner = None;
101 if self.fRestoreGC:
102 self.fRestoreGC = False;
103 gc.enable();
104
105 self.oLock.release();
106
107## Reporter lock.
108g_oLock = ReporterLock('reporter');
109
110
111
112class PythonLoggingStream(object):
113 """
114 Python logging => testdriver/reporter.py stream.
115 """
116
117 def write(self, sText):
118 """Writes python log message to our stream."""
119 if g_oReporter != None:
120 sText = sText.rstrip("\r\n");
121 #g_oReporter.log(0, 'python: %s' % (sText), utils.getCallerName(), utils.getTimePrefix());
122 return True;
123
124 def flush(self):
125 """Flushes the stream."""
126 return True;
127
128
129class ReporterBase(object):
130 """
131 Base class for the reporters.
132 """
133
134 def __init__(self):
135 self.iVerbose = 1;
136 self.iDebug = 0;
137 self.cErrors = 0;
138 self.fTimedOut = False; # Once set, it trickles all the way up.
139 self.atTests = [];
140 self.sName = os.path.splitext(os.path.basename(sys.argv[0]))[0];
141
142 # Hook into the python logging.
143 import logging;
144 logging.basicConfig(stream = PythonLoggingStream(),
145 level = logging.DEBUG,
146 format = '%(name)-12s %(levelname)-8s %(message)s');
147 #
148 # Introspection and configuration.
149 #
150
151 def isLocal(self):
152 """Is this a local reporter?"""
153 return False;
154
155 def incVerbosity(self):
156 """Increases the verbosity level."""
157 self.iVerbose += 1;
158
159 def incDebug(self):
160 """Increases the debug level."""
161 self.iDebug += 1;
162
163 def appendToProcessName(self, sAppend):
164 """
165 Appends sAppend to the base process name.
166 Returns the new process name.
167 """
168 self.sName = os.path.splitext(os.path.basename(sys.argv[0]))[0] + sAppend;
169 return self.sName;
170
171
172 #
173 # Generic logging.
174 #
175
176 def log(self, iLevel, sText, sCaller, sTsPrf):
177 """
178 Writes the specfied text to the log if iLevel is less or requal
179 to iVerbose.
180 """
181 _ = iLevel; _ = sText; _ = sCaller; _ = sTsPrf;
182 return 0;
183
184 #
185 # XML output from the reporter.
186 #
187
188 def _xmlEscAttr(self, sValue):
189 """Escapes an XML attribute value."""
190 sValue = sValue.replace('&', '&amp;');
191 sValue = sValue.replace('<', '&lt;');
192 sValue = sValue.replace('>', '&gt;');
193 #sValue = sValue.replace('\'', '&apos;');
194 sValue = sValue.replace('"', '&quot;');
195 sValue = sValue.replace('\n', '&#xA');
196 sValue = sValue.replace('\r', '&#xD');
197 return sValue;
198
199 def _xmlWrite(self, asText, fIndent = True):
200 """XML output function for the reporter."""
201 _ = asText; _ = fIndent;
202 return None;
203
204 def xmlFlush(self, fRetry = False, fForce = False):
205 """Flushes XML output if buffered."""
206 _ = fRetry; _ = fForce;
207 return None;
208
209 #
210 # XML output from child.
211 #
212
213 def subXmlStart(self, oFileWrapper):
214 """Called by the file wrapper when the first bytes are written to the test pipe."""
215 _ = oFileWrapper;
216 return None;
217
218 def subXmlWrite(self, oFileWrapper, sRawXml, sCaller):
219 """Called by the file wrapper write method for test pipes."""
220 return self.log(0, 'raw xml%s: %s' % (oFileWrapper.sPrefix, sRawXml), sCaller, utils.getTimePrefix());
221
222 def subXmlEnd(self, oFileWrapper):
223 """Called by the file wrapper __del__ method for test pipes."""
224 _ = oFileWrapper;
225 return None;
226
227 #
228 # File output.
229 #
230
231 def addLogFile(self, oSrcFile, sSrcFilename, sAltName, sDescription, sKind, sCaller, sTsPrf):
232 """
233 Adds the file to the report.
234 Returns True on success, False on failure.
235 """
236 _ = oSrcFile; _ = sSrcFilename; _ = sAltName; _ = sDescription; _ = sKind; _ = sCaller; _ = sTsPrf;
237 return True;
238
239 def addLogString(self, sLog, sLogName, sDescription, sKind, sCaller, sTsPrf):
240 """
241 Adds the file to the report.
242 Returns True on success, False on failure.
243 """
244 _ = sLog; _ = sLogName; _ = sDescription; _ = sKind; _ = sCaller; _ = sTsPrf;
245 return True;
246
247 #
248 # Test reporting
249 #
250
251 def _testGetFullName(self):
252 """
253 Mangles the test names in atTest into a single name to make it easier
254 to spot where we are.
255 """
256 sName = '';
257 for t in self.atTests:
258 if sName != '':
259 sName += ', ';
260 sName += t[0];
261 return sName;
262
263 def testIncErrors(self):
264 """Increates the error count."""
265 self.cErrors += 1;
266 return self.cErrors;
267
268 def testSetTimedOut(self):
269 """Sets time out indicator for the current test and increases the error counter."""
270 self.fTimedOut = True;
271 self.cErrors += 1;
272 return None;
273
274 def testStart(self, sName, sCaller):
275 """ Starts a new test, may be nested. """
276 (sTsPrf, sTsIso) = utils.getTimePrefixAndIsoTimestamp();
277 self._xmlWrite([ '<Test timestamp="%s" name="%s">' % (sTsIso, self._xmlEscAttr(sName),), ]);
278 self.atTests.append((sName, self.cErrors, self.fTimedOut));
279 self.fTimedOut = False;
280 return self.log(1, ' %-50s: TESTING' % (self._testGetFullName()), sCaller, sTsPrf);
281
282 def testValue(self, sName, sValue, sUnit, sCaller):
283 """ Reports a benchmark value or something simiarlly useful. """
284 (sTsPrf, sTsIso) = utils.getTimePrefixAndIsoTimestamp();
285 self._xmlWrite([ '<Value timestamp="%s" name="%s" unit="%s" value="%s"/>'
286 % (sTsIso, self._xmlEscAttr(sName), self._xmlEscAttr(sUnit), self._xmlEscAttr(sValue)), ]);
287 return self.log(0, '** %-48s: %12s %s' % (sName, sValue, sUnit), sCaller, sTsPrf);
288
289 def testFailure(self, sDetails, sCaller):
290 """ Reports a failure. """
291 (sTsPrf, sTsIso) = utils.getTimePrefixAndIsoTimestamp();
292 self.cErrors = self.cErrors + 1;
293 self._xmlWrite([ '<FailureDetails timestamp="%s" text="%s"/>' % (sTsIso, self._xmlEscAttr(sDetails),), ]);
294 return self.log(0, sDetails, sCaller, sTsPrf);
295
296 def testDone(self, fSkipped, sCaller):
297 """
298 Marks the current test as DONE, pops it and maks the next test on the
299 stack current.
300 Returns (name, errors).
301 """
302 (sTsPrf, sTsIso) = utils.getTimePrefixAndIsoTimestamp();
303 sFullName = self._testGetFullName();
304
305 # safe pop
306 if len(self.atTests) <= 0:
307 self.log(0, 'testDone on empty test stack!', sCaller, sTsPrf);
308 return ('internal error', 0);
309 fTimedOut = self.fTimedOut;
310 sName, cErrorsStart, self.fTimedOut = self.atTests.pop();
311
312 # log + xml.
313 cErrors = self.cErrors - cErrorsStart;
314 if cErrors == 0:
315 if fSkipped is not True:
316 self._xmlWrite([ ' <Passed timestamp="%s"/>' % (sTsIso,), '</Test>' ],);
317 self.log(1, '** %-50s: PASSED' % (sFullName,), sCaller, sTsPrf);
318 else:
319 self._xmlWrite([ ' <Skipped timestamp="%s"/>' % (sTsIso,), '</Test>' ]);
320 self.log(1, '** %-50s: SKIPPED' % (sFullName,), sCaller, sTsPrf);
321 elif fTimedOut:
322 self._xmlWrite([ ' <TimedOut timestamp="%s" errors="%d"/>' % (sTsIso, cErrors), '</Test>' ]);
323 self.log(0, '** %-50s: TIMED-OUT - %d errors' % (sFullName, cErrors), sCaller, sTsPrf);
324 else:
325 self._xmlWrite([ ' <Failed timestamp="%s" errors="%d"/>' % (sTsIso, cErrors), '</Test>' ]);
326 self.log(0, '** %-50s: FAILED - %d errors' % (sFullName, cErrors), sCaller, sTsPrf);
327
328 # Flush buffers when reaching the last test.
329 if not self.atTests:
330 self.xmlFlush(fRetry = True);
331
332 return (sName, cErrors);
333
334 def testErrorCount(self):
335 """
336 Returns the number of errors accumulated by the current test.
337 """
338 cTests = len(self.atTests);
339 if cTests <= 0:
340 return self.cErrors;
341 return self.cErrors - self.atTests[cTests - 1][1];
342
343 def testCleanup(self, sCaller):
344 """
345 Closes all open test as failed.
346 Returns True if no open tests, False if there were open tests.
347 """
348 if not self.atTests:
349 return True;
350 for _ in range(len(self.atTests)):
351 self.testFailure('Test not closed by test drver', sCaller)
352 self.testDone(False, sCaller);
353 return False;
354
355 #
356 # Misc.
357 #
358
359 def doPollWork(self, sDebug = None):
360 """
361 Check if any pending stuff expired and needs doing.
362 """
363 _ = sDebug;
364 return None;
365
366
367
368
369class LocalReporter(ReporterBase):
370 """
371 Local reporter instance.
372 """
373
374 def __init__(self):
375 ReporterBase.__init__(self);
376 self.oLogFile = None;
377 self.oXmlFile = None;
378 self.fXmlOk = True;
379 self.iSubXml = 0;
380 self.iOtherFile = 0;
381 self.fnGetIsoTimestamp = utils.getIsoTimestamp; # Hack to get a timestamp in __del__.
382 self.oStdErr = sys.stderr; # Hack for __del__ output.
383
384 #
385 # Figure the main log directory.
386 #
387 try:
388 self.sDefLogDir = os.path.abspath(os.path.expanduser(os.path.join('~', 'VBoxTestLogs')));
389 except:
390 self.sDefLogDir = os.path.abspath("VBoxTestLogs");
391 try:
392 sLogDir = os.path.abspath(os.environ.get('TESTBOX_REPORTER_LOG_DIR', self.sDefLogDir));
393 if not os.path.isdir(sLogDir):
394 os.makedirs(sLogDir, 0o750);
395 except:
396 sLogDir = self.sDefLogDir;
397 if not os.path.isdir(sLogDir):
398 os.makedirs(sLogDir, 0o750);
399
400 #
401 # Make a subdirectory for this test run.
402 #
403 sTs = datetime.datetime.utcnow().strftime('%Y-%m-%dT%H-%M-%S.log');
404 self.sLogDir = sLogDir = os.path.join(sLogDir, '%s-%s' % (sTs, self.sName));
405 try:
406 os.makedirs(self.sLogDir, 0o750);
407 except:
408 self.sLogDir = '%s-%s' % (self.sLogDir, os.getpid());
409 os.makedirs(self.sLogDir, 0o750);
410
411 #
412 # Open the log file and write a header.
413 #
414 sLogName = os.path.join(self.sLogDir, 'testsuite.log');
415 sTsIso = utils.getIsoTimestamp();
416 if sys.version_info[0] >= 3: # Add 'b' to prevent write taking issue with encode('utf-8') not returning a string.
417 self.oLogFile = utils.openNoInherit(sLogName, "wb");
418 else:
419 self.oLogFile = utils.openNoInherit(sLogName, "w");
420 self.oLogFile.write(('Created log file at %s.\nRunning: %s' % (sTsIso, sys.argv)).encode('utf-8'));
421
422 #
423 # Open the xml log file and write the mandatory introduction.
424 #
425 # Note! This is done here and not in the base class because the remote
426 # logger doesn't really need this. It doesn't need the outer
427 # test wrapper either.
428 #
429 sXmlName = os.path.join(self.sLogDir, 'testsuite.xml');
430 if sys.version_info[0] >= 3: # Add 'b' to prevent write taking issue with encode('utf-8') not returning a string.
431 self.oXmlFile = utils.openNoInherit(sXmlName, "wb");
432 else:
433 self.oXmlFile = utils.openNoInherit(sXmlName, "w");
434 self._xmlWrite([ '<?xml version="1.0" encoding="UTF-8" ?>',
435 '<Test timestamp="%s" name="%s">' % (sTsIso, self._xmlEscAttr(self.sName),), ],
436 fIndent = False);
437
438 def __del__(self):
439 """Ends and completes the log files."""
440 try: sTsIso = self.fnGetIsoTimestamp();
441 except Exception as oXcpt:
442 sTsIso = str(oXcpt);
443
444 if self.oLogFile is not None:
445 try:
446 self.oLogFile.write(('\nThe End %s\n' % (sTsIso,)).encode('utf-8'));
447 self.oLogFile.close();
448 except: pass;
449 self.oLogFile = None;
450
451 if self.oXmlFile is not None:
452 self._closeXml(sTsIso);
453 self.oXmlFile = None;
454
455 def _closeXml(self, sTsIso):
456 """Closes the XML file."""
457 if self.oXmlFile is not None:
458 # pop the test stack
459 while self.atTests:
460 sName, cErrorsStart, self.fTimedOut = self.atTests.pop();
461 self._xmlWrite([ '<End timestamp="%s" errors="%d"/>' % (sTsIso, self.cErrors - cErrorsStart,),
462 '</%s>' % (sName,), ]);
463
464 # The outer one is not on the stack.
465 self._xmlWrite([ ' <End timestamp="%s"/>' % (sTsIso,),
466 '</Test>', ], fIndent = False);
467 try:
468 self.oXmlFile.close();
469 self.oXmlFile = None;
470 except:
471 pass;
472
473 def _xmlWrite(self, asText, fIndent = True):
474 """Writes to the XML file."""
475 for sText in asText:
476 if fIndent:
477 sIndent = ''.ljust((len(self.atTests) + 1) * 2);
478 sText = sIndent + sText;
479 sText += '\n';
480
481 try:
482 self.oXmlFile.write(sText.encode('utf-8'));
483 except:
484 if self.fXmlOk:
485 traceback.print_exc();
486 self.fXmlOk = False;
487 return False;
488 return True;
489
490 #
491 # Overridden methods.
492 #
493
494 def isLocal(self):
495 """Is this a local reporter?"""
496 return True;
497
498 def log(self, iLevel, sText, sCaller, sTsPrf):
499 if iLevel <= self.iVerbose:
500 # format it.
501 if self.iDebug > 0:
502 sLogText = '%s %30s: %s' % (sTsPrf, sCaller, sText);
503 else:
504 sLogText = '%s %s' % (sTsPrf, sText);
505
506 # output it.
507 if sys.version_info[0] >= 3:
508 sAscii = sLogText;
509 else:
510 sAscii = sLogText.encode('ascii', 'replace');
511 if self.iDebug == 0:
512 print('%s: %s' % (self.sName, sAscii), file = self.oStdErr);
513 else:
514 print('%s' % (sAscii), file = self.oStdErr);
515 sLogText += '\n';
516 try:
517 self.oLogFile.write(sLogText.encode('utf-8'));
518 except:
519 pass;
520 return 0;
521
522 def addLogFile(self, oSrcFile, sSrcFilename, sAltName, sDescription, sKind, sCaller, sTsPrf):
523 # Figure the destination filename.
524 iOtherFile = self.iOtherFile;
525 self.iOtherFile += 1;
526 sDstFilename = os.path.join(self.sLogDir, 'other-%d-%s.log' \
527 % (iOtherFile, os.path.splitext(os.path.basename(sSrcFilename))[0]));
528 self.log(0, '** Other log file: %s - %s (%s)' % (sDstFilename, sDescription, sSrcFilename), sCaller, sTsPrf);
529
530 # Open the destination file and copy over the data.
531 fRc = True;
532 try:
533 oDstFile = utils.openNoInherit(sDstFilename, 'wb');
534 except Exception as oXcpt:
535 self.log(0, 'error opening %s: %s' % (sDstFilename, oXcpt), sCaller, sTsPrf);
536 else:
537 while True:
538 try:
539 abBuf = oSrcFile.read(65536);
540 except Exception as oXcpt:
541 fRc = False;
542 self.log(0, 'error reading %s: %s' % (sSrcFilename, oXcpt), sCaller, sTsPrf);
543 else:
544 try:
545 oDstFile.write(abBuf);
546 except Exception as oXcpt:
547 fRc = False;
548 self.log(0, 'error writing %s: %s' % (sDstFilename, oXcpt), sCaller, sTsPrf);
549 else:
550 if abBuf:
551 continue;
552 break;
553 oDstFile.close();
554
555 # Leave a mark in the XML log.
556 self._xmlWrite(['<LogFile timestamp="%s" filename="%s" source="%s" kind="%s" ok="%s">%s</LogFile>\n'
557 % (utils.getIsoTimestamp(), self._xmlEscAttr(os.path.basename(sDstFilename)), self._xmlEscAttr(sSrcFilename), \
558 self._xmlEscAttr(sKind), fRc, self._xmlEscAttr(sDescription))] );
559 _ = sAltName;
560 return fRc;
561
562 def addLogString(self, sLog, sLogName, sDescription, sKind, sCaller, sTsPrf):
563 # Figure the destination filename.
564 iOtherFile = self.iOtherFile;
565 self.iOtherFile += 1;
566 sDstFilename = os.path.join(self.sLogDir, 'other-%d-%s.log' \
567 % (iOtherFile, os.path.splitext(os.path.basename(sLogName))[0]));
568 self.log(0, '** Other log file: %s - %s (%s)' % (sDstFilename, sDescription, sLogName), sCaller, sTsPrf);
569
570 # Open the destination file and copy over the data.
571 fRc = True;
572 try:
573 oDstFile = utils.openNoInherit(sDstFilename, 'w');
574 except Exception as oXcpt:
575 self.log(0, 'error opening %s: %s' % (sDstFilename, oXcpt), sCaller, sTsPrf);
576 else:
577 try:
578 oDstFile.write(sLog);
579 except Exception as oXcpt:
580 fRc = False;
581 self.log(0, 'error writing %s: %s' % (sDstFilename, oXcpt), sCaller, sTsPrf);
582
583 oDstFile.close();
584
585 # Leave a mark in the XML log.
586 self._xmlWrite(['<LogFile timestamp="%s" filename="%s" source="%s" kind="%s" ok="%s">%s</LogFile>\n'
587 % (utils.getIsoTimestamp(), self._xmlEscAttr(os.path.basename(sDstFilename)), self._xmlEscAttr(sLogName), \
588 self._xmlEscAttr(sKind), fRc, self._xmlEscAttr(sDescription))] );
589 return fRc;
590
591 def subXmlStart(self, oFileWrapper):
592 # Open a new file and just include it from the main XML.
593 iSubXml = self.iSubXml;
594 self.iSubXml += 1;
595 sSubXmlName = os.path.join(self.sLogDir, 'sub-%d.xml' % (iSubXml,));
596 try:
597 oFileWrapper.oSubXmlFile = utils.openNoInherit(sSubXmlName, "w");
598 except:
599 errorXcpt('open(%s)' % oFileWrapper.oSubXmlName);
600 oFileWrapper.oSubXmlFile = None;
601 else:
602 self._xmlWrite(['<Include timestamp="%s" filename="%s"/>\n'
603 % (utils.getIsoTimestamp(), self._xmlEscAttr(os.path.basename(sSubXmlName)))]);
604 return None;
605
606 def subXmlWrite(self, oFileWrapper, sRawXml, sCaller):
607 if oFileWrapper.oSubXmlFile is not None:
608 try:
609 oFileWrapper.oSubXmlFile.write(sRawXml);
610 except:
611 pass;
612 if sCaller is None: pass; # pychecker - NOREF
613 return None;
614
615 def subXmlEnd(self, oFileWrapper):
616 if oFileWrapper.oSubXmlFile is not None:
617 try:
618 oFileWrapper.oSubXmlFile.close();
619 oFileWrapper.oSubXmlFile = None;
620 except:
621 pass;
622 return None;
623
624
625
626class RemoteReporter(ReporterBase):
627 """
628 Reporter that talks to the test manager server.
629 """
630
631
632 ## The XML sync min time (seconds).
633 kcSecXmlFlushMin = 30;
634 ## The XML sync max time (seconds).
635 kcSecXmlFlushMax = 120;
636 ## The XML sync idle time before flushing (seconds).
637 kcSecXmlFlushIdle = 5;
638 ## The XML sync line count threshold.
639 kcLinesXmlFlush = 512;
640
641 ## The retry timeout.
642 kcSecTestManagerRetryTimeout = 120;
643 ## The request timeout.
644 kcSecTestManagerRequestTimeout = 30;
645
646
647 def __init__(self):
648 ReporterBase.__init__(self);
649 self.sTestManagerUrl = os.environ.get('TESTBOX_MANAGER_URL');
650 self.sTestBoxUuid = os.environ.get('TESTBOX_UUID');
651 self.idTestBox = int(os.environ.get('TESTBOX_ID'));
652 self.idTestSet = int(os.environ.get('TESTBOX_TEST_SET_ID'));
653 self._asXml = [];
654 self._secTsXmlFlush = utils.timestampSecond();
655 self._secTsXmlLast = self._secTsXmlFlush;
656 self._fXmlFlushing = False;
657 self.oOutput = sys.stdout; # Hack for __del__ output.
658 self.fFlushEachLine = True;
659 self.fDebugXml = 'TESTDRIVER_REPORTER_DEBUG_XML' in os.environ;
660
661 # Prepare the TM connecting.
662 from common import constants;
663 if sys.version_info[0] >= 3:
664 import urllib;
665 self._fnUrlEncode = urllib.parse.urlencode; # pylint: disable=no-member
666 self._fnUrlParseQs = urllib.parse.parse_qs; # pylint: disable=no-member
667 self._oParsedTmUrl = urllib.parse.urlparse(self.sTestManagerUrl); # pylint: disable=no-member
668 import http.client as httplib; # pylint: disable=no-name-in-module,import-error
669 else:
670 import urllib;
671 self._fnUrlEncode = urllib.urlencode; # pylint: disable=no-member
672 import urlparse; # pylint: disable=import-error
673 self._fnUrlParseQs = urlparse.parse_qs; # pylint: disable=no-member
674 self._oParsedTmUrl = urlparse.urlparse(self.sTestManagerUrl); # pylint: disable=no-member
675 import httplib; # pylint: disable=no-name-in-module,import-error
676
677 if sys.version_info[0] >= 3 \
678 or (sys.version_info[0] == 2 and sys.version_info[1] >= 6):
679 if self._oParsedTmUrl.scheme == 'https': # pylint: disable=E1101
680 self._fnTmConnect = lambda: httplib.HTTPSConnection(self._oParsedTmUrl.hostname,
681 timeout = self.kcSecTestManagerRequestTimeout);
682 else:
683 self._fnTmConnect = lambda: httplib.HTTPConnection( self._oParsedTmUrl.hostname,
684 timeout = self.kcSecTestManagerRequestTimeout);
685 else:
686 if self._oParsedTmUrl.scheme == 'https': # pylint: disable=E1101
687 self._fnTmConnect = lambda: httplib.HTTPSConnection(self._oParsedTmUrl.hostname);
688 else:
689 self._fnTmConnect = lambda: httplib.HTTPConnection( self._oParsedTmUrl.hostname);
690 self._dHttpHeader = \
691 {
692 'Content-Type': 'application/x-www-form-urlencoded; charset=utf-8',
693 'User-Agent': 'TestDriverReporter/%s.0 (%s, %s)' % (__version__, utils.getHostOs(), utils.getHostArch(),),
694 'Accept': 'text/plain,application/x-www-form-urlencoded',
695 'Accept-Encoding': 'identity',
696 'Cache-Control': 'max-age=0',
697 #'Connection': 'keep-alive',
698 };
699
700 dParams = {
701 constants.tbreq.ALL_PARAM_TESTBOX_UUID: self.sTestBoxUuid,
702 constants.tbreq.ALL_PARAM_TESTBOX_ID: self.idTestBox,
703 constants.tbreq.RESULT_PARAM_TEST_SET_ID: self.idTestSet,
704 };
705 self._sTmServerPath = '/%s/testboxdisp.py?%s' \
706 % ( self._oParsedTmUrl.path.strip('/'), # pylint: disable=E1101
707 self._fnUrlEncode(dParams), );
708
709 def __del__(self):
710 """Flush pending log messages?"""
711 if self._asXml:
712 self._xmlDoFlush(self._asXml, fRetry = True, fDtor = True);
713
714 def _writeOutput(self, sText):
715 """ Does the actual writing and flushing. """
716 if sys.version_info[0] >= 3:
717 print(sText, file = self.oOutput);
718 else:
719 print(sText.encode('ascii', 'replace'), file = self.oOutput);
720 if self.fFlushEachLine: self.oOutput.flush();
721 return None;
722
723 #
724 # Talking to TM.
725 #
726
727 def _processTmStatusResponse(self, oConn, sOperation, fClose = True):
728 """
729 Processes HTTP reponse from the test manager.
730 Returns True, False or None. None should be retried, the others not.
731 May raise exception on HTTP issue (retry ok).
732 """
733 if sys.version_info[0] >= 3: import http.client as httplib; # pylint: disable=no-name-in-module,import-error
734 else: import httplib; # pylint: disable=import-error
735 from common import constants;
736
737 # Read the response and (optionally) close the connection.
738 oResponse = oConn.getresponse();
739 try:
740 sRspBody = oResponse.read();
741 except httplib.IncompleteRead as oXcpt:
742 self._writeOutput('%s: %s: Warning: httplib.IncompleteRead: %s [expected %s, got %s]'
743 % (utils.getTimePrefix(), sOperation, oXcpt, oXcpt.expected, len(oXcpt.partial),));
744 sRspBody = oXcpt.partial;
745 if fClose is True:
746 try: oConn.close();
747 except: pass;
748
749 # Make sure it's a string which encoding we grok.
750 if hasattr(sRspBody, 'decode'):
751 sRspBody = sRspBody.decode('utf-8', 'ignore');
752
753 # Check the content type.
754 sContentType = oResponse.getheader('Content-Type');
755 if sContentType is not None and sContentType == 'application/x-www-form-urlencoded; charset=utf-8':
756
757 # Parse the body and check the RESULT parameter.
758 dResponse = self._fnUrlParseQs(sRspBody, strict_parsing = True);
759 sResult = dResponse.get(constants.tbresp.ALL_PARAM_RESULT, None);
760 if isinstance(sResult, list):
761 sResult = sResult[0] if len(sResult) == 1 else '%d results' % (len(sResult),);
762
763 if sResult is not None:
764 if sResult == constants.tbresp.STATUS_ACK:
765 return True;
766 if sResult == constants.tbresp.STATUS_NACK:
767 self._writeOutput('%s: %s: Failed (%s). (dResponse=%s)'
768 % (utils.getTimePrefix(), sOperation, sResult, dResponse,));
769 return False;
770
771 self._writeOutput('%s: %s: Failed - dResponse=%s' % (utils.getTimePrefix(), sOperation, dResponse,));
772 else:
773 self._writeOutput('%s: %s: Unexpected Content-Type: %s' % (utils.getTimePrefix(), sOperation, sContentType,));
774 self._writeOutput('%s: %s: Body: %s' % (utils.getTimePrefix(), sOperation, sRspBody,));
775 return None;
776
777 def _doUploadFile(self, oSrcFile, sSrcFilename, sDescription, sKind, sMime):
778 """ Uploads the given file to the test manager. """
779
780 # Prepare header and url.
781 dHeader = dict(self._dHttpHeader);
782 dHeader['Content-Type'] = 'application/octet-stream';
783 self._writeOutput('%s: _doUploadFile: sHeader=%s' % (utils.getTimePrefix(), dHeader,));
784 oSrcFile.seek(0, 2);
785 self._writeOutput('%s: _doUploadFile: size=%d' % (utils.getTimePrefix(), oSrcFile.tell(),));
786 oSrcFile.seek(0);
787
788 from common import constants;
789 sUrl = self._sTmServerPath + '&' \
790 + self._fnUrlEncode({ constants.tbreq.UPLOAD_PARAM_NAME: os.path.basename(sSrcFilename),
791 constants.tbreq.UPLOAD_PARAM_DESC: sDescription,
792 constants.tbreq.UPLOAD_PARAM_KIND: sKind,
793 constants.tbreq.UPLOAD_PARAM_MIME: sMime,
794 constants.tbreq.ALL_PARAM_ACTION: constants.tbreq.UPLOAD,
795 });
796
797 # Retry loop.
798 secStart = utils.timestampSecond();
799 while True:
800 try:
801 oConn = self._fnTmConnect();
802 oConn.request('POST', sUrl, oSrcFile.read(), dHeader);
803 fRc = self._processTmStatusResponse(oConn, '_doUploadFile', fClose = True);
804 oConn.close();
805 if fRc is not None:
806 return fRc;
807 except:
808 logXcpt('warning: exception during UPLOAD request');
809
810 if utils.timestampSecond() - secStart >= self.kcSecTestManagerRetryTimeout:
811 self._writeOutput('%s: _doUploadFile: Timed out.' % (utils.getTimePrefix(),));
812 break;
813 try: oSrcFile.seek(0);
814 except:
815 logXcpt();
816 break;
817 self._writeOutput('%s: _doUploadFile: Retrying...' % (utils.getTimePrefix(), ));
818 time.sleep(2);
819
820 return False;
821
822 def _doUploadString(self, sSrc, sSrcName, sDescription, sKind, sMime):
823 """ Uploads the given string as a separate file to the test manager. """
824
825 # Prepare header and url.
826 dHeader = dict(self._dHttpHeader);
827 dHeader['Content-Type'] = 'application/octet-stream';
828 self._writeOutput('%s: _doUploadString: sHeader=%s' % (utils.getTimePrefix(), dHeader,));
829 self._writeOutput('%s: _doUploadString: size=%d' % (utils.getTimePrefix(), sys.getsizeof(sSrc),));
830
831 from common import constants;
832 sUrl = self._sTmServerPath + '&' \
833 + self._fnUrlEncode({ constants.tbreq.UPLOAD_PARAM_NAME: os.path.basename(sSrcName),
834 constants.tbreq.UPLOAD_PARAM_DESC: sDescription,
835 constants.tbreq.UPLOAD_PARAM_KIND: sKind,
836 constants.tbreq.UPLOAD_PARAM_MIME: sMime,
837 constants.tbreq.ALL_PARAM_ACTION: constants.tbreq.UPLOAD,
838 });
839
840 # Retry loop.
841 secStart = utils.timestampSecond();
842 while True:
843 try:
844 oConn = self._fnTmConnect();
845 oConn.request('POST', sUrl, sSrc, dHeader);
846 fRc = self._processTmStatusResponse(oConn, '_doUploadString', fClose = True);
847 oConn.close();
848 if fRc is not None:
849 return fRc;
850 except:
851 logXcpt('warning: exception during UPLOAD request');
852
853 if utils.timestampSecond() - secStart >= self.kcSecTestManagerRetryTimeout:
854 self._writeOutput('%s: _doUploadString: Timed out.' % (utils.getTimePrefix(),));
855 break;
856 self._writeOutput('%s: _doUploadString: Retrying...' % (utils.getTimePrefix(), ));
857 time.sleep(2);
858
859 return False;
860
861 def _xmlDoFlush(self, asXml, fRetry = False, fDtor = False):
862 """
863 The code that does the actual talking to the server.
864 Used by both xmlFlush and __del__.
865 """
866 secStart = utils.timestampSecond();
867 while True:
868 fRc = None;
869 try:
870 # Post.
871 from common import constants;
872 sPostBody = self._fnUrlEncode({constants.tbreq.XML_RESULT_PARAM_BODY: '\n'.join(asXml),});
873 oConn = self._fnTmConnect();
874 oConn.request('POST',
875 self._sTmServerPath + ('&%s=%s' % (constants.tbreq.ALL_PARAM_ACTION, constants.tbreq.XML_RESULTS)),
876 sPostBody,
877 self._dHttpHeader);
878
879 fRc = self._processTmStatusResponse(oConn, '_xmlDoFlush', fClose = True);
880 if fRc is True:
881 if self.fDebugXml:
882 self._writeOutput('_xmlDoFlush:\n%s' % ('\n'.join(asXml),));
883 return (None, False);
884 if fRc is False:
885 self._writeOutput('_xmlDoFlush: Failed - we should abort the test, really.');
886 return (None, True);
887 except Exception as oXcpt:
888 if not fDtor:
889 logXcpt('warning: exception during XML_RESULTS request');
890 else:
891 self._writeOutput('warning: exception during XML_RESULTS request: %s' % (oXcpt,));
892
893 if fRetry is not True \
894 or utils.timestampSecond() - secStart >= self.kcSecTestManagerRetryTimeout:
895 break;
896 time.sleep(2);
897
898 return (asXml, False);
899
900
901 #
902 # Overridden methods.
903 #
904
905 def isLocal(self):
906 return False;
907
908 def log(self, iLevel, sText, sCaller, sTsPrf):
909 if iLevel <= self.iVerbose:
910 if self.iDebug > 0:
911 sLogText = '%s %30s: %s' % (sTsPrf, sCaller, sText);
912 else:
913 sLogText = '%s %s: %s' % (sTsPrf, self.sName, sText);
914 self._writeOutput(sLogText);
915 return 0;
916
917 def addLogFile(self, oSrcFile, sSrcFilename, sAltName, sDescription, sKind, sCaller, sTsPrf):
918 fRc = True;
919 if sKind in [ 'text', 'log', 'process'] \
920 or sKind.startswith('log/') \
921 or sKind.startswith('info/') \
922 or sKind.startswith('process/'):
923 self.log(0, '*** Uploading "%s" - KIND: "%s" - DESC: "%s" ***'
924 % (sSrcFilename, sKind, sDescription), sCaller, sTsPrf);
925 self.xmlFlush();
926 g_oLock.release();
927 try:
928 self._doUploadFile(oSrcFile, sAltName, sDescription, sKind, 'text/plain');
929 finally:
930 g_oLock.acquire();
931 elif sKind.startswith('screenshot/'):
932 self.log(0, '*** Uploading "%s" - KIND: "%s" - DESC: "%s" ***'
933 % (sSrcFilename, sKind, sDescription), sCaller, sTsPrf);
934 self.xmlFlush();
935 g_oLock.release();
936 try:
937 self._doUploadFile(oSrcFile, sAltName, sDescription, sKind, 'image/png');
938 finally:
939 g_oLock.acquire();
940 elif sKind.startswith('misc/'):
941 self.log(0, '*** Uploading "%s" - KIND: "%s" - DESC: "%s" ***'
942 % (sSrcFilename, sKind, sDescription), sCaller, sTsPrf);
943 self.xmlFlush();
944 g_oLock.release();
945 try:
946 self._doUploadFile(oSrcFile, sAltName, sDescription, sKind, 'application/octet-stream');
947 finally:
948 g_oLock.acquire();
949 else:
950 self.log(0, '*** UNKNOWN FILE "%s" - KIND "%s" - DESC "%s" ***'
951 % (sSrcFilename, sKind, sDescription), sCaller, sTsPrf);
952 return fRc;
953
954 def addLogString(self, sLog, sLogName, sDescription, sKind, sCaller, sTsPrf):
955 fRc = True;
956 if sKind in [ 'text', 'log', 'process'] \
957 or sKind.startswith('log/') \
958 or sKind.startswith('info/') \
959 or sKind.startswith('process/'):
960 self.log(0, '*** Uploading "%s" - KIND: "%s" - DESC: "%s" ***'
961 % (sLogName, sKind, sDescription), sCaller, sTsPrf);
962 self.xmlFlush();
963 g_oLock.release();
964 try:
965 self._doUploadString(sLog, sLogName, sDescription, sKind, 'text/plain');
966 finally:
967 g_oLock.acquire();
968 else:
969 self.log(0, '*** UNKNOWN FILE "%s" - KIND "%s" - DESC "%s" ***'
970 % (sLogName, sKind, sDescription), sCaller, sTsPrf);
971 return fRc;
972
973 def xmlFlush(self, fRetry = False, fForce = False):
974 """
975 Flushes the XML back log. Called with the lock held, may leave it
976 while communicating with the server.
977 """
978 if not self._fXmlFlushing:
979 asXml = self._asXml;
980 self._asXml = [];
981 if asXml or fForce is True:
982 self._fXmlFlushing = True;
983
984 g_oLock.release();
985 try:
986 (asXml, fIncErrors) = self._xmlDoFlush(asXml, fRetry = fRetry);
987 finally:
988 g_oLock.acquire();
989
990 if fIncErrors:
991 self.testIncErrors();
992
993 self._fXmlFlushing = False;
994 if asXml is None:
995 self._secTsXmlFlush = utils.timestampSecond();
996 else:
997 self._asXml = asXml + self._asXml;
998 return True;
999
1000 self._secTsXmlFlush = utils.timestampSecond();
1001 return False;
1002
1003 def _xmlFlushIfNecessary(self, fPolling = False, sDebug = None):
1004 """Flushes the XML back log if necessary."""
1005 tsNow = utils.timestampSecond();
1006 cSecs = tsNow - self._secTsXmlFlush;
1007 cSecsLast = tsNow - self._secTsXmlLast;
1008 if fPolling is not True:
1009 self._secTsXmlLast = tsNow;
1010
1011 # Absolute flush thresholds.
1012 if cSecs >= self.kcSecXmlFlushMax:
1013 return self.xmlFlush();
1014 if len(self._asXml) >= self.kcLinesXmlFlush:
1015 return self.xmlFlush();
1016
1017 # Flush if idle long enough.
1018 if cSecs >= self.kcSecXmlFlushMin \
1019 and cSecsLast >= self.kcSecXmlFlushIdle:
1020 return self.xmlFlush();
1021
1022 _ = sDebug;
1023 return False;
1024
1025 def _xmlWrite(self, asText, fIndent = True):
1026 """XML output function for the reporter."""
1027 self._asXml += asText;
1028 self._xmlFlushIfNecessary();
1029 _ = fIndent; # No pretty printing, thank you.
1030 return None;
1031
1032 def subXmlStart(self, oFileWrapper):
1033 oFileWrapper.sXmlBuffer = '';
1034 return None;
1035
1036 def subXmlWrite(self, oFileWrapper, sRawXml, sCaller):
1037 oFileWrapper.sXmlBuffer += sRawXml;
1038 _ = sCaller;
1039 return None;
1040
1041 def subXmlEnd(self, oFileWrapper):
1042 sRawXml = oFileWrapper.sXmlBuffer;
1043 ## @todo should validate the document here and maybe auto terminate things. Adding some hints to have the server do
1044 # this instead.
1045 g_oLock.acquire();
1046 try:
1047 self._asXml += [ '<PushHint testdepth="%d"/>' % (len(self.atTests),),
1048 sRawXml,
1049 '<PopHint testdepth="%d"/>' % (len(self.atTests),),];
1050 self._xmlFlushIfNecessary();
1051 finally:
1052 g_oLock.release();
1053 return None;
1054
1055 def doPollWork(self, sDebug = None):
1056 if self._asXml:
1057 g_oLock.acquire();
1058 try:
1059 self._xmlFlushIfNecessary(fPolling = True, sDebug = sDebug);
1060 finally:
1061 g_oLock.release();
1062 return None;
1063
1064
1065#
1066# Helpers
1067#
1068
1069def logXcptWorker(iLevel, fIncErrors, sPrefix="", sText=None, cFrames=1):
1070 """
1071 Log an exception, optionally with a preceeding message and more than one
1072 call frame.
1073 """
1074 g_oLock.acquire();
1075 try:
1076
1077 if fIncErrors:
1078 g_oReporter.testIncErrors();
1079
1080 ## @todo skip all this if iLevel is too high!
1081
1082 # Try get exception info.
1083 sTsPrf = utils.getTimePrefix();
1084 try:
1085 oType, oValue, oTraceback = sys.exc_info();
1086 except:
1087 oType = oValue = oTraceback = None;
1088 if oType is not None:
1089
1090 # Try format the info
1091 try:
1092 rc = 0;
1093 sCaller = utils.getCallerName(oTraceback.tb_frame);
1094 if sText is not None:
1095 rc = g_oReporter.log(iLevel, "%s%s" % (sPrefix, sText), sCaller, sTsPrf);
1096 asInfo = [];
1097 try:
1098 asInfo = asInfo + traceback.format_exception_only(oType, oValue);
1099 if cFrames is not None and cFrames <= 1:
1100 asInfo = asInfo + traceback.format_tb(oTraceback, 1);
1101 else:
1102 asInfo.append('Traceback:')
1103 asInfo = asInfo + traceback.format_tb(oTraceback, cFrames);
1104 asInfo.append('Stack:')
1105 asInfo = asInfo + traceback.format_stack(oTraceback.tb_frame.f_back, cFrames);
1106 except:
1107 g_oReporter.log(0, '** internal-error: Hit exception #2! %s' % (traceback.format_exc()), sCaller, sTsPrf);
1108
1109 if asInfo:
1110 # Do the logging.
1111 for sItem in asInfo:
1112 asLines = sItem.splitlines();
1113 for sLine in asLines:
1114 rc = g_oReporter.log(iLevel, '%s%s' % (sPrefix, sLine), sCaller, sTsPrf);
1115
1116 else:
1117 g_oReporter.log(iLevel, 'No exception info...', sCaller, sTsPrf);
1118 rc = -3;
1119 except:
1120 g_oReporter.log(0, '** internal-error: Hit exception! %s' % (traceback.format_exc()), None, sTsPrf);
1121 rc = -2;
1122 else:
1123 g_oReporter.log(0, '** internal-error: No exception! %s'
1124 % (utils.getCallerName(iFrame=3)), utils.getCallerName(iFrame=3), sTsPrf);
1125 rc = -1;
1126
1127 finally:
1128 g_oLock.release();
1129 return rc;
1130
1131#
1132# The public Classes
1133#
1134class FileWrapper(object):
1135 """ File like class for TXS EXEC and similar. """
1136 def __init__(self, sPrefix):
1137 self.sPrefix = sPrefix;
1138
1139 def __del__(self):
1140 self.close();
1141
1142 def close(self):
1143 """ file.close """
1144 # Nothing to be done.
1145 return;
1146
1147 def read(self, cb):
1148 """file.read"""
1149 _ = cb;
1150 return "";
1151
1152 def write(self, sText):
1153 """file.write"""
1154 if isinstance(sText, array.array):
1155 try:
1156 sText = sText.tostring();
1157 except:
1158 pass;
1159 g_oLock.acquire();
1160 try:
1161 sTsPrf = utils.getTimePrefix();
1162 sCaller = utils.getCallerName();
1163 asLines = sText.splitlines();
1164 for sLine in asLines:
1165 g_oReporter.log(0, '%s: %s' % (self.sPrefix, sLine), sCaller, sTsPrf);
1166 except:
1167 traceback.print_exc();
1168 finally:
1169 g_oLock.release();
1170 return None;
1171
1172class FileWrapperTestPipe(object):
1173 """ File like class for the test pipe (TXS EXEC and similar). """
1174 def __init__(self):
1175 self.sPrefix = '';
1176 self.fStarted = False;
1177 self.fClosed = False;
1178 self.sTagBuffer = None;
1179
1180 def __del__(self):
1181 self.close();
1182
1183 def close(self):
1184 """ file.close """
1185 if self.fStarted is True and self.fClosed is False:
1186 self.fClosed = True;
1187 try: g_oReporter.subXmlEnd(self);
1188 except:
1189 try: traceback.print_exc();
1190 except: pass;
1191 return True;
1192
1193 def read(self, cb = None):
1194 """file.read"""
1195 _ = cb;
1196 return "";
1197
1198 def write(self, sText):
1199 """file.write"""
1200 # lazy start.
1201 if self.fStarted is not True:
1202 try:
1203 g_oReporter.subXmlStart(self);
1204 except:
1205 traceback.print_exc();
1206 self.fStarted = True;
1207
1208 # Turn non-string stuff into strings.
1209 if not utils.isString(sText):
1210 if isinstance(sText, array.array):
1211 try: sText = sText.tostring();
1212 except: pass;
1213 if not utils.isString(sText) and hasattr(sText, 'decode'):
1214 try: sText = sText.decode('utf-8', 'ignore');
1215 except: pass;
1216
1217 try:
1218 g_oReporter.subXmlWrite(self, sText, utils.getCallerName());
1219 # Parse the supplied text and look for <Failed.../> tags to keep track of the
1220 # error counter. This is only a very lazy aproach.
1221 sText.strip();
1222 idxText = 0;
1223 while sText:
1224 if self.sTagBuffer is None:
1225 # Look for the start of a tag.
1226 idxStart = sText[idxText:].find('<');
1227 if idxStart != -1:
1228 # Look for the end of the tag.
1229 idxEnd = sText[idxStart:].find('>');
1230
1231 # If the end was found inside the current buffer, parse the line,
1232 # else we have to save it for later.
1233 if idxEnd != -1:
1234 idxEnd += idxStart + 1;
1235 self._processXmlElement(sText[idxStart:idxEnd]);
1236 idxText = idxEnd;
1237 else:
1238 self.sTagBuffer = sText[idxStart:];
1239 idxText = len(sText);
1240 else:
1241 idxText = len(sText);
1242 else:
1243 # Search for the end of the tag and parse the whole tag.
1244 idxEnd = sText[idxText:].find('>');
1245 if idxEnd != -1:
1246 idxEnd += idxStart + 1;
1247 self._processXmlElement(self.sTagBuffer + sText[idxText:idxEnd]);
1248 self.sTagBuffer = None;
1249 idxText = idxEnd;
1250 else:
1251 self.sTagBuffer = self.sTagBuffer + sText[idxText:];
1252 idxText = len(sText);
1253
1254 sText = sText[idxText:];
1255 sText = sText.lstrip();
1256 except:
1257 traceback.print_exc();
1258 return None;
1259
1260 def _processXmlElement(self, sElement):
1261 """
1262 Processes a complete XML tag (so far we only search for the Failed to tag
1263 to keep track of the error counter.
1264 """
1265 # Make sure we don't parse any space between < and the element name.
1266 sElement = sElement.strip();
1267
1268 # Find the end of the name
1269 idxEndName = sElement.find(' ');
1270 if idxEndName == -1:
1271 idxEndName = sElement.find('/');
1272 if idxEndName == -1:
1273 idxEndName = sElement.find('>');
1274
1275 if idxEndName != -1:
1276 if sElement[1:idxEndName] == 'Failed':
1277 g_oLock.acquire();
1278 try:
1279 g_oReporter.testIncErrors();
1280 finally:
1281 g_oLock.release();
1282 else:
1283 error('_processXmlElement(%s)' % sElement);
1284
1285
1286#
1287# The public APIs.
1288#
1289
1290def log(sText):
1291 """Writes the specfied text to the log."""
1292 g_oLock.acquire();
1293 try:
1294 rc = g_oReporter.log(1, sText, utils.getCallerName(), utils.getTimePrefix());
1295 except:
1296 rc = -1;
1297 finally:
1298 g_oLock.release();
1299 return rc;
1300
1301def logXcpt(sText=None, cFrames=1):
1302 """
1303 Log an exception, optionally with a preceeding message and more than one
1304 call frame.
1305 """
1306 return logXcptWorker(1, False, "", sText, cFrames);
1307
1308def log2(sText):
1309 """Log level 2: Writes the specfied text to the log."""
1310 g_oLock.acquire();
1311 try:
1312 rc = g_oReporter.log(2, sText, utils.getCallerName(), utils.getTimePrefix());
1313 except:
1314 rc = -1;
1315 finally:
1316 g_oLock.release();
1317 return rc;
1318
1319def log2Xcpt(sText=None, cFrames=1):
1320 """
1321 Log level 2: Log an exception, optionally with a preceeding message and
1322 more than one call frame.
1323 """
1324 return logXcptWorker(2, False, "", sText, cFrames);
1325
1326def maybeErr(fIsError, sText):
1327 """ Maybe error or maybe normal log entry. """
1328 if fIsError is True:
1329 return error(sText);
1330 return log(sText);
1331
1332def maybeErrXcpt(fIsError, sText=None, cFrames=1):
1333 """ Maybe error or maybe normal log exception entry. """
1334 if fIsError is True:
1335 return errorXcpt(sText, cFrames);
1336 return logXcpt(sText, cFrames);
1337
1338def maybeLog(fIsNotError, sText):
1339 """ Maybe error or maybe normal log entry. """
1340 if fIsNotError is not True:
1341 return error(sText);
1342 return log(sText);
1343
1344def maybeLogXcpt(fIsNotError, sText=None, cFrames=1):
1345 """ Maybe error or maybe normal log exception entry. """
1346 if fIsNotError is not True:
1347 return errorXcpt(sText, cFrames);
1348 return logXcpt(sText, cFrames);
1349
1350def error(sText):
1351 """
1352 Writes the specfied error message to the log.
1353
1354 This will add an error to the current test.
1355
1356 Always returns False for the convenience of methods returning boolean
1357 success indicators.
1358 """
1359 g_oLock.acquire();
1360 try:
1361 g_oReporter.testIncErrors();
1362 g_oReporter.log(0, '** error: %s' % (sText), utils.getCallerName(), utils.getTimePrefix());
1363 except:
1364 pass;
1365 finally:
1366 g_oLock.release();
1367 return False;
1368
1369def errorXcpt(sText=None, cFrames=1):
1370 """
1371 Log an error caused by an exception. If sText is given, it will preceed
1372 the exception information. cFrames can be used to display more stack.
1373
1374 This will add an error to the current test.
1375
1376 Always returns False for the convenience of methods returning boolean
1377 success indicators.
1378 """
1379 logXcptWorker(0, True, '** error: ', sText, cFrames);
1380 return False;
1381
1382def errorTimeout(sText):
1383 """
1384 Flags the current test as having timed out and writes the specified message to the log.
1385
1386 This will add an error to the current test.
1387
1388 Always returns False for the convenience of methods returning boolean
1389 success indicators.
1390 """
1391 g_oLock.acquire();
1392 try:
1393 g_oReporter.testSetTimedOut();
1394 g_oReporter.log(0, '** timeout-error: %s' % (sText), utils.getCallerName(), utils.getTimePrefix());
1395 except:
1396 pass;
1397 finally:
1398 g_oLock.release();
1399 return False;
1400
1401def fatal(sText):
1402 """
1403 Writes a fatal error to the log.
1404
1405 This will add an error to the current test.
1406
1407 Always returns False for the convenience of methods returning boolean
1408 success indicators.
1409 """
1410 g_oLock.acquire();
1411 try:
1412 g_oReporter.testIncErrors();
1413 g_oReporter.log(0, '** fatal error: %s' % (sText), utils.getCallerName(), utils.getTimePrefix());
1414 except:
1415 pass
1416 finally:
1417 g_oLock.release();
1418 return False;
1419
1420def fatalXcpt(sText=None, cFrames=1):
1421 """
1422 Log a fatal error caused by an exception. If sText is given, it will
1423 preceed the exception information. cFrames can be used to display more
1424 stack.
1425
1426 This will add an error to the current test.
1427
1428 Always returns False for the convenience of methods returning boolean
1429 success indicators.
1430 """
1431 logXcptWorker(0, True, "** fatal error: ", sText, cFrames);
1432 return False;
1433
1434def addLogFile(sFilename, sKind, sDescription = '', sAltName = None):
1435 """
1436 Adds the specified log file to the report if the file exists.
1437
1438 The sDescription is a free form description of the log file.
1439
1440 The sKind parameter is for adding some machine parsable hint what kind of
1441 log file this really is.
1442
1443 Returns True on success, False on failure (no ENOENT errors are logged).
1444 """
1445 sTsPrf = utils.getTimePrefix();
1446 sCaller = utils.getCallerName();
1447 fRc = False;
1448 if sAltName is None:
1449 sAltName = sFilename;
1450
1451 try:
1452 oSrcFile = utils.openNoInherit(sFilename, 'rb');
1453 except IOError as oXcpt:
1454 if oXcpt.errno != errno.ENOENT:
1455 logXcpt('addLogFile(%s,%s,%s)' % (sFilename, sDescription, sKind));
1456 else:
1457 logXcpt('addLogFile(%s,%s,%s) IOError' % (sFilename, sDescription, sKind));
1458 except:
1459 logXcpt('addLogFile(%s,%s,%s)' % (sFilename, sDescription, sKind));
1460 else:
1461 g_oLock.acquire();
1462 try:
1463 fRc = g_oReporter.addLogFile(oSrcFile, sFilename, sAltName, sDescription, sKind, sCaller, sTsPrf);
1464 finally:
1465 g_oLock.release();
1466 oSrcFile.close();
1467 return fRc;
1468
1469def addLogString(sLog, sLogName, sKind, sDescription = ''):
1470 """
1471 Adds the specified log string to the report.
1472
1473 The sLog parameter sets the name of the log file.
1474
1475 The sDescription is a free form description of the log file.
1476
1477 The sKind parameter is for adding some machine parsable hint what kind of
1478 log file this really is.
1479
1480 Returns True on success, False on failure (no ENOENT errors are logged).
1481 """
1482 sTsPrf = utils.getTimePrefix();
1483 sCaller = utils.getCallerName();
1484 fRc = False;
1485
1486 g_oLock.acquire();
1487 try:
1488 fRc = g_oReporter.addLogString(sLog, sLogName, sDescription, sKind, sCaller, sTsPrf);
1489 finally:
1490 g_oLock.release();
1491 return fRc;
1492
1493def isLocal():
1494 """Is this a local reporter?"""
1495 return g_oReporter.isLocal()
1496
1497def incVerbosity():
1498 """Increases the verbosity level."""
1499 return g_oReporter.incVerbosity()
1500
1501def incDebug():
1502 """Increases the debug level."""
1503 return g_oReporter.incDebug()
1504
1505def appendToProcessName(sAppend):
1506 """
1507 Appends sAppend to the base process name.
1508 Returns the new process name.
1509 """
1510 return g_oReporter.appendToProcessName(sAppend);
1511
1512def getErrorCount():
1513 """
1514 Get the current error count for the entire test run.
1515 """
1516 g_oLock.acquire();
1517 try:
1518 cErrors = g_oReporter.cErrors;
1519 finally:
1520 g_oLock.release();
1521 return cErrors;
1522
1523def doPollWork(sDebug = None):
1524 """
1525 This can be called from wait loops and similar to make the reporter call
1526 home with pending XML and such.
1527 """
1528 g_oReporter.doPollWork(sDebug);
1529 return None;
1530
1531
1532#
1533# Test reporting, a bit similar to RTTestI*.
1534#
1535
1536def testStart(sName):
1537 """
1538 Starts a new test (pushes it).
1539 """
1540 g_oLock.acquire();
1541 try:
1542 rc = g_oReporter.testStart(sName, utils.getCallerName());
1543 finally:
1544 g_oLock.release();
1545 return rc;
1546
1547def testValue(sName, sValue, sUnit):
1548 """
1549 Reports a benchmark value or something simiarlly useful.
1550 """
1551 g_oLock.acquire();
1552 try:
1553 rc = g_oReporter.testValue(sName, str(sValue), sUnit, utils.getCallerName());
1554 finally:
1555 g_oLock.release();
1556 return rc;
1557
1558def testFailure(sDetails):
1559 """
1560 Reports a failure.
1561 We count these calls and testDone will use them to report PASSED or FAILED.
1562
1563 Returns False so that a return False line can be saved.
1564 """
1565 g_oLock.acquire();
1566 try:
1567 g_oReporter.testFailure(sDetails, utils.getCallerName());
1568 finally:
1569 g_oLock.release();
1570 return False;
1571
1572def testFailureXcpt(sDetails = ''):
1573 """
1574 Reports a failure with exception.
1575 We count these calls and testDone will use them to report PASSED or FAILED.
1576
1577 Returns False so that a return False line can be saved.
1578 """
1579 # Extract exception info.
1580 try:
1581 oType, oValue, oTraceback = sys.exc_info();
1582 except:
1583 oType = oValue, oTraceback = None;
1584 if oType is not None:
1585 sCaller = utils.getCallerName(oTraceback.tb_frame);
1586 sXcpt = ' '.join(traceback.format_exception_only(oType, oValue));
1587 else:
1588 sCaller = utils.getCallerName();
1589 sXcpt = 'No exception at %s' % (sCaller,);
1590
1591 # Use testFailure to do the work.
1592 g_oLock.acquire();
1593 try:
1594 if sDetails == '':
1595 g_oReporter.testFailure('Exception: %s' % (sXcpt,), sCaller);
1596 else:
1597 g_oReporter.testFailure('%s: %s' % (sDetails, sXcpt), sCaller);
1598 finally:
1599 g_oLock.release();
1600 return False;
1601
1602def testDone(fSkipped = False):
1603 """
1604 Completes the current test (pops it), logging PASSED / FAILURE.
1605
1606 Returns a tuple with the name of the test and its error count.
1607 """
1608 g_oLock.acquire();
1609 try:
1610 rc = g_oReporter.testDone(fSkipped, utils.getCallerName());
1611 finally:
1612 g_oLock.release();
1613 return rc;
1614
1615def testErrorCount():
1616 """
1617 Gets the error count of the current test.
1618
1619 Returns the number of errors.
1620 """
1621 g_oLock.acquire();
1622 try:
1623 cErrors = g_oReporter.testErrorCount();
1624 finally:
1625 g_oLock.release();
1626 return cErrors;
1627
1628def testCleanup():
1629 """
1630 Closes all open tests with a generic error condition.
1631
1632 Returns True if no open tests, False if something had to be closed with failure.
1633 """
1634 g_oLock.acquire();
1635 try:
1636 fRc = g_oReporter.testCleanup(utils.getCallerName());
1637 g_oReporter.xmlFlush(fRetry = False, fForce = True);
1638 finally:
1639 g_oLock.release();
1640 return fRc;
1641
1642
1643#
1644# Sub XML stuff.
1645#
1646
1647def addSubXmlFile(sFilename):
1648 """
1649 Adds a sub-xml result file to the party.
1650 """
1651 fRc = False;
1652 try:
1653 oSrcFile = utils.openNoInherit(sFilename, 'r');
1654 except IOError as oXcpt:
1655 if oXcpt.errno != errno.ENOENT:
1656 logXcpt('addSubXmlFile(%s)' % (sFilename,));
1657 except:
1658 logXcpt('addSubXmlFile(%s)' % (sFilename,));
1659 else:
1660 try:
1661 oWrapper = FileWrapperTestPipe()
1662 oWrapper.write(oSrcFile.read());
1663 oWrapper.close();
1664 except:
1665 logXcpt('addSubXmlFile(%s)' % (sFilename,));
1666 oSrcFile.close();
1667
1668 return fRc;
1669
1670
1671#
1672# Other useful debugging tools.
1673#
1674
1675def logAllStacks(cFrames = None):
1676 """
1677 Logs the stacks of all python threads.
1678 """
1679 sTsPrf = utils.getTimePrefix();
1680 sCaller = utils.getCallerName();
1681 g_oLock.acquire();
1682
1683 cThread = 0;
1684 for idThread, oStack in sys._current_frames().items(): # >=2.5, a bit ugly - pylint: disable=W0212
1685 try:
1686 if cThread > 0:
1687 g_oReporter.log(1, '', sCaller, sTsPrf);
1688 g_oReporter.log(1, 'Thread %s (%#x)' % (idThread, idThread), sCaller, sTsPrf);
1689 try:
1690 asInfo = traceback.format_stack(oStack, cFrames);
1691 except:
1692 g_oReporter.log(1, ' Stack formatting failed w/ exception', sCaller, sTsPrf);
1693 else:
1694 for sInfo in asInfo:
1695 asLines = sInfo.splitlines();
1696 for sLine in asLines:
1697 g_oReporter.log(1, sLine, sCaller, sTsPrf);
1698 except:
1699 pass;
1700 cThread += 1;
1701
1702 g_oLock.release();
1703 return None;
1704
1705def checkTestManagerConnection():
1706 """
1707 Checks the connection to the test manager.
1708
1709 Returns True if the connection is fine, False if not, None if not remote
1710 reporter.
1711
1712 Note! This as the sideeffect of flushing XML.
1713 """
1714 g_oLock.acquire();
1715 try:
1716 fRc = g_oReporter.xmlFlush(fRetry = False, fForce = True);
1717 finally:
1718 g_oLock.release();
1719 return fRc;
1720
1721def flushall(fSkipXml = False):
1722 """
1723 Flushes all output streams, both standard and logger related.
1724 This may also push data to the remote test manager.
1725 """
1726 try: sys.stdout.flush();
1727 except: pass;
1728 try: sys.stderr.flush();
1729 except: pass;
1730
1731 if fSkipXml is not True:
1732 g_oLock.acquire();
1733 try:
1734 g_oReporter.xmlFlush(fRetry = False);
1735 finally:
1736 g_oLock.release();
1737
1738 return True;
1739
1740
1741#
1742# Module initialization.
1743#
1744
1745def _InitReporterModule():
1746 """
1747 Instantiate the test reporter.
1748 """
1749 global g_oReporter, g_sReporterName
1750
1751 g_sReporterName = os.getenv("TESTBOX_REPORTER", "local");
1752 if g_sReporterName == "local":
1753 g_oReporter = LocalReporter();
1754 elif g_sReporterName == "remote":
1755 g_oReporter = RemoteReporter(); # Correct, but still plain stupid. pylint: disable=redefined-variable-type
1756 else:
1757 print(os.path.basename(__file__) + ": Unknown TESTBOX_REPORTER value: '" + g_sReporterName + "'", file = sys.stderr);
1758 raise Exception("Unknown TESTBOX_REPORTER value '" + g_sReporterName + "'");
1759
1760if __name__ != "checker": # pychecker avoidance.
1761 _InitReporterModule();
1762
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette