VirtualBox

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

Last change on this file since 79142 was 79138, checked in by vboxsync, 6 years ago

ValKit: Hack to try make COM and XPCOM exceptions more readable.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 177.7 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: vbox.py 79138 2019-06-14 01:00:00Z vboxsync $
3# pylint: disable=too-many-lines
4
5"""
6VirtualBox Specific base testdriver.
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: 79138 $"
31
32# pylint: disable=unnecessary-semicolon
33
34# Standard Python imports.
35import datetime
36import os
37import platform
38import re;
39import sys
40import threading
41import time
42import traceback
43
44# Figure out where the validation kit lives and make sure it's in the path.
45try: __file__
46except: __file__ = sys.argv[0];
47g_ksValidationKitDir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)));
48if g_ksValidationKitDir not in sys.path:
49 sys.path.append(g_ksValidationKitDir);
50
51# Validation Kit imports.
52from common import utils;
53from testdriver import base;
54from testdriver import btresolver;
55from testdriver import reporter;
56from testdriver import vboxcon;
57from testdriver import vboxtestvms;
58
59# Python 3 hacks:
60if sys.version_info[0] >= 3:
61 xrange = range; # pylint: disable=redefined-builtin,invalid-name
62 long = int; # pylint: disable=redefined-builtin,invalid-name
63
64#
65# Exception and Error Unification Hacks.
66# Note! This is pretty gross stuff. Be warned!
67# TODO: Find better ways of doing these things, preferrably in vboxapi.
68#
69
70ComException = None; # pylint: disable=invalid-name
71__fnComExceptionGetAttr__ = None; # pylint: disable=invalid-name
72
73def __MyDefaultGetAttr(oSelf, sName):
74 """ __getattribute__/__getattr__ default fake."""
75 try:
76 oAttr = oSelf.__dict__[sName];
77 except:
78 oAttr = dir(oSelf)[sName];
79 return oAttr;
80
81def __MyComExceptionGetAttr(oSelf, sName):
82 """ ComException.__getattr__ wrapper - both XPCOM and COM. """
83 try:
84 oAttr = __fnComExceptionGetAttr__(oSelf, sName);
85 except AttributeError:
86 if platform.system() == 'Windows':
87 if sName == 'errno':
88 oAttr = __fnComExceptionGetAttr__(oSelf, 'hresult');
89 elif sName == 'msg':
90 oAttr = __fnComExceptionGetAttr__(oSelf, 'strerror');
91 else:
92 raise;
93 else:
94 if sName == 'hresult':
95 oAttr = __fnComExceptionGetAttr__(oSelf, 'errno');
96 elif sName == 'strerror':
97 oAttr = __fnComExceptionGetAttr__(oSelf, 'msg');
98 elif sName == 'excepinfo':
99 oAttr = None;
100 elif sName == 'argerror':
101 oAttr = None;
102 else:
103 raise;
104 #print '__MyComExceptionGetAttr(,%s) -> "%s"' % (sName, oAttr);
105 return oAttr;
106
107def __deployExceptionHacks__(oNativeComExceptionClass):
108 """
109 Deploys the exception and error hacks that helps unifying COM and XPCOM
110 exceptions and errors.
111 """
112 global ComException # pylint: disable=invalid-name
113 global __fnComExceptionGetAttr__ # pylint: disable=invalid-name
114
115 # Hook up our attribute getter for the exception class (ASSUMES new-style).
116 if __fnComExceptionGetAttr__ is None:
117 try:
118 __fnComExceptionGetAttr__ = getattr(oNativeComExceptionClass, '__getattr__');
119 except:
120 try:
121 __fnComExceptionGetAttr__ = getattr(oNativeComExceptionClass, '__getattribute__');
122 except:
123 __fnComExceptionGetAttr__ = __MyDefaultGetAttr;
124 setattr(oNativeComExceptionClass, '__getattr__', __MyComExceptionGetAttr)
125
126 # Make the modified classes accessible (are there better ways to do this?)
127 ComException = oNativeComExceptionClass
128 return None;
129
130
131
132#
133# Utility functions.
134#
135
136def isIpAddrValid(sIpAddr):
137 """
138 Checks if a IPv4 address looks valid. This will return false for
139 localhost and similar.
140 Returns True / False.
141 """
142 if sIpAddr is None: return False;
143 if len(sIpAddr.split('.')) != 4: return False;
144 if sIpAddr.endswith('.0'): return False;
145 if sIpAddr.endswith('.255'): return False;
146 if sIpAddr.startswith('127.'): return False;
147 if sIpAddr.startswith('169.254.'): return False;
148 if sIpAddr.startswith('192.0.2.'): return False;
149 if sIpAddr.startswith('224.0.0.'): return False;
150 return True;
151
152def stringifyErrorInfo(oErrInfo):
153 """
154 Stringifies the error information in a IVirtualBoxErrorInfo object.
155
156 Returns string with error info.
157 """
158 try:
159 rc = oErrInfo.resultCode;
160 sText = oErrInfo.text;
161 sIid = oErrInfo.interfaceID;
162 sComponent = oErrInfo.component;
163 except:
164 sRet = 'bad error object (%s)?' % (oErrInfo,);
165 traceback.print_exc();
166 else:
167 sRet = 'rc=%s text="%s" IID=%s component=%s' % (ComError.toString(rc), sText, sIid, sComponent);
168 return sRet;
169
170def reportError(oErr, sText):
171 """
172 Report a VirtualBox error on oErr. oErr can be IVirtualBoxErrorInfo
173 or IProgress. Anything else is ignored.
174
175 Returns the same a reporter.error().
176 """
177 try:
178 oErrObj = oErr.errorInfo; # IProgress.
179 except:
180 oErrObj = oErr;
181 reporter.error(sText);
182 return reporter.error(stringifyErrorInfo(oErrObj));
183
184def formatComOrXpComException(oType, oXcpt):
185 """
186 Callback installed with the reporter to better format COM exceptions.
187 Similar to format_exception_only, only it returns None if not interested.
188 """
189 _ = oType;
190 oVBoxMgr = vboxcon.goHackModuleClass.oVBoxMgr;
191 if oVBoxMgr is None:
192 return None;
193 if not oVBoxMgr.xcptIsOurXcptKind(oXcpt): # pylint: disable=not-callable
194 return None;
195
196 if platform.system() == 'Windows':
197 hrc = oXcpt.hresult;
198 if hrc == ComError.DISP_E_EXCEPTION and oXcpt.excepinfo is not None and len(oXcpt.excepinfo) > 5:
199 hrc = oXcpt.excepinfo[5];
200 sWhere = oXcpt.excepinfo[1];
201 sMsg = oXcpt.excepinfo[2];
202 else:
203 sWhere = None;
204 sMsg = oXcpt.strerror;
205 else:
206 hrc = oXcpt.errno;
207 sWhere = None;
208 sMsg = oXcpt.msg;
209
210 sHrc = oVBoxMgr.xcptToString(hrc); # pylint: disable=not-callable
211 if sHrc.find('(') < 0:
212 sHrc = '%s (%#x)' % (sHrc, hrc & 0xffffffff);
213
214 asRet = ['COM-Xcpt: %s' % (sHrc,)];
215 if sMsg and sWhere:
216 asRet.append('--------- %s: %s' % (sWhere, sMsg,));
217 elif sMsg:
218 asRet.append('--------- %s' % (sMsg,));
219 return asRet;
220 #if sMsg and sWhere:
221 # return ['COM-Xcpt: %s - %s: %s' % (sHrc, sWhere, sMsg,)];
222 #if sMsg:
223 # return ['COM-Xcpt: %s - %s' % (sHrc, sMsg,)];
224 #return ['COM-Xcpt: %s' % (sHrc,)];
225
226#
227# Classes
228#
229
230class ComError(object):
231 """
232 Unified COM and XPCOM status code repository.
233 This works more like a module than a class since it's replacing a module.
234 """
235
236 # The VBOX_E_XXX bits:
237 __VBOX_E_BASE = -2135228416;
238 VBOX_E_OBJECT_NOT_FOUND = __VBOX_E_BASE + 1;
239 VBOX_E_INVALID_VM_STATE = __VBOX_E_BASE + 2;
240 VBOX_E_VM_ERROR = __VBOX_E_BASE + 3;
241 VBOX_E_FILE_ERROR = __VBOX_E_BASE + 4;
242 VBOX_E_IPRT_ERROR = __VBOX_E_BASE + 5;
243 VBOX_E_PDM_ERROR = __VBOX_E_BASE + 6;
244 VBOX_E_INVALID_OBJECT_STATE = __VBOX_E_BASE + 7;
245 VBOX_E_HOST_ERROR = __VBOX_E_BASE + 8;
246 VBOX_E_NOT_SUPPORTED = __VBOX_E_BASE + 9;
247 VBOX_E_XML_ERROR = __VBOX_E_BASE + 10;
248 VBOX_E_INVALID_SESSION_STATE = __VBOX_E_BASE + 11;
249 VBOX_E_OBJECT_IN_USE = __VBOX_E_BASE + 12;
250 VBOX_E_DONT_CALL_AGAIN = __VBOX_E_BASE + 13;
251
252 # Reverse lookup table.
253 dDecimalToConst = {}; # pylint: disable=invalid-name
254
255 def __init__(self):
256 raise base.GenError('No instances, please');
257
258 @staticmethod
259 def copyErrors(oNativeComErrorClass):
260 """
261 Copy all error codes from oNativeComErrorClass to this class and
262 install compatability mappings.
263 """
264
265 # First, add the VBOX_E_XXX constants to dDecimalToConst.
266 for sAttr in dir(ComError):
267 if sAttr.startswith('VBOX_E'):
268 oAttr = getattr(ComError, sAttr);
269 ComError.dDecimalToConst[oAttr] = sAttr;
270
271 # Copy all error codes from oNativeComErrorClass to this class.
272 for sAttr in dir(oNativeComErrorClass):
273 if sAttr[0].isupper():
274 oAttr = getattr(oNativeComErrorClass, sAttr);
275 setattr(ComError, sAttr, oAttr);
276 if isinstance(oAttr, int):
277 ComError.dDecimalToConst[oAttr] = sAttr;
278
279 # Install mappings to the other platform.
280 if platform.system() == 'Windows':
281 ComError.NS_OK = ComError.S_OK;
282 ComError.NS_ERROR_FAILURE = ComError.E_FAIL;
283 ComError.NS_ERROR_ABORT = ComError.E_ABORT;
284 ComError.NS_ERROR_NULL_POINTER = ComError.E_POINTER;
285 ComError.NS_ERROR_NO_INTERFACE = ComError.E_NOINTERFACE;
286 ComError.NS_ERROR_INVALID_ARG = ComError.E_INVALIDARG;
287 ComError.NS_ERROR_OUT_OF_MEMORY = ComError.E_OUTOFMEMORY;
288 ComError.NS_ERROR_NOT_IMPLEMENTED = ComError.E_NOTIMPL;
289 ComError.NS_ERROR_UNEXPECTED = ComError.E_UNEXPECTED;
290 else:
291 ComError.E_ACCESSDENIED = -2147024891; # see VBox/com/defs.h
292 ComError.S_OK = ComError.NS_OK;
293 ComError.E_FAIL = ComError.NS_ERROR_FAILURE;
294 ComError.E_ABORT = ComError.NS_ERROR_ABORT;
295 ComError.E_POINTER = ComError.NS_ERROR_NULL_POINTER;
296 ComError.E_NOINTERFACE = ComError.NS_ERROR_NO_INTERFACE;
297 ComError.E_INVALIDARG = ComError.NS_ERROR_INVALID_ARG;
298 ComError.E_OUTOFMEMORY = ComError.NS_ERROR_OUT_OF_MEMORY;
299 ComError.E_NOTIMPL = ComError.NS_ERROR_NOT_IMPLEMENTED;
300 ComError.E_UNEXPECTED = ComError.NS_ERROR_UNEXPECTED;
301 ComError.DISP_E_EXCEPTION = -2147352567; # For COM compatability only.
302 return True;
303
304 @staticmethod
305 def getXcptResult(oXcpt):
306 """
307 Gets the result code for an exception.
308 Returns COM status code (or E_UNEXPECTED).
309 """
310 if platform.system() == 'Windows':
311 # The DISP_E_EXCEPTION + excptinfo fun needs checking up, only
312 # empirical info on it so far.
313 try:
314 hrXcpt = oXcpt.hresult;
315 except AttributeError:
316 hrXcpt = ComError.E_UNEXPECTED;
317 if hrXcpt == ComError.DISP_E_EXCEPTION and oXcpt.excepinfo is not None:
318 hrXcpt = oXcpt.excepinfo[5];
319 else:
320 try:
321 hrXcpt = oXcpt.errno;
322 except AttributeError:
323 hrXcpt = ComError.E_UNEXPECTED;
324 return hrXcpt;
325
326 @staticmethod
327 def equal(oXcpt, hr):
328 """
329 Checks if the ComException e is not equal to the COM status code hr.
330 This takes DISP_E_EXCEPTION & excepinfo into account.
331
332 This method can be used with any Exception derivate, however it will
333 only return True for classes similar to the two ComException variants.
334 """
335 if platform.system() == 'Windows':
336 # The DISP_E_EXCEPTION + excptinfo fun needs checking up, only
337 # empirical info on it so far.
338 try:
339 hrXcpt = oXcpt.hresult;
340 except AttributeError:
341 return False;
342 if hrXcpt == ComError.DISP_E_EXCEPTION and oXcpt.excepinfo is not None:
343 hrXcpt = oXcpt.excepinfo[5];
344 else:
345 try:
346 hrXcpt = oXcpt.errno;
347 except AttributeError:
348 return False;
349 return hrXcpt == hr;
350
351 @staticmethod
352 def notEqual(oXcpt, hr):
353 """
354 Checks if the ComException e is not equal to the COM status code hr.
355 See equal() for more details.
356 """
357 return not ComError.equal(oXcpt, hr)
358
359 @staticmethod
360 def toString(hr):
361 """
362 Converts the specified COM status code to a string.
363 """
364 try:
365 sStr = ComError.dDecimalToConst[int(hr)];
366 except KeyError:
367 hrLong = long(hr);
368 sStr = '%#x (%d)' % (hrLong, hrLong);
369 return sStr;
370
371
372class Build(object): # pylint: disable=too-few-public-methods
373 """
374 A VirtualBox build.
375
376 Note! After dropping the installation of VBox from this code and instead
377 realizing that with the vboxinstall.py wrapper driver, this class is
378 of much less importance and contains unnecessary bits and pieces.
379 """
380
381 def __init__(self, oDriver, strInstallPath):
382 """
383 Construct a build object from a build file name and/or install path.
384 """
385 # Initialize all members first.
386 self.oDriver = oDriver;
387 self.sInstallPath = strInstallPath;
388 self.sSdkPath = None;
389 self.sSrcRoot = None;
390 self.sKind = None;
391 self.sDesignation = None;
392 self.sType = None;
393 self.sOs = None;
394 self.sArch = None;
395 self.sGuestAdditionsIso = None;
396
397 # Figure out the values as best we can.
398 if strInstallPath is None:
399 #
400 # Both parameters are None, which means we're falling back on a
401 # build in the development tree.
402 #
403 self.sKind = "development";
404
405 if self.sType is None:
406 self.sType = os.environ.get("KBUILD_TYPE", os.environ.get("BUILD_TYPE", "release"));
407 if self.sOs is None:
408 self.sOs = os.environ.get("KBUILD_TARGET", os.environ.get("BUILD_TARGET", oDriver.sHost));
409 if self.sArch is None:
410 self.sArch = os.environ.get("KBUILD_TARGET_ARCH", os.environ.get("BUILD_TARGET_ARCH", oDriver.sHostArch));
411
412 sOut = os.path.join('out', self.sOs + '.' + self.sArch, self.sType);
413 sSearch = os.environ.get('VBOX_TD_DEV_TREE', os.path.dirname(__file__)); # Env.var. for older trees or testboxscript.
414 sCandidat = None;
415 for i in range(0, 10): # pylint: disable=unused-variable
416 sBldDir = os.path.join(sSearch, sOut);
417 if os.path.isdir(sBldDir):
418 sCandidat = os.path.join(sBldDir, 'bin', 'VBoxSVC' + base.exeSuff());
419 if os.path.isfile(sCandidat):
420 self.sSdkPath = os.path.join(sBldDir, 'bin/sdk');
421 break;
422 sCandidat = os.path.join(sBldDir, 'dist/VirtualBox.app/Contents/MacOS/VBoxSVC');
423 if os.path.isfile(sCandidat):
424 self.sSdkPath = os.path.join(sBldDir, 'dist/sdk');
425 break;
426 sSearch = os.path.abspath(os.path.join(sSearch, '..'));
427 if sCandidat is None or not os.path.isfile(sCandidat):
428 raise base.GenError();
429 self.sInstallPath = os.path.abspath(os.path.dirname(sCandidat));
430 self.sSrcRoot = os.path.abspath(sSearch);
431
432 self.sDesignation = os.environ.get('TEST_BUILD_DESIGNATION', None);
433 if self.sDesignation is None:
434 try:
435 oFile = utils.openNoInherit(os.path.join(self.sSrcRoot, sOut, 'revision.kmk'), 'r');
436 except:
437 pass;
438 else:
439 s = oFile.readline();
440 oFile.close();
441 oMatch = re.search("VBOX_SVN_REV=(\\d+)", s);
442 if oMatch is not None:
443 self.sDesignation = oMatch.group(1);
444
445 if self.sDesignation is None:
446 self.sDesignation = 'XXXXX'
447 else:
448 #
449 # We've been pointed to an existing installation, this could be
450 # in the out dir of a svn checkout, untarred VBoxAll or a real
451 # installation directory.
452 #
453 self.sKind = "preinstalled";
454 self.sType = "release";
455 self.sOs = oDriver.sHost;
456 self.sArch = oDriver.sHostArch;
457 self.sInstallPath = os.path.abspath(strInstallPath);
458 self.sSdkPath = os.path.join(self.sInstallPath, 'sdk');
459 self.sSrcRoot = None;
460 self.sDesignation = os.environ.get('TEST_BUILD_DESIGNATION', 'XXXXX');
461 ## @todo Much more work is required here.
462
463 # Do some checks.
464 sVMMR0 = os.path.join(self.sInstallPath, 'VMMR0.r0');
465 if not os.path.isfile(sVMMR0) and utils.getHostOs() == 'solaris': # solaris is special.
466 sVMMR0 = os.path.join(self.sInstallPath, 'amd64' if utils.getHostArch() == 'amd64' else 'i386', 'VMMR0.r0');
467 if not os.path.isfile(sVMMR0):
468 raise base.GenError('%s is missing' % (sVMMR0,));
469
470 # Guest additions location is different on windows for some _stupid_ reason.
471 if self.sOs == 'win' and self.sKind != 'development':
472 self.sGuestAdditionsIso = '%s/VBoxGuestAdditions.iso' % (self.sInstallPath,);
473 elif self.sOs == 'darwin':
474 self.sGuestAdditionsIso = '%s/VBoxGuestAdditions.iso' % (self.sInstallPath,);
475 elif self.sOs == 'solaris':
476 self.sGuestAdditionsIso = '%s/VBoxGuestAdditions.iso' % (self.sInstallPath,);
477 else:
478 self.sGuestAdditionsIso = '%s/additions/VBoxGuestAdditions.iso' % (self.sInstallPath,);
479
480 # __init__ end;
481
482 def isDevBuild(self):
483 """ Returns True if it's development build (kind), otherwise False. """
484 return self.sKind == 'development';
485
486
487class EventHandlerBase(object):
488 """
489 Base class for both Console and VirtualBox event handlers.
490 """
491
492 def __init__(self, dArgs, fpApiVer, sName = None):
493 self.oVBoxMgr = dArgs['oVBoxMgr'];
494 self.oEventSrc = dArgs['oEventSrc']; # Console/VirtualBox for < 3.3
495 self.oListener = dArgs['oListener'];
496 self.fPassive = self.oListener is not None;
497 self.sName = sName
498 self.fShutdown = False;
499 self.oThread = None;
500 self.fpApiVer = fpApiVer;
501 self.dEventNo2Name = {};
502 for sKey, iValue in self.oVBoxMgr.constants.all_values('VBoxEventType').items():
503 self.dEventNo2Name[iValue] = sKey;
504
505 def threadForPassiveMode(self):
506 """
507 The thread procedure for the event processing thread.
508 """
509 assert self.fPassive is not None;
510 while not self.fShutdown:
511 try:
512 oEvt = self.oEventSrc.getEvent(self.oListener, 500);
513 except:
514 if not self.oVBoxMgr.xcptIsDeadInterface(): reporter.logXcpt();
515 else: reporter.log('threadForPassiveMode/%s: interface croaked (ignored)' % (self.sName,));
516 break;
517 if oEvt:
518 self.handleEvent(oEvt);
519 if not self.fShutdown:
520 try:
521 self.oEventSrc.eventProcessed(self.oListener, oEvt);
522 except:
523 reporter.logXcpt();
524 break;
525 self.unregister(fWaitForThread = False);
526 return None;
527
528 def startThreadForPassiveMode(self):
529 """
530 Called when working in passive mode.
531 """
532 self.oThread = threading.Thread(target = self.threadForPassiveMode, \
533 args=(), name=('PAS-%s' % (self.sName,)));
534 self.oThread.setDaemon(True)
535 self.oThread.start();
536 return None;
537
538 def unregister(self, fWaitForThread = True):
539 """
540 Unregister the event handler.
541 """
542 fRc = False;
543 if not self.fShutdown:
544 self.fShutdown = True;
545
546 if self.oEventSrc is not None:
547 if self.fpApiVer < 3.3:
548 try:
549 self.oEventSrc.unregisterCallback(self.oListener);
550 fRc = True;
551 except:
552 reporter.errorXcpt('unregisterCallback failed on %s' % (self.oListener,));
553 else:
554 try:
555 self.oEventSrc.unregisterListener(self.oListener);
556 fRc = True;
557 except:
558 if self.oVBoxMgr.xcptIsDeadInterface():
559 reporter.log('unregisterListener failed on %s because of dead interface (%s)'
560 % (self.oListener, self.oVBoxMgr.xcptToString(),));
561 else:
562 reporter.errorXcpt('unregisterListener failed on %s' % (self.oListener,));
563
564 if self.oThread is not None \
565 and self.oThread != threading.current_thread():
566 self.oThread.join();
567 self.oThread = None;
568
569 _ = fWaitForThread;
570 return fRc;
571
572 def handleEvent(self, oEvt):
573 """
574 Compatibility wrapper that child classes implement.
575 """
576 _ = oEvt;
577 return None;
578
579 @staticmethod
580 def registerDerivedEventHandler(oVBoxMgr, fpApiVer, oSubClass, dArgsCopy, # pylint: disable=too-many-arguments
581 oSrcParent, sSrcParentNm, sICallbackNm,
582 fMustSucceed = True, sLogSuffix = '', aenmEvents = None):
583 """
584 Registers the callback / event listener.
585 """
586 dArgsCopy['oVBoxMgr'] = oVBoxMgr;
587 dArgsCopy['oListener'] = None;
588 if fpApiVer < 3.3:
589 dArgsCopy['oEventSrc'] = oSrcParent;
590 try:
591 oRet = oVBoxMgr.createCallback(sICallbackNm, oSubClass, dArgsCopy);
592 except:
593 reporter.errorXcpt('%s::registerCallback(%s) failed%s' % (sSrcParentNm, oRet, sLogSuffix));
594 else:
595 try:
596 oSrcParent.registerCallback(oRet);
597 return oRet;
598 except Exception as oXcpt:
599 if fMustSucceed or ComError.notEqual(oXcpt, ComError.E_UNEXPECTED):
600 reporter.errorXcpt('%s::registerCallback(%s)%s' % (sSrcParentNm, oRet, sLogSuffix));
601 else:
602 #
603 # Scalable event handling introduced in VBox 4.0.
604 #
605 fPassive = sys.platform == 'win32'; # or webservices.
606
607 if not aenmEvents:
608 aenmEvents = (vboxcon.VBoxEventType_Any,);
609
610 try:
611 oEventSrc = oSrcParent.eventSource;
612 dArgsCopy['oEventSrc'] = oEventSrc;
613 if not fPassive:
614 oListener = oRet = oVBoxMgr.createListener(oSubClass, dArgsCopy);
615 else:
616 oListener = oEventSrc.createListener();
617 dArgsCopy['oListener'] = oListener;
618 oRet = oSubClass(dArgsCopy);
619 except:
620 reporter.errorXcpt('%s::eventSource.createListener(%s) failed%s' % (sSrcParentNm, oListener, sLogSuffix));
621 else:
622 try:
623 oEventSrc.registerListener(oListener, aenmEvents, not fPassive);
624 except Exception as oXcpt:
625 if fMustSucceed or ComError.notEqual(oXcpt, ComError.E_UNEXPECTED):
626 reporter.errorXcpt('%s::eventSource.registerListener(%s) failed%s'
627 % (sSrcParentNm, oListener, sLogSuffix));
628 else:
629 if not fPassive:
630 if sys.platform == 'win32':
631 from win32com.server.util import unwrap # pylint: disable=import-error
632 oRet = unwrap(oRet);
633 oRet.oListener = oListener;
634 else:
635 oRet.startThreadForPassiveMode();
636 return oRet;
637 return None;
638
639
640
641
642class ConsoleEventHandlerBase(EventHandlerBase):
643 """
644 Base class for handling IConsole events.
645
646 The class has IConsoleCallback (<=3.2) compatible callback methods which
647 the user can override as needed.
648
649 Note! This class must not inherit from object or we'll get type errors in VBoxPython.
650 """
651 def __init__(self, dArgs, sName = None):
652 self.oSession = dArgs['oSession'];
653 self.oConsole = dArgs['oConsole'];
654 if sName is None:
655 sName = self.oSession.sName;
656 EventHandlerBase.__init__(self, dArgs, self.oSession.fpApiVer, sName);
657
658
659 # pylint: disable=missing-docstring,too-many-arguments,unused-argument
660 def onMousePointerShapeChange(self, fVisible, fAlpha, xHot, yHot, cx, cy, abShape):
661 reporter.log2('onMousePointerShapeChange/%s' % (self.sName));
662 def onMouseCapabilityChange(self, fSupportsAbsolute, *aArgs): # Extra argument was added in 3.2.
663 reporter.log2('onMouseCapabilityChange/%s' % (self.sName));
664 def onKeyboardLedsChange(self, fNumLock, fCapsLock, fScrollLock):
665 reporter.log2('onKeyboardLedsChange/%s' % (self.sName));
666 def onStateChange(self, eState):
667 reporter.log2('onStateChange/%s' % (self.sName));
668 def onAdditionsStateChange(self):
669 reporter.log2('onAdditionsStateChange/%s' % (self.sName));
670 def onNetworkAdapterChange(self, oNic):
671 reporter.log2('onNetworkAdapterChange/%s' % (self.sName));
672 def onSerialPortChange(self, oPort):
673 reporter.log2('onSerialPortChange/%s' % (self.sName));
674 def onParallelPortChange(self, oPort):
675 reporter.log2('onParallelPortChange/%s' % (self.sName));
676 def onStorageControllerChange(self):
677 reporter.log2('onStorageControllerChange/%s' % (self.sName));
678 def onMediumChange(self, attachment):
679 reporter.log2('onMediumChange/%s' % (self.sName));
680 def onCPUChange(self, iCpu, fAdd):
681 reporter.log2('onCPUChange/%s' % (self.sName));
682 def onVRDPServerChange(self):
683 reporter.log2('onVRDPServerChange/%s' % (self.sName));
684 def onRemoteDisplayInfoChange(self):
685 reporter.log2('onRemoteDisplayInfoChange/%s' % (self.sName));
686 def onUSBControllerChange(self):
687 reporter.log2('onUSBControllerChange/%s' % (self.sName));
688 def onUSBDeviceStateChange(self, oDevice, fAttached, oError):
689 reporter.log2('onUSBDeviceStateChange/%s' % (self.sName));
690 def onSharedFolderChange(self, fGlobal):
691 reporter.log2('onSharedFolderChange/%s' % (self.sName));
692 def onRuntimeError(self, fFatal, sErrId, sMessage):
693 reporter.log2('onRuntimeError/%s' % (self.sName));
694 def onCanShowWindow(self):
695 reporter.log2('onCanShowWindow/%s' % (self.sName));
696 return True
697 def onShowWindow(self):
698 reporter.log2('onShowWindow/%s' % (self.sName));
699 return None;
700 # pylint: enable=missing-docstring,too-many-arguments,unused-argument
701
702 def handleEvent(self, oEvt):
703 """
704 Compatibility wrapper.
705 """
706 try:
707 oEvtBase = self.oVBoxMgr.queryInterface(oEvt, 'IEvent');
708 eType = oEvtBase.type;
709 except:
710 reporter.logXcpt();
711 return None;
712 if eType == vboxcon.VBoxEventType_OnRuntimeError:
713 try:
714 oEvtIt = self.oVBoxMgr.queryInterface(oEvtBase, 'IRuntimeErrorEvent');
715 return self.onRuntimeError(oEvtIt.fatal, oEvtIt.id, oEvtIt.message)
716 except:
717 reporter.logXcpt();
718 ## @todo implement the other events.
719 if eType != vboxcon.VBoxEventType_OnMousePointerShapeChanged:
720 if eType in self.dEventNo2Name:
721 reporter.log2('%s(%s)/%s' % (self.dEventNo2Name[eType], str(eType), self.sName));
722 else:
723 reporter.log2('%s/%s' % (str(eType), self.sName));
724 return None;
725
726
727class VirtualBoxEventHandlerBase(EventHandlerBase):
728 """
729 Base class for handling IVirtualBox events.
730
731 The class has IConsoleCallback (<=3.2) compatible callback methods which
732 the user can override as needed.
733
734 Note! This class must not inherit from object or we'll get type errors in VBoxPython.
735 """
736 def __init__(self, dArgs, sName = "emanon"):
737 self.oVBoxMgr = dArgs['oVBoxMgr'];
738 self.oVBox = dArgs['oVBox'];
739 EventHandlerBase.__init__(self, dArgs, self.oVBox.fpApiVer, sName);
740
741 # pylint: disable=missing-docstring,unused-argument
742 def onMachineStateChange(self, sMachineId, eState):
743 pass;
744 def onMachineDataChange(self, sMachineId):
745 pass;
746 def onExtraDataCanChange(self, sMachineId, sKey, sValue):
747 # The COM bridge does tuples differently. Not very funny if you ask me... ;-)
748 if self.oVBoxMgr.type == 'MSCOM':
749 return '', 0, True;
750 return True, ''
751 def onExtraDataChange(self, sMachineId, sKey, sValue):
752 pass;
753 def onMediumRegistered(self, sMediumId, eMediumType, fRegistered):
754 pass;
755 def onMachineRegistered(self, sMachineId, fRegistered):
756 pass;
757 def onSessionStateChange(self, sMachineId, eState):
758 pass;
759 def onSnapshotTaken(self, sMachineId, sSnapshotId):
760 pass;
761 def onSnapshotDiscarded(self, sMachineId, sSnapshotId):
762 pass;
763 def onSnapshotChange(self, sMachineId, sSnapshotId):
764 pass;
765 def onGuestPropertyChange(self, sMachineId, sName, sValue, sFlags):
766 pass;
767 # pylint: enable=missing-docstring,unused-argument
768
769 def handleEvent(self, oEvt):
770 """
771 Compatibility wrapper.
772 """
773 try:
774 oEvtBase = self.oVBoxMgr.queryInterface(oEvt, 'IEvent');
775 eType = oEvtBase.type;
776 except:
777 reporter.logXcpt();
778 return None;
779 if eType == vboxcon.VBoxEventType_OnMachineStateChanged:
780 try:
781 oEvtIt = self.oVBoxMgr.queryInterface(oEvtBase, 'IMachineStateChangedEvent');
782 return self.onMachineStateChange(oEvtIt.machineId, oEvtIt.state)
783 except:
784 reporter.logXcpt();
785 elif eType == vboxcon.VBoxEventType_OnGuestPropertyChanged:
786 try:
787 oEvtIt = self.oVBoxMgr.queryInterface(oEvtBase, 'IGuestPropertyChangedEvent');
788 return self.onGuestPropertyChange(oEvtIt.machineId, oEvtIt.name, oEvtIt.value, oEvtIt.flags);
789 except:
790 reporter.logXcpt();
791 ## @todo implement the other events.
792 if eType in self.dEventNo2Name:
793 reporter.log2('%s(%s)/%s' % (self.dEventNo2Name[eType], str(eType), self.sName));
794 else:
795 reporter.log2('%s/%s' % (str(eType), self.sName));
796 return None;
797
798
799class SessionConsoleEventHandler(ConsoleEventHandlerBase):
800 """
801 For catching machine state changes and waking up the task machinery at that point.
802 """
803 def __init__(self, dArgs):
804 ConsoleEventHandlerBase.__init__(self, dArgs);
805
806 def onMachineStateChange(self, sMachineId, eState): # pylint: disable=unused-argument
807 """ Just interrupt the wait loop here so it can check again. """
808 _ = sMachineId; _ = eState;
809 self.oVBoxMgr.interruptWaitEvents();
810
811 def onRuntimeError(self, fFatal, sErrId, sMessage):
812 reporter.log('onRuntimeError/%s: fFatal=%d sErrId=%s sMessage=%s' % (self.sName, fFatal, sErrId, sMessage));
813 oSession = self.oSession;
814 if oSession is not None: # paranoia
815 if sErrId == 'HostMemoryLow':
816 oSession.signalHostMemoryLow();
817 if sys.platform == 'win32':
818 from testdriver import winbase;
819 winbase.logMemoryStats();
820 oSession.signalTask();
821 self.oVBoxMgr.interruptWaitEvents();
822
823
824
825class TestDriver(base.TestDriver): # pylint: disable=too-many-instance-attributes
826 """
827 This is the VirtualBox test driver.
828 """
829
830 def __init__(self):
831 base.TestDriver.__init__(self);
832 self.fImportedVBoxApi = False;
833 self.fpApiVer = 3.2;
834 self.uRevision = 0;
835 self.oBuild = None;
836 self.oVBoxMgr = None;
837 self.oVBox = None;
838 self.aoRemoteSessions = [];
839 self.aoVMs = []; ## @todo not sure if this list will be of any use.
840 self.oTestVmManager = vboxtestvms.TestVmManager(self.sResourcePath);
841 self.oTestVmSet = vboxtestvms.TestVmSet();
842 self.sSessionTypeDef = 'headless';
843 self.sSessionType = self.sSessionTypeDef;
844 self.fEnableVrdp = True;
845 self.uVrdpBasePortDef = 6000;
846 self.uVrdpBasePort = self.uVrdpBasePortDef;
847 self.sDefBridgedNic = None;
848 self.fUseDefaultSvc = False;
849 self.sLogSelfGroups = '';
850 self.sLogSelfFlags = 'time';
851 self.sLogSelfDest = '';
852 self.sLogSessionGroups = '';
853 self.sLogSessionFlags = 'time';
854 self.sLogSessionDest = '';
855 self.sLogSvcGroups = '';
856 self.sLogSvcFlags = 'time';
857 self.sLogSvcDest = '';
858 self.sSelfLogFile = None;
859 self.sVBoxSvcLogFile = None;
860 self.oVBoxSvcProcess = None;
861 self.sVBoxSvcPidFile = None;
862 self.fVBoxSvcInDebugger = False;
863 self.sVBoxValidationKit = None;
864 self.sVBoxValidationKitIso = None;
865 self.sVBoxBootSectors = None;
866 self.fAlwaysUploadLogs = False;
867 self.fAlwaysUploadScreenshots = False;
868 self.fEnableDebugger = True;
869
870 # Quietly detect build and validation kit.
871 self._detectBuild(False);
872 self._detectValidationKit(False);
873
874 # Make sure all debug logs goes to the scratch area unless
875 # specified otherwise (more of this later on).
876 if 'VBOX_LOG_DEST' not in os.environ:
877 os.environ['VBOX_LOG_DEST'] = 'nodeny dir=%s' % (self.sScratchPath);
878
879
880 def _detectBuild(self, fQuiet = False):
881 """
882 This is used internally to try figure a locally installed build when
883 running tests manually.
884 """
885 if self.oBuild is not None:
886 return True;
887
888 # Try dev build first since that's where I'll be using it first...
889 if True is True: # pylint: disable=comparison-with-itself
890 try:
891 self.oBuild = Build(self, None);
892 return True;
893 except base.GenError:
894 pass;
895
896 # Try default installation locations.
897 if self.sHost == 'win':
898 sProgFiles = os.environ.get('ProgramFiles', 'C:\\Program Files');
899 asLocs = [
900 os.path.join(sProgFiles, 'Oracle', 'VirtualBox'),
901 os.path.join(sProgFiles, 'OracleVM', 'VirtualBox'),
902 os.path.join(sProgFiles, 'Sun', 'VirtualBox'),
903 ];
904 elif self.sHost == 'solaris':
905 asLocs = [ '/opt/VirtualBox-3.2', '/opt/VirtualBox-3.1', '/opt/VirtualBox-3.0', '/opt/VirtualBox' ];
906 elif self.sHost == 'darwin':
907 asLocs = [ '/Applications/VirtualBox.app/Contents/MacOS' ];
908 elif self.sHost == 'linux':
909 asLocs = [ '/opt/VirtualBox-3.2', '/opt/VirtualBox-3.1', '/opt/VirtualBox-3.0', '/opt/VirtualBox' ];
910 else:
911 asLocs = [ '/opt/VirtualBox' ];
912 if 'VBOX_INSTALL_PATH' in os.environ:
913 asLocs.insert(0, os.environ['VBOX_INSTALL_PATH']);
914
915 for sLoc in asLocs:
916 try:
917 self.oBuild = Build(self, sLoc);
918 return True;
919 except base.GenError:
920 pass;
921
922 if not fQuiet:
923 reporter.error('failed to find VirtualBox installation');
924 return False;
925
926 def _detectValidationKit(self, fQuiet = False):
927 """
928 This is used internally by the constructor to try locate an unzipped
929 VBox Validation Kit somewhere in the immediate proximity.
930 """
931 if self.sVBoxValidationKit is not None:
932 return True;
933
934 #
935 # Normally it's found where we're running from, which is the same as
936 # the script directly on the testboxes.
937 #
938 asCandidates = [self.sScriptPath, ];
939 if g_ksValidationKitDir not in asCandidates:
940 asCandidates.append(g_ksValidationKitDir);
941 if os.getcwd() not in asCandidates:
942 asCandidates.append(os.getcwd());
943 if self.oBuild is not None and self.oBuild.sInstallPath not in asCandidates:
944 asCandidates.append(self.oBuild.sInstallPath);
945
946 #
947 # When working out of the tree, we'll search the current directory
948 # as well as parent dirs.
949 #
950 for sDir in list(asCandidates):
951 for i in range(10):
952 sDir = os.path.dirname(sDir);
953 if sDir not in asCandidates:
954 asCandidates.append(sDir);
955
956 #
957 # Do the searching.
958 #
959 sCandidate = None;
960 for i, _ in enumerate(asCandidates):
961 sCandidate = asCandidates[i];
962 if os.path.isfile(os.path.join(sCandidate, 'VBoxValidationKit.iso')):
963 break;
964 sCandidate = os.path.join(sCandidate, 'validationkit');
965 if os.path.isfile(os.path.join(sCandidate, 'VBoxValidationKit.iso')):
966 break;
967 sCandidate = None;
968
969 fRc = sCandidate is not None;
970 if fRc is False:
971 if not fQuiet:
972 reporter.error('failed to find VBox Validation Kit installation (candidates: %s)' % (asCandidates,));
973 sCandidate = os.path.join(self.sScriptPath, 'validationkit'); # Don't leave the values as None.
974
975 #
976 # Set the member values.
977 #
978 self.sVBoxValidationKit = sCandidate;
979 self.sVBoxValidationKitIso = os.path.join(sCandidate, 'VBoxValidationKit.iso');
980 self.sVBoxBootSectors = os.path.join(sCandidate, 'bootsectors');
981 return fRc;
982
983 def _makeEnvironmentChanges(self):
984 """
985 Make the necessary VBox related environment changes.
986 Children not importing the VBox API should call this.
987 """
988 # Make sure we've got our own VirtualBox config and VBoxSVC (on XPCOM at least).
989 if not self.fUseDefaultSvc:
990 os.environ['VBOX_USER_HOME'] = os.path.join(self.sScratchPath, 'VBoxUserHome');
991 sUser = os.environ.get('USERNAME', os.environ.get('USER', os.environ.get('LOGNAME', 'unknown')));
992 os.environ['VBOX_IPC_SOCKETID'] = sUser + '-VBoxTest';
993 return True;
994
995 def importVBoxApi(self):
996 """
997 Import the 'vboxapi' module from the VirtualBox build we're using and
998 instantiate the two basic objects.
999
1000 This will try detect an development or installed build if no build has
1001 been associated with the driver yet.
1002 """
1003 if self.fImportedVBoxApi:
1004 return True;
1005
1006 self._makeEnvironmentChanges();
1007
1008 # Do the detecting.
1009 self._detectBuild();
1010 if self.oBuild is None:
1011 return False;
1012
1013 # Avoid crashing when loading the 32-bit module (or whatever it is that goes bang).
1014 if self.oBuild.sArch == 'x86' \
1015 and self.sHost == 'darwin' \
1016 and platform.architecture()[0] == '64bit' \
1017 and self.oBuild.sKind == 'development' \
1018 and os.getenv('VERSIONER_PYTHON_PREFER_32_BIT') != 'yes':
1019 reporter.log("WARNING: 64-bit python on darwin, 32-bit VBox development build => crash");
1020 reporter.log("WARNING: bash-3.2$ /usr/bin/python2.5 ./testdriver");
1021 reporter.log("WARNING: or");
1022 reporter.log("WARNING: bash-3.2$ VERSIONER_PYTHON_PREFER_32_BIT=yes ./testdriver");
1023 return False;
1024
1025 # Start VBoxSVC and load the vboxapi bits.
1026 if self._startVBoxSVC() is True:
1027 assert(self.oVBoxSvcProcess is not None);
1028
1029 sSavedSysPath = sys.path;
1030 self._setupVBoxApi();
1031 sys.path = sSavedSysPath;
1032
1033 # Adjust the default machine folder.
1034 if self.fImportedVBoxApi and not self.fUseDefaultSvc and self.fpApiVer >= 4.0:
1035 sNewFolder = os.path.join(self.sScratchPath, 'VBoxUserHome', 'Machines');
1036 try:
1037 self.oVBox.systemProperties.defaultMachineFolder = sNewFolder;
1038 except:
1039 self.fImportedVBoxApi = False;
1040 self.oVBoxMgr = None;
1041 self.oVBox = None;
1042 reporter.logXcpt("defaultMachineFolder exception (sNewFolder=%s)" % (sNewFolder,));
1043
1044 # Kill VBoxSVC on failure.
1045 if self.oVBoxMgr is None:
1046 self._stopVBoxSVC();
1047 else:
1048 assert(self.oVBoxSvcProcess is None);
1049 return self.fImportedVBoxApi;
1050
1051 def _startVBoxSVC(self): # pylint: disable=too-many-statements
1052 """ Starts VBoxSVC. """
1053 assert(self.oVBoxSvcProcess is None);
1054
1055 # Setup vbox logging for VBoxSVC now and start it manually. This way
1056 # we can control both logging and shutdown.
1057 self.sVBoxSvcLogFile = '%s/VBoxSVC-debug.log' % (self.sScratchPath,);
1058 try: os.remove(self.sVBoxSvcLogFile);
1059 except: pass;
1060 os.environ['VBOX_LOG'] = self.sLogSvcGroups;
1061 os.environ['VBOX_LOG_FLAGS'] = '%s append' % (self.sLogSvcFlags,); # Append becuse of VBoxXPCOMIPCD.
1062 if self.sLogSvcDest:
1063 os.environ['VBOX_LOG_DEST'] = 'nodeny ' + self.sLogSvcDest;
1064 else:
1065 os.environ['VBOX_LOG_DEST'] = 'nodeny file=%s' % (self.sVBoxSvcLogFile,);
1066 os.environ['VBOXSVC_RELEASE_LOG_FLAGS'] = 'time append';
1067
1068 # Always leave a pid file behind so we can kill it during cleanup-before.
1069 self.sVBoxSvcPidFile = '%s/VBoxSVC.pid' % (self.sScratchPath,);
1070 fWritePidFile = True;
1071
1072 cMsFudge = 1;
1073 sVBoxSVC = '%s/VBoxSVC' % (self.oBuild.sInstallPath,); ## @todo .exe and stuff.
1074 if self.fVBoxSvcInDebugger:
1075 if self.sHost in ('darwin', 'freebsd', 'linux', 'solaris', ):
1076 # Start VBoxSVC in gdb in a new terminal.
1077 #sTerm = '/usr/bin/gnome-terminal'; - doesn't work, some fork+exec stuff confusing us.
1078 sTerm = '/usr/bin/xterm';
1079 if not os.path.isfile(sTerm): sTerm = '/usr/X11/bin/xterm';
1080 if not os.path.isfile(sTerm): sTerm = '/usr/X11R6/bin/xterm';
1081 if not os.path.isfile(sTerm): sTerm = '/usr/bin/xterm';
1082 if not os.path.isfile(sTerm): sTerm = 'xterm';
1083 sGdb = '/usr/bin/gdb';
1084 if not os.path.isfile(sGdb): sGdb = '/usr/local/bin/gdb';
1085 if not os.path.isfile(sGdb): sGdb = '/usr/sfw/bin/gdb';
1086 if not os.path.isfile(sGdb): sGdb = 'gdb';
1087 sGdbCmdLine = '%s --args %s --pidfile %s' % (sGdb, sVBoxSVC, self.sVBoxSvcPidFile);
1088 reporter.log('term="%s" gdb="%s"' % (sTerm, sGdbCmdLine));
1089 os.environ['SHELL'] = self.sOrgShell; # Non-working shell may cause gdb and/or the term problems.
1090 ## @todo -e is deprecated; use "-- <args>".
1091 self.oVBoxSvcProcess = base.Process.spawnp(sTerm, sTerm, '-e', sGdbCmdLine);
1092 os.environ['SHELL'] = self.sOurShell;
1093 if self.oVBoxSvcProcess is not None:
1094 reporter.log('Press enter or return after starting VBoxSVC in the debugger...');
1095 sys.stdin.read(1);
1096 fWritePidFile = False;
1097
1098 elif self.sHost == 'win':
1099 sWinDbg = 'c:\\Program Files\\Debugging Tools for Windows\\windbg.exe';
1100 if not os.path.isfile(sWinDbg): sWinDbg = 'c:\\Program Files\\Debugging Tools for Windows (x64)\\windbg.exe';
1101 if not os.path.isfile(sWinDbg): sWinDbg = 'c:\\Programme\\Debugging Tools for Windows\\windbg.exe'; # Localization rulez! pylint: disable=line-too-long
1102 if not os.path.isfile(sWinDbg): sWinDbg = 'c:\\Programme\\Debugging Tools for Windows (x64)\\windbg.exe';
1103 if not os.path.isfile(sWinDbg): sWinDbg = 'windbg'; # WinDbg must be in the path; better than nothing.
1104 # Assume that everything WinDbg needs is defined using the environment variables.
1105 # See WinDbg help for more information.
1106 reporter.log('windbg="%s"' % (sWinDbg));
1107 self.oVBoxSvcProcess = base.Process.spawn(sWinDbg, sWinDbg, sVBoxSVC + base.exeSuff());
1108 if self.oVBoxSvcProcess is not None:
1109 reporter.log('Press enter or return after starting VBoxSVC in the debugger...');
1110 sys.stdin.read(1);
1111 fWritePidFile = False;
1112 ## @todo add a pipe interface similar to xpcom if feasible, i.e. if
1113 # we can get actual handle values for pipes in python.
1114
1115 else:
1116 reporter.error('Port me!');
1117 else: # Run without a debugger attached.
1118 if self.sHost in ('darwin', 'freebsd', 'linux', 'solaris', ):
1119 #
1120 # XPCOM - We can use a pipe to let VBoxSVC notify us when it's ready.
1121 #
1122 iPipeR, iPipeW = os.pipe();
1123 if hasattr(os, 'set_inheritable'):
1124 os.set_inheritable(iPipeW, True); # pylint: disable=no-member
1125 os.environ['NSPR_INHERIT_FDS'] = 'vboxsvc:startup-pipe:5:0x%x' % (iPipeW,);
1126 reporter.log2("NSPR_INHERIT_FDS=%s" % (os.environ['NSPR_INHERIT_FDS']));
1127
1128 self.oVBoxSvcProcess = base.Process.spawn(sVBoxSVC, sVBoxSVC, '--auto-shutdown'); # SIGUSR1 requirement.
1129 try: # Try make sure we get the SIGINT and not VBoxSVC.
1130 os.setpgid(self.oVBoxSvcProcess.getPid(), 0); # pylint: disable=no-member
1131 os.setpgid(0, 0); # pylint: disable=no-member
1132 except:
1133 reporter.logXcpt();
1134
1135 os.close(iPipeW);
1136 try:
1137 sResponse = os.read(iPipeR, 32);
1138 except:
1139 reporter.logXcpt();
1140 sResponse = None;
1141 os.close(iPipeR);
1142
1143 if hasattr(sResponse, 'decode'):
1144 sResponse = sResponse.decode('utf-8', 'ignore');
1145
1146 if sResponse is None or sResponse.strip() != 'READY':
1147 reporter.error('VBoxSVC failed starting up... (sResponse=%s)' % (sResponse,));
1148 if not self.oVBoxSvcProcess.wait(5000):
1149 self.oVBoxSvcProcess.terminate();
1150 self.oVBoxSvcProcess.wait(5000);
1151 self.oVBoxSvcProcess = None;
1152
1153 elif self.sHost == 'win':
1154 #
1155 # Windows - Just fudge it for now.
1156 #
1157 cMsFudge = 2000;
1158 self.oVBoxSvcProcess = base.Process.spawn(sVBoxSVC, sVBoxSVC);
1159
1160 else:
1161 reporter.error('Port me!');
1162
1163 #
1164 # Enable automatic crash reporting if we succeeded.
1165 #
1166 if self.oVBoxSvcProcess is not None:
1167 self.oVBoxSvcProcess.enableCrashReporting('crash/report/svc', 'crash/dump/svc');
1168
1169 #
1170 # Fudge and pid file.
1171 #
1172 if self.oVBoxSvcProcess is not None and not self.oVBoxSvcProcess.wait(cMsFudge):
1173 if fWritePidFile:
1174 iPid = self.oVBoxSvcProcess.getPid();
1175 try:
1176 oFile = utils.openNoInherit(self.sVBoxSvcPidFile, "w+");
1177 oFile.write('%s' % (iPid,));
1178 oFile.close();
1179 except:
1180 reporter.logXcpt('sPidFile=%s' % (self.sVBoxSvcPidFile,));
1181 reporter.log('VBoxSVC PID=%u' % (iPid,));
1182
1183 #
1184 # Finally add the task so we'll notice when it dies in a relatively timely manner.
1185 #
1186 self.addTask(self.oVBoxSvcProcess);
1187 else:
1188 self.oVBoxSvcProcess = None;
1189 try: os.remove(self.sVBoxSvcPidFile);
1190 except: pass;
1191
1192 return self.oVBoxSvcProcess is not None;
1193
1194
1195 def _killVBoxSVCByPidFile(self, sPidFile):
1196 """ Kill a VBoxSVC given the pid from it's pid file. """
1197
1198 # Read the pid file.
1199 if not os.path.isfile(sPidFile):
1200 return False;
1201 try:
1202 oFile = utils.openNoInherit(sPidFile, "r");
1203 sPid = oFile.readline().strip();
1204 oFile.close();
1205 except:
1206 reporter.logXcpt('sPidfile=%s' % (sPidFile,));
1207 return False;
1208
1209 # Convert the pid to an integer and validate the range a little bit.
1210 try:
1211 iPid = long(sPid);
1212 except:
1213 reporter.logXcpt('sPidfile=%s sPid="%s"' % (sPidFile, sPid));
1214 return False;
1215 if iPid <= 0:
1216 reporter.log('negative pid - sPidfile=%s sPid="%s" iPid=%d' % (sPidFile, sPid, iPid));
1217 return False;
1218
1219 # Take care checking that it's VBoxSVC we're about to inhume.
1220 if base.processCheckPidAndName(iPid, "VBoxSVC") is not True:
1221 reporter.log('Ignoring stale VBoxSVC pid file (pid=%s)' % (iPid,));
1222 return False;
1223
1224 # Loop thru our different ways of getting VBoxSVC to terminate.
1225 for aHow in [ [ base.sendUserSignal1, 5000, 'Dropping VBoxSVC a SIGUSR1 hint...'], \
1226 [ base.processInterrupt, 5000, 'Dropping VBoxSVC a SIGINT hint...'], \
1227 [ base.processTerminate, 7500, 'VBoxSVC is still around, killing it...'] ]:
1228 reporter.log(aHow[2]);
1229 if aHow[0](iPid) is True:
1230 msStart = base.timestampMilli();
1231 while base.timestampMilli() - msStart < 5000 \
1232 and base.processExists(iPid):
1233 time.sleep(0.2);
1234
1235 fRc = not base.processExists(iPid);
1236 if fRc is True:
1237 break;
1238 if fRc:
1239 reporter.log('Successfully killed VBoxSVC (pid=%s)' % (iPid,));
1240 else:
1241 reporter.log('Failed to kill VBoxSVC (pid=%s)' % (iPid,));
1242 return fRc;
1243
1244 def _stopVBoxSVC(self):
1245 """
1246 Stops VBoxSVC. Try the polite way first.
1247 """
1248
1249 if self.oVBoxSvcProcess:
1250 self.removeTask(self.oVBoxSvcProcess);
1251 self.oVBoxSvcProcess.enableCrashReporting(None, None); # Disables it.
1252
1253 fRc = False;
1254 if self.oVBoxSvcProcess is not None \
1255 and not self.fVBoxSvcInDebugger:
1256 # by process object.
1257 if self.oVBoxSvcProcess.isRunning():
1258 reporter.log('Dropping VBoxSVC a SIGUSR1 hint...');
1259 if not self.oVBoxSvcProcess.sendUserSignal1() \
1260 or not self.oVBoxSvcProcess.wait(5000):
1261 reporter.log('Dropping VBoxSVC a SIGINT hint...');
1262 if not self.oVBoxSvcProcess.interrupt() \
1263 or not self.oVBoxSvcProcess.wait(5000):
1264 reporter.log('VBoxSVC is still around, killing it...');
1265 self.oVBoxSvcProcess.terminate();
1266 self.oVBoxSvcProcess.wait(7500);
1267 else:
1268 reporter.log('VBoxSVC is no longer running...');
1269 if not self.oVBoxSvcProcess.isRunning():
1270 self.oVBoxSvcProcess = None;
1271 else:
1272 # by pid file.
1273 self._killVBoxSVCByPidFile('%s/VBoxSVC.pid' % (self.sScratchPath,));
1274 return fRc;
1275
1276 def _setupVBoxApi(self):
1277 """
1278 Import and set up the vboxapi.
1279 The caller saves and restores sys.path.
1280 """
1281
1282 # Setup vbox logging for self (the test driver).
1283 self.sSelfLogFile = '%s/VBoxTestDriver.log' % (self.sScratchPath,);
1284 try: os.remove(self.sSelfLogFile);
1285 except: pass;
1286 os.environ['VBOX_LOG'] = self.sLogSelfGroups;
1287 os.environ['VBOX_LOG_FLAGS'] = '%s append' % (self.sLogSelfFlags, );
1288 if self.sLogSelfDest:
1289 os.environ['VBOX_LOG_DEST'] = 'nodeny ' + self.sLogSelfDest;
1290 else:
1291 os.environ['VBOX_LOG_DEST'] = 'nodeny file=%s' % (self.sSelfLogFile,);
1292 os.environ['VBOX_RELEASE_LOG_FLAGS'] = 'time append';
1293
1294 # Hack the sys.path + environment so the vboxapi can be found.
1295 sys.path.insert(0, self.oBuild.sInstallPath);
1296 if self.oBuild.sSdkPath is not None:
1297 sys.path.insert(0, os.path.join(self.oBuild.sSdkPath, 'installer'))
1298 sys.path.insert(1, os.path.join(self.oBuild.sSdkPath, 'install')); # stupid stupid windows installer!
1299 sys.path.insert(2, os.path.join(self.oBuild.sSdkPath, 'bindings', 'xpcom', 'python'))
1300 os.environ['VBOX_PROGRAM_PATH'] = self.oBuild.sInstallPath;
1301 reporter.log("sys.path: %s" % (sys.path));
1302
1303 try:
1304 from vboxapi import VirtualBoxManager; # pylint: disable=import-error
1305 except:
1306 reporter.logXcpt('Error importing vboxapi');
1307 return False;
1308
1309 # Exception and error hacks.
1310 try:
1311 # pylint: disable=import-error
1312 if self.sHost == 'win':
1313 from pythoncom import com_error as NativeComExceptionClass # pylint: disable=no-name-in-module
1314 import winerror as NativeComErrorClass
1315 else:
1316 from xpcom import Exception as NativeComExceptionClass
1317 from xpcom import nsError as NativeComErrorClass
1318 # pylint: enable=import-error
1319 except:
1320 reporter.logXcpt('Error importing (XP)COM related stuff for exception hacks and errors');
1321 return False;
1322 __deployExceptionHacks__(NativeComExceptionClass)
1323 ComError.copyErrors(NativeComErrorClass);
1324
1325 # Create the manager.
1326 try:
1327 self.oVBoxMgr = VirtualBoxManager(None, None)
1328 except:
1329 self.oVBoxMgr = None;
1330 reporter.logXcpt('VirtualBoxManager exception');
1331 return False;
1332
1333 # Figure the API version.
1334 try:
1335 oVBox = self.oVBoxMgr.getVirtualBox();
1336
1337 try:
1338 sVer = oVBox.version;
1339 except:
1340 reporter.logXcpt('Failed to get VirtualBox version, assuming 4.0.0');
1341 sVer = "4.0.0";
1342 reporter.log("IVirtualBox.version=%s" % (sVer,));
1343
1344 # Convert the string to three integer values and check ranges.
1345 asVerComponents = sVer.split('.');
1346 try:
1347 sLast = asVerComponents[2].split('_')[0].split('r')[0];
1348 aiVerComponents = (int(asVerComponents[0]), int(asVerComponents[1]), int(sLast));
1349 except:
1350 raise base.GenError('Malformed version "%s"' % (sVer,));
1351 if aiVerComponents[0] < 3 or aiVerComponents[0] > 19:
1352 raise base.GenError('Malformed version "%s" - 1st component is out of bounds 3..19: %u'
1353 % (sVer, aiVerComponents[0]));
1354 if aiVerComponents[1] < 0 or aiVerComponents[1] > 9:
1355 raise base.GenError('Malformed version "%s" - 2nd component is out of bounds 0..9: %u'
1356 % (sVer, aiVerComponents[1]));
1357 if aiVerComponents[2] < 0 or aiVerComponents[2] > 99:
1358 raise base.GenError('Malformed version "%s" - 3rd component is out of bounds 0..99: %u'
1359 % (sVer, aiVerComponents[2]));
1360
1361 # Convert the three integers into a floating point value. The API is table witin a
1362 # x.y release, so the third component only indicates whether it's a stable or
1363 # development build of the next release.
1364 self.fpApiVer = aiVerComponents[0] + 0.1 * aiVerComponents[1];
1365 if aiVerComponents[2] >= 51:
1366 if self.fpApiVer not in [4.3, 3.2,]:
1367 self.fpApiVer += 0.1;
1368 else:
1369 self.fpApiVer += 1.1;
1370
1371 try:
1372 self.uRevision = oVBox.revision;
1373 except:
1374 reporter.logXcpt('Failed to get VirtualBox revision, assuming 0');
1375 self.uRevision = 0;
1376 reporter.log("IVirtualBox.revision=%u" % (self.uRevision,));
1377
1378 # Patch VBox manage to gloss over portability issues (error constants, etc).
1379 self._patchVBoxMgr();
1380
1381 # Wrap oVBox.
1382 from testdriver.vboxwrappers import VirtualBoxWrapper;
1383 self.oVBox = VirtualBoxWrapper(oVBox, self.oVBoxMgr, self.fpApiVer, self);
1384
1385 # Install the constant wrapping hack.
1386 vboxcon.goHackModuleClass.oVBoxMgr = self.oVBoxMgr; # VBoxConstantWrappingHack.
1387 vboxcon.fpApiVer = self.fpApiVer;
1388 reporter.setComXcptFormatter(formatComOrXpComException);
1389
1390 except:
1391 self.oVBoxMgr = None;
1392 self.oVBox = None;
1393 reporter.logXcpt("getVirtualBox / API version exception");
1394 return False;
1395
1396 # Done
1397 self.fImportedVBoxApi = True;
1398 reporter.log('Found version %s (%s)' % (self.fpApiVer, sVer));
1399 return True;
1400
1401 def _patchVBoxMgr(self):
1402 """
1403 Glosses over missing self.oVBoxMgr methods on older VBox versions.
1404 """
1405
1406 def _xcptGetResult(oSelf, oXcpt = None):
1407 """ See vboxapi. """
1408 _ = oSelf;
1409 if oXcpt is None: oXcpt = sys.exc_info()[1];
1410 if sys.platform == 'win32':
1411 import winerror; # pylint: disable=import-error
1412 hrXcpt = oXcpt.hresult;
1413 if hrXcpt == winerror.DISP_E_EXCEPTION:
1414 hrXcpt = oXcpt.excepinfo[5];
1415 else:
1416 hrXcpt = oXcpt.error;
1417 return hrXcpt;
1418
1419 def _xcptIsDeadInterface(oSelf, oXcpt = None):
1420 """ See vboxapi. """
1421 return oSelf.xcptGetStatus(oXcpt) in [
1422 0x80004004, -2147467260, # NS_ERROR_ABORT
1423 0x800706be, -2147023170, # NS_ERROR_CALL_FAILED (RPC_S_CALL_FAILED)
1424 0x800706ba, -2147023174, # RPC_S_SERVER_UNAVAILABLE.
1425 0x800706be, -2147023170, # RPC_S_CALL_FAILED.
1426 0x800706bf, -2147023169, # RPC_S_CALL_FAILED_DNE.
1427 0x80010108, -2147417848, # RPC_E_DISCONNECTED.
1428 0x800706b5, -2147023179, # RPC_S_UNKNOWN_IF
1429 ];
1430
1431 def _xcptIsOurXcptKind(oSelf, oXcpt = None):
1432 """ See vboxapi. """
1433 _ = oSelf;
1434 if oXcpt is None: oXcpt = sys.exc_info()[1];
1435 if sys.platform == 'win32':
1436 from pythoncom import com_error as NativeComExceptionClass # pylint: disable=import-error,no-name-in-module
1437 else:
1438 from xpcom import Exception as NativeComExceptionClass # pylint: disable=import-error
1439 return isinstance(oXcpt, NativeComExceptionClass);
1440
1441 def _xcptIsEqual(oSelf, oXcpt, hrStatus):
1442 """ See vboxapi. """
1443 hrXcpt = oSelf.xcptGetResult(oXcpt);
1444 return hrXcpt == hrStatus or hrXcpt == hrStatus - 0x100000000; # pylint: disable=consider-using-in
1445
1446 def _xcptToString(oSelf, oXcpt):
1447 """ See vboxapi. """
1448 _ = oSelf;
1449 if oXcpt is None: oXcpt = sys.exc_info()[1];
1450 return str(oXcpt);
1451
1452 # Add utilities found in newer vboxapi revision.
1453 if not hasattr(self.oVBoxMgr, 'xcptIsDeadInterface'):
1454 import types;
1455 self.oVBoxMgr.xcptGetResult = types.MethodType(_xcptGetResult, self.oVBoxMgr);
1456 self.oVBoxMgr.xcptIsDeadInterface = types.MethodType(_xcptIsDeadInterface, self.oVBoxMgr);
1457 self.oVBoxMgr.xcptIsOurXcptKind = types.MethodType(_xcptIsOurXcptKind, self.oVBoxMgr);
1458 self.oVBoxMgr.xcptIsEqual = types.MethodType(_xcptIsEqual, self.oVBoxMgr);
1459 self.oVBoxMgr.xcptToString = types.MethodType(_xcptToString, self.oVBoxMgr);
1460
1461
1462 def _teardownVBoxApi(self): # pylint: disable=too-many-statements
1463 """
1464 Drop all VBox object references and shutdown com/xpcom.
1465 """
1466 if not self.fImportedVBoxApi:
1467 return True;
1468 import gc;
1469
1470 # Drop all references we've have to COM objects.
1471 self.aoRemoteSessions = [];
1472 self.aoVMs = [];
1473 self.oVBoxMgr = None;
1474 self.oVBox = None;
1475 vboxcon.goHackModuleClass.oVBoxMgr = None; # VBoxConstantWrappingHack.
1476 reporter.setComXcptFormatter(None);
1477
1478 # Do garbage collection to try get rid of those objects.
1479 try:
1480 gc.collect();
1481 except:
1482 reporter.logXcpt();
1483 self.fImportedVBoxApi = False;
1484
1485 # Check whether the python is still having any COM objects/interfaces around.
1486 cVBoxMgrs = 0;
1487 aoObjsLeftBehind = [];
1488 if self.sHost == 'win':
1489 import pythoncom; # pylint: disable=import-error
1490 try:
1491 cIfs = pythoncom._GetInterfaceCount(); # pylint: disable=no-member,protected-access
1492 cObjs = pythoncom._GetGatewayCount(); # pylint: disable=no-member,protected-access
1493 if cObjs == 0 and cIfs == 0:
1494 reporter.log('_teardownVBoxApi: no interfaces or objects left behind.');
1495 else:
1496 reporter.log('_teardownVBoxApi: Python COM still has %s objects and %s interfaces...' % ( cObjs, cIfs));
1497
1498 from win32com.client import DispatchBaseClass; # pylint: disable=import-error
1499 for oObj in gc.get_objects():
1500 if isinstance(oObj, DispatchBaseClass):
1501 reporter.log('_teardownVBoxApi: %s' % (oObj,));
1502 aoObjsLeftBehind.append(oObj);
1503 elif utils.getObjectTypeName(oObj) == 'VirtualBoxManager':
1504 reporter.log('_teardownVBoxApi: %s' % (oObj,));
1505 cVBoxMgrs += 1;
1506 aoObjsLeftBehind.append(oObj);
1507 oObj = None;
1508 except:
1509 reporter.logXcpt();
1510
1511 # If not being used, we can safely uninitialize COM.
1512 if cIfs == 0 and cObjs == 0 and cVBoxMgrs == 0 and not aoObjsLeftBehind:
1513 reporter.log('_teardownVBoxApi: Calling CoUninitialize...');
1514 try: pythoncom.CoUninitialize(); # pylint: disable=no-member
1515 except: reporter.logXcpt();
1516 else:
1517 reporter.log('_teardownVBoxApi: Returned from CoUninitialize.');
1518 else:
1519 try:
1520 # XPCOM doesn't crash and burn like COM if you shut it down with interfaces and objects around.
1521 # Also, it keeps a number of internal objects and interfaces around to do its job, so shutting
1522 # it down before we go looking for dangling interfaces is more or less required.
1523 from xpcom import _xpcom as _xpcom; # pylint: disable=import-error,useless-import-alias
1524 hrc = _xpcom.DeinitCOM();
1525 cIfs = _xpcom._GetInterfaceCount(); # pylint: disable=protected-access
1526 cObjs = _xpcom._GetGatewayCount(); # pylint: disable=protected-access
1527
1528 if cObjs == 0 and cIfs == 0:
1529 reporter.log('_teardownVBoxApi: No XPCOM interfaces or objects active. (hrc=%#x)' % (hrc,));
1530 else:
1531 reporter.log('_teardownVBoxApi: %s XPCOM objects and %s interfaces still around! (hrc=%#x)'
1532 % (cObjs, cIfs, hrc));
1533 if hasattr(_xpcom, '_DumpInterfaces'):
1534 try: _xpcom._DumpInterfaces(); # pylint: disable=protected-access
1535 except: reporter.logXcpt('_teardownVBoxApi: _DumpInterfaces failed');
1536
1537 from xpcom.client import Component; # pylint: disable=import-error
1538 for oObj in gc.get_objects():
1539 if isinstance(oObj, Component):
1540 reporter.log('_teardownVBoxApi: %s' % (oObj,));
1541 aoObjsLeftBehind.append(oObj);
1542 if utils.getObjectTypeName(oObj) == 'VirtualBoxManager':
1543 reporter.log('_teardownVBoxApi: %s' % (oObj,));
1544 cVBoxMgrs += 1;
1545 aoObjsLeftBehind.append(oObj);
1546 oObj = None;
1547 except:
1548 reporter.logXcpt();
1549
1550 # Try get the referrers to (XP)COM interfaces and objects that was left behind.
1551 for iObj in range(len(aoObjsLeftBehind)): # pylint: disable=consider-using-enumerate
1552 try:
1553 aoReferrers = gc.get_referrers(aoObjsLeftBehind[iObj]);
1554 reporter.log('_teardownVBoxApi: Found %u referrers to %s:' % (len(aoReferrers), aoObjsLeftBehind[iObj],));
1555 for oReferrer in aoReferrers:
1556 oMyFrame = sys._getframe(0); # pylint: disable=protected-access
1557 if oReferrer is oMyFrame:
1558 reporter.log('_teardownVBoxApi: - frame of this function');
1559 elif oReferrer is aoObjsLeftBehind:
1560 reporter.log('_teardownVBoxApi: - aoObjsLeftBehind');
1561 else:
1562 fPrinted = False;
1563 if isinstance(oReferrer, (dict, list, tuple)):
1564 try:
1565 aoSubReferreres = gc.get_referrers(oReferrer);
1566 for oSubRef in aoSubReferreres:
1567 if not isinstance(oSubRef, list) \
1568 and not isinstance(oSubRef, dict) \
1569 and oSubRef is not oMyFrame \
1570 and oSubRef is not aoSubReferreres:
1571 reporter.log('_teardownVBoxApi: - %s :: %s:'
1572 % (utils.getObjectTypeName(oSubRef), utils.getObjectTypeName(oReferrer)));
1573 fPrinted = True;
1574 break;
1575 del aoSubReferreres;
1576 except:
1577 reporter.logXcpt('subref');
1578 if not fPrinted:
1579 reporter.log('_teardownVBoxApi: - %s:' % (utils.getObjectTypeName(oReferrer),));
1580 try:
1581 import pprint;
1582 for sLine in pprint.pformat(oReferrer, width = 130).split('\n'):
1583 reporter.log('_teardownVBoxApi: %s' % (sLine,));
1584 except:
1585 reporter.log('_teardownVBoxApi: %s' % (oReferrer,));
1586 except:
1587 reporter.logXcpt();
1588 del aoObjsLeftBehind;
1589
1590 # Force garbage collection again, just for good measure.
1591 try:
1592 gc.collect();
1593 time.sleep(0.5); # fudge factor
1594 except:
1595 reporter.logXcpt();
1596 return True;
1597
1598 def _powerOffAllVms(self):
1599 """
1600 Tries to power off all running VMs.
1601 """
1602 for oSession in self.aoRemoteSessions:
1603 uPid = oSession.getPid();
1604 if uPid is not None:
1605 reporter.log('_powerOffAllVms: PID is %s for %s, trying to kill it.' % (uPid, oSession.sName,));
1606 base.processKill(uPid);
1607 else:
1608 reporter.log('_powerOffAllVms: No PID for %s' % (oSession.sName,));
1609 oSession.close();
1610 return None;
1611
1612
1613
1614 #
1615 # Build type, OS and arch getters.
1616 #
1617
1618 def getBuildType(self):
1619 """
1620 Get the build type.
1621 """
1622 if not self._detectBuild():
1623 return 'release';
1624 return self.oBuild.sType;
1625
1626 def getBuildOs(self):
1627 """
1628 Get the build OS.
1629 """
1630 if not self._detectBuild():
1631 return self.sHost;
1632 return self.oBuild.sOs;
1633
1634 def getBuildArch(self):
1635 """
1636 Get the build arch.
1637 """
1638 if not self._detectBuild():
1639 return self.sHostArch;
1640 return self.oBuild.sArch;
1641
1642 def getGuestAdditionsIso(self):
1643 """
1644 Get the path to the guest addition iso.
1645 """
1646 if not self._detectBuild():
1647 return None;
1648 return self.oBuild.sGuestAdditionsIso;
1649
1650 #
1651 # Override everything from the base class so the testdrivers don't have to
1652 # check whether we have overridden a method or not.
1653 #
1654
1655 def showUsage(self):
1656 rc = base.TestDriver.showUsage(self);
1657 reporter.log('');
1658 reporter.log('Generic VirtualBox Options:');
1659 reporter.log(' --vbox-session-type <type>');
1660 reporter.log(' Sets the session type. Typical values are: gui, headless, sdl');
1661 reporter.log(' Default: %s' % (self.sSessionTypeDef));
1662 reporter.log(' --vrdp, --no-vrdp');
1663 reporter.log(' Enables VRDP, ports starting at 6000');
1664 reporter.log(' Default: --vrdp');
1665 reporter.log(' --vrdp-base-port <port>');
1666 reporter.log(' Sets the base for VRDP port assignments.');
1667 reporter.log(' Default: %s' % (self.uVrdpBasePortDef));
1668 reporter.log(' --vbox-default-bridged-nic <interface>');
1669 reporter.log(' Sets the default interface for bridged networking.');
1670 reporter.log(' Default: autodetect');
1671 reporter.log(' --vbox-use-svc-defaults');
1672 reporter.log(' Use default locations and files for VBoxSVC. This is useful');
1673 reporter.log(' for automatically configuring the test VMs for debugging.');
1674 reporter.log(' --vbox-self-log');
1675 reporter.log(' The VBox logger group settings for the testdriver.');
1676 reporter.log(' --vbox-self-log-flags');
1677 reporter.log(' The VBox logger flags settings for the testdriver.');
1678 reporter.log(' --vbox-self-log-dest');
1679 reporter.log(' The VBox logger destination settings for the testdriver.');
1680 reporter.log(' --vbox-session-log');
1681 reporter.log(' The VM session logger group settings.');
1682 reporter.log(' --vbox-session-log-flags');
1683 reporter.log(' The VM session logger flags.');
1684 reporter.log(' --vbox-session-log-dest');
1685 reporter.log(' The VM session logger destination settings.');
1686 reporter.log(' --vbox-svc-log');
1687 reporter.log(' The VBoxSVC logger group settings.');
1688 reporter.log(' --vbox-svc-log-flags');
1689 reporter.log(' The VBoxSVC logger flag settings.');
1690 reporter.log(' --vbox-svc-log-dest');
1691 reporter.log(' The VBoxSVC logger destination settings.');
1692 reporter.log(' --vbox-log');
1693 reporter.log(' The VBox logger group settings for everyone.');
1694 reporter.log(' --vbox-log-flags');
1695 reporter.log(' The VBox logger flags settings for everyone.');
1696 reporter.log(' --vbox-log-dest');
1697 reporter.log(' The VBox logger destination settings for everyone.');
1698 reporter.log(' --vbox-svc-debug');
1699 reporter.log(' Start VBoxSVC in a debugger');
1700 reporter.log(' --vbox-always-upload-logs');
1701 reporter.log(' Whether to always upload log files, or only do so on failure.');
1702 reporter.log(' --vbox-always-upload-screenshots');
1703 reporter.log(' Whether to always upload final screen shots, or only do so on failure.');
1704 reporter.log(' --vbox-debugger, --no-vbox-debugger');
1705 reporter.log(' Enables the VBox debugger, port at 5000');
1706 reporter.log(' Default: --vbox-debugger');
1707 if self.oTestVmSet is not None:
1708 self.oTestVmSet.showUsage();
1709 return rc;
1710
1711 def parseOption(self, asArgs, iArg): # pylint: disable=too-many-statements
1712 if asArgs[iArg] == '--vbox-session-type':
1713 iArg += 1;
1714 if iArg >= len(asArgs):
1715 raise base.InvalidOption('The "--vbox-session-type" takes an argument');
1716 self.sSessionType = asArgs[iArg];
1717 elif asArgs[iArg] == '--vrdp':
1718 self.fEnableVrdp = True;
1719 elif asArgs[iArg] == '--no-vrdp':
1720 self.fEnableVrdp = False;
1721 elif asArgs[iArg] == '--vrdp-base-port':
1722 iArg += 1;
1723 if iArg >= len(asArgs):
1724 raise base.InvalidOption('The "--vrdp-base-port" takes an argument');
1725 try: self.uVrdpBasePort = int(asArgs[iArg]);
1726 except: raise base.InvalidOption('The "--vrdp-base-port" value "%s" is not a valid integer' % (asArgs[iArg],));
1727 if self.uVrdpBasePort <= 0 or self.uVrdpBasePort >= 65530:
1728 raise base.InvalidOption('The "--vrdp-base-port" value "%s" is not in the valid range (1..65530)'
1729 % (asArgs[iArg],));
1730 elif asArgs[iArg] == '--vbox-default-bridged-nic':
1731 iArg += 1;
1732 if iArg >= len(asArgs):
1733 raise base.InvalidOption('The "--vbox-default-bridged-nic" takes an argument');
1734 self.sDefBridgedNic = asArgs[iArg];
1735 elif asArgs[iArg] == '--vbox-use-svc-defaults':
1736 self.fUseDefaultSvc = True;
1737 elif asArgs[iArg] == '--vbox-self-log':
1738 iArg += 1;
1739 if iArg >= len(asArgs):
1740 raise base.InvalidOption('The "--vbox-self-log" takes an argument');
1741 self.sLogSelfGroups = asArgs[iArg];
1742 elif asArgs[iArg] == '--vbox-self-log-flags':
1743 iArg += 1;
1744 if iArg >= len(asArgs):
1745 raise base.InvalidOption('The "--vbox-self-log-flags" takes an argument');
1746 self.sLogSelfFlags = asArgs[iArg];
1747 elif asArgs[iArg] == '--vbox-self-log-dest':
1748 iArg += 1;
1749 if iArg >= len(asArgs):
1750 raise base.InvalidOption('The "--vbox-self-log-dest" takes an argument');
1751 self.sLogSelfDest = asArgs[iArg];
1752 elif asArgs[iArg] == '--vbox-session-log':
1753 iArg += 1;
1754 if iArg >= len(asArgs):
1755 raise base.InvalidOption('The "--vbox-session-log" takes an argument');
1756 self.sLogSessionGroups = asArgs[iArg];
1757 elif asArgs[iArg] == '--vbox-session-log-flags':
1758 iArg += 1;
1759 if iArg >= len(asArgs):
1760 raise base.InvalidOption('The "--vbox-session-log-flags" takes an argument');
1761 self.sLogSessionFlags = asArgs[iArg];
1762 elif asArgs[iArg] == '--vbox-session-log-dest':
1763 iArg += 1;
1764 if iArg >= len(asArgs):
1765 raise base.InvalidOption('The "--vbox-session-log-dest" takes an argument');
1766 self.sLogSessionDest = asArgs[iArg];
1767 elif asArgs[iArg] == '--vbox-svc-log':
1768 iArg += 1;
1769 if iArg >= len(asArgs):
1770 raise base.InvalidOption('The "--vbox-svc-log" takes an argument');
1771 self.sLogSvcGroups = asArgs[iArg];
1772 elif asArgs[iArg] == '--vbox-svc-log-flags':
1773 iArg += 1;
1774 if iArg >= len(asArgs):
1775 raise base.InvalidOption('The "--vbox-svc-log-flags" takes an argument');
1776 self.sLogSvcFlags = asArgs[iArg];
1777 elif asArgs[iArg] == '--vbox-svc-log-dest':
1778 iArg += 1;
1779 if iArg >= len(asArgs):
1780 raise base.InvalidOption('The "--vbox-svc-log-dest" takes an argument');
1781 self.sLogSvcDest = asArgs[iArg];
1782 elif asArgs[iArg] == '--vbox-log':
1783 iArg += 1;
1784 if iArg >= len(asArgs):
1785 raise base.InvalidOption('The "--vbox-log" takes an argument');
1786 self.sLogSelfGroups = asArgs[iArg];
1787 self.sLogSessionGroups = asArgs[iArg];
1788 self.sLogSvcGroups = asArgs[iArg];
1789 elif asArgs[iArg] == '--vbox-log-flags':
1790 iArg += 1;
1791 if iArg >= len(asArgs):
1792 raise base.InvalidOption('The "--vbox-svc-flags" takes an argument');
1793 self.sLogSelfFlags = asArgs[iArg];
1794 self.sLogSessionFlags = asArgs[iArg];
1795 self.sLogSvcFlags = asArgs[iArg];
1796 elif asArgs[iArg] == '--vbox-log-dest':
1797 iArg += 1;
1798 if iArg >= len(asArgs):
1799 raise base.InvalidOption('The "--vbox-log-dest" takes an argument');
1800 self.sLogSelfDest = asArgs[iArg];
1801 self.sLogSessionDest = asArgs[iArg];
1802 self.sLogSvcDest = asArgs[iArg];
1803 elif asArgs[iArg] == '--vbox-svc-debug':
1804 self.fVBoxSvcInDebugger = True;
1805 elif asArgs[iArg] == '--vbox-always-upload-logs':
1806 self.fAlwaysUploadLogs = True;
1807 elif asArgs[iArg] == '--vbox-always-upload-screenshots':
1808 self.fAlwaysUploadScreenshots = True;
1809 elif asArgs[iArg] == '--vbox-debugger':
1810 self.fEnableDebugger = True;
1811 elif asArgs[iArg] == '--no-vbox-debugger':
1812 self.fEnableDebugger = False;
1813 else:
1814 # Relevant for selecting VMs to test?
1815 if self.oTestVmSet is not None:
1816 iRc = self.oTestVmSet.parseOption(asArgs, iArg);
1817 if iRc != iArg:
1818 return iRc;
1819
1820 # Hand it to the base class.
1821 return base.TestDriver.parseOption(self, asArgs, iArg);
1822 return iArg + 1;
1823
1824 def completeOptions(self):
1825 return base.TestDriver.completeOptions(self);
1826
1827 def getResourceSet(self):
1828 asRsrcs = [];
1829 if self.oTestVmSet is not None:
1830 asRsrcs.extend(self.oTestVmSet.getResourceSet());
1831 asRsrcs.extend(base.TestDriver.getResourceSet(self));
1832 return asRsrcs;
1833
1834 def actionExtract(self):
1835 return base.TestDriver.actionExtract(self);
1836
1837 def actionVerify(self):
1838 return base.TestDriver.actionVerify(self);
1839
1840 def actionConfig(self):
1841 return base.TestDriver.actionConfig(self);
1842
1843 def actionExecute(self):
1844 return base.TestDriver.actionExecute(self);
1845
1846 def actionCleanupBefore(self):
1847 """
1848 Kill any VBoxSVC left behind by a previous test run.
1849 """
1850 self._killVBoxSVCByPidFile('%s/VBoxSVC.pid' % (self.sScratchPath,));
1851 return base.TestDriver.actionCleanupBefore(self);
1852
1853 def actionCleanupAfter(self):
1854 """
1855 Clean up the VBox bits and then call the base driver.
1856
1857 If your test driver overrides this, it should normally call us at the
1858 end of the job.
1859 """
1860
1861 # Kill any left over VM processes.
1862 self._powerOffAllVms();
1863
1864 # Drop all VBox object references and shutdown xpcom then
1865 # terminating VBoxSVC, with extreme prejudice if need be.
1866 self._teardownVBoxApi();
1867 self._stopVBoxSVC();
1868
1869 # Add the VBoxSVC and testdriver debug+release log files.
1870 if self.fAlwaysUploadLogs or reporter.getErrorCount() > 0:
1871 if self.sVBoxSvcLogFile is not None and os.path.isfile(self.sVBoxSvcLogFile):
1872 reporter.addLogFile(self.sVBoxSvcLogFile, 'log/debug/svc', 'Debug log file for VBoxSVC');
1873 self.sVBoxSvcLogFile = None;
1874
1875 if self.sSelfLogFile is not None and os.path.isfile(self.sSelfLogFile):
1876 reporter.addLogFile(self.sSelfLogFile, 'log/debug/client', 'Debug log file for the test driver');
1877 self.sSelfLogFile = None;
1878
1879 sVBoxSvcRelLog = os.path.join(self.sScratchPath, 'VBoxUserHome', 'VBoxSVC.log');
1880 if os.path.isfile(sVBoxSvcRelLog):
1881 reporter.addLogFile(sVBoxSvcRelLog, 'log/release/svc', 'Release log file for VBoxSVC');
1882 for sSuff in [ '.1', '.2', '.3', '.4', '.5', '.6', '.7', '.8' ]:
1883 if os.path.isfile(sVBoxSvcRelLog + sSuff):
1884 reporter.addLogFile(sVBoxSvcRelLog + sSuff, 'log/release/svc', 'Release log file for VBoxSVC');
1885 # Testbox debugging - START - TEMPORARY, REMOVE ASAP.
1886 if self.sHost in ('darwin', 'freebsd', 'linux', 'solaris', ):
1887 try:
1888 reporter.log('> ls -R -la %s' % (self.sScratchPath,));
1889 utils.processCall(['ls', '-R', '-la', self.sScratchPath]);
1890 except: pass;
1891 # Testbox debugging - END - TEMPORARY, REMOVE ASAP.
1892
1893 # Finally, call the base driver to wipe the scratch space.
1894 return base.TestDriver.actionCleanupAfter(self);
1895
1896 def actionAbort(self):
1897 """
1898 Terminate VBoxSVC if we've got a pid file.
1899 """
1900 #
1901 # Take default action first, then kill VBoxSVC. The other way around
1902 # is problematic since the testscript would continue running and possibly
1903 # trigger a new VBoxSVC to start.
1904 #
1905 fRc1 = base.TestDriver.actionAbort(self);
1906 fRc2 = self._killVBoxSVCByPidFile('%s/VBoxSVC.pid' % (self.sScratchPath,));
1907 return fRc1 is True and fRc2 is True;
1908
1909 def onExit(self, iRc):
1910 """
1911 Stop VBoxSVC if we've started it.
1912 """
1913 if self.oVBoxSvcProcess is not None:
1914 reporter.log('*** Shutting down the VBox API... (iRc=%s)' % (iRc,));
1915 self._powerOffAllVms();
1916 self._teardownVBoxApi();
1917 self._stopVBoxSVC();
1918 reporter.log('*** VBox API shutdown done.');
1919 return base.TestDriver.onExit(self, iRc);
1920
1921
1922 #
1923 # Task wait method override.
1924 #
1925
1926 def notifyAboutReadyTask(self, oTask):
1927 """
1928 Overriding base.TestDriver.notifyAboutReadyTask.
1929 """
1930 try:
1931 self.oVBoxMgr.interruptWaitEvents();
1932 reporter.log2('vbox.notifyAboutReadyTask: called interruptWaitEvents');
1933 except:
1934 reporter.logXcpt('vbox.notifyAboutReadyTask');
1935 return base.TestDriver.notifyAboutReadyTask(self, oTask);
1936
1937 def waitForTasksSleepWorker(self, cMsTimeout):
1938 """
1939 Overriding base.TestDriver.waitForTasksSleepWorker.
1940 """
1941 try:
1942 rc = self.oVBoxMgr.waitForEvents(int(cMsTimeout));
1943 _ = rc; #reporter.log2('vbox.waitForTasksSleepWorker(%u): true (waitForEvents -> %s)' % (cMsTimeout, rc));
1944 reporter.doPollWork('vbox.TestDriver.waitForTasksSleepWorker');
1945 return True;
1946 except KeyboardInterrupt:
1947 raise;
1948 except:
1949 reporter.logXcpt('vbox.waitForTasksSleepWorker');
1950 return False;
1951
1952 #
1953 # Utility methods.
1954 #
1955
1956 def processEvents(self, cMsTimeout = 0):
1957 """
1958 Processes events, returning after the first batch has been processed
1959 or the time limit has been reached.
1960
1961 Only Ctrl-C exception, no return.
1962 """
1963 try:
1964 self.oVBoxMgr.waitForEvents(cMsTimeout);
1965 except KeyboardInterrupt:
1966 raise;
1967 except:
1968 pass;
1969 return None;
1970
1971 def processPendingEvents(self):
1972 """ processEvents(0) - no waiting. """
1973 return self.processEvents(0);
1974
1975 def sleep(self, cSecs):
1976 """
1977 Sleep for a specified amount of time, processing XPCOM events all the while.
1978 """
1979 cMsTimeout = long(cSecs * 1000);
1980 msStart = base.timestampMilli();
1981 self.processEvents(0);
1982 while True:
1983 cMsElapsed = base.timestampMilli() - msStart;
1984 if cMsElapsed > cMsTimeout:
1985 break;
1986 #reporter.log2('cMsTimeout=%s - cMsElapsed=%d => %s' % (cMsTimeout, cMsElapsed, cMsTimeout - cMsElapsed));
1987 self.processEvents(cMsTimeout - cMsElapsed);
1988 return None;
1989
1990 def _logVmInfoUnsafe(self, oVM): # pylint: disable=too-many-statements,too-many-branches
1991 """
1992 Internal worker for logVmInfo that is wrapped in try/except.
1993
1994 This is copy, paste, search, replace and edit of infoCmd from vboxshell.py.
1995 """
1996 oOsType = self.oVBox.getGuestOSType(oVM.OSTypeId)
1997 reporter.log(" Name: %s" % (oVM.name));
1998 reporter.log(" ID: %s" % (oVM.id));
1999 reporter.log(" OS Type: %s - %s" % (oVM.OSTypeId, oOsType.description));
2000 reporter.log(" Machine state: %s" % (oVM.state));
2001 reporter.log(" Session state: %s" % (oVM.sessionState));
2002 if self.fpApiVer >= 4.2:
2003 reporter.log(" Session PID: %u (%#x)" % (oVM.sessionPID, oVM.sessionPID));
2004 else:
2005 reporter.log(" Session PID: %u (%#x)" % (oVM.sessionPid, oVM.sessionPid));
2006 if self.fpApiVer >= 5.0:
2007 reporter.log(" Session Name: %s" % (oVM.sessionName));
2008 else:
2009 reporter.log(" Session Name: %s" % (oVM.sessionType));
2010 reporter.log(" CPUs: %s" % (oVM.CPUCount));
2011 reporter.log(" RAM: %sMB" % (oVM.memorySize));
2012 reporter.log(" VRAM: %sMB" % (oVM.VRAMSize));
2013 reporter.log(" Monitors: %s" % (oVM.monitorCount));
2014 if oVM.chipsetType == vboxcon.ChipsetType_PIIX3: sType = "PIIX3";
2015 elif oVM.chipsetType == vboxcon.ChipsetType_ICH9: sType = "ICH9";
2016 else: sType = "unknown %s" % (oVM.chipsetType);
2017 reporter.log(" Chipset: %s" % (sType));
2018 if oVM.firmwareType == vboxcon.FirmwareType_BIOS: sType = "BIOS";
2019 elif oVM.firmwareType == vboxcon.FirmwareType_EFI: sType = "EFI";
2020 elif oVM.firmwareType == vboxcon.FirmwareType_EFI32: sType = "EFI32";
2021 elif oVM.firmwareType == vboxcon.FirmwareType_EFI64: sType = "EFI64";
2022 elif oVM.firmwareType == vboxcon.FirmwareType_EFIDUAL: sType = "EFIDUAL";
2023 else: sType = "unknown %s" % (oVM.firmwareType);
2024 reporter.log(" Firmware: %s" % (sType));
2025 reporter.log(" HwVirtEx: %s" % (oVM.getHWVirtExProperty(vboxcon.HWVirtExPropertyType_Enabled)));
2026 reporter.log(" VPID support: %s" % (oVM.getHWVirtExProperty(vboxcon.HWVirtExPropertyType_VPID)));
2027 reporter.log(" Nested paging: %s" % (oVM.getHWVirtExProperty(vboxcon.HWVirtExPropertyType_NestedPaging)));
2028 if self.fpApiVer >= 4.2 and hasattr(vboxcon, 'CPUPropertyType_LongMode'):
2029 reporter.log(" Long-mode: %s" % (oVM.getCPUProperty(vboxcon.CPUPropertyType_LongMode)));
2030 if self.fpApiVer >= 3.2:
2031 reporter.log(" PAE: %s" % (oVM.getCPUProperty(vboxcon.CPUPropertyType_PAE)));
2032 if self.fpApiVer < 5.0:
2033 reporter.log(" Synthetic CPU: %s" % (oVM.getCPUProperty(vboxcon.CPUPropertyType_Synthetic)));
2034 else:
2035 reporter.log(" PAE: %s" % (oVM.getCpuProperty(vboxcon.CpuPropertyType_PAE)));
2036 reporter.log(" Synthetic CPU: %s" % (oVM.getCpuProperty(vboxcon.CpuPropertyType_Synthetic)));
2037 if self.fpApiVer >= 5.3 and hasattr(vboxcon, 'CPUPropertyType_HWVirt'):
2038 reporter.log(" Nested VT-x/AMD-V: %s" % (oVM.getCPUProperty(vboxcon.CPUPropertyType_HWVirt)));
2039 reporter.log(" ACPI: %s" % (oVM.BIOSSettings.ACPIEnabled));
2040 reporter.log(" IO-APIC: %s" % (oVM.BIOSSettings.IOAPICEnabled));
2041 if self.fpApiVer >= 3.2:
2042 if self.fpApiVer >= 4.2:
2043 reporter.log(" HPET: %s" % (oVM.HPETEnabled));
2044 else:
2045 reporter.log(" HPET: %s" % (oVM.hpetEnabled));
2046 reporter.log(" 3D acceleration: %s" % (oVM.accelerate3DEnabled));
2047 reporter.log(" 2D acceleration: %s" % (oVM.accelerate2DVideoEnabled));
2048 reporter.log(" TeleporterEnabled: %s" % (oVM.teleporterEnabled));
2049 reporter.log(" TeleporterPort: %s" % (oVM.teleporterPort));
2050 reporter.log(" TeleporterAddress: %s" % (oVM.teleporterAddress));
2051 reporter.log(" TeleporterPassword: %s" % (oVM.teleporterPassword));
2052 reporter.log(" Clipboard mode: %s" % (oVM.clipboardMode));
2053 if self.fpApiVer >= 5.0:
2054 reporter.log(" Drag and drop mode: %s" % (oVM.dnDMode));
2055 elif self.fpApiVer >= 4.3:
2056 reporter.log(" Drag and drop mode: %s" % (oVM.dragAndDropMode));
2057 if self.fpApiVer >= 4.0:
2058 reporter.log(" VRDP server: %s" % (oVM.VRDEServer.enabled));
2059 try: sPorts = oVM.VRDEServer.getVRDEProperty("TCP/Ports");
2060 except: sPorts = "";
2061 reporter.log(" VRDP server ports: %s" % (sPorts));
2062 reporter.log(" VRDP auth: %s (%s)" % (oVM.VRDEServer.authType, oVM.VRDEServer.authLibrary));
2063 else:
2064 reporter.log(" VRDP server: %s" % (oVM.VRDPServer.enabled));
2065 reporter.log(" VRDP server ports: %s" % (oVM.VRDPServer.ports));
2066 reporter.log(" Last changed: %s" % (oVM.lastStateChange));
2067
2068 aoControllers = self.oVBoxMgr.getArray(oVM, 'storageControllers')
2069 if aoControllers:
2070 reporter.log(" Controllers:");
2071 for oCtrl in aoControllers:
2072 reporter.log(" %s %s bus: %s type: %s" % (oCtrl.name, oCtrl.controllerType, oCtrl.bus, oCtrl.controllerType));
2073 oAudioAdapter = oVM.audioAdapter;
2074 if oAudioAdapter.audioController == vboxcon.AudioControllerType_AC97: sType = "AC97";
2075 elif oAudioAdapter.audioController == vboxcon.AudioControllerType_SB16: sType = "SB16";
2076 elif oAudioAdapter.audioController == vboxcon.AudioControllerType_HDA: sType = "HDA";
2077 else: sType = "unknown %s" % (oAudioAdapter.audioController);
2078 reporter.log(" AudioController: %s" % (sType));
2079 reporter.log(" AudioEnabled: %s" % (oAudioAdapter.enabled));
2080 if oAudioAdapter.audioDriver == vboxcon.AudioDriverType_CoreAudio: sType = "CoreAudio";
2081 elif oAudioAdapter.audioDriver == vboxcon.AudioDriverType_DirectSound: sType = "DirectSound";
2082 elif oAudioAdapter.audioDriver == vboxcon.AudioDriverType_Pulse: sType = "PulseAudio";
2083 elif oAudioAdapter.audioDriver == vboxcon.AudioDriverType_OSS: sType = "OSS";
2084 elif oAudioAdapter.audioDriver == vboxcon.AudioDriverType_Null: sType = "NULL";
2085 else: sType = "unknown %s" % (oAudioAdapter.audioDriver);
2086 reporter.log(" Host AudioDriver: %s" % (sType));
2087
2088 self.processPendingEvents();
2089 aoAttachments = self.oVBoxMgr.getArray(oVM, 'mediumAttachments')
2090 if aoAttachments:
2091 reporter.log(" Attachments:");
2092 for oAtt in aoAttachments:
2093 sCtrl = "Controller: %s port: %s device: %s type: %s" % (oAtt.controller, oAtt.port, oAtt.device, oAtt.type);
2094 oMedium = oAtt.medium
2095 if oAtt.type == vboxcon.DeviceType_HardDisk:
2096 reporter.log(" %s: HDD" % sCtrl);
2097 reporter.log(" Id: %s" % (oMedium.id));
2098 reporter.log(" Name: %s" % (oMedium.name));
2099 reporter.log(" Format: %s" % (oMedium.format));
2100 reporter.log(" Location: %s" % (oMedium.location));
2101
2102 if oAtt.type == vboxcon.DeviceType_DVD:
2103 reporter.log(" %s: DVD" % sCtrl);
2104 if oMedium:
2105 reporter.log(" Id: %s" % (oMedium.id));
2106 reporter.log(" Name: %s" % (oMedium.name));
2107 if oMedium.hostDrive:
2108 reporter.log(" Host DVD %s" % (oMedium.location));
2109 if oAtt.passthrough:
2110 reporter.log(" [passthrough mode]");
2111 else:
2112 reporter.log(" Virtual image: %s" % (oMedium.location));
2113 reporter.log(" Size: %s" % (oMedium.size));
2114 else:
2115 reporter.log(" empty");
2116
2117 if oAtt.type == vboxcon.DeviceType_Floppy:
2118 reporter.log(" %s: Floppy" % sCtrl);
2119 if oMedium:
2120 reporter.log(" Id: %s" % (oMedium.id));
2121 reporter.log(" Name: %s" % (oMedium.name));
2122 if oMedium.hostDrive:
2123 reporter.log(" Host floppy: %s" % (oMedium.location));
2124 else:
2125 reporter.log(" Virtual image: %s" % (oMedium.location));
2126 reporter.log(" Size: %s" % (oMedium.size));
2127 else:
2128 reporter.log(" empty");
2129 self.processPendingEvents();
2130
2131 reporter.log(" Network Adapter:");
2132 for iSlot in range(0, 32):
2133 try: oNic = oVM.getNetworkAdapter(iSlot)
2134 except: break;
2135 if not oNic.enabled:
2136 reporter.log2(" slot #%d found but not enabled, skipping" % (iSlot,));
2137 continue;
2138 if oNic.adapterType == vboxcon.NetworkAdapterType_Am79C973: sType = "PCNet";
2139 elif oNic.adapterType == vboxcon.NetworkAdapterType_Am79C970A: sType = "PCNetOld";
2140 elif oNic.adapterType == vboxcon.NetworkAdapterType_I82545EM: sType = "E1000";
2141 elif oNic.adapterType == vboxcon.NetworkAdapterType_I82540EM: sType = "E1000Desk";
2142 elif oNic.adapterType == vboxcon.NetworkAdapterType_I82543GC: sType = "E1000Srv2";
2143 elif oNic.adapterType == vboxcon.NetworkAdapterType_Virtio: sType = "Virtio";
2144 else: sType = "unknown %s" % (oNic.adapterType);
2145 reporter.log(" slot #%d: type: %s (%s) MAC Address: %s lineSpeed: %s" % \
2146 (iSlot, sType, oNic.adapterType, oNic.MACAddress, oNic.lineSpeed) );
2147
2148 if oNic.attachmentType == vboxcon.NetworkAttachmentType_NAT:
2149 reporter.log(" attachmentType: NAT (%s)" % (oNic.attachmentType));
2150 if self.fpApiVer >= 4.1:
2151 reporter.log(" nat-network: %s" % (oNic.NATNetwork,));
2152 elif oNic.attachmentType == vboxcon.NetworkAttachmentType_Bridged:
2153 reporter.log(" attachmentType: Bridged (%s)" % (oNic.attachmentType));
2154 if self.fpApiVer >= 4.1:
2155 reporter.log(" hostInterface: %s" % (oNic.bridgedInterface));
2156 else:
2157 reporter.log(" hostInterface: %s" % (oNic.hostInterface));
2158 elif oNic.attachmentType == vboxcon.NetworkAttachmentType_Internal:
2159 reporter.log(" attachmentType: Internal (%s)" % (oNic.attachmentType));
2160 reporter.log(" intnet-name: %s" % (oNic.internalNetwork,));
2161 elif oNic.attachmentType == vboxcon.NetworkAttachmentType_HostOnly:
2162 reporter.log(" attachmentType: HostOnly (%s)" % (oNic.attachmentType));
2163 if self.fpApiVer >= 4.1:
2164 reporter.log(" hostInterface: %s" % (oNic.hostOnlyInterface));
2165 else:
2166 reporter.log(" hostInterface: %s" % (oNic.hostInterface));
2167 else:
2168 if self.fpApiVer >= 4.1:
2169 if oNic.attachmentType == vboxcon.NetworkAttachmentType_Generic:
2170 reporter.log(" attachmentType: Generic (%s)" % (oNic.attachmentType));
2171 reporter.log(" generic-driver: %s" % (oNic.GenericDriver));
2172 else:
2173 reporter.log(" attachmentType: unknown-%s" % (oNic.attachmentType));
2174 else:
2175 reporter.log(" attachmentType: unknown-%s" % (oNic.attachmentType));
2176 if oNic.traceEnabled:
2177 reporter.log(" traceFile: %s" % (oNic.traceFile));
2178 self.processPendingEvents();
2179
2180 reporter.log(" Serial ports:");
2181 for iSlot in range(0, 8):
2182 try: oPort = oVM.getSerialPort(iSlot)
2183 except: break;
2184 if oPort is not None and oPort.enabled:
2185 enmHostMode = oPort.hostMode;
2186 if enmHostMode == vboxcon.PortMode_Disconnected: sType = "Disconnected";
2187 elif enmHostMode == vboxcon.PortMode_HostPipe: sType = "HostPipe";
2188 elif enmHostMode == vboxcon.PortMode_HostDevice: sType = "HostDevice";
2189 elif enmHostMode == vboxcon.PortMode_RawFile: sType = "RawFile";
2190 elif enmHostMode == vboxcon.PortMode_TCP: sType = "TCP";
2191 else: sType = "unknown %s" % (enmHostMode);
2192 reporter.log(" slot #%d: hostMode: %s (%s) I/O port: %s IRQ: %s server: %s path: %s" %
2193 (iSlot, sType, enmHostMode, oPort.IOBase, oPort.IRQ, oPort.server, oPort.path,) );
2194 self.processPendingEvents();
2195
2196 return True;
2197
2198 def logVmInfo(self, oVM): # pylint: disable=too-many-statements,too-many-branches
2199 """
2200 Logs VM configuration details.
2201
2202 This is copy, past, search, replace and edit of infoCmd from vboxshell.py.
2203 """
2204 try:
2205 fRc = self._logVmInfoUnsafe(oVM);
2206 except:
2207 reporter.logXcpt();
2208 fRc = False;
2209 return fRc;
2210
2211 def logVmInfoByName(self, sName):
2212 """
2213 logVmInfo + getVmByName.
2214 """
2215 return self.logVmInfo(self.getVmByName(sName));
2216
2217 def tryFindGuestOsId(self, sIdOrDesc):
2218 """
2219 Takes a guest OS ID or Description and returns the ID.
2220 If nothing matching it is found, the input is returned unmodified.
2221 """
2222
2223 if self.fpApiVer >= 4.0:
2224 if sIdOrDesc == 'Solaris (64 bit)':
2225 sIdOrDesc = 'Oracle Solaris 10 5/09 and earlier (64 bit)';
2226
2227 try:
2228 aoGuestTypes = self.oVBoxMgr.getArray(self.oVBox, 'GuestOSTypes');
2229 except:
2230 reporter.logXcpt();
2231 else:
2232 for oGuestOS in aoGuestTypes:
2233 try:
2234 sId = oGuestOS.id;
2235 sDesc = oGuestOS.description;
2236 except:
2237 reporter.logXcpt();
2238 else:
2239 if sIdOrDesc in (sId, sDesc,):
2240 sIdOrDesc = sId;
2241 break;
2242 self.processPendingEvents();
2243 return sIdOrDesc
2244
2245 def resourceFindVmHd(self, sVmName, sFlavor):
2246 """
2247 Search the test resources for the most recent VM HD.
2248
2249 Returns path relative to the test resource root.
2250 """
2251 ## @todo implement a proper search algo here.
2252 return '4.2/' + sFlavor + '/' + sVmName + '/t-' + sVmName + '.vdi';
2253
2254
2255 #
2256 # VM Api wrappers that logs errors, hides exceptions and other details.
2257 #
2258
2259 def createTestVMOnly(self, sName, sKind):
2260 """
2261 Creates and register a test VM without doing any kind of configuration.
2262
2263 Returns VM object (IMachine) on success, None on failure.
2264 """
2265 if not self.importVBoxApi():
2266 return None;
2267
2268 # create + register the VM
2269 try:
2270 if self.fpApiVer >= 4.2: # Introduces grouping (third parameter, empty for now).
2271 oVM = self.oVBox.createMachine("", sName, [], self.tryFindGuestOsId(sKind), "");
2272 elif self.fpApiVer >= 4.0:
2273 oVM = self.oVBox.createMachine("", sName, self.tryFindGuestOsId(sKind), "", False);
2274 elif self.fpApiVer >= 3.2:
2275 oVM = self.oVBox.createMachine(sName, self.tryFindGuestOsId(sKind), "", "", False);
2276 else:
2277 oVM = self.oVBox.createMachine(sName, self.tryFindGuestOsId(sKind), "", "");
2278 try:
2279 oVM.saveSettings();
2280 try:
2281 self.oVBox.registerMachine(oVM);
2282 return oVM;
2283 except:
2284 reporter.logXcpt();
2285 raise;
2286 except:
2287 reporter.logXcpt();
2288 if self.fpApiVer >= 4.0:
2289 try:
2290 if self.fpApiVer >= 4.3:
2291 oProgress = oVM.deleteConfig([]);
2292 else:
2293 oProgress = oVM.delete(None);
2294 self.waitOnProgress(oProgress);
2295 except:
2296 reporter.logXcpt();
2297 else:
2298 try: oVM.deleteSettings();
2299 except: reporter.logXcpt();
2300 raise;
2301 except:
2302 reporter.errorXcpt('failed to create vm "%s"' % (sName));
2303 return None;
2304
2305 # pylint: disable=too-many-arguments,too-many-locals,too-many-statements
2306 def createTestVM(self,
2307 sName,
2308 iGroup,
2309 sHd = None,
2310 cMbRam = None,
2311 cCpus = 1,
2312 fVirtEx = None,
2313 fNestedPaging = None,
2314 sDvdImage = None,
2315 sKind = "Other",
2316 fIoApic = None,
2317 fNstHwVirt = None,
2318 fPae = None,
2319 fFastBootLogo = True,
2320 eNic0Type = None,
2321 eNic0AttachType = None,
2322 sNic0NetName = 'default',
2323 sNic0MacAddr = 'grouped',
2324 sFloppy = None,
2325 fNatForwardingForTxs = None,
2326 sHddControllerType = 'IDE Controller',
2327 fVmmDevTestingPart = None,
2328 fVmmDevTestingMmio = False,
2329 sFirmwareType = 'bios',
2330 sChipsetType = 'piix3',
2331 sDvdControllerType = 'IDE Controller',
2332 sCom1RawFile = None):
2333 """
2334 Creates a test VM with a immutable HD from the test resources.
2335 """
2336 # create + register the VM
2337 oVM = self.createTestVMOnly(sName, sKind);
2338 if not oVM:
2339 return None;
2340
2341 # Configure the VM.
2342 fRc = True;
2343 oSession = self.openSession(oVM);
2344 if oSession is not None:
2345 fRc = oSession.setupPreferredConfig();
2346
2347 if fRc and cMbRam is not None :
2348 fRc = oSession.setRamSize(cMbRam);
2349 if fRc and cCpus is not None:
2350 fRc = oSession.setCpuCount(cCpus);
2351 if fRc and fVirtEx is not None:
2352 fRc = oSession.enableVirtEx(fVirtEx);
2353 if fRc and fNestedPaging is not None:
2354 fRc = oSession.enableNestedPaging(fNestedPaging);
2355 if fRc and fIoApic is not None:
2356 fRc = oSession.enableIoApic(fIoApic);
2357 if fRc and fNstHwVirt is not None:
2358 fRc = oSession.enableNestedHwVirt(fNstHwVirt);
2359 if fRc and fPae is not None:
2360 fRc = oSession.enablePae(fPae);
2361 if fRc and sDvdImage is not None:
2362 fRc = oSession.attachDvd(sDvdImage, sDvdControllerType);
2363 if fRc and sHd is not None:
2364 fRc = oSession.attachHd(sHd, sHddControllerType);
2365 if fRc and sFloppy is not None:
2366 fRc = oSession.attachFloppy(sFloppy);
2367 if fRc and eNic0Type is not None:
2368 fRc = oSession.setNicType(eNic0Type, 0);
2369 if fRc and (eNic0AttachType is not None or (sNic0NetName is not None and sNic0NetName != 'default')):
2370 fRc = oSession.setNicAttachment(eNic0AttachType, sNic0NetName, 0);
2371 if fRc and sNic0MacAddr is not None:
2372 if sNic0MacAddr == 'grouped':
2373 sNic0MacAddr = '%02X' % (iGroup);
2374 fRc = oSession.setNicMacAddress(sNic0MacAddr, 0);
2375 if fRc and fNatForwardingForTxs is True:
2376 fRc = oSession.setupNatForwardingForTxs();
2377 if fRc and fFastBootLogo is not None:
2378 fRc = oSession.setupBootLogo(fFastBootLogo);
2379 if fRc and self.fEnableVrdp:
2380 fRc = oSession.setupVrdp(True, self.uVrdpBasePort + iGroup);
2381 if fRc and fVmmDevTestingPart is not None:
2382 fRc = oSession.enableVmmDevTestingPart(fVmmDevTestingPart, fVmmDevTestingMmio);
2383 if fRc and sFirmwareType == 'bios':
2384 fRc = oSession.setFirmwareType(vboxcon.FirmwareType_BIOS);
2385 elif sFirmwareType == 'efi':
2386 fRc = oSession.setFirmwareType(vboxcon.FirmwareType_EFI);
2387 if fRc and self.fEnableDebugger:
2388 fRc = oSession.setExtraData('VBoxInternal/DBGC/Enabled', '1');
2389 if fRc and sChipsetType == 'piix3':
2390 fRc = oSession.setChipsetType(vboxcon.ChipsetType_PIIX3);
2391 elif sChipsetType == 'ich9':
2392 fRc = oSession.setChipsetType(vboxcon.ChipsetType_ICH9);
2393 if fRc and sCom1RawFile:
2394 fRc = oSession.setupSerialToRawFile(0, sCom1RawFile);
2395
2396 if fRc: fRc = oSession.saveSettings();
2397 if not fRc: oSession.discardSettings(True);
2398 oSession.close();
2399 if not fRc:
2400 try: self.oVBox.unregisterMachine(oVM.id);
2401 except: pass;
2402 if self.fpApiVer >= 4.0:
2403 try:
2404 if self.fpApiVer >= 4.3:
2405 oProgress = oVM.deleteConfig([]);
2406 else:
2407 oProgress = oVM.delete(None);
2408 self.waitOnProgress(oProgress);
2409 except:
2410 reporter.logXcpt();
2411 else:
2412 try: oVM.deleteSettings();
2413 except: reporter.logXcpt();
2414 return None;
2415
2416 # success.
2417 reporter.log('created "%s" with name "%s"' % (oVM.id, sName));
2418 self.aoVMs.append(oVM);
2419 self.logVmInfo(oVM); # testing...
2420 return oVM;
2421 # pylint: enable=too-many-arguments,too-many-locals,too-many-statements
2422
2423 def createTestVmWithDefaults(self, # pylint: disable=too-many-arguments
2424 sName,
2425 iGroup,
2426 sKind,
2427 sDvdImage = None,
2428 fFastBootLogo = True,
2429 eNic0AttachType = None,
2430 sNic0NetName = 'default',
2431 sNic0MacAddr = 'grouped',
2432 fVmmDevTestingPart = None,
2433 fVmmDevTestingMmio = False,
2434 sCom1RawFile = None):
2435 """
2436 Creates a test VM with all defaults and no HDs.
2437 """
2438 # create + register the VM
2439 oVM = self.createTestVMOnly(sName, sKind);
2440 if oVM is not None:
2441 # Configure the VM with defaults according to sKind.
2442 fRc = True;
2443 oSession = self.openSession(oVM);
2444 if oSession is not None:
2445 if self.fpApiVer >= 6.0:
2446 try:
2447 oSession.o.machine.applyDefaults('');
2448 except:
2449 reporter.errorXcpt('failed to apply defaults to vm "%s"' % (sName,));
2450 fRc = False;
2451 else:
2452 reporter.error("Implement applyDefaults for vbox version %s" % (self.fpApiVer,));
2453 #fRc = oSession.setupPreferredConfig();
2454 fRc = False;
2455
2456 # Apply the specified configuration:
2457 if fRc and sDvdImage is not None:
2458 #fRc = oSession.insertDvd(sDvdImage); # attachDvd
2459 reporter.error('Implement: oSession.insertDvd(%s)' % (sDvdImage,));
2460 fRc = False;
2461
2462 if fRc and fFastBootLogo is not None:
2463 fRc = oSession.setupBootLogo(fFastBootLogo);
2464
2465 if fRc and (eNic0AttachType is not None or (sNic0NetName is not None and sNic0NetName != 'default')):
2466 fRc = oSession.setNicAttachment(eNic0AttachType, sNic0NetName, 0);
2467 if fRc and sNic0MacAddr is not None:
2468 if sNic0MacAddr == 'grouped':
2469 sNic0MacAddr = '%02X' % (iGroup,);
2470 fRc = oSession.setNicMacAddress(sNic0MacAddr, 0);
2471
2472 if fRc and fVmmDevTestingPart is not None:
2473 fRc = oSession.enableVmmDevTestingPart(fVmmDevTestingPart, fVmmDevTestingMmio);
2474
2475 if fRc and sCom1RawFile:
2476 fRc = oSession.setupSerialToRawFile(0, sCom1RawFile);
2477
2478 # Save the settings if we were successfull, otherwise discard them.
2479 if fRc:
2480 fRc = oSession.saveSettings();
2481 if not fRc:
2482 oSession.discardSettings(True);
2483 oSession.close();
2484
2485 if fRc is True:
2486 # If we've been successful, add the VM to the list and return it.
2487 # success.
2488 reporter.log('created "%s" with name "%s"' % (oVM.id, sName, ));
2489 self.aoVMs.append(oVM);
2490 self.logVmInfo(oVM); # testing...
2491 return oVM;
2492
2493 # Failed. Unregister the machine and delete it.
2494 try: self.oVBox.unregisterMachine(oVM.id);
2495 except: pass;
2496
2497 if self.fpApiVer >= 4.0:
2498 try:
2499 if self.fpApiVer >= 4.3:
2500 oProgress = oVM.deleteConfig([]);
2501 else:
2502 oProgress = oVM.delete(None);
2503 self.waitOnProgress(oProgress);
2504 except:
2505 reporter.logXcpt();
2506 else:
2507 try: oVM.deleteSettings();
2508 except: reporter.logXcpt();
2509 return None;
2510
2511 def addTestMachine(self, sNameOrId, fQuiet = False):
2512 """
2513 Adds an already existing (that is, configured) test VM to the
2514 test VM list.
2515 """
2516 # find + add the VM to the list.
2517 try:
2518 if self.fpApiVer >= 4.0:
2519 oVM = self.oVBox.findMachine(sNameOrId);
2520 else:
2521 reporter.error('fpApiVer=%s - did you remember to initialize the API' % (self.fpApiVer,));
2522 except:
2523 reporter.errorXcpt('could not find vm "%s"' % (sNameOrId,));
2524 return None;
2525
2526 self.aoVMs.append(oVM);
2527 if not fQuiet:
2528 reporter.log('Added "%s" with name "%s"' % (oVM.id, sNameOrId));
2529 self.logVmInfo(oVM);
2530 return oVM;
2531
2532 def openSession(self, oVM):
2533 """
2534 Opens a session for the VM. Returns the a Session wrapper object that
2535 will automatically close the session when the wrapper goes out of scope.
2536
2537 On failure None is returned and an error is logged.
2538 """
2539 try:
2540 sUuid = oVM.id;
2541 except:
2542 reporter.errorXcpt('failed to get the UUID for VM "%s"' % (oVM,));
2543 return None;
2544
2545 # This loop is a kludge to deal with us racing the closing of the
2546 # direct session of a previous VM run. See waitOnDirectSessionClose.
2547 for i in range(10):
2548 try:
2549 if self.fpApiVer <= 3.2:
2550 oSession = self.oVBoxMgr.openMachineSession(sUuid);
2551 else:
2552 oSession = self.oVBoxMgr.openMachineSession(oVM);
2553 break;
2554 except:
2555 if i == 9:
2556 reporter.errorXcpt('failed to open session for "%s" ("%s")' % (sUuid, oVM));
2557 return None;
2558 if i > 0:
2559 reporter.logXcpt('warning: failed to open session for "%s" ("%s") - retrying in %u secs' % (sUuid, oVM, i));
2560 self.waitOnDirectSessionClose(oVM, 5000 + i * 1000);
2561 from testdriver.vboxwrappers import SessionWrapper;
2562 return SessionWrapper(oSession, oVM, self.oVBox, self.oVBoxMgr, self, False);
2563
2564 def getVmByName(self, sName):
2565 """
2566 Get a test VM by name. Returns None if not found, logged.
2567 """
2568 # Look it up in our 'cache'.
2569 for oVM in self.aoVMs:
2570 try:
2571 #reporter.log2('cur: %s / %s (oVM=%s)' % (oVM.name, oVM.id, oVM));
2572 if oVM.name == sName:
2573 return oVM;
2574 except:
2575 reporter.errorXcpt('failed to get the name from the VM "%s"' % (oVM));
2576
2577 # Look it up the standard way.
2578 return self.addTestMachine(sName, fQuiet = True);
2579
2580 def getVmByUuid(self, sUuid):
2581 """
2582 Get a test VM by uuid. Returns None if not found, logged.
2583 """
2584 # Look it up in our 'cache'.
2585 for oVM in self.aoVMs:
2586 try:
2587 if oVM.id == sUuid:
2588 return oVM;
2589 except:
2590 reporter.errorXcpt('failed to get the UUID from the VM "%s"' % (oVM));
2591
2592 # Look it up the standard way.
2593 return self.addTestMachine(sUuid, fQuiet = True);
2594
2595 def waitOnProgress(self, oProgress, cMsTimeout = 1000000, fErrorOnTimeout = True, cMsInterval = 1000):
2596 """
2597 Waits for a progress object to complete. Returns the status code.
2598 """
2599 # Wait for progress no longer than cMsTimeout time period.
2600 tsStart = datetime.datetime.now()
2601 while True:
2602 self.processPendingEvents();
2603 try:
2604 if oProgress.completed:
2605 break;
2606 except:
2607 return -1;
2608 self.processPendingEvents();
2609
2610 tsNow = datetime.datetime.now()
2611 tsDelta = tsNow - tsStart
2612 if ((tsDelta.microseconds + tsDelta.seconds * 1000000) // 1000) > cMsTimeout:
2613 if fErrorOnTimeout:
2614 reporter.errorTimeout('Timeout while waiting for progress.')
2615 return -1
2616
2617 reporter.doPollWork('vbox.TestDriver.waitOnProgress');
2618 try: oProgress.waitForCompletion(cMsInterval);
2619 except: return -2;
2620
2621 try: rc = oProgress.resultCode;
2622 except: rc = -2;
2623 self.processPendingEvents();
2624 return rc;
2625
2626 def waitOnDirectSessionClose(self, oVM, cMsTimeout):
2627 """
2628 Waits for the VM process to close it's current direct session.
2629
2630 Returns None.
2631 """
2632 # Get the original values so we're not subject to
2633 try:
2634 eCurState = oVM.sessionState;
2635 if self.fpApiVer >= 5.0:
2636 sCurName = sOrgName = oVM.sessionName;
2637 else:
2638 sCurName = sOrgName = oVM.sessionType;
2639 if self.fpApiVer >= 4.2:
2640 iCurPid = iOrgPid = oVM.sessionPID;
2641 else:
2642 iCurPid = iOrgPid = oVM.sessionPid;
2643 except Exception as oXcpt:
2644 if ComError.notEqual(oXcpt, ComError.E_ACCESSDENIED):
2645 reporter.logXcpt();
2646 self.processPendingEvents();
2647 return None;
2648 self.processPendingEvents();
2649
2650 msStart = base.timestampMilli();
2651 while iCurPid == iOrgPid \
2652 and sCurName == sOrgName \
2653 and sCurName != '' \
2654 and base.timestampMilli() - msStart < cMsTimeout \
2655 and eCurState in (vboxcon.SessionState_Unlocking, vboxcon.SessionState_Spawning, vboxcon.SessionState_Locked,):
2656 self.processEvents(1000);
2657 try:
2658 eCurState = oVM.sessionState;
2659 sCurName = oVM.sessionName if self.fpApiVer >= 5.0 else oVM.sessionType;
2660 iCurPid = oVM.sessionPID if self.fpApiVer >= 4.2 else oVM.sessionPid;
2661 except Exception as oXcpt:
2662 if ComError.notEqual(oXcpt, ComError.E_ACCESSDENIED):
2663 reporter.logXcpt();
2664 break;
2665 self.processPendingEvents();
2666 self.processPendingEvents();
2667 return None;
2668
2669 def uploadStartupLogFile(self, oVM, sVmName):
2670 """
2671 Uploads the VBoxStartup.log when present.
2672 """
2673 fRc = True;
2674 try:
2675 sLogFile = os.path.join(oVM.logFolder, 'VBoxHardening.log');
2676 except:
2677 reporter.logXcpt();
2678 fRc = False;
2679 else:
2680 if os.path.isfile(sLogFile):
2681 reporter.addLogFile(sLogFile, 'log/release/vm', '%s hardening log' % (sVmName, ),
2682 sAltName = '%s-%s' % (sVmName, os.path.basename(sLogFile),));
2683 return fRc;
2684
2685 def annotateAndUploadProcessReport(self, sProcessReport, sFilename, sKind, sDesc):
2686 """
2687 Annotates the given VM process report and uploads it if successfull.
2688 """
2689 fRc = False;
2690 if self.oBuild is not None and self.oBuild.sInstallPath is not None:
2691 oResolver = btresolver.BacktraceResolver(self.sScratchPath, self.oBuild.sInstallPath,
2692 self.getBuildOs(), self.getBuildArch(),
2693 fnLog = reporter.log);
2694 fRcTmp = oResolver.prepareEnv();
2695 if fRcTmp:
2696 reporter.log('Successfully prepared environment');
2697 sReportDbgSym = oResolver.annotateReport(sProcessReport);
2698 if sReportDbgSym is not None:
2699 reporter.addLogString(sReportDbgSym, sFilename, sKind, sDesc);
2700 fRc = True;
2701 else:
2702 reporter.log('Annotating report failed');
2703 oResolver.cleanupEnv();
2704 return fRc;
2705
2706 def startVmEx(self, oVM, fWait = True, sType = None, sName = None, asEnv = None): # pylint: disable=too-many-locals,too-many-statements
2707 """
2708 Start the VM, returning the VM session and progress object on success.
2709 The session is also added to the task list and to the aoRemoteSessions set.
2710
2711 asEnv is a list of string on the putenv() form.
2712
2713 On failure (None, None) is returned and an error is logged.
2714 """
2715 # Massage and check the input.
2716 if sType is None:
2717 sType = self.sSessionType;
2718 if sName is None:
2719 try: sName = oVM.name;
2720 except: sName = 'bad-vm-handle';
2721 reporter.log('startVmEx: sName=%s fWait=%s sType=%s' % (sName, fWait, sType));
2722 if oVM is None:
2723 return (None, None);
2724
2725 ## @todo Do this elsewhere.
2726 # Hack alert. Disables all annoying GUI popups.
2727 if sType == 'gui' and not self.aoRemoteSessions:
2728 try:
2729 self.oVBox.setExtraData('GUI/Input/AutoCapture', 'false');
2730 if self.fpApiVer >= 3.2:
2731 self.oVBox.setExtraData('GUI/LicenseAgreed', '8');
2732 else:
2733 self.oVBox.setExtraData('GUI/LicenseAgreed', '7');
2734 self.oVBox.setExtraData('GUI/RegistrationData', 'triesLeft=0');
2735 self.oVBox.setExtraData('GUI/SUNOnlineData', 'triesLeft=0');
2736 self.oVBox.setExtraData('GUI/SuppressMessages', 'confirmVMReset,remindAboutMouseIntegrationOn,'
2737 'remindAboutMouseIntegrationOff,remindAboutPausedVMInput,confirmInputCapture,'
2738 'confirmGoingFullscreen,remindAboutInaccessibleMedia,remindAboutWrongColorDepth,'
2739 'confirmRemoveMedium,allPopupPanes,allMessageBoxes,all');
2740 self.oVBox.setExtraData('GUI/UpdateDate', 'never');
2741 self.oVBox.setExtraData('GUI/PreventBetaWarning', self.oVBox.version);
2742 except:
2743 reporter.logXcpt();
2744
2745 # The UUID for the name.
2746 try:
2747 sUuid = oVM.id;
2748 except:
2749 reporter.errorXcpt('failed to get the UUID for VM "%s"' % (oVM));
2750 return (None, None);
2751 self.processPendingEvents();
2752
2753 # Construct the environment.
2754 sLogFile = '%s/VM-%s.log' % (self.sScratchPath, sUuid);
2755 try: os.remove(sLogFile);
2756 except: pass;
2757 if self.sLogSessionDest:
2758 sLogDest = self.sLogSessionDest;
2759 else:
2760 sLogDest = 'file=%s' % sLogFile;
2761 sEnv = 'VBOX_LOG=%s\nVBOX_LOG_FLAGS=%s\nVBOX_LOG_DEST=nodeny %s\nVBOX_RELEASE_LOG_FLAGS=append time' \
2762 % (self.sLogSessionGroups, self.sLogSessionFlags, sLogDest,);
2763 if sType == 'gui':
2764 sEnv += '\nVBOX_GUI_DBG_ENABLED=1'
2765 if asEnv is not None and asEnv:
2766 sEnv += '\n' + ('\n'.join(asEnv));
2767
2768 # Shortcuts for local testing.
2769 oProgress = oWrapped = None;
2770 oTestVM = self.oTestVmSet.findTestVmByName(sName) if self.oTestVmSet is not None else None;
2771 try:
2772 if oTestVM is not None \
2773 and oTestVM.fSnapshotRestoreCurrent is True:
2774 if oVM.state is vboxcon.MachineState_Running:
2775 reporter.log2('Machine "%s" already running.' % (sName,));
2776 oProgress = None;
2777 oWrapped = self.openSession(oVM);
2778 else:
2779 reporter.log2('Checking if snapshot for machine "%s" exists.' % (sName,));
2780 oSessionWrapperRestore = self.openSession(oVM);
2781 if oSessionWrapperRestore is not None:
2782 oSnapshotCur = oVM.currentSnapshot;
2783 if oSnapshotCur is not None:
2784 reporter.log2('Restoring snapshot for machine "%s".' % (sName,));
2785 oSessionWrapperRestore.restoreSnapshot(oSnapshotCur);
2786 reporter.log2('Current snapshot for machine "%s" restored.' % (sName,));
2787 else:
2788 reporter.log('warning: no current snapshot for machine "%s" found.' % (sName,));
2789 oSessionWrapperRestore.close();
2790 except:
2791 reporter.errorXcpt();
2792 return (None, None);
2793
2794 # Open a remote session, wait for this operation to complete.
2795 # (The loop is a kludge to deal with us racing the closing of the
2796 # direct session of a previous VM run. See waitOnDirectSessionClose.)
2797 if oWrapped is None:
2798 for i in range(10):
2799 try:
2800 if self.fpApiVer < 4.3 \
2801 or (self.fpApiVer == 4.3 and not hasattr(self.oVBoxMgr, 'getSessionObject')):
2802 oSession = self.oVBoxMgr.mgr.getSessionObject(self.oVBox); # pylint: disable=no-member
2803 elif self.fpApiVer < 5.2 \
2804 or (self.fpApiVer == 5.2 and hasattr(self.oVBoxMgr, 'vbox')):
2805 oSession = self.oVBoxMgr.getSessionObject(self.oVBox); # pylint: disable=no-member
2806 else:
2807 oSession = self.oVBoxMgr.getSessionObject(); # pylint: disable=no-member,no-value-for-parameter
2808 if self.fpApiVer < 3.3:
2809 oProgress = self.oVBox.openRemoteSession(oSession, sUuid, sType, sEnv);
2810 else:
2811 oProgress = oVM.launchVMProcess(oSession, sType, sEnv);
2812 break;
2813 except:
2814 if i == 9:
2815 reporter.errorXcpt('failed to start VM "%s" ("%s"), aborting.' % (sUuid, sName));
2816 return (None, None);
2817 oSession = None;
2818 if i >= 0:
2819 reporter.logXcpt('warning: failed to start VM "%s" ("%s") - retrying in %u secs.' % (sUuid, oVM, i)); # pylint: disable=line-too-long
2820 self.waitOnDirectSessionClose(oVM, 5000 + i * 1000);
2821 if fWait and oProgress is not None:
2822 rc = self.waitOnProgress(oProgress);
2823 if rc < 0:
2824 self.waitOnDirectSessionClose(oVM, 5000);
2825
2826 # VM failed to power up, still collect VBox.log, need to wrap the session object
2827 # in order to use the helper for adding the log files to the report.
2828 from testdriver.vboxwrappers import SessionWrapper;
2829 oTmp = SessionWrapper(oSession, oVM, self.oVBox, self.oVBoxMgr, self, True, sName, sLogFile);
2830 oTmp.addLogsToReport();
2831 try:
2832 if oSession is not None:
2833 oSession.close();
2834 except: pass;
2835 reportError(oProgress, 'failed to open session for "%s"' % (sName));
2836 self.uploadStartupLogFile(oVM, sName);
2837 return (None, None);
2838 reporter.log2('waitOnProgress -> %s' % (rc,));
2839
2840 # Wrap up the session object and push on to the list before returning it.
2841 if oWrapped is None:
2842 from testdriver.vboxwrappers import SessionWrapper;
2843 oWrapped = SessionWrapper(oSession, oVM, self.oVBox, self.oVBoxMgr, self, True, sName, sLogFile);
2844
2845 oWrapped.registerEventHandlerForTask();
2846 self.aoRemoteSessions.append(oWrapped);
2847 if oWrapped is not self.aoRemoteSessions[len(self.aoRemoteSessions) - 1]:
2848 reporter.error('not by reference: oWrapped=%s aoRemoteSessions[%s]=%s'
2849 % (oWrapped, len(self.aoRemoteSessions) - 1,
2850 self.aoRemoteSessions[len(self.aoRemoteSessions) - 1]));
2851 self.addTask(oWrapped);
2852
2853 reporter.log2('startVmEx: oSession=%s, oSessionWrapper=%s, oProgress=%s' % (oSession, oWrapped, oProgress));
2854
2855 from testdriver.vboxwrappers import ProgressWrapper;
2856 return (oWrapped, ProgressWrapper(oProgress, self.oVBoxMgr, self,
2857 'starting %s' % (sName,)) if oProgress else None);
2858
2859 def startVm(self, oVM, sType=None, sName = None, asEnv = None):
2860 """ Simplified version of startVmEx. """
2861 oSession, _ = self.startVmEx(oVM, True, sType, sName, asEnv = asEnv);
2862 return oSession;
2863
2864 def startVmByNameEx(self, sName, fWait=True, sType=None, asEnv = None):
2865 """
2866 Start the VM, returning the VM session and progress object on success.
2867 The session is also added to the task list and to the aoRemoteSessions set.
2868
2869 On failure (None, None) is returned and an error is logged.
2870 """
2871 oVM = self.getVmByName(sName);
2872 if oVM is None:
2873 return (None, None);
2874 return self.startVmEx(oVM, fWait, sType, sName, asEnv = asEnv);
2875
2876 def startVmByName(self, sName, sType=None, asEnv = None):
2877 """
2878 Start the VM, returning the VM session on success. The session is
2879 also added to the task list and to the aoRemoteSessions set.
2880
2881 On failure None is returned and an error is logged.
2882 """
2883 oSession, _ = self.startVmByNameEx(sName, True, sType, asEnv = asEnv);
2884 return oSession;
2885
2886 def terminateVmBySession(self, oSession, oProgress = None, fTakeScreenshot = None): # pylint: disable=too-many-statements
2887 """
2888 Terminates the VM specified by oSession and adds the release logs to
2889 the test report.
2890
2891 This will try achieve this by using powerOff, but will resort to
2892 tougher methods if that fails.
2893
2894 The session will always be removed from the task list.
2895 The session will be closed unless we fail to kill the process.
2896 The session will be removed from the remote session list if closed.
2897
2898 The progress object (a wrapper!) is for teleportation and similar VM
2899 operations, it will be attempted canceled before powering off the VM.
2900 Failures are logged but ignored.
2901 The progress object will always be removed from the task list.
2902
2903 Returns True if powerOff and session close both succeed.
2904 Returns False if on failure (logged), including when we successfully
2905 kill the VM process.
2906 """
2907 reporter.log2('terminateVmBySession: oSession=%s (pid=%s) oProgress=%s' % (oSession.sName, oSession.getPid(), oProgress));
2908
2909 # Call getPid first to make sure the PID is cached in the wrapper.
2910 oSession.getPid();
2911
2912 #
2913 # If the host is out of memory, just skip all the info collection as it
2914 # requires memory too and seems to wedge.
2915 #
2916 sHostProcessInfo = None;
2917 sHostProcessInfoHung = None;
2918 sLastScreenshotPath = None;
2919 sOsKernelLog = None;
2920 sVgaText = None;
2921 asMiscInfos = [];
2922
2923 if not oSession.fHostMemoryLow:
2924 # Try to fetch the VM process info before meddling with its state.
2925 if self.fAlwaysUploadLogs or reporter.testErrorCount() > 0:
2926 sHostProcessInfo = utils.processGetInfo(oSession.getPid(), fSudo = True);
2927
2928 #
2929 # Pause the VM if we're going to take any screenshots or dig into the
2930 # guest. Failures are quitely ignored.
2931 #
2932 if self.fAlwaysUploadLogs or reporter.testErrorCount() > 0:
2933 try:
2934 if oSession.oVM.state in [ vboxcon.MachineState_Running,
2935 vboxcon.MachineState_LiveSnapshotting,
2936 vboxcon.MachineState_Teleporting ]:
2937 oSession.o.console.pause();
2938 except:
2939 reporter.logXcpt();
2940
2941 #
2942 # Take Screenshot and upload it (see below) to Test Manager if appropriate/requested.
2943 #
2944 if fTakeScreenshot is True or self.fAlwaysUploadScreenshots or reporter.testErrorCount() > 0:
2945 sLastScreenshotPath = os.path.join(self.sScratchPath, "LastScreenshot-%s.png" % oSession.sName);
2946 fRc = oSession.takeScreenshot(sLastScreenshotPath);
2947 if fRc is not True:
2948 sLastScreenshotPath = None;
2949
2950 # Query the OS kernel log from the debugger if appropriate/requested.
2951 if self.fAlwaysUploadLogs or reporter.testErrorCount() > 0:
2952 sOsKernelLog = oSession.queryOsKernelLog();
2953
2954 # Do "info vgatext all" separately.
2955 if self.fAlwaysUploadLogs or reporter.testErrorCount() > 0:
2956 sVgaText = oSession.queryDbgInfoVgaText();
2957
2958 # Various infos (do after kernel because of symbols).
2959 if self.fAlwaysUploadLogs or reporter.testErrorCount() > 0:
2960 # Dump the guest stack for all CPUs.
2961 cCpus = oSession.getCpuCount();
2962 if cCpus > 0:
2963 for iCpu in xrange(0, cCpus):
2964 sThis = oSession.queryDbgGuestStack(iCpu);
2965 if sThis:
2966 asMiscInfos += [
2967 '================ start guest stack VCPU %s ================\n' % (iCpu,),
2968 sThis,
2969 '================ end guest stack VCPU %s ==================\n' % (iCpu,),
2970 ];
2971
2972 for sInfo, sArg in [ ('mode', 'all'),
2973 ('fflags', ''),
2974 ('cpumguest', 'verbose all'),
2975 ('cpumguestinstr', 'symbol all'),
2976 ('exits', ''),
2977 ('pic', ''),
2978 ('apic', ''),
2979 ('apiclvt', ''),
2980 ('apictimer', ''),
2981 ('ioapic', ''),
2982 ('pit', ''),
2983 ('phys', ''),
2984 ('clocks', ''),
2985 ('timers', ''),
2986 ('gdtguest', ''),
2987 ('ldtguest', ''),
2988 ]:
2989 if sInfo in ['apic',] and self.fpApiVer < 5.1: # asserts and burns
2990 continue;
2991 sThis = oSession.queryDbgInfo(sInfo, sArg);
2992 if sThis:
2993 if sThis[-1] != '\n':
2994 sThis += '\n';
2995 asMiscInfos += [
2996 '================ start %s %s ================\n' % (sInfo, sArg),
2997 sThis,
2998 '================ end %s %s ==================\n' % (sInfo, sArg),
2999 ];
3000
3001 #
3002 # Terminate the VM
3003 #
3004
3005 # Cancel the progress object if specified.
3006 if oProgress is not None:
3007 if not oProgress.isCompleted() and oProgress.isCancelable():
3008 reporter.log2('terminateVmBySession: canceling "%s"...' % (oProgress.sName));
3009 try:
3010 oProgress.o.cancel();
3011 except:
3012 reporter.logXcpt();
3013 else:
3014 oProgress.wait();
3015 self.removeTask(oProgress);
3016
3017 # Check if the VM has terminated by itself before powering it off.
3018 fClose = True;
3019 fRc = True;
3020 if oSession.needsPoweringOff():
3021 reporter.log('terminateVmBySession: powering off "%s"...' % (oSession.sName,));
3022 fRc = oSession.powerOff(fFudgeOnFailure = False);
3023 if fRc is not True:
3024 # power off failed, try terminate it in a nice manner.
3025 fRc = False;
3026 uPid = oSession.getPid();
3027 if uPid is not None:
3028 #
3029 # Collect some information about the VM process first to have
3030 # some state information for further investigation why powering off failed.
3031 #
3032 sHostProcessInfoHung = utils.processGetInfo(uPid, fSudo = True);
3033
3034 # Exterminate...
3035 reporter.error('terminateVmBySession: Terminating PID %u (VM %s)' % (uPid, oSession.sName));
3036 fClose = base.processTerminate(uPid);
3037 if fClose is True:
3038 self.waitOnDirectSessionClose(oSession.oVM, 5000);
3039 fClose = oSession.waitForTask(1000);
3040
3041 if fClose is not True:
3042 # Being nice failed...
3043 reporter.error('terminateVmBySession: Termination failed, trying to kill PID %u (VM %s) instead' \
3044 % (uPid, oSession.sName));
3045 fClose = base.processKill(uPid);
3046 if fClose is True:
3047 self.waitOnDirectSessionClose(oSession.oVM, 5000);
3048 fClose = oSession.waitForTask(1000);
3049 if fClose is not True:
3050 reporter.error('terminateVmBySession: Failed to kill PID %u (VM %s)' % (uPid, oSession.sName));
3051
3052 # The final steps.
3053 if fClose is True:
3054 reporter.log('terminateVmBySession: closing session "%s"...' % (oSession.sName,));
3055 oSession.close();
3056 self.waitOnDirectSessionClose(oSession.oVM, 10000);
3057 try:
3058 eState = oSession.oVM.state;
3059 except:
3060 reporter.logXcpt();
3061 else:
3062 if eState == vboxcon.MachineState_Aborted:
3063 reporter.error('terminateVmBySession: The VM "%s" aborted!' % (oSession.sName,));
3064 self.removeTask(oSession);
3065
3066 #
3067 # Add the release log, debug log and a screenshot of the VM to the test report.
3068 #
3069 if self.fAlwaysUploadLogs or reporter.testErrorCount() > 0:
3070 oSession.addLogsToReport();
3071
3072 # Add a screenshot if it has been requested and taken successfully.
3073 if sLastScreenshotPath is not None:
3074 if reporter.testErrorCount() > 0:
3075 reporter.addLogFile(sLastScreenshotPath, 'screenshot/failure', 'Last VM screenshot');
3076 else:
3077 reporter.addLogFile(sLastScreenshotPath, 'screenshot/success', 'Last VM screenshot');
3078
3079 # Add the guest OS log if it has been requested and taken successfully.
3080 if sOsKernelLog is not None:
3081 reporter.addLogString(sOsKernelLog, 'kernel.log', 'log/guest/kernel', 'Guest OS kernel log');
3082
3083 # Add "info vgatext all" if we've got it.
3084 if sVgaText is not None:
3085 reporter.addLogString(sVgaText, 'vgatext.txt', 'info/vgatext', 'info vgatext all');
3086
3087 # Add the "info xxxx" items if we've got any.
3088 if asMiscInfos:
3089 reporter.addLogString(u''.join(asMiscInfos), 'info.txt', 'info/collection', 'A bunch of info items.');
3090
3091 # Add the host process info if we were able to retrieve it.
3092 if sHostProcessInfo is not None:
3093 reporter.log('Trying to annotate the VM process report, please stand by...');
3094 fRcTmp = self.annotateAndUploadProcessReport(sHostProcessInfo, 'vmprocess.log',
3095 'process/report/vm', 'Annotated VM process state');
3096 # Upload the raw log for manual annotation in case resolving failed.
3097 if not fRcTmp:
3098 reporter.log('Failed to annotate VM process report, uploading raw report');
3099 reporter.addLogString(sHostProcessInfo, 'vmprocess.log', 'process/report/vm', 'VM process state');
3100
3101 # Add the host process info for failed power off attempts if we were able to retrieve it.
3102 if sHostProcessInfoHung is not None:
3103 reporter.log('Trying to annotate the hung VM process report, please stand by...');
3104 fRcTmp = self.annotateAndUploadProcessReport(sHostProcessInfoHung, 'vmprocess-hung.log',
3105 'process/report/vm', 'Annotated hung VM process state');
3106 # Upload the raw log for manual annotation in case resolving failed.
3107 if not fRcTmp:
3108 reporter.log('Failed to annotate hung VM process report, uploading raw report');
3109 reporter.addLogString(sHostProcessInfoHung, 'vmprocess-hung.log', 'process/report/vm',
3110 'Hung VM process state');
3111
3112 return fRc;
3113
3114
3115 #
3116 # Some information query functions (mix).
3117 #
3118 # Methods require the VBox API. If the information is provided by both
3119 # the testboxscript as well as VBox API, we'll check if it matches.
3120 #
3121
3122 def _hasHostCpuFeature(self, sEnvVar, sEnum, fpApiMinVer, fQuiet):
3123 """
3124 Common Worker for hasHostNestedPaging() and hasHostHwVirt().
3125
3126 Returns True / False.
3127 Raises exception on environment / host mismatch.
3128 """
3129 fEnv = os.environ.get(sEnvVar, None);
3130 if fEnv is not None:
3131 fEnv = fEnv.lower() not in [ 'false', 'f', 'not', 'no', 'n', '0', ];
3132
3133 fVBox = None;
3134 self.importVBoxApi();
3135 if self.fpApiVer >= fpApiMinVer and hasattr(vboxcon, sEnum):
3136 try:
3137 fVBox = self.oVBox.host.getProcessorFeature(getattr(vboxcon, sEnum));
3138 except:
3139 if not fQuiet:
3140 reporter.logXcpt();
3141
3142 if fVBox is not None:
3143 if fEnv is not None:
3144 if fEnv != fVBox and not fQuiet:
3145 reporter.log('TestBox configuration overwritten: fVBox=%s (%s) vs. fEnv=%s (%s)'
3146 % (fVBox, sEnum, fEnv, sEnvVar));
3147 return fEnv;
3148 return fVBox;
3149 if fEnv is not None:
3150 return fEnv;
3151 return False;
3152
3153 def hasHostHwVirt(self, fQuiet = False):
3154 """
3155 Checks if hardware assisted virtualization is supported by the host.
3156
3157 Returns True / False.
3158 Raises exception on environment / host mismatch.
3159 """
3160 return self._hasHostCpuFeature('TESTBOX_HAS_HW_VIRT', 'ProcessorFeature_HWVirtEx', 3.1, fQuiet);
3161
3162 def hasHostNestedPaging(self, fQuiet = False):
3163 """
3164 Checks if nested paging is supported by the host.
3165
3166 Returns True / False.
3167 Raises exception on environment / host mismatch.
3168 """
3169 return self._hasHostCpuFeature('TESTBOX_HAS_NESTED_PAGING', 'ProcessorFeature_NestedPaging', 4.2, fQuiet) \
3170 and self.hasHostHwVirt(fQuiet);
3171
3172 def hasHostLongMode(self, fQuiet = False):
3173 """
3174 Checks if the host supports 64-bit guests.
3175
3176 Returns True / False.
3177 Raises exception on environment / host mismatch.
3178 """
3179 # Note that the testboxscript doesn't export this variable atm.
3180 return self._hasHostCpuFeature('TESTBOX_HAS_LONG_MODE', 'ProcessorFeature_LongMode', 3.1, fQuiet);
3181
3182 def getHostCpuCount(self, fQuiet = False):
3183 """
3184 Returns the number of CPUs on the host.
3185
3186 Returns True / False.
3187 Raises exception on environment / host mismatch.
3188 """
3189 cEnv = os.environ.get('TESTBOX_CPU_COUNT', None);
3190 if cEnv is not None:
3191 cEnv = int(cEnv);
3192
3193 try:
3194 cVBox = self.oVBox.host.processorOnlineCount;
3195 except:
3196 if not fQuiet:
3197 reporter.logXcpt();
3198 cVBox = None;
3199
3200 if cVBox is not None:
3201 if cEnv is not None:
3202 assert cVBox == cEnv, 'Misconfigured TestBox: VBox: %u CPUs, testboxscript: %u CPUs' % (cVBox, cEnv);
3203 return cVBox;
3204 if cEnv is not None:
3205 return cEnv;
3206 return 1;
3207
3208 def _getHostCpuDesc(self, fQuiet = False):
3209 """
3210 Internal method used for getting the host CPU description from VBoxSVC.
3211 Returns description string, on failure an empty string is returned.
3212 """
3213 try:
3214 return self.oVBox.host.getProcessorDescription(0);
3215 except:
3216 if not fQuiet:
3217 reporter.logXcpt();
3218 return '';
3219
3220 def isHostCpuAmd(self, fQuiet = False):
3221 """
3222 Checks if the host CPU vendor is AMD.
3223
3224 Returns True / False.
3225 """
3226 sCpuDesc = self._getHostCpuDesc(fQuiet);
3227 return 'AMD' in sCpuDesc or sCpuDesc == 'AuthenticAMD';
3228
3229 def isHostCpuIntel(self, fQuiet = False):
3230 """
3231 Checks if the host CPU vendor is Intel.
3232
3233 Returns True / False.
3234 """
3235 sCpuDesc = self._getHostCpuDesc(fQuiet);
3236 return sCpuDesc.startswith("Intel") or sCpuDesc == 'GenuineIntel';
3237
3238 def isHostCpuVia(self, fQuiet = False):
3239 """
3240 Checks if the host CPU vendor is VIA (or Centaur).
3241
3242 Returns True / False.
3243 """
3244 sCpuDesc = self._getHostCpuDesc(fQuiet);
3245 return sCpuDesc.startswith("VIA") or sCpuDesc == 'CentaurHauls';
3246
3247 def isHostCpuShanghai(self, fQuiet = False):
3248 """
3249 Checks if the host CPU vendor is Shanghai (or Zhaoxin).
3250
3251 Returns True / False.
3252 """
3253 sCpuDesc = self._getHostCpuDesc(fQuiet);
3254 return sCpuDesc.startswith("ZHAOXIN") or sCpuDesc.strip(' ') == 'Shanghai';
3255
3256 def isHostCpuP4(self, fQuiet = False):
3257 """
3258 Checks if the host CPU is a Pentium 4 / Pentium D.
3259
3260 Returns True / False.
3261 """
3262 if not self.isHostCpuIntel(fQuiet):
3263 return False;
3264
3265 (uFamilyModel, _, _, _) = self.oVBox.host.getProcessorCPUIDLeaf(0, 0x1, 0);
3266 return ((uFamilyModel >> 8) & 0xf) == 0xf;
3267
3268 def hasRawModeSupport(self, fQuiet = False):
3269 """
3270 Checks if raw-mode is supported by VirtualBox that the testbox is
3271 configured for it.
3272
3273 Returns True / False.
3274 Raises no exceptions.
3275
3276 Note! Differs from the rest in that we don't require the
3277 TESTBOX_WITH_RAW_MODE value to match the API. It is
3278 sometimes helpful to disable raw-mode on individual
3279 test boxes. (This probably goes for
3280 """
3281 # The environment variable can be used to disable raw-mode.
3282 fEnv = os.environ.get('TESTBOX_WITH_RAW_MODE', None);
3283 if fEnv is not None:
3284 fEnv = fEnv.lower() not in [ 'false', 'f', 'not', 'no', 'n', '0', ];
3285 if fEnv is False:
3286 return False;
3287
3288 # Starting with 5.0 GA / RC2 the API can tell us whether VBox was built
3289 # with raw-mode support or not.
3290 self.importVBoxApi();
3291 if self.fpApiVer >= 5.0:
3292 try:
3293 fVBox = self.oVBox.systemProperties.rawModeSupported;
3294 except:
3295 if not fQuiet:
3296 reporter.logXcpt();
3297 fVBox = True;
3298 if fVBox is False:
3299 return False;
3300
3301 return True;
3302
3303 #
3304 # Testdriver execution methods.
3305 #
3306
3307 def handleTask(self, oTask, sMethod):
3308 """
3309 Callback method for handling unknown tasks in the various run loops.
3310
3311 The testdriver should override this if it already tasks running when
3312 calling startVmAndConnectToTxsViaTcp, txsRunTest or similar methods.
3313 Call super to handle unknown tasks.
3314
3315 Returns True if handled, False if not.
3316 """
3317 reporter.error('%s: unknown task %s' % (sMethod, oTask));
3318 return False;
3319
3320 def txsDoTask(self, oSession, oTxsSession, fnAsync, aArgs):
3321 """
3322 Generic TXS task wrapper which waits both on the TXS and the session tasks.
3323
3324 Returns False on error, logged.
3325
3326 Returns task result on success.
3327 """
3328 # All async methods ends with the following two args.
3329 cMsTimeout = aArgs[-2];
3330 fIgnoreErrors = aArgs[-1];
3331
3332 fRemoveVm = self.addTask(oSession);
3333 fRemoveTxs = self.addTask(oTxsSession);
3334
3335 rc = fnAsync(*aArgs); # pylint: disable=star-args
3336 if rc is True:
3337 rc = False;
3338 oTask = self.waitForTasks(cMsTimeout + 1);
3339 if oTask is oTxsSession:
3340 if oTxsSession.isSuccess():
3341 rc = oTxsSession.getResult();
3342 elif fIgnoreErrors is True:
3343 reporter.log( 'txsDoTask: task failed (%s)' % (oTxsSession.getLastReply()[1],));
3344 else:
3345 reporter.error('txsDoTask: task failed (%s)' % (oTxsSession.getLastReply()[1],));
3346 else:
3347 oTxsSession.cancelTask();
3348 if oTask is None:
3349 if fIgnoreErrors is True:
3350 reporter.log( 'txsDoTask: The task timed out.');
3351 else:
3352 reporter.errorTimeout('txsDoTask: The task timed out.');
3353 elif oTask is oSession:
3354 reporter.error('txsDoTask: The VM terminated unexpectedly');
3355 else:
3356 if fIgnoreErrors is True:
3357 reporter.log( 'txsDoTask: An unknown task %s was returned' % (oTask,));
3358 else:
3359 reporter.error('txsDoTask: An unknown task %s was returned' % (oTask,));
3360 else:
3361 reporter.error('txsDoTask: fnAsync returned %s' % (rc,));
3362
3363 if fRemoveTxs:
3364 self.removeTask(oTxsSession);
3365 if fRemoveVm:
3366 self.removeTask(oSession);
3367 return rc;
3368
3369 # pylint: disable=missing-docstring
3370
3371 def txsDisconnect(self, oSession, oTxsSession, cMsTimeout = 30000, fIgnoreErrors = False):
3372 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncDisconnect,
3373 (self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3374
3375 def txsUuid(self, oSession, oTxsSession, cMsTimeout = 30000, fIgnoreErrors = False):
3376 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncUuid,
3377 (self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3378
3379 def txsMkDir(self, oSession, oTxsSession, sRemoteDir, fMode = 0o700, cMsTimeout = 30000, fIgnoreErrors = False):
3380 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncMkDir,
3381 (sRemoteDir, fMode, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3382
3383 def txsMkDirPath(self, oSession, oTxsSession, sRemoteDir, fMode = 0o700, cMsTimeout = 30000, fIgnoreErrors = False):
3384 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncMkDirPath,
3385 (sRemoteDir, fMode, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3386
3387 def txsMkSymlink(self, oSession, oTxsSession, sLinkTarget, sLink, cMsTimeout = 30000, fIgnoreErrors = False):
3388 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncMkSymlink,
3389 (sLinkTarget, sLink, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3390
3391 def txsRmDir(self, oSession, oTxsSession, sRemoteDir, cMsTimeout = 30000, fIgnoreErrors = False):
3392 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncRmDir,
3393 (sRemoteDir, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3394
3395 def txsRmFile(self, oSession, oTxsSession, sRemoteFile, cMsTimeout = 30000, fIgnoreErrors = False):
3396 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncRmFile,
3397 (sRemoteFile, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3398
3399 def txsRmSymlink(self, oSession, oTxsSession, sRemoteSymlink, cMsTimeout = 30000, fIgnoreErrors = False):
3400 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncRmSymlink,
3401 (sRemoteSymlink, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3402
3403 def txsRmTree(self, oSession, oTxsSession, sRemoteTree, cMsTimeout = 30000, fIgnoreErrors = False):
3404 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncRmTree,
3405 (sRemoteTree, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3406
3407 def txsIsDir(self, oSession, oTxsSession, sRemoteDir, cMsTimeout = 30000, fIgnoreErrors = False):
3408 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncIsDir,
3409 (sRemoteDir, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3410
3411 def txsIsFile(self, oSession, oTxsSession, sRemoteFile, cMsTimeout = 30000, fIgnoreErrors = False):
3412 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncIsFile,
3413 (sRemoteFile, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3414
3415 def txsIsSymlink(self, oSession, oTxsSession, sRemoteSymlink, cMsTimeout = 30000, fIgnoreErrors = False):
3416 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncIsSymlink,
3417 (sRemoteSymlink, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3418
3419 def txsUploadFile(self, oSession, oTxsSession, sLocalFile, sRemoteFile, cMsTimeout = 30000, fIgnoreErrors = False):
3420 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncUploadFile, \
3421 (sLocalFile, sRemoteFile, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3422
3423 def txsUploadString(self, oSession, oTxsSession, sContent, sRemoteFile, cMsTimeout = 30000, fIgnoreErrors = False):
3424 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncUploadString, \
3425 (sContent, sRemoteFile, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3426
3427 def txsDownloadFile(self, oSession, oTxsSession, sRemoteFile, sLocalFile, cMsTimeout = 30000, fIgnoreErrors = False):
3428 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncDownloadFile, \
3429 (sRemoteFile, sLocalFile, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3430
3431 def txsDownloadFiles(self, oSession, oTxsSession, asFiles, fIgnoreErrors = False):
3432 """
3433 Convenience function to get files from the guest and stores it
3434 into the scratch directory for later (manual) review.
3435
3436 Returns True on success.
3437
3438 Returns False on failure, logged.
3439 """
3440 fRc = True;
3441 for sGstFile in asFiles:
3442 sTmpFile = os.path.join(self.sScratchPath, 'tmp-' + os.path.basename(sGstFile));
3443 reporter.log2('Downloading file "%s" to "%s" ...' % (sGstFile, sTmpFile));
3444 # First try to remove (unlink) an existing temporary file, as we don't truncate the file.
3445 try: os.unlink(sTmpFile);
3446 except: pass;
3447 ## @todo Check for already existing files on the host and create a new
3448 # name for the current file to download.
3449 fRc = self.txsDownloadFile(oSession, oTxsSession, sGstFile, sTmpFile, 30 * 1000, fIgnoreErrors);
3450 if fRc:
3451 reporter.addLogFile(sTmpFile, 'misc/other', 'guest - ' + sGstFile);
3452 else:
3453 if fIgnoreErrors is not True:
3454 reporter.error('error downloading file "%s" to "%s"' % (sGstFile, sTmpFile));
3455 return fRc;
3456 reporter.log('warning: file "%s" was not downloaded, ignoring.' % (sGstFile,));
3457 return True;
3458
3459 def txsDownloadString(self, oSession, oTxsSession, sRemoteFile, sEncoding = 'utf-8', fIgnoreEncodingErrors = True,
3460 cMsTimeout = 30000, fIgnoreErrors = False):
3461 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncDownloadString,
3462 (sRemoteFile, sEncoding, fIgnoreEncodingErrors, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3463
3464 def txsUnpackFile(self, oSession, oTxsSession, sRemoteFile, sRemoteDir, cMsTimeout = 30000, fIgnoreErrors = False):
3465 return self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncUnpackFile, \
3466 (sRemoteFile, sRemoteDir, self.adjustTimeoutMs(cMsTimeout), fIgnoreErrors));
3467
3468 # pylint: enable=missing-docstring
3469
3470 def txsCdWait(self,
3471 oSession, # type: vboxwrappers.SessionWrapper
3472 oTxsSession, # type: txsclient.Session
3473 cMsTimeout = 30000, # type: int
3474 sFile = None # type: String
3475 ): # -> bool
3476 """
3477 Mostly an internal helper for txsRebootAndReconnectViaTcp and
3478 startVmAndConnectToTxsViaTcp that waits for the CDROM drive to become
3479 ready. It does this by polling for a file it knows to exist on the CD.
3480
3481 Returns True on success.
3482
3483 Returns False on failure, logged.
3484 """
3485
3486 if sFile is None:
3487 sFile = 'valkit.txt';
3488 fRemoveVm = self.addTask(oSession);
3489 fRemoveTxs = self.addTask(oTxsSession);
3490 cMsTimeout = self.adjustTimeoutMs(cMsTimeout);
3491 msStart = base.timestampMilli();
3492 cMsTimeout2 = cMsTimeout;
3493 fRc = oTxsSession.asyncIsFile('${CDROM}/%s' % (sFile,), cMsTimeout2);
3494 if fRc is True:
3495 while True:
3496 # wait for it to complete.
3497 oTask = self.waitForTasks(cMsTimeout2 + 1);
3498 if oTask is not oTxsSession:
3499 oTxsSession.cancelTask();
3500 if oTask is None:
3501 reporter.errorTimeout('txsCdWait: The task timed out (after %s ms).'
3502 % (base.timestampMilli() - msStart,));
3503 elif oTask is oSession:
3504 reporter.error('txsCdWait: The VM terminated unexpectedly');
3505 else:
3506 reporter.error('txsCdWait: An unknown task %s was returned' % (oTask,));
3507 fRc = False;
3508 break;
3509 if oTxsSession.isSuccess():
3510 break;
3511
3512 # Check for timeout.
3513 cMsElapsed = base.timestampMilli() - msStart;
3514 if cMsElapsed >= cMsTimeout:
3515 reporter.error('txsCdWait: timed out');
3516 fRc = False;
3517 break;
3518
3519 # delay.
3520 self.sleep(1);
3521
3522 # resubmitt the task.
3523 cMsTimeout2 = msStart + cMsTimeout - base.timestampMilli();
3524 if cMsTimeout2 < 500:
3525 cMsTimeout2 = 500;
3526 fRc = oTxsSession.asyncIsFile('${CDROM}/%s' % (sFile,), cMsTimeout2);
3527 if fRc is not True:
3528 reporter.error('txsCdWait: asyncIsFile failed');
3529 break;
3530 else:
3531 reporter.error('txsCdWait: asyncIsFile failed');
3532
3533 if fRemoveTxs:
3534 self.removeTask(oTxsSession);
3535 if fRemoveVm:
3536 self.removeTask(oSession);
3537 return fRc;
3538
3539 def txsDoConnectViaTcp(self, oSession, cMsTimeout, fNatForwardingForTxs = False):
3540 """
3541 Mostly an internal worker for connecting to TXS via TCP used by the
3542 *ViaTcp methods.
3543
3544 Returns a tuplet with True/False and TxsSession/None depending on the
3545 result. Errors are logged.
3546 """
3547
3548 reporter.log2('txsDoConnectViaTcp: oSession=%s, cMsTimeout=%s, fNatForwardingForTxs=%s'
3549 % (oSession, cMsTimeout, fNatForwardingForTxs));
3550
3551 cMsTimeout = self.adjustTimeoutMs(cMsTimeout);
3552 oTxsConnect = oSession.txsConnectViaTcp(cMsTimeout, fNatForwardingForTxs = fNatForwardingForTxs);
3553 if oTxsConnect is not None:
3554 self.addTask(oTxsConnect);
3555 fRemoveVm = self.addTask(oSession);
3556 oTask = self.waitForTasks(cMsTimeout + 1);
3557 reporter.log2('txsDoConnectViaTcp: waitForTasks returned %s' % (oTask,));
3558 self.removeTask(oTxsConnect);
3559 if oTask is oTxsConnect:
3560 oTxsSession = oTxsConnect.getResult();
3561 if oTxsSession is not None:
3562 reporter.log('txsDoConnectViaTcp: Connected to TXS on %s.' % (oTxsSession.oTransport.sHostname,));
3563 return (True, oTxsSession);
3564
3565 reporter.error('txsDoConnectViaTcp: failed to connect to TXS.');
3566 else:
3567 oTxsConnect.cancelTask();
3568 if oTask is None:
3569 reporter.errorTimeout('txsDoConnectViaTcp: connect stage 1 timed out');
3570 elif oTask is oSession:
3571 oSession.reportPrematureTermination('txsDoConnectViaTcp: ');
3572 else:
3573 reporter.error('txsDoConnectViaTcp: unknown/wrong task %s' % (oTask,));
3574 if fRemoveVm:
3575 self.removeTask(oSession);
3576 else:
3577 reporter.error('txsDoConnectViaTcp: txsConnectViaTcp failed');
3578 return (False, None);
3579
3580 def startVmAndConnectToTxsViaTcp(self, sVmName, fCdWait = False, cMsTimeout = 15*60000, \
3581 cMsCdWait = 30000, sFileCdWait = None, \
3582 fNatForwardingForTxs = False):
3583 """
3584 Starts the specified VM and tries to connect to its TXS via TCP.
3585 The VM will be powered off if TXS doesn't respond before the specified
3586 time has elapsed.
3587
3588 Returns a the VM and TXS sessions (a two tuple) on success. The VM
3589 session is in the task list, the TXS session is not.
3590 Returns (None, None) on failure, fully logged.
3591 """
3592
3593 # Zap the guest IP to make sure we're not getting a stale entry
3594 # (unless we're restoring the VM of course).
3595 oTestVM = self.oTestVmSet.findTestVmByName(sVmName) if self.oTestVmSet is not None else None;
3596 if oTestVM is None \
3597 or oTestVM.fSnapshotRestoreCurrent is False:
3598 try:
3599 oSession1 = self.openSession(self.getVmByName(sVmName));
3600 oSession1.delGuestPropertyValue('/VirtualBox/GuestInfo/Net/0/V4/IP');
3601 oSession1.saveSettings(True);
3602 del oSession1;
3603 except:
3604 reporter.logXcpt();
3605
3606 # Start the VM.
3607 reporter.log('startVmAndConnectToTxsViaTcp: Starting(/preparing) "%s" (timeout %s s)...' % (sVmName, cMsTimeout / 1000));
3608 reporter.flushall();
3609 oSession = self.startVmByName(sVmName);
3610 if oSession is not None:
3611 # Connect to TXS.
3612 reporter.log2('startVmAndConnectToTxsViaTcp: Started(/prepared) "%s", connecting to TXS ...' % (sVmName,));
3613 (fRc, oTxsSession) = self.txsDoConnectViaTcp(oSession, cMsTimeout, fNatForwardingForTxs);
3614 if fRc is True:
3615 if fCdWait:
3616 # Wait for CD?
3617 fRc = self.txsCdWait(oSession, oTxsSession, cMsCdWait, sFileCdWait);
3618 if fRc is not True:
3619 reporter.error('startVmAndConnectToTxsViaTcp: txsCdWait failed');
3620 if fRc is True:
3621 # Success!
3622 return (oSession, oTxsSession);
3623 else:
3624 reporter.error('startVmAndConnectToTxsViaTcp: txsDoConnectViaTcp failed');
3625 # If something went wrong while waiting for TXS to be started - take VM screenshot before terminate it
3626 self.terminateVmBySession(oSession);
3627 return (None, None);
3628
3629 def txsRebootAndReconnectViaTcp(self, oSession, oTxsSession, fCdWait = False, cMsTimeout = 15*60000, \
3630 cMsCdWait = 30000, sFileCdWait = None, fNatForwardingForTxs = False):
3631 """
3632 Executes the TXS reboot command
3633
3634 Returns A tuple of True and the new TXS session on success.
3635
3636 Returns A tuple of False and either the old TXS session or None on failure.
3637 """
3638 reporter.log2('txsRebootAndReconnect: cMsTimeout=%u' % (cMsTimeout,));
3639
3640 #
3641 # This stuff is a bit complicated because of rebooting being kind of
3642 # disruptive to the TXS and such... The protocol is that TXS will:
3643 # - ACK the reboot command.
3644 # - Shutdown the transport layer, implicitly disconnecting us.
3645 # - Execute the reboot operation.
3646 # - On failure, it will be re-init the transport layer and be
3647 # available pretty much immediately. UUID unchanged.
3648 # - On success, it will be respawed after the reboot (hopefully),
3649 # with a different UUID.
3650 #
3651 fRc = False;
3652 iStart = base.timestampMilli();
3653
3654 # Get UUID.
3655 cMsTimeout2 = min(60000, cMsTimeout);
3656 sUuidBefore = self.txsUuid(oSession, oTxsSession, self.adjustTimeoutMs(cMsTimeout2, 60000));
3657 if sUuidBefore is not False:
3658 # Reboot.
3659 cMsElapsed = base.timestampMilli() - iStart;
3660 cMsTimeout2 = cMsTimeout - cMsElapsed;
3661 fRc = self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncReboot,
3662 (self.adjustTimeoutMs(cMsTimeout2, 60000), False));
3663 if fRc is True:
3664 # Reconnect.
3665 if fNatForwardingForTxs is True:
3666 self.sleep(22); # NAT fudge - Two fixes are wanted: 1. TXS connect retries. 2. Main API reboot/reset hint.
3667 cMsElapsed = base.timestampMilli() - iStart;
3668 (fRc, oTxsSession) = self.txsDoConnectViaTcp(oSession, cMsTimeout - cMsElapsed, fNatForwardingForTxs);
3669 if fRc is True:
3670 # Check the UUID.
3671 cMsElapsed = base.timestampMilli() - iStart;
3672 cMsTimeout2 = min(60000, cMsTimeout - cMsElapsed);
3673 sUuidAfter = self.txsDoTask(oSession, oTxsSession, oTxsSession.asyncUuid,
3674 (self.adjustTimeoutMs(cMsTimeout2, 60000), False));
3675 if sUuidBefore is not False:
3676 if sUuidAfter != sUuidBefore:
3677 reporter.log('The guest rebooted (UUID %s -> %s)' % (sUuidBefore, sUuidAfter))
3678
3679 # Do CD wait if specified.
3680 if fCdWait:
3681 fRc = self.txsCdWait(oSession, oTxsSession, cMsCdWait, sFileCdWait);
3682 if fRc is not True:
3683 reporter.error('txsRebootAndReconnectViaTcp: txsCdWait failed');
3684 else:
3685 reporter.error('txsRebootAndReconnectViaTcp: failed to get UUID (after)');
3686 else:
3687 reporter.error('txsRebootAndReconnectViaTcp: did not reboot (UUID %s)' % (sUuidBefore,));
3688 else:
3689 reporter.error('txsRebootAndReconnectViaTcp: txsDoConnectViaTcp failed');
3690 else:
3691 reporter.error('txsRebootAndReconnectViaTcp: reboot failed');
3692 else:
3693 reporter.error('txsRebootAndReconnectViaTcp: failed to get UUID (before)');
3694 return (fRc, oTxsSession);
3695
3696 # pylint: disable=too-many-locals,too-many-arguments
3697
3698 def txsRunTest(self, oTxsSession, sTestName, cMsTimeout, sExecName, asArgs = (), asAddEnv = (), sAsUser = ""):
3699 """
3700 Executes the specified test task, waiting till it completes or times out.
3701
3702 The VM session (if any) must be in the task list.
3703
3704 Returns True if we executed the task and nothing abnormal happend.
3705 Query the process status from the TXS session.
3706
3707 Returns False if some unexpected task was signalled or we failed to
3708 submit the job.
3709 """
3710 reporter.testStart(sTestName);
3711 reporter.log2('txsRunTest: cMsTimeout=%u sExecName=%s asArgs=%s' % (cMsTimeout, sExecName, asArgs));
3712
3713 # Submit the job.
3714 fRc = False;
3715 if oTxsSession.asyncExec(sExecName, asArgs, asAddEnv, sAsUser, cMsTimeout = self.adjustTimeoutMs(cMsTimeout)):
3716 self.addTask(oTxsSession);
3717
3718 # Wait for the job to complete.
3719 while True:
3720 oTask = self.waitForTasks(cMsTimeout + 1);
3721 if oTask is None:
3722 reporter.log('txsRunTest: waitForTasks timed out');
3723 break;
3724 if oTask is oTxsSession:
3725 fRc = True;
3726 reporter.log('txsRunTest: isSuccess=%s getResult=%s' % (oTxsSession.isSuccess(), oTxsSession.getResult()));
3727 break;
3728 if not self.handleTask(oTask, 'txsRunTest'):
3729 break;
3730
3731 self.removeTask(oTxsSession);
3732 if not oTxsSession.pollTask():
3733 oTxsSession.cancelTask();
3734 else:
3735 reporter.error('txsRunTest: asyncExec failed');
3736
3737 reporter.testDone();
3738 return fRc;
3739
3740 def txsRunTestRedirectStd(self, oTxsSession, sTestName, cMsTimeout, sExecName, asArgs = (), asAddEnv = (), sAsUser = "",
3741 oStdIn = '/dev/null', oStdOut = '/dev/null', oStdErr = '/dev/null', oTestPipe = '/dev/null'):
3742 """
3743 Executes the specified test task, waiting till it completes or times out,
3744 redirecting stdin, stdout and stderr to the given objects.
3745
3746 The VM session (if any) must be in the task list.
3747
3748 Returns True if we executed the task and nothing abnormal happend.
3749 Query the process status from the TXS session.
3750
3751 Returns False if some unexpected task was signalled or we failed to
3752 submit the job.
3753 """
3754 reporter.testStart(sTestName);
3755 reporter.log2('txsRunTestRedirectStd: cMsTimeout=%u sExecName=%s asArgs=%s' % (cMsTimeout, sExecName, asArgs));
3756
3757 # Submit the job.
3758 fRc = False;
3759 if oTxsSession.asyncExecEx(sExecName, asArgs, asAddEnv, oStdIn, oStdOut, oStdErr,
3760 oTestPipe, sAsUser, cMsTimeout = self.adjustTimeoutMs(cMsTimeout)):
3761 self.addTask(oTxsSession);
3762
3763 # Wait for the job to complete.
3764 while True:
3765 oTask = self.waitForTasks(cMsTimeout + 1);
3766 if oTask is None:
3767 reporter.log('txsRunTestRedirectStd: waitForTasks timed out');
3768 break;
3769 if oTask is oTxsSession:
3770 fRc = True;
3771 reporter.log('txsRunTestRedirectStd: isSuccess=%s getResult=%s'
3772 % (oTxsSession.isSuccess(), oTxsSession.getResult()));
3773 break;
3774 if not self.handleTask(oTask, 'txsRunTestRedirectStd'):
3775 break;
3776
3777 self.removeTask(oTxsSession);
3778 if not oTxsSession.pollTask():
3779 oTxsSession.cancelTask();
3780 else:
3781 reporter.error('txsRunTestRedirectStd: asyncExec failed');
3782
3783 reporter.testDone();
3784 return fRc;
3785
3786 def txsRunTest2(self, oTxsSession1, oTxsSession2, sTestName, cMsTimeout,
3787 sExecName1, asArgs1,
3788 sExecName2, asArgs2,
3789 asAddEnv1 = (), sAsUser1 = '', fWithTestPipe1 = True,
3790 asAddEnv2 = (), sAsUser2 = '', fWithTestPipe2 = True):
3791 """
3792 Executes the specified test tasks, waiting till they complete or
3793 times out. The 1st task is started after the 2nd one.
3794
3795 The VM session (if any) must be in the task list.
3796
3797 Returns True if we executed the task and nothing abnormal happend.
3798 Query the process status from the TXS sessions.
3799
3800 Returns False if some unexpected task was signalled or we failed to
3801 submit the job.
3802 """
3803 reporter.testStart(sTestName);
3804
3805 # Submit the jobs.
3806 fRc = False;
3807 if oTxsSession1.asyncExec(sExecName1, asArgs1, asAddEnv1, sAsUser1, fWithTestPipe1, '1-',
3808 self.adjustTimeoutMs(cMsTimeout)):
3809 self.addTask(oTxsSession1);
3810
3811 self.sleep(2); # fudge! grr
3812
3813 if oTxsSession2.asyncExec(sExecName2, asArgs2, asAddEnv2, sAsUser2, fWithTestPipe2, '2-',
3814 self.adjustTimeoutMs(cMsTimeout)):
3815 self.addTask(oTxsSession2);
3816
3817 # Wait for the jobs to complete.
3818 cPendingJobs = 2;
3819 while True:
3820 oTask = self.waitForTasks(cMsTimeout + 1);
3821 if oTask is None:
3822 reporter.log('txsRunTest2: waitForTasks timed out');
3823 break;
3824
3825 if oTask is oTxsSession1 or oTask is oTxsSession2:
3826 if oTask is oTxsSession1: iTask = 1;
3827 else: iTask = 2;
3828 reporter.log('txsRunTest2: #%u - isSuccess=%s getResult=%s' \
3829 % (iTask, oTask.isSuccess(), oTask.getResult()));
3830 self.removeTask(oTask);
3831 cPendingJobs -= 1;
3832 if cPendingJobs <= 0:
3833 fRc = True;
3834 break;
3835
3836 elif not self.handleTask(oTask, 'txsRunTest'):
3837 break;
3838
3839 self.removeTask(oTxsSession2);
3840 if not oTxsSession2.pollTask():
3841 oTxsSession2.cancelTask();
3842 else:
3843 reporter.error('txsRunTest2: asyncExec #2 failed');
3844
3845 self.removeTask(oTxsSession1);
3846 if not oTxsSession1.pollTask():
3847 oTxsSession1.cancelTask();
3848 else:
3849 reporter.error('txsRunTest2: asyncExec #1 failed');
3850
3851 reporter.testDone();
3852 return fRc;
3853
3854 # pylint: enable=too-many-locals,too-many-arguments
3855
3856
3857 #
3858 # Working with test results via serial port.
3859 #
3860
3861 class TxsMonitorComFile(base.TdTaskBase):
3862 """
3863 Class that monitors a COM output file.
3864 """
3865
3866 def __init__(self, sComRawFile, asStopWords = None):
3867 base.TdTaskBase.__init__(self, utils.getCallerName());
3868 self.sComRawFile = sComRawFile;
3869 self.oStopRegExp = re.compile('\\b(' + '|'.join(asStopWords if asStopWords else ('PASSED', 'FAILED',)) + ')\\b');
3870 self.sResult = None; ##< The result.
3871 self.cchDisplayed = 0; ##< Offset into the file string of what we've already fed to the logger.
3872
3873 def toString(self):
3874 return '<%s sComRawFile=%s oStopRegExp=%s sResult=%s cchDisplayed=%s>' \
3875 % (base.TdTaskBase.toString(self), self.sComRawFile, self.oStopRegExp, self.sResult, self.cchDisplayed,);
3876
3877 def pollTask(self, fLocked = False):
3878 """
3879 Overrides TdTaskBase.pollTask() for the purpose of polling the file.
3880 """
3881 if not fLocked:
3882 self.lockTask();
3883
3884 sFile = utils.noxcptReadFile(self.sComRawFile, '', 'rU');
3885 if len(sFile) > self.cchDisplayed:
3886 sNew = sFile[self.cchDisplayed:];
3887 oMatch = self.oStopRegExp.search(sNew);
3888 if oMatch:
3889 # Done! Get result, flush all the output and signal the task.
3890 self.sResult = oMatch.group(1);
3891 for sLine in sNew.split('\n'):
3892 reporter.log('COM OUTPUT: %s' % (sLine,));
3893 self.cchDisplayed = len(sFile);
3894 self.signalTaskLocked();
3895 else:
3896 # Output whole lines only.
3897 offNewline = sFile.find('\n', self.cchDisplayed);
3898 while offNewline >= 0:
3899 reporter.log('COM OUTPUT: %s' % (sFile[self.cchDisplayed:offNewline]))
3900 self.cchDisplayed = offNewline + 1;
3901 offNewline = sFile.find('\n', self.cchDisplayed);
3902
3903 fRet = self.fSignalled;
3904 if not fLocked:
3905 self.unlockTask();
3906 return fRet;
3907
3908 # Our stuff.
3909 def getResult(self):
3910 """
3911 Returns the connected TXS session object on success.
3912 Returns None on failure or if the task has not yet completed.
3913 """
3914 self.oCv.acquire();
3915 sResult = self.sResult;
3916 self.oCv.release();
3917 return sResult;
3918
3919 def cancelTask(self):
3920 """ Cancels the task. """
3921 self.signalTask();
3922 return True;
3923
3924
3925 def monitorComRawFile(self, oSession, sComRawFile, cMsTimeout = 15*60000, asStopWords = None):
3926 """
3927 Monitors the COM output file for stop words (PASSED and FAILED by default).
3928
3929 Returns the stop word.
3930 Returns None on VM error and timeout.
3931 """
3932
3933 reporter.log2('monitorComRawFile: oSession=%s, cMsTimeout=%s, sComRawFile=%s' % (oSession, cMsTimeout, sComRawFile));
3934
3935 oMonitorTask = self.TxsMonitorComFile(sComRawFile, asStopWords);
3936 self.addTask(oMonitorTask);
3937
3938 cMsTimeout = self.adjustTimeoutMs(cMsTimeout);
3939 oTask = self.waitForTasks(cMsTimeout + 1);
3940 reporter.log2('monitorComRawFile: waitForTasks returned %s' % (oTask,));
3941
3942 if oTask is not oMonitorTask:
3943 oMonitorTask.cancelTask();
3944 self.removeTask(oMonitorTask);
3945
3946 oMonitorTask.pollTask();
3947 return oMonitorTask.getResult();
3948
3949
3950 def runVmAndMonitorComRawFile(self, sVmName, sComRawFile, cMsTimeout = 15*60000, asStopWords = None):
3951 """
3952 Runs the specified VM and monitors the given COM output file for stop
3953 words (PASSED and FAILED by default).
3954
3955 The caller is assumed to have configured the VM to use the given
3956 file. The method will take no action to verify this.
3957
3958 Returns the stop word.
3959 Returns None on VM error and timeout.
3960 """
3961
3962 # Start the VM.
3963 reporter.log('runVmAndMonitorComRawFile: Starting(/preparing) "%s" (timeout %s s)...' % (sVmName, cMsTimeout / 1000));
3964 reporter.flushall();
3965 oSession = self.startVmByName(sVmName);
3966 if oSession is not None:
3967 # Let it run and then terminate it.
3968 sRet = self.monitorComRawFile(oSession, sComRawFile, cMsTimeout, asStopWords);
3969 self.terminateVmBySession(oSession);
3970 else:
3971 sRet = None;
3972 return sRet;
3973
3974 #
3975 # Other stuff
3976 #
3977
3978 def waitForGAs(self,
3979 oSession, # type: vboxwrappers.SessionWrapper
3980 cMsTimeout = 120000, aenmWaitForRunLevels = None, aenmWaitForActive = None, aenmWaitForInactive = None):
3981 """
3982 Waits for the guest additions to enter a certain state.
3983
3984 aenmWaitForRunLevels - List of run level values to wait for (success if one matches).
3985 aenmWaitForActive - List facilities (type values) that must be active.
3986 aenmWaitForInactive - List facilities (type values) that must be inactive.
3987
3988 Defaults to wait for AdditionsRunLevelType_Userland if nothing else is given.
3989
3990 Returns True on success, False w/ error logging on timeout or failure.
3991 """
3992 reporter.log2('waitForGAs: oSession=%s, cMsTimeout=%s' % (oSession, cMsTimeout,));
3993
3994 #
3995 # Get IGuest:
3996 #
3997 try:
3998 oIGuest = oSession.o.console.guest;
3999 except:
4000 return reporter.errorXcpt();
4001
4002 #
4003 # Create a wait task:
4004 #
4005 from testdriver.vboxwrappers import AdditionsStatusTask;
4006 try:
4007 oGaStatusTask = AdditionsStatusTask(oSession = oSession,
4008 oIGuest = oIGuest,
4009 cMsTimeout = cMsTimeout,
4010 aenmWaitForRunLevels = aenmWaitForRunLevels,
4011 aenmWaitForActive = aenmWaitForActive,
4012 aenmWaitForInactive = aenmWaitForInactive);
4013 except:
4014 return reporter.errorXcpt();
4015
4016 #
4017 # Add the task and make sure the VM session is also present.
4018 #
4019 self.addTask(oGaStatusTask);
4020 fRemoveSession = self.addTask(oSession);
4021 oTask = self.waitForTasks(cMsTimeout + 1);
4022 reporter.log2('waitForGAs: returned %s (oGaStatusTask=%s, oSession=%s)' % (oTask, oGaStatusTask, oSession,));
4023 self.removeTask(oGaStatusTask);
4024 if fRemoveSession:
4025 self.removeTask(oSession);
4026
4027 #
4028 # Digest the result.
4029 #
4030 if oTask is oGaStatusTask:
4031 fSucceeded = oGaStatusTask.getResult();
4032 if fSucceeded is True:
4033 reporter.log('waitForGAs: Succeeded.');
4034 else:
4035 reporter.error('waitForGAs: Failed.');
4036 else:
4037 oGaStatusTask.cancelTask();
4038 if oTask is None:
4039 reporter.error('waitForGAs: Timed out.');
4040 elif oTask is oSession:
4041 oSession.reportPrematureTermination('waitForGAs: ');
4042 else:
4043 reporter.error('waitForGAs: unknown/wrong task %s' % (oTask,));
4044 fSucceeded = False;
4045 return fSucceeded;
4046
4047 @staticmethod
4048 def controllerTypeToName(eControllerType):
4049 """
4050 Translate a controller type to a standard controller name.
4051 """
4052 if eControllerType in (vboxcon.StorageControllerType_PIIX3, vboxcon.StorageControllerType_PIIX4,):
4053 sName = "IDE Controller";
4054 elif eControllerType == vboxcon.StorageControllerType_IntelAhci:
4055 sName = "SATA Controller";
4056 elif eControllerType == vboxcon.StorageControllerType_LsiLogicSas:
4057 sName = "SAS Controller";
4058 elif eControllerType in (vboxcon.StorageControllerType_LsiLogic, vboxcon.StorageControllerType_BusLogic,):
4059 sName = "SCSI Controller";
4060 elif eControllerType == vboxcon.StorageControllerType_NVMe:
4061 sName = "NVMe Controller";
4062 else:
4063 sName = "Storage Controller";
4064 return sName;
4065
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