VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testdriver/base.py@ 82894

Last change on this file since 82894 was 80237, checked in by vboxsync, 5 years ago

ValKit/vboxinstaller.py,++: Hack for letting vboxinstaller report a 'bad testbox' status rather than a simple 'skipped' so vsheriff can more easily spot the problem with stuck drivers.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 63.6 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: base.py 80237 2019-08-12 21:16:08Z vboxsync $
3# pylint: disable=too-many-lines
4
5"""
6Base testdriver module.
7"""
8
9__copyright__ = \
10"""
11Copyright (C) 2010-2019 Oracle Corporation
12
13This file is part of VirtualBox Open Source Edition (OSE), as
14available from http://www.virtualbox.org. This file is free software;
15you can redistribute it and/or modify it under the terms of the GNU
16General Public License (GPL) as published by the Free Software
17Foundation, in version 2 as it comes in the "COPYING" file of the
18VirtualBox OSE distribution. VirtualBox OSE is distributed in the
19hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
20
21The contents of this file may alternatively be used under the terms
22of the Common Development and Distribution License Version 1.0
23(CDDL) only, as it comes in the "COPYING.CDDL" file of the
24VirtualBox OSE distribution, in which case the provisions of the
25CDDL are applicable instead of those of the GPL.
26
27You may elect to license modified versions of this file under the
28terms and conditions of either the GPL or the CDDL or both.
29"""
30__version__ = "$Revision: 80237 $"
31
32
33# Standard Python imports.
34import os
35import os.path
36import signal
37import socket
38import stat
39import subprocess
40import sys
41import time
42if sys.version_info[0] < 3: import thread; # pylint: disable=import-error
43else: import _thread as thread; # pylint: disable=import-error
44import threading
45import traceback
46import tempfile;
47import unittest;
48
49# Validation Kit imports.
50from common import utils;
51from common.constants import rtexitcode;
52from testdriver import reporter;
53if sys.platform == 'win32':
54 from testdriver import winbase;
55
56# Figure where we are.
57try: __file__
58except: __file__ = sys.argv[0];
59g_ksValidationKitDir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)));
60
61# Python 3 hacks:
62if sys.version_info[0] >= 3:
63 long = int; # pylint: disable=redefined-builtin,invalid-name
64
65
66#
67# Some utility functions.
68#
69
70def exeSuff():
71 """
72 Returns the executable suffix.
73 """
74 if os.name == 'nt' or os.name == 'os2':
75 return '.exe';
76 return '';
77
78def searchPath(sExecName):
79 """
80 Searches the PATH for the specified executable name, returning the first
81 existing file/directory/whatever. The return is abspath'ed.
82 """
83 sSuff = exeSuff();
84
85 sPath = os.getenv('PATH', os.getenv('Path', os.path.defpath));
86 aPaths = sPath.split(os.path.pathsep)
87 for sDir in aPaths:
88 sFullExecName = os.path.join(sDir, sExecName);
89 if os.path.exists(sFullExecName):
90 return os.path.abspath(sFullExecName);
91 sFullExecName += sSuff;
92 if os.path.exists(sFullExecName):
93 return os.path.abspath(sFullExecName);
94 return sExecName;
95
96def getEnv(sVar, sLocalAlternative = None):
97 """
98 Tries to get an environment variable, optionally with a local run alternative.
99 Will raise an exception if sLocalAlternative is None and the variable is
100 empty or missing.
101 """
102 try:
103 sVal = os.environ.get(sVar, None);
104 if sVal is None:
105 raise GenError('environment variable "%s" is missing' % (sVar));
106 if sVal == "":
107 raise GenError('environment variable "%s" is empty' % (sVar));
108 except:
109 if sLocalAlternative is None or not reporter.isLocal():
110 raise
111 sVal = sLocalAlternative;
112 return sVal;
113
114def getDirEnv(sVar, sAlternative = None, fLocalReq = False, fTryCreate = False):
115 """
116 Tries to get an environment variable specifying a directory path.
117
118 Resolves it into an absolute path and verifies its existance before
119 returning it.
120
121 If the environment variable is empty or isn't set, or if the directory
122 doesn't exist or isn't a directory, sAlternative is returned instead.
123 If sAlternative is None, then we'll raise a GenError. For local runs we'll
124 only do this if fLocalReq is True.
125 """
126 assert sAlternative is None or fTryCreate is False;
127 try:
128 sVal = os.environ.get(sVar, None);
129 if sVal is None:
130 raise GenError('environment variable "%s" is missing' % (sVar));
131 if sVal == "":
132 raise GenError('environment variable "%s" is empty' % (sVar));
133
134 sVal = os.path.abspath(sVal);
135 if not os.path.isdir(sVal):
136 if not fTryCreate or os.path.exists(sVal):
137 reporter.error('the value of env.var. "%s" is not a dir: "%s"' % (sVar, sVal));
138 raise GenError('the value of env.var. "%s" is not a dir: "%s"' % (sVar, sVal));
139 try:
140 os.makedirs(sVal, 0o700);
141 except:
142 reporter.error('makedirs failed on the value of env.var. "%s": "%s"' % (sVar, sVal));
143 raise GenError('makedirs failed on the value of env.var. "%s": "%s"' % (sVar, sVal));
144 except:
145 if sAlternative is None:
146 if reporter.isLocal() and fLocalReq:
147 raise;
148 sVal = None;
149 else:
150 sVal = os.path.abspath(sAlternative);
151 return sVal;
152
153def timestampMilli():
154 """
155 Gets a millisecond timestamp.
156 """
157 if sys.platform == 'win32':
158 return long(time.clock() * 1000);
159 return long(time.time() * 1000);
160
161def timestampNano():
162 """
163 Gets a nanosecond timestamp.
164 """
165 if sys.platform == 'win32':
166 return long(time.clock() * 1000000000);
167 return long(time.time() * 1000000000);
168
169def tryGetHostByName(sName):
170 """
171 Wrapper around gethostbyname.
172 """
173 if sName is not None:
174 try:
175 sIpAddr = socket.gethostbyname(sName);
176 except:
177 reporter.errorXcpt('gethostbyname(%s)' % (sName));
178 else:
179 if sIpAddr != '0.0.0.0':
180 sName = sIpAddr;
181 else:
182 reporter.error('gethostbyname(%s) -> %s' % (sName, sIpAddr));
183 return sName;
184
185def __processSudoKill(uPid, iSignal, fSudo):
186 """
187 Does the sudo kill -signal pid thing if fSudo is true, else uses os.kill.
188 """
189 try:
190 if fSudo:
191 return utils.sudoProcessCall(['/bin/kill', '-%s' % (iSignal,), str(uPid)]) == 0;
192 os.kill(uPid, iSignal);
193 return True;
194 except:
195 reporter.logXcpt('uPid=%s' % (uPid,));
196 return False;
197
198def processInterrupt(uPid, fSudo = False):
199 """
200 Sends a SIGINT or equivalent to interrupt the specified process.
201 Returns True on success, False on failure.
202
203 On Windows hosts this may not work unless the process happens to be a
204 process group leader.
205 """
206 if sys.platform == 'win32':
207 fRc = winbase.processInterrupt(uPid)
208 else:
209 fRc = __processSudoKill(uPid, signal.SIGINT, fSudo);
210 return fRc;
211
212def sendUserSignal1(uPid, fSudo = False):
213 """
214 Sends a SIGUSR1 or equivalent to nudge the process into shutting down
215 (VBoxSVC) or something.
216 Returns True on success, False on failure or if not supported (win).
217
218 On Windows hosts this may not work unless the process happens to be a
219 process group leader.
220 """
221 if sys.platform == 'win32':
222 fRc = False;
223 else:
224 fRc = __processSudoKill(uPid, signal.SIGUSR1, fSudo); # pylint: disable=no-member
225 return fRc;
226
227def processTerminate(uPid, fSudo = False):
228 """
229 Terminates the process in a nice manner (SIGTERM or equivalent).
230 Returns True on success, False on failure (logged).
231 """
232 fRc = False;
233 if sys.platform == 'win32':
234 fRc = winbase.processTerminate(uPid);
235 else:
236 fRc = __processSudoKill(uPid, signal.SIGTERM, fSudo);
237 return fRc;
238
239def processKill(uPid, fSudo = False):
240 """
241 Terminates the process with extreme prejudice (SIGKILL).
242 Returns True on success, False on failure.
243 """
244 fRc = False;
245 if sys.platform == 'win32':
246 fRc = winbase.processKill(uPid);
247 else:
248 fRc = __processSudoKill(uPid, signal.SIGKILL, fSudo); # pylint: disable=no-member
249 return fRc;
250
251def processKillWithNameCheck(uPid, sName):
252 """
253 Like processKill(), but checks if the process name matches before killing
254 it. This is intended for killing using potentially stale pid values.
255
256 Returns True on success, False on failure.
257 """
258
259 if processCheckPidAndName(uPid, sName) is not True:
260 return False;
261 return processKill(uPid);
262
263
264def processExists(uPid):
265 """
266 Checks if the specified process exits.
267 This will only work if we can signal/open the process.
268
269 Returns True if it positively exists, False otherwise.
270 """
271 if sys.platform == 'win32':
272 fRc = winbase.processExists(uPid);
273 else:
274 try:
275 os.kill(uPid, 0);
276 fRc = True;
277 except:
278 reporter.logXcpt('uPid=%s' % (uPid,));
279 fRc = False;
280 return fRc;
281
282def processCheckPidAndName(uPid, sName):
283 """
284 Checks if a process PID and NAME matches.
285 """
286 if sys.platform == 'win32':
287 fRc = winbase.processCheckPidAndName(uPid, sName);
288 else:
289 sOs = utils.getHostOs();
290 if sOs == 'linux':
291 asPsCmd = ['/bin/ps', '-p', '%u' % (uPid,), '-o', 'fname='];
292 elif sOs == 'solaris':
293 asPsCmd = ['/usr/bin/ps', '-p', '%u' % (uPid,), '-o', 'fname='];
294 elif sOs == 'darwin':
295 asPsCmd = ['/bin/ps', '-p', '%u' % (uPid,), '-o', 'ucomm='];
296 else:
297 asPsCmd = None;
298
299 if asPsCmd is not None:
300 try:
301 oPs = subprocess.Popen(asPsCmd, stdout=subprocess.PIPE);
302 sCurName = oPs.communicate()[0];
303 iExitCode = oPs.wait();
304 except:
305 reporter.logXcpt();
306 return False;
307
308 # ps fails with non-zero exit code if the pid wasn't found.
309 if iExitCode != 0:
310 return False;
311 if sCurName is None:
312 return False;
313 sCurName = sCurName.strip();
314 if sCurName == '':
315 return False;
316
317 if os.path.basename(sName) == sName:
318 sCurName = os.path.basename(sCurName);
319 elif os.path.basename(sCurName) == sCurName:
320 sName = os.path.basename(sName);
321
322 if sCurName != sName:
323 return False;
324
325 fRc = True;
326 return fRc;
327
328def wipeDirectory(sDir):
329 """
330 Deletes all file and sub-directories in sDir, leaving sDir in empty afterwards.
331 Returns the number of errors after logging them as errors.
332 """
333 if not os.path.exists(sDir):
334 return 0;
335
336 try:
337 asNames = os.listdir(sDir);
338 except:
339 return reporter.errorXcpt('os.listdir("%s")' % (sDir));
340
341 cErrors = 0;
342 for sName in asNames:
343 # Build full path and lstat the object.
344 sFullName = os.path.join(sDir, sName)
345 try:
346 oStat = os.lstat(sFullName);
347 except:
348 reporter.errorXcpt('lstat("%s")' % (sFullName,));
349 cErrors = cErrors + 1;
350 continue;
351
352 if stat.S_ISDIR(oStat.st_mode):
353 # Directory - recurse and try remove it.
354 cErrors = cErrors + wipeDirectory(sFullName);
355 try:
356 os.rmdir(sFullName);
357 except:
358 reporter.errorXcpt('rmdir("%s")' % (sFullName,));
359 cErrors = cErrors + 1;
360 else:
361 # File, symlink, fifo or something - remove/unlink.
362 try:
363 os.remove(sFullName);
364 except:
365 reporter.errorXcpt('remove("%s")' % (sFullName,));
366 cErrors = cErrors + 1;
367 return cErrors;
368
369
370#
371# Classes
372#
373
374class GenError(Exception):
375 """
376 Exception class which only purpose it is to allow us to only catch our own
377 exceptions. Better design later.
378 """
379
380 def __init__(self, sWhat = "whatever"):
381 Exception.__init__(self);
382 self.sWhat = sWhat
383
384 def str(self):
385 """Get the message string."""
386 return self.sWhat;
387
388
389class InvalidOption(GenError):
390 """
391 Exception thrown by TestDriverBase.parseOption(). It contains the error message.
392 """
393 def __init__(self, sWhat):
394 GenError.__init__(self, sWhat);
395
396
397class QuietInvalidOption(GenError):
398 """
399 Exception thrown by TestDriverBase.parseOption(). Error already printed, just
400 return failure.
401 """
402 def __init__(self):
403 GenError.__init__(self, "");
404
405
406class TdTaskBase(object):
407 """
408 The base task.
409 """
410
411 def __init__(self, sCaller, fnProcessEvents = None):
412 self.sDbgCreated = '%s: %s' % (utils.getTimePrefix(), sCaller);
413 self.fSignalled = False;
414 self.__oRLock = threading.RLock();
415 self.oCv = threading.Condition(self.__oRLock);
416 self.oOwner = None;
417 self.msStart = timestampMilli();
418 self.oLocker = None;
419
420 ## Callback function that takes no parameters and will not be called holding the lock.
421 ## It is a hack to work the XPCOM and COM event queues, so we won't hold back events
422 ## that could block task progress (i.e. hangs VM).
423 self.fnProcessEvents = fnProcessEvents;
424
425 def __del__(self):
426 """In case we need it later on."""
427 pass; # pylint: disable=unnecessary-pass
428
429 def toString(self):
430 """
431 Stringifies the object, mostly as a debug aid.
432 """
433 return '<%s: fSignalled=%s, __oRLock=%s, oCv=%s, oOwner=%s, oLocker=%s, msStart=%s, sDbgCreated=%s>' \
434 % (type(self).__name__, self.fSignalled, self.__oRLock, self.oCv, repr(self.oOwner), self.oLocker, self.msStart,
435 self.sDbgCreated,);
436
437 def __str__(self):
438 return self.toString();
439
440 def lockTask(self):
441 """ Wrapper around oCv.acquire(). """
442 if True is True: # change to False for debugging deadlocks. # pylint: disable=comparison-with-itself
443 self.oCv.acquire();
444 else:
445 msStartWait = timestampMilli();
446 while self.oCv.acquire(0) is False:
447 if timestampMilli() - msStartWait > 30*1000:
448 reporter.error('!!! timed out waiting for %s' % (self, ));
449 traceback.print_stack();
450 reporter.logAllStacks()
451 self.oCv.acquire();
452 break;
453 time.sleep(0.5);
454 self.oLocker = thread.get_ident()
455 return None;
456
457 def unlockTask(self):
458 """ Wrapper around oCv.release(). """
459 self.oLocker = None;
460 self.oCv.release();
461 return None;
462
463 def getAgeAsMs(self):
464 """
465 Returns the number of milliseconds the task has existed.
466 """
467 return timestampMilli() - self.msStart;
468
469 def setTaskOwner(self, oOwner):
470 """
471 Sets or clears the task owner. (oOwner can be None.)
472
473 Returns the previous owner, this means None if not owned.
474 """
475 self.lockTask();
476 oOldOwner = self.oOwner;
477 self.oOwner = oOwner;
478 self.unlockTask();
479 return oOldOwner;
480
481 def signalTaskLocked(self):
482 """
483 Variant of signalTask that can be called while owning the lock.
484 """
485 fOld = self.fSignalled;
486 if not fOld:
487 reporter.log2('signalTaskLocked(%s)' % (self,));
488 self.fSignalled = True;
489 self.oCv.notifyAll()
490 if self.oOwner is not None:
491 self.oOwner.notifyAboutReadyTask(self);
492 return fOld;
493
494 def signalTask(self):
495 """
496 Signals the task, internal use only.
497
498 Returns the previous state.
499 """
500 self.lockTask();
501 fOld = self.signalTaskLocked();
502 self.unlockTask();
503 return fOld
504
505 def resetTaskLocked(self):
506 """
507 Variant of resetTask that can be called while owning the lock.
508 """
509 fOld = self.fSignalled;
510 self.fSignalled = False;
511 return fOld;
512
513 def resetTask(self):
514 """
515 Resets the task signal, internal use only.
516
517 Returns the previous state.
518 """
519 self.lockTask();
520 fOld = self.resetTaskLocked();
521 self.unlockTask();
522 return fOld
523
524 def pollTask(self, fLocked = False):
525 """
526 Poll the signal status of the task.
527 Returns True if signalled, False if not.
528
529 Override this method.
530 """
531 if not fLocked:
532 self.lockTask();
533 fState = self.fSignalled;
534 if not fLocked:
535 self.unlockTask();
536 return fState
537
538 def waitForTask(self, cMsTimeout = 0):
539 """
540 Waits for the task to be signalled.
541
542 Returns True if the task is/became ready before the timeout expired.
543 Returns False if the task is still not after cMsTimeout have elapsed.
544
545 Overriable.
546 """
547 if self.fnProcessEvents:
548 self.fnProcessEvents();
549
550 self.lockTask();
551
552 fState = self.pollTask(True);
553 if not fState:
554 # Don't wait more than 1s. This allow lazy state polling and avoid event processing trouble.
555 msStart = timestampMilli();
556 while not fState:
557 cMsElapsed = timestampMilli() - msStart;
558 if cMsElapsed >= cMsTimeout:
559 break;
560
561 cMsWait = cMsTimeout - cMsElapsed
562 if cMsWait > 1000:
563 cMsWait = 1000;
564 try:
565 self.oCv.wait(cMsWait / 1000.0);
566 except:
567 pass;
568
569 if self.fnProcessEvents:
570 self.unlockTask();
571 self.fnProcessEvents();
572 self.lockTask();
573
574 reporter.doPollWork('TdTaskBase.waitForTask');
575 fState = self.pollTask(True);
576
577 self.unlockTask();
578
579 if self.fnProcessEvents:
580 self.fnProcessEvents();
581
582 return fState;
583
584
585class Process(TdTaskBase):
586 """
587 Child Process.
588 """
589
590 def __init__(self, sName, asArgs, uPid, hWin = None, uTid = None):
591 TdTaskBase.__init__(self, utils.getCallerName());
592 self.sName = sName;
593 self.asArgs = asArgs;
594 self.uExitCode = -127;
595 self.uPid = uPid;
596 self.hWin = hWin;
597 self.uTid = uTid;
598 self.sKindCrashReport = None;
599 self.sKindCrashDump = None;
600
601 def toString(self):
602 return '<%s uExitcode=%s, uPid=%s, sName=%s, asArgs=%s, hWin=%s, uTid=%s>' \
603 % (TdTaskBase.toString(self), self.uExitCode, self.uPid, self.sName, self.asArgs, self.hWin, self.uTid);
604
605 #
606 # Instantiation methods.
607 #
608
609 @staticmethod
610 def spawn(sName, *asArgsIn):
611 """
612 Similar to os.spawnl(os.P_NOWAIT,).
613
614 """
615 # Make argument array (can probably use asArgsIn directly, but wtf).
616 asArgs = [];
617 for sArg in asArgsIn:
618 asArgs.append(sArg);
619
620 # Special case: Windows.
621 if sys.platform == 'win32':
622 (uPid, hProcess, uTid) = winbase.processCreate(searchPath(sName), asArgs);
623 if uPid == -1:
624 return None;
625 return Process(sName, asArgs, uPid, hProcess, uTid);
626
627 # Unixy.
628 try:
629 uPid = os.spawnv(os.P_NOWAIT, sName, asArgs);
630 except:
631 reporter.logXcpt('sName=%s' % (sName,));
632 return None;
633 return Process(sName, asArgs, uPid);
634
635 @staticmethod
636 def spawnp(sName, *asArgsIn):
637 """
638 Similar to os.spawnlp(os.P_NOWAIT,).
639
640 """
641 return Process.spawn(searchPath(sName), *asArgsIn);
642
643 #
644 # Task methods
645 #
646
647 def pollTask(self, fLocked = False):
648 """
649 Overridden pollTask method.
650 """
651 if not fLocked:
652 self.lockTask();
653
654 fRc = self.fSignalled;
655 if not fRc:
656 if sys.platform == 'win32':
657 if winbase.processPollByHandle(self.hWin):
658 try:
659 (uPid, uStatus) = os.waitpid(self.hWin, 0);
660 if uPid in (self.hWin, self.uPid,):
661 self.hWin.Detach(); # waitpid closed it, so it's now invalid.
662 self.hWin = None;
663 uPid = self.uPid;
664 except:
665 reporter.logXcpt();
666 uPid = self.uPid;
667 uStatus = 0xffffffff;
668 else:
669 uPid = 0;
670 uStatus = 0; # pylint: disable=redefined-variable-type
671 else:
672 try:
673 (uPid, uStatus) = os.waitpid(self.uPid, os.WNOHANG); # pylint: disable=no-member
674 except:
675 reporter.logXcpt();
676 uPid = self.uPid;
677 uStatus = 0xffffffff;
678
679 # Got anything?
680 if uPid == self.uPid:
681 self.uExitCode = uStatus;
682 reporter.log('Process %u -> %u (%#x)' % (uPid, uStatus, uStatus));
683 self.signalTaskLocked();
684 if self.uExitCode != 0 and (self.sKindCrashReport is not None or self.sKindCrashDump is not None):
685 reporter.error('Process "%s" returned/crashed with a non-zero status code!! rc=%u sig=%u%s (raw=%#x)'
686 % ( self.sName, self.uExitCode >> 8, self.uExitCode & 0x7f,
687 ' w/ core' if self.uExitCode & 0x80 else '', self.uExitCode))
688 utils.processCollectCrashInfo(self.uPid, reporter.log, self._addCrashFile);
689
690 fRc = self.fSignalled;
691 if not fLocked:
692 self.unlockTask();
693 return fRc;
694
695 def _addCrashFile(self, sFile, fBinary):
696 """
697 Helper for adding a crash report or dump to the test report.
698 """
699 sKind = self.sKindCrashDump if fBinary else self.sKindCrashReport;
700 if sKind is not None:
701 reporter.addLogFile(sFile, sKind);
702 return None;
703
704
705 #
706 # Methods
707 #
708
709 def enableCrashReporting(self, sKindCrashReport, sKindCrashDump):
710 """
711 Enabling (or disables) automatic crash reporting on systems where that
712 is possible. The two file kind parameters are on the form
713 'crash/log/client' and 'crash/dump/client'. If both are None,
714 reporting will be disabled.
715 """
716 self.sKindCrashReport = sKindCrashReport;
717 self.sKindCrashDump = sKindCrashDump;
718 return True;
719
720 def isRunning(self):
721 """
722 Returns True if the process is still running, False if not.
723 """
724 return not self.pollTask();
725
726 def wait(self, cMsTimeout = 0):
727 """
728 Wait for the process to exit.
729
730 Returns True if the process exited withint the specified wait period.
731 Returns False if still running.
732 """
733 return self.waitForTask(cMsTimeout);
734
735 def getExitCode(self):
736 """
737 Returns the exit code of the process.
738 The process must have exited or the result will be wrong.
739 """
740 if self.isRunning():
741 return -127;
742 return self.uExitCode >> 8;
743
744 def interrupt(self):
745 """
746 Sends a SIGINT or equivalent to interrupt the process.
747 Returns True on success, False on failure.
748
749 On Windows hosts this may not work unless the process happens to be a
750 process group leader.
751 """
752 if sys.platform == 'win32':
753 return winbase.postThreadMesssageQuit(self.uTid);
754 return processInterrupt(self.uPid);
755
756 def sendUserSignal1(self):
757 """
758 Sends a SIGUSR1 or equivalent to nudge the process into shutting down
759 (VBoxSVC) or something.
760 Returns True on success, False on failure.
761
762 On Windows hosts this may not work unless the process happens to be a
763 process group leader.
764 """
765 #if sys.platform == 'win32':
766 # return winbase.postThreadMesssageClose(self.uTid);
767 return sendUserSignal1(self.uPid);
768
769 def terminate(self):
770 """
771 Terminates the process in a nice manner (SIGTERM or equivalent).
772 Returns True on success, False on failure (logged).
773 """
774 if sys.platform == 'win32':
775 return winbase.processTerminateByHandle(self.hWin);
776 return processTerminate(self.uPid);
777
778 def getPid(self):
779 """ Returns the process id. """
780 return self.uPid;
781
782
783class SubTestDriverBase(object):
784 """
785 The base sub-test driver.
786
787 It helps thinking of these as units/sets/groups of tests, where the test
788 cases are (mostly) realized in python.
789
790 The sub-test drivers are subordinates of one or more test drivers. They
791 can be viewed as test code libraries that is responsible for parts of a
792 test driver run in different setups. One example would be testing a guest
793 additions component, which is applicable both to freshly installed guest
794 additions and VMs with old guest.
795
796 The test drivers invokes the sub-test drivers in a private manner during
797 test execution, but some of the generic bits are done automagically by the
798 base class: options, help, resources, various other actions.
799 """
800
801 def __init__(self, oTstDrv, sName, sTestName):
802 self.oTstDrv = oTstDrv # type: TestDriverBase
803 self.sName = sName; # For use with options (--enable-sub-driver sName:sName2)
804 self.sTestName = sTestName; # More descriptive for passing to reporter.testStart().
805 self.asRsrcs = [] # type: List(str)
806 self.fEnabled = True; # TestDriverBase --enable-sub-driver and --disable-sub-driver.
807
808 def showUsage(self):
809 """
810 Show usage information if any.
811
812 The default implementation only prints the name.
813 """
814 reporter.log('');
815 reporter.log('Options for sub-test driver %s (%s):' % (self.sTestName, self.sName,));
816 return True;
817
818 def parseOption(self, asArgs, iArg):
819 """
820 Parse an option. Override this.
821
822 @param asArgs The argument vector.
823 @param iArg The index of the current argument.
824
825 @returns The index of the next argument if consumed, @a iArg if not.
826
827 @throws InvalidOption or QuietInvalidOption on syntax error or similar.
828 """
829 _ = asArgs;
830 return iArg;
831
832
833class TestDriverBase(object): # pylint: disable=too-many-instance-attributes
834 """
835 The base test driver.
836 """
837
838 def __init__(self):
839 self.fInterrupted = False;
840
841 # Actions.
842 self.asSpecialActions = ['extract', 'abort'];
843 self.asNormalActions = ['cleanup-before', 'verify', 'config', 'execute', 'cleanup-after' ];
844 self.asActions = [];
845 self.sExtractDstPath = None;
846
847 # Options.
848 self.fNoWipeClean = False;
849
850 # Tasks - only accessed by one thread atm, so no need for locking.
851 self.aoTasks = [];
852
853 # Host info.
854 self.sHost = utils.getHostOs();
855 self.sHostArch = utils.getHostArch();
856
857 # Skipped status modifier (see end of innerMain()).
858 self.fBadTestbox = False;
859
860 #
861 # Get our bearings and adjust the environment.
862 #
863 if not utils.isRunningFromCheckout():
864 self.sBinPath = os.path.join(g_ksValidationKitDir, utils.getHostOs(), utils.getHostArch());
865 else:
866 self.sBinPath = os.path.join(g_ksValidationKitDir, os.pardir, os.pardir, os.pardir, 'out', utils.getHostOsDotArch(),
867 os.environ.get('KBUILD_TYPE', os.environ.get('BUILD_TYPE', 'debug')),
868 'validationkit', utils.getHostOs(), utils.getHostArch());
869 self.sOrgShell = os.environ.get('SHELL');
870 self.sOurShell = os.path.join(self.sBinPath, 'vts_shell' + exeSuff()); # No shell yet.
871 os.environ['SHELL'] = self.sOurShell;
872
873 self.sScriptPath = getDirEnv('TESTBOX_PATH_SCRIPTS');
874 if self.sScriptPath is None:
875 self.sScriptPath = os.path.abspath(os.path.join(os.getcwd(), '..'));
876 os.environ['TESTBOX_PATH_SCRIPTS'] = self.sScriptPath;
877
878 self.sScratchPath = getDirEnv('TESTBOX_PATH_SCRATCH', fTryCreate = True);
879 if self.sScratchPath is None:
880 sTmpDir = tempfile.gettempdir();
881 if sTmpDir == '/tmp': # /var/tmp is generally more suitable on all platforms.
882 sTmpDir = '/var/tmp';
883 self.sScratchPath = os.path.abspath(os.path.join(sTmpDir, 'VBoxTestTmp'));
884 if not os.path.isdir(self.sScratchPath):
885 os.makedirs(self.sScratchPath, 0o700);
886 os.environ['TESTBOX_PATH_SCRATCH'] = self.sScratchPath;
887
888 self.sTestBoxName = getEnv( 'TESTBOX_NAME', 'local');
889 self.sTestSetId = getEnv( 'TESTBOX_TEST_SET_ID', 'local');
890 self.sBuildPath = getDirEnv('TESTBOX_PATH_BUILDS');
891 self.sUploadPath = getDirEnv('TESTBOX_PATH_UPLOAD');
892 self.sResourcePath = getDirEnv('TESTBOX_PATH_RESOURCES');
893 if self.sResourcePath is None:
894 if self.sHost == 'darwin': self.sResourcePath = "/Volumes/testrsrc/";
895 elif self.sHost == 'freebsd': self.sResourcePath = "/mnt/testrsrc/";
896 elif self.sHost == 'linux': self.sResourcePath = "/mnt/testrsrc/";
897 elif self.sHost == 'os2': self.sResourcePath = "T:/";
898 elif self.sHost == 'solaris': self.sResourcePath = "/mnt/testrsrc/";
899 elif self.sHost == 'win': self.sResourcePath = "T:/";
900 else: raise GenError('unknown host OS "%s"' % (self.sHost));
901
902 # PID file for the testdriver.
903 self.sPidFile = os.path.join(self.sScratchPath, 'testdriver.pid');
904
905 # Some stuff for the log...
906 reporter.log('scratch: %s' % (self.sScratchPath,));
907
908 # Get the absolute timeout (seconds since epoch, see
909 # utils.timestampSecond()). None if not available.
910 self.secTimeoutAbs = os.environ.get('TESTBOX_TIMEOUT_ABS', None);
911 if self.secTimeoutAbs is not None:
912 self.secTimeoutAbs = long(self.secTimeoutAbs);
913 reporter.log('secTimeoutAbs: %s' % (self.secTimeoutAbs,));
914 else:
915 reporter.log('TESTBOX_TIMEOUT_ABS not found in the environment');
916
917 # Distance from secTimeoutAbs that timeouts should be adjusted to.
918 self.secTimeoutFudge = 30;
919
920 # List of sub-test drivers (SubTestDriverBase derivatives).
921 self.aoSubTstDrvs = [] # type: list(SubTestDriverBase)
922
923 # Use the scratch path for temporary files.
924 if self.sHost in ['win', 'os2']:
925 os.environ['TMP'] = self.sScratchPath;
926 os.environ['TEMP'] = self.sScratchPath;
927 os.environ['TMPDIR'] = self.sScratchPath;
928 os.environ['IPRT_TMPDIR'] = self.sScratchPath; # IPRT/VBox specific.
929
930
931 #
932 # Resource utility methods.
933 #
934
935 def isResourceFile(self, sFile):
936 """
937 Checks if sFile is in in the resource set.
938 """
939 ## @todo need to deal with stuff in the validationkit.zip and similar.
940 asRsrcs = self.getResourceSet();
941 if sFile in asRsrcs:
942 return os.path.isfile(os.path.join(self.sResourcePath, sFile));
943 for sRsrc in asRsrcs:
944 if sFile.startswith(sRsrc):
945 sFull = os.path.join(self.sResourcePath, sRsrc);
946 if os.path.isdir(sFull):
947 return os.path.isfile(os.path.join(self.sResourcePath, sRsrc));
948 return False;
949
950 def getFullResourceName(self, sName):
951 """
952 Returns the full resource name.
953 """
954 if os.path.isabs(sName): ## @todo Hack. Need to deal properly with stuff in the validationkit.zip and similar.
955 return sName;
956 return os.path.join(self.sResourcePath, sName);
957
958 #
959 # Scratch related utility methods.
960 #
961
962 def wipeScratch(self):
963 """
964 Removes the content of the scratch directory.
965 Returns True on no errors, False + log entries on errors.
966 """
967 cErrors = wipeDirectory(self.sScratchPath);
968 return cErrors == 0;
969
970 #
971 # Sub-test driver related methods.
972 #
973
974 def addSubTestDriver(self, oSubTstDrv):
975 """
976 Adds a sub-test driver.
977
978 Returns True on success, false on failure.
979 """
980 assert isinstance(oSubTstDrv, SubTestDriverBase);
981 if oSubTstDrv in self.aoSubTstDrvs:
982 reporter.error('Attempt at adding sub-test driver %s twice.' % (oSubTstDrv.sName,));
983 return False;
984 self.aoSubTstDrvs.append(oSubTstDrv);
985 return True;
986
987 def showSubTstDrvUsage(self):
988 """
989 Shows the usage of the sub-test drivers.
990 """
991 for oSubTstDrv in self.aoSubTstDrvs:
992 oSubTstDrv.showUsage();
993 return True;
994
995 def subTstDrvParseOption(self, asArgs, iArgs):
996 """
997 Lets the sub-test drivers have a go at the option.
998 Returns the index of the next option if handled, otherwise iArgs.
999 """
1000 for oSubTstDrv in self.aoSubTstDrvs:
1001 iNext = oSubTstDrv.parseOption(asArgs, iArgs)
1002 if iNext != iArgs:
1003 assert iNext > iArgs;
1004 assert iNext <= len(asArgs);
1005 return iNext;
1006 return iArgs;
1007
1008 def findSubTstDrvByShortName(self, sShortName):
1009 """
1010 Locates a sub-test driver by it's short name.
1011 Returns sub-test driver object reference if found, None if not.
1012 """
1013 for oSubTstDrv in self.aoSubTstDrvs:
1014 if oSubTstDrv.sName == sShortName:
1015 return oSubTstDrv;
1016 return None;
1017
1018
1019 #
1020 # Task related methods.
1021 #
1022
1023 def addTask(self, oTask):
1024 """
1025 Adds oTask to the task list.
1026
1027 Returns True if the task was added.
1028
1029 Returns False if the task was already in the task list.
1030 """
1031 if oTask in self.aoTasks:
1032 return False;
1033 #reporter.log2('adding task %s' % (oTask,));
1034 self.aoTasks.append(oTask);
1035 oTask.setTaskOwner(self);
1036 #reporter.log2('tasks now in list: %d - %s' % (len(self.aoTasks), self.aoTasks));
1037 return True;
1038
1039 def removeTask(self, oTask):
1040 """
1041 Removes oTask to the task list.
1042
1043 Returns oTask on success and None on failure.
1044 """
1045 try:
1046 #reporter.log2('removing task %s' % (oTask,));
1047 self.aoTasks.remove(oTask);
1048 except:
1049 return None;
1050 else:
1051 oTask.setTaskOwner(None);
1052 #reporter.log2('tasks left: %d - %s' % (len(self.aoTasks), self.aoTasks));
1053 return oTask;
1054
1055 def removeAllTasks(self):
1056 """
1057 Removes all the task from the task list.
1058
1059 Returns None.
1060 """
1061 aoTasks = self.aoTasks;
1062 self.aoTasks = [];
1063 for oTask in aoTasks:
1064 oTask.setTaskOwner(None);
1065 return None;
1066
1067 def notifyAboutReadyTask(self, oTask):
1068 """
1069 Notificiation that there is a ready task. May be called owning the
1070 task lock, so be careful wrt deadlocks.
1071
1072 Remember to call super when overriding this.
1073 """
1074 if oTask is None: pass; # lint
1075 return None;
1076
1077 def pollTasks(self):
1078 """
1079 Polls the task to see if any of them are ready.
1080 Returns the ready task, None if none are ready.
1081 """
1082 for oTask in self.aoTasks:
1083 if oTask.pollTask():
1084 return oTask;
1085 return None;
1086
1087 def waitForTasksSleepWorker(self, cMsTimeout):
1088 """
1089 Overridable method that does the sleeping for waitForTask().
1090
1091 cMsTimeout will not be larger than 1000, so there is normally no need
1092 to do any additional splitting up of the polling interval.
1093
1094 Returns True if cMillieSecs elapsed.
1095 Returns False if some exception was raised while we waited or
1096 there turned out to be nothing to wait on.
1097 """
1098 try:
1099 self.aoTasks[0].waitForTask(cMsTimeout);
1100 return True;
1101 except Exception as oXcpt:
1102 reporter.log("waitForTasksSleepWorker: %s" % (str(oXcpt),));
1103 return False;
1104
1105 def waitForTasks(self, cMsTimeout):
1106 """
1107 Waits for any of the tasks to require attention or a KeyboardInterrupt.
1108 Returns the ready task on success, None on timeout or interrupt.
1109 """
1110 try:
1111 #reporter.log2('waitForTasks: cMsTimeout=%d' % (cMsTimeout,));
1112
1113 if cMsTimeout == 0:
1114 return self.pollTasks();
1115
1116 if not self.aoTasks:
1117 return None;
1118
1119 fMore = True;
1120 if cMsTimeout < 0:
1121 while fMore:
1122 oTask = self.pollTasks();
1123 if oTask is not None:
1124 return oTask;
1125 fMore = self.waitForTasksSleepWorker(1000);
1126 else:
1127 msStart = timestampMilli();
1128 while fMore:
1129 oTask = self.pollTasks();
1130 if oTask is not None:
1131 #reporter.log2('waitForTasks: returning %s, msStart=%d' % \
1132 # (oTask, msStart));
1133 return oTask;
1134
1135 cMsElapsed = timestampMilli() - msStart;
1136 if cMsElapsed > cMsTimeout: # not ==, we want the final waitForEvents.
1137 break;
1138 cMsSleep = cMsTimeout - cMsElapsed;
1139 if cMsSleep > 1000:
1140 cMsSleep = 1000;
1141 fMore = self.waitForTasksSleepWorker(cMsSleep);
1142 except KeyboardInterrupt:
1143 self.fInterrupted = True;
1144 reporter.errorXcpt('KeyboardInterrupt', 6);
1145 except:
1146 reporter.errorXcpt(None, 6);
1147 return None;
1148
1149 #
1150 # PID file management methods.
1151 #
1152
1153 def pidFileRead(self):
1154 """
1155 Worker that reads the PID file.
1156 Returns dictionary of PID with value (sName, fSudo), empty if no file.
1157 """
1158 dPids = {};
1159 if os.path.isfile(self.sPidFile):
1160 try:
1161 oFile = utils.openNoInherit(self.sPidFile, 'r');
1162 sContent = str(oFile.read());
1163 oFile.close();
1164 except:
1165 reporter.errorXcpt();
1166 return dPids;
1167
1168 sContent = str(sContent).strip().replace('\n', ' ').replace('\r', ' ').replace('\t', ' ');
1169 for sProcess in sContent.split(' '):
1170 asFields = sProcess.split(':');
1171 if len(asFields) == 3 and asFields[0].isdigit():
1172 try:
1173 dPids[int(asFields[0])] = (asFields[2], asFields[1] == 'sudo');
1174 except:
1175 reporter.logXcpt('sProcess=%s' % (sProcess,));
1176 else:
1177 reporter.log('%s: "%s"' % (self.sPidFile, sProcess));
1178
1179 return dPids;
1180
1181 def pidFileAdd(self, iPid, sName, fSudo = False):
1182 """
1183 Adds a PID to the PID file, creating the file if necessary.
1184 """
1185 try:
1186 oFile = utils.openNoInherit(self.sPidFile, 'a');
1187 oFile.write('%s:%s:%s\n'
1188 % ( iPid,
1189 'sudo' if fSudo else 'normal',
1190 sName.replace(' ', '_').replace(':','_').replace('\n','_').replace('\r','_').replace('\t','_'),));
1191 oFile.close();
1192 except:
1193 reporter.errorXcpt();
1194 return False;
1195 ## @todo s/log/log2/
1196 reporter.log('pidFileAdd: added %s (%#x) %s fSudo=%s (new content: %s)'
1197 % (iPid, iPid, sName, fSudo, self.pidFileRead(),));
1198 return True;
1199
1200 def pidFileRemove(self, iPid, fQuiet = False):
1201 """
1202 Removes a PID from the PID file.
1203 """
1204 dPids = self.pidFileRead();
1205 if iPid not in dPids:
1206 if not fQuiet:
1207 reporter.log('pidFileRemove could not find %s in the PID file (content: %s)' % (iPid, dPids));
1208 return False;
1209
1210 sName = dPids[iPid][0];
1211 del dPids[iPid];
1212
1213 sPid = '';
1214 for iPid2 in dPids:
1215 sPid += '%s:%s:%s\n' % (iPid2, 'sudo' if dPids[iPid2][1] else 'normal', dPids[iPid2][0]);
1216
1217 try:
1218 oFile = utils.openNoInherit(self.sPidFile, 'w');
1219 oFile.write(sPid);
1220 oFile.close();
1221 except:
1222 reporter.errorXcpt();
1223 return False;
1224 ## @todo s/log/log2/
1225 reporter.log('pidFileRemove: removed PID %d [%s] (new content: %s)' % (iPid, sName, self.pidFileRead(),));
1226 return True;
1227
1228 def pidFileDelete(self):
1229 """Creates the testdriver PID file."""
1230 if os.path.isfile(self.sPidFile):
1231 try:
1232 os.unlink(self.sPidFile);
1233 except:
1234 reporter.logXcpt();
1235 return False;
1236 ## @todo s/log/log2/
1237 reporter.log('pidFileDelete: deleted "%s"' % (self.sPidFile,));
1238 return True;
1239
1240 #
1241 # Misc helper methods.
1242 #
1243
1244 def requireMoreArgs(self, cMinNeeded, asArgs, iArg):
1245 """
1246 Checks that asArgs has at least cMinNeeded args following iArg.
1247
1248 Returns iArg + 1 if it checks out fine.
1249 Raise appropritate exception if not, ASSUMING that the current argument
1250 is found at iArg.
1251 """
1252 assert cMinNeeded >= 1;
1253 if iArg + cMinNeeded > len(asArgs):
1254 if cMinNeeded > 1:
1255 raise InvalidOption('The "%s" option takes %s values' % (asArgs[iArg], cMinNeeded,));
1256 raise InvalidOption('The "%s" option takes 1 value' % (asArgs[iArg],));
1257 return iArg + 1;
1258
1259 def getBinTool(self, sName):
1260 """
1261 Returns the full path to the given binary validation kit tool.
1262 """
1263 return os.path.join(self.sBinPath, sName) + exeSuff();
1264
1265 def adjustTimeoutMs(self, cMsTimeout, cMsMinimum = None):
1266 """
1267 Adjusts the given timeout (milliseconds) to take TESTBOX_TIMEOUT_ABS
1268 and cMsMinimum (optional) into account.
1269
1270 Returns adjusted timeout.
1271 Raises no exceptions.
1272 """
1273 if self.secTimeoutAbs is not None:
1274 cMsToDeadline = self.secTimeoutAbs * 1000 - utils.timestampMilli();
1275 if cMsToDeadline >= 0:
1276 # Adjust for fudge and enforce the minimum timeout
1277 cMsToDeadline -= self.secTimeoutFudge * 1000;
1278 if cMsToDeadline < (cMsMinimum if cMsMinimum is not None else 10000):
1279 cMsToDeadline = cMsMinimum if cMsMinimum is not None else 10000;
1280
1281 # Is the timeout beyond the (adjusted) deadline, if so change it.
1282 if cMsTimeout > cMsToDeadline:
1283 reporter.log('adjusting timeout: %s ms -> %s ms (deadline)\n' % (cMsTimeout, cMsToDeadline,));
1284 return cMsToDeadline;
1285 reporter.log('adjustTimeoutMs: cMsTimeout (%s) > cMsToDeadline (%s)' % (cMsTimeout, cMsToDeadline,));
1286 else:
1287 # Don't bother, we've passed the deadline.
1288 reporter.log('adjustTimeoutMs: ooops! cMsToDeadline=%s (%s), timestampMilli()=%s, timestampSecond()=%s'
1289 % (cMsToDeadline, cMsToDeadline*1000, utils.timestampMilli(), utils.timestampSecond()));
1290
1291 # Only enforce the minimum timeout if specified.
1292 if cMsMinimum is not None and cMsTimeout < cMsMinimum:
1293 reporter.log('adjusting timeout: %s ms -> %s ms (minimum)\n' % (cMsTimeout, cMsMinimum,));
1294 cMsTimeout = cMsMinimum;
1295
1296 return cMsTimeout;
1297
1298 def prepareResultFile(self, sName = 'results.xml'):
1299 """
1300 Given a base name (no path, but extension if required), a scratch file
1301 name is computed and any previous file removed.
1302
1303 Returns the full path to the file sName.
1304 Raises exception on failure.
1305 """
1306 sXmlFile = os.path.join(self.sScratchPath, sName);
1307 if os.path.exists(sXmlFile):
1308 os.unlink(sXmlFile);
1309 return sXmlFile;
1310
1311
1312 #
1313 # Overridable methods.
1314 #
1315
1316 def showUsage(self):
1317 """
1318 Shows the usage.
1319
1320 When overriding this, call super first.
1321 """
1322 sName = os.path.basename(sys.argv[0]);
1323 reporter.log('Usage: %s [options] <action(s)>' % (sName,));
1324 reporter.log('');
1325 reporter.log('Actions (in execution order):');
1326 reporter.log(' cleanup-before');
1327 reporter.log(' Cleanups done at the start of testing.');
1328 reporter.log(' verify');
1329 reporter.log(' Verify that all necessary resources are present.');
1330 reporter.log(' config');
1331 reporter.log(' Configure the tests.');
1332 reporter.log(' execute');
1333 reporter.log(' Execute the tests.');
1334 reporter.log(' cleanup-after');
1335 reporter.log(' Cleanups done at the end of the testing.');
1336 reporter.log('');
1337 reporter.log('Special Actions:');
1338 reporter.log(' all');
1339 reporter.log(' Alias for: %s' % (' '.join(self.asNormalActions),));
1340 reporter.log(' extract <path>');
1341 reporter.log(' Extract the test resources and put them in the specified');
1342 reporter.log(' path for off side/line testing.');
1343 reporter.log(' abort');
1344 reporter.log(' Aborts the test.');
1345 reporter.log('');
1346 reporter.log('Base Options:');
1347 reporter.log(' -h, --help');
1348 reporter.log(' Show this help message.');
1349 reporter.log(' -v, --verbose');
1350 reporter.log(' Increase logging verbosity, repeat for more logging.');
1351 reporter.log(' -d, --debug');
1352 reporter.log(' Increase the debug logging level, repeat for more info.');
1353 reporter.log(' --no-wipe-clean');
1354 reporter.log(' Do not wipe clean the scratch area during the two clean up');
1355 reporter.log(' actions. This is for facilitating nested test driver execution.');
1356 if self.aoSubTstDrvs:
1357 reporter.log(' --enable-sub-driver <sub1>[:..]');
1358 reporter.log(' --disable-sub-driver <sub1>[:..]');
1359 reporter.log(' Enables or disables one or more of the sub drivers: %s'
1360 % (', '.join([oSubTstDrv.sName for oSubTstDrv in self.aoSubTstDrvs]),));
1361 return True;
1362
1363 def parseOption(self, asArgs, iArg):
1364 """
1365 Parse an option. Override this.
1366
1367 Keyword arguments:
1368 asArgs -- The argument vector.
1369 iArg -- The index of the current argument.
1370
1371 Returns iArg if the option was not recognized.
1372 Returns the index of the next argument when something is consumed.
1373 In the event of a syntax error, a InvalidOption or QuietInvalidOption
1374 should be thrown.
1375 """
1376
1377 if asArgs[iArg] in ('--help', '-help', '-h', '-?', '/?', '/help', '/H', '-H'):
1378 self.showUsage();
1379 self.showSubTstDrvUsage();
1380 raise QuietInvalidOption();
1381
1382 # options
1383 if asArgs[iArg] in ('--verbose', '-v'):
1384 reporter.incVerbosity()
1385 elif asArgs[iArg] in ('--debug', '-d'):
1386 reporter.incDebug()
1387 elif asArgs[iArg] == '--no-wipe-clean':
1388 self.fNoWipeClean = True;
1389 elif asArgs[iArg] in ('--enable-sub-driver', '--disable-sub-driver') and self.aoSubTstDrvs:
1390 sOption = asArgs[iArg];
1391 iArg = self.requireMoreArgs(1, asArgs, iArg);
1392 for sSubTstDrvName in asArgs[iArg].split(':'):
1393 oSubTstDrv = self.findSubTstDrvByShortName(sSubTstDrvName);
1394 if oSubTstDrv is None:
1395 raise InvalidOption('Unknown sub-test driver given to %s: %s' % (sOption, sSubTstDrvName,));
1396 oSubTstDrv.fEnabled = sOption == '--enable-sub-driver';
1397 elif (asArgs[iArg] == 'all' or asArgs[iArg] in self.asNormalActions) \
1398 and self.asActions in self.asSpecialActions:
1399 raise InvalidOption('selected special action "%s" already' % (self.asActions[0], ));
1400 # actions
1401 elif asArgs[iArg] == 'all':
1402 self.asActions = [ 'all' ];
1403 elif asArgs[iArg] in self.asNormalActions:
1404 self.asActions.append(asArgs[iArg])
1405 elif asArgs[iArg] in self.asSpecialActions:
1406 if self.asActions != []:
1407 raise InvalidOption('selected special action "%s" already' % (self.asActions[0], ));
1408 self.asActions = [ asArgs[iArg] ];
1409 # extact <destination>
1410 if asArgs[iArg] == 'extract':
1411 iArg = iArg + 1;
1412 if iArg >= len(asArgs): raise InvalidOption('The "extract" action requires a destination directory');
1413 self.sExtractDstPath = asArgs[iArg];
1414 else:
1415 return iArg;
1416 return iArg + 1;
1417
1418 def completeOptions(self):
1419 """
1420 This method is called after parsing all the options.
1421 Returns success indicator. Use the reporter to complain.
1422
1423 Overriable, call super.
1424 """
1425 return True;
1426
1427 def getResourceSet(self):
1428 """
1429 Returns a set of file and/or directory names relative to
1430 TESTBOX_PATH_RESOURCES.
1431
1432 Override this, call super when using sub-test drivers.
1433 """
1434 asRsrcs = [];
1435 for oSubTstDrv in self.aoSubTstDrvs:
1436 asRsrcs.extend(oSubTstDrv.asRsrcs);
1437 return asRsrcs;
1438
1439 def actionExtract(self):
1440 """
1441 Handle the action that extracts the test resources for off site use.
1442 Returns a success indicator and error details with the reporter.
1443
1444 There is usually no need to override this.
1445 """
1446 fRc = True;
1447 asRsrcs = self.getResourceSet();
1448 for iRsrc, sRsrc in enumerate(asRsrcs):
1449 reporter.log('Resource #%s: "%s"' % (iRsrc, sRsrc));
1450 sSrcPath = os.path.normpath(os.path.abspath(os.path.join(self.sResourcePath, sRsrc.replace('/', os.path.sep))));
1451 sDstPath = os.path.normpath(os.path.join(self.sExtractDstPath, sRsrc.replace('/', os.path.sep)));
1452
1453 sDstDir = os.path.dirname(sDstPath);
1454 if not os.path.exists(sDstDir):
1455 try: os.makedirs(sDstDir, 0o775);
1456 except: fRc = reporter.errorXcpt('Error creating directory "%s":' % (sDstDir,));
1457
1458 if os.path.isfile(sSrcPath):
1459 try: utils.copyFileSimple(sSrcPath, sDstPath);
1460 except: fRc = reporter.errorXcpt('Error copying "%s" to "%s":' % (sSrcPath, sDstPath,));
1461 elif os.path.isdir(sSrcPath):
1462 fRc = reporter.error('Extracting directories have not been implemented yet');
1463 else:
1464 fRc = reporter.error('Missing or unsupported resource type: %s' % (sSrcPath,));
1465 return fRc;
1466
1467 def actionVerify(self):
1468 """
1469 Handle the action that verify the test resources.
1470 Returns a success indicator and error details with the reporter.
1471
1472 There is usually no need to override this.
1473 """
1474
1475 asRsrcs = self.getResourceSet();
1476 for sRsrc in asRsrcs:
1477 # Go thru some pain to catch escape sequences.
1478 if sRsrc.find("//") >= 0:
1479 reporter.error('Double slash test resource name: "%s"' % (sRsrc));
1480 return False;
1481 if sRsrc == ".." \
1482 or sRsrc.startswith("../") \
1483 or sRsrc.find("/../") >= 0 \
1484 or sRsrc.endswith("/.."):
1485 reporter.error('Relative path in test resource name: "%s"' % (sRsrc));
1486 return False;
1487
1488 sFull = os.path.normpath(os.path.abspath(os.path.join(self.sResourcePath, sRsrc)));
1489 if not sFull.startswith(os.path.normpath(self.sResourcePath)):
1490 reporter.error('sFull="%s" self.sResourcePath=%s' % (sFull, self.sResourcePath));
1491 reporter.error('The resource "%s" seems to specify a relative path' % (sRsrc));
1492 return False;
1493
1494 reporter.log2('Checking for resource "%s" at "%s" ...' % (sRsrc, sFull));
1495 if os.path.isfile(sFull):
1496 try:
1497 oFile = utils.openNoInherit(sFull, "rb");
1498 oFile.close();
1499 except Exception as oXcpt:
1500 reporter.error('The file resource "%s" cannot be accessed: %s' % (sFull, oXcpt));
1501 return False;
1502 elif os.path.isdir(sFull):
1503 if not os.path.isdir(os.path.join(sFull, '.')):
1504 reporter.error('The directory resource "%s" cannot be accessed' % (sFull));
1505 return False;
1506 elif os.path.exists(sFull):
1507 reporter.error('The resource "%s" is not a file or directory' % (sFull));
1508 return False;
1509 else:
1510 reporter.error('The resource "%s" was not found' % (sFull));
1511 return False;
1512 return True;
1513
1514 def actionConfig(self):
1515 """
1516 Handle the action that configures the test.
1517 Returns True (success), False (failure) or None (skip the test),
1518 posting complaints and explanations with the reporter.
1519
1520 Override this.
1521 """
1522 return True;
1523
1524 def actionExecute(self):
1525 """
1526 Handle the action that executes the test.
1527
1528 Returns True (success), False (failure) or None (skip the test),
1529 posting complaints and explanations with the reporter.
1530
1531 Override this.
1532 """
1533 return True;
1534
1535 def actionCleanupBefore(self):
1536 """
1537 Handle the action that cleans up spills from previous tests before
1538 starting the tests. This is mostly about wiping the scratch space
1539 clean in local runs. On a testbox the testbox script will use the
1540 cleanup-after if the test is interrupted.
1541
1542 Returns True (success), False (failure) or None (skip the test),
1543 posting complaints and explanations with the reporter.
1544
1545 Override this, but call super to wipe the scratch directory.
1546 """
1547 if self.fNoWipeClean is False:
1548 self.wipeScratch();
1549 return True;
1550
1551 def actionCleanupAfter(self):
1552 """
1553 Handle the action that cleans up all spills from executing the test.
1554
1555 Returns True (success) or False (failure) posting complaints and
1556 explanations with the reporter.
1557
1558 Override this, but call super to wipe the scratch directory.
1559 """
1560 if self.fNoWipeClean is False:
1561 self.wipeScratch();
1562 return True;
1563
1564 def actionAbort(self):
1565 """
1566 Handle the action that aborts a (presumed) running testdriver, making
1567 sure to include all it's children.
1568
1569 Returns True (success) or False (failure) posting complaints and
1570 explanations with the reporter.
1571
1572 Override this, but call super to kill the testdriver script and any
1573 other process covered by the testdriver PID file.
1574 """
1575
1576 dPids = self.pidFileRead();
1577 reporter.log('The pid file contained: %s' % (dPids,));
1578
1579 #
1580 # Try convince the processes to quit with increasing impoliteness.
1581 #
1582 if sys.platform == 'win32':
1583 afnMethods = [ processInterrupt, processTerminate ];
1584 else:
1585 afnMethods = [ sendUserSignal1, processInterrupt, processTerminate, processKill ];
1586 for fnMethod in afnMethods:
1587 for iPid in dPids:
1588 fnMethod(iPid, fSudo = dPids[iPid][1]);
1589
1590 for i in range(10):
1591 if i > 0:
1592 time.sleep(1);
1593
1594 for iPid in dPids:
1595 if not processExists(iPid):
1596 reporter.log('%s (%s) terminated' % (dPids[iPid][0], iPid,));
1597 self.pidFileRemove(iPid, fQuiet = True);
1598 del dPids[iPid];
1599
1600 if not dPids:
1601 reporter.log('All done.');
1602 return True;
1603
1604 if i in [4, 8]:
1605 reporter.log('Still waiting for: %s (method=%s)' % (dPids, fnMethod,));
1606
1607 reporter.log('Failed to terminate the following processes: %s' % (dPids,));
1608 return False;
1609
1610
1611 def onExit(self, iRc):
1612 """
1613 Hook for doing very important cleanups on the way out.
1614
1615 iRc is the exit code or -1 in the case of an unhandled exception.
1616 Returns nothing and shouldn't raise exceptions (will be muted+ignored).
1617 """
1618 _ = iRc;
1619 return None;
1620
1621
1622 #
1623 # main() - don't override anything!
1624 #
1625
1626 def main(self, asArgs = None):
1627 """
1628 The main function of the test driver.
1629
1630 Keyword arguments:
1631 asArgs -- The argument vector. Defaults to sys.argv.
1632
1633 Returns exit code. No exceptions.
1634 """
1635
1636 #
1637 # Wrap worker in exception handler and always call a 'finally' like
1638 # method to do crucial cleanups on the way out.
1639 #
1640 try:
1641 iRc = self.innerMain(asArgs);
1642 except:
1643 reporter.logXcpt(cFrames = None);
1644 try:
1645 self.onExit(-1);
1646 except:
1647 reporter.logXcpt();
1648 raise;
1649 self.onExit(iRc);
1650 return iRc;
1651
1652
1653 def innerMain(self, asArgs = None): # pylint: disable=too-many-statements
1654 """
1655 Exception wrapped main() worker.
1656 """
1657
1658 # parse the arguments.
1659 if asArgs is None:
1660 asArgs = list(sys.argv);
1661 iArg = 1;
1662 try:
1663 while iArg < len(asArgs):
1664 iNext = self.parseOption(asArgs, iArg);
1665 if iNext == iArg:
1666 iNext = self.subTstDrvParseOption(asArgs, iArg);
1667 if iNext == iArg:
1668 raise InvalidOption('unknown option: %s' % (asArgs[iArg]))
1669 iArg = iNext;
1670 except QuietInvalidOption as oXcpt:
1671 return rtexitcode.RTEXITCODE_SYNTAX;
1672 except InvalidOption as oXcpt:
1673 reporter.error(oXcpt.str());
1674 return rtexitcode.RTEXITCODE_SYNTAX;
1675 except:
1676 reporter.error('unexpected exception while parsing argument #%s' % (iArg));
1677 traceback.print_exc();
1678 return rtexitcode.RTEXITCODE_SYNTAX;
1679
1680 if not self.completeOptions():
1681 return rtexitcode.RTEXITCODE_SYNTAX;
1682
1683 if self.asActions == []:
1684 reporter.error('no action was specified');
1685 reporter.error('valid actions: %s' % (self.asNormalActions + self.asSpecialActions + ['all']));
1686 return rtexitcode.RTEXITCODE_SYNTAX;
1687
1688 # execte the actions.
1689 fRc = True; # Tristate - True (success), False (failure), None (skipped).
1690 asActions = self.asActions;
1691 if 'extract' in asActions:
1692 reporter.log('*** extract action ***');
1693 asActions.remove('extract');
1694 fRc = self.actionExtract();
1695 reporter.log('*** extract action completed (fRc=%s) ***' % (fRc));
1696 elif 'abort' in asActions:
1697 reporter.appendToProcessName('/abort'); # Make it easier to spot in the log.
1698 reporter.log('*** abort action ***');
1699 asActions.remove('abort');
1700 fRc = self.actionAbort();
1701 reporter.log('*** abort action completed (fRc=%s) ***' % (fRc));
1702 else:
1703 if asActions == [ 'all' ]:
1704 asActions = self.asNormalActions;
1705
1706 if 'verify' in asActions:
1707 reporter.log('*** verify action ***');
1708 asActions.remove('verify');
1709 fRc = self.actionVerify();
1710 if fRc is True: reporter.log("verified succeeded");
1711 else: reporter.log("verified failed (fRc=%s)" % (fRc,));
1712 reporter.log('*** verify action completed (fRc=%s) ***' % (fRc,));
1713
1714 if 'cleanup-before' in asActions:
1715 reporter.log('*** cleanup-before action ***');
1716 asActions.remove('cleanup-before');
1717 fRc2 = self.actionCleanupBefore();
1718 if fRc2 is not True: reporter.log("cleanup-before failed");
1719 if fRc2 is not True and fRc is True: fRc = fRc2;
1720 reporter.log('*** cleanup-before action completed (fRc2=%s, fRc=%s) ***' % (fRc2, fRc,));
1721
1722 self.pidFileAdd(os.getpid(), os.path.basename(sys.argv[0]));
1723
1724 if 'config' in asActions and fRc is True:
1725 asActions.remove('config');
1726 reporter.log('*** config action ***');
1727 fRc = self.actionConfig();
1728 if fRc is True: reporter.log("config succeeded");
1729 elif fRc is None: reporter.log("config skipping test");
1730 else: reporter.log("config failed");
1731 reporter.log('*** config action completed (fRc=%s) ***' % (fRc,));
1732
1733 if 'execute' in asActions and fRc is True:
1734 asActions.remove('execute');
1735 reporter.log('*** execute action ***');
1736 fRc = self.actionExecute();
1737 if fRc is True: reporter.log("execute succeeded");
1738 elif fRc is None: reporter.log("execute skipping test");
1739 else: reporter.log("execute failed (fRc=%s)" % (fRc,));
1740 reporter.testCleanup();
1741 reporter.log('*** execute action completed (fRc=%s) ***' % (fRc,));
1742
1743 if 'cleanup-after' in asActions:
1744 reporter.log('*** cleanup-after action ***');
1745 asActions.remove('cleanup-after');
1746 fRc2 = self.actionCleanupAfter();
1747 if fRc2 is not True: reporter.log("cleanup-after failed");
1748 if fRc2 is not True and fRc is True: fRc = fRc2;
1749 reporter.log('*** cleanup-after action completed (fRc2=%s, fRc=%s) ***' % (fRc2, fRc,));
1750
1751 self.pidFileRemove(os.getpid());
1752
1753 if asActions != [] and fRc is True:
1754 reporter.error('unhandled actions: %s' % (asActions,));
1755 fRc = False;
1756
1757 # Done
1758 if fRc is None:
1759 if self.fBadTestbox:
1760 reporter.log('****************************************************************');
1761 reporter.log('*** The test driver SKIPPED the test because of BAD_TESTBOX. ***');
1762 reporter.log('****************************************************************');
1763 return rtexitcode.RTEXITCODE_BAD_TESTBOX;
1764 reporter.log('*****************************************');
1765 reporter.log('*** The test driver SKIPPED the test. ***');
1766 reporter.log('*****************************************');
1767 return rtexitcode.RTEXITCODE_SKIPPED;
1768 if fRc is not True:
1769 reporter.error('!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!');
1770 reporter.error('!!! The test driver FAILED (in case we forgot to mention it). !!!');
1771 reporter.error('!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!');
1772 return rtexitcode.RTEXITCODE_FAILURE;
1773 reporter.log('*******************************************');
1774 reporter.log('*** The test driver exits successfully. ***');
1775 reporter.log('*******************************************');
1776 return rtexitcode.RTEXITCODE_SUCCESS;
1777
1778# The old, deprecated name.
1779TestDriver = TestDriverBase; # pylint: disable=invalid-name
1780
1781
1782#
1783# Unit testing.
1784#
1785
1786# pylint: disable=missing-docstring
1787class TestDriverBaseTestCase(unittest.TestCase):
1788 def setUp(self):
1789 self.oTstDrv = TestDriverBase();
1790 self.oTstDrv.pidFileDelete();
1791
1792 def tearDown(self):
1793 pass; # clean up scratch dir and such.
1794
1795 def testPidFile(self):
1796
1797 iPid1 = os.getpid() + 1;
1798 iPid2 = os.getpid() + 2;
1799
1800 self.assertTrue(self.oTstDrv.pidFileAdd(iPid1, 'test1'));
1801 self.assertEqual(self.oTstDrv.pidFileRead(), {iPid1:('test1',False)});
1802
1803 self.assertTrue(self.oTstDrv.pidFileAdd(iPid2, 'test2', fSudo = True));
1804 self.assertEqual(self.oTstDrv.pidFileRead(), {iPid1:('test1',False), iPid2:('test2',True)});
1805
1806 self.assertTrue(self.oTstDrv.pidFileRemove(iPid1));
1807 self.assertEqual(self.oTstDrv.pidFileRead(), {iPid2:('test2',True)});
1808
1809 self.assertTrue(self.oTstDrv.pidFileRemove(iPid2));
1810 self.assertEqual(self.oTstDrv.pidFileRead(), {});
1811
1812 self.assertTrue(self.oTstDrv.pidFileDelete());
1813
1814if __name__ == '__main__':
1815 unittest.main();
1816 # not reached.
1817
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