VirtualBox

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

Last change on this file since 62497 was 62484, checked in by vboxsync, 8 years ago

(C) 2016

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 113.9 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: vboxwrappers.py 62484 2016-07-22 18:35:33Z vboxsync $
3# pylint: disable=C0302
4
5"""
6VirtualBox Wrapper Classes
7"""
8
9__copyright__ = \
10"""
11Copyright (C) 2010-2016 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: 62484 $"
31
32
33# Standard Python imports.
34import array
35import os
36import socket
37
38# Validation Kit imports.
39from common import utils;
40from testdriver import base;
41from testdriver import reporter;
42from testdriver import txsclient;
43from testdriver import vboxcon;
44from testdriver import vbox;
45from testdriver.base import TdTaskBase;
46
47
48def _ControllerNameToBus(sController):
49 """ Translate a controller name to a storage bus. """
50 if sController == "IDE Controller":
51 iType = vboxcon.StorageBus_IDE;
52 elif sController == "SATA Controller":
53 iType = vboxcon.StorageBus_SATA;
54 elif sController == "Floppy Controller":
55 iType = vboxcon.StorageBus_Floppy;
56 elif sController == "SAS Controller":
57 iType = vboxcon.StorageBus_SAS;
58 elif sController == "SCSI Controller":
59 iType = vboxcon.StorageBus_SCSI;
60 elif sController == "NVMe Controller":
61 iType = vboxcon.StorageBus_PCIe;
62 else:
63 iType = vboxcon.StorageBus_Null;
64 return iType;
65
66def _nameMachineState(eState):
67 """ Gets the name (string) of a machine state."""
68 if eState == vboxcon.MachineState_PoweredOff: return 'PoweredOff';
69 if eState == vboxcon.MachineState_Saved: return 'Saved';
70 if eState == vboxcon.MachineState_Teleported: return 'Teleported';
71 if eState == vboxcon.MachineState_Aborted: return 'Aborted';
72 if eState == vboxcon.MachineState_Running: return 'Running';
73 if eState == vboxcon.MachineState_Paused: return 'Paused';
74 if eState == vboxcon.MachineState_Stuck: return 'GuruMeditation';
75 if eState == vboxcon.MachineState_Teleporting: return 'Teleporting';
76 if eState == vboxcon.MachineState_LiveSnapshotting: return 'LiveSnapshotting';
77 if eState == vboxcon.MachineState_Starting: return 'Starting';
78 if eState == vboxcon.MachineState_Stopping: return 'Stopping';
79 if eState == vboxcon.MachineState_Saving: return 'Saving';
80 if eState == vboxcon.MachineState_Restoring: return 'Restoring';
81 if eState == vboxcon.MachineState_TeleportingPausedVM: return 'TeleportingPausedVM';
82 if eState == vboxcon.MachineState_TeleportingIn: return 'TeleportingIn';
83 if eState == vboxcon.MachineState_FaultTolerantSyncing: return 'FaultTolerantSyncing';
84 if eState == vboxcon.MachineState_DeletingSnapshotOnline: return 'DeletingSnapshotOnline';
85 if eState == vboxcon.MachineState_DeletingSnapshotPaused: return 'DeletingSnapshotPaused';
86 if eState == vboxcon.MachineState_RestoringSnapshot: return 'RestoringSnapshot';
87 if eState == vboxcon.MachineState_DeletingSnapshot: return 'DeletingSnapshot';
88 if eState == vboxcon.MachineState_SettingUp: return 'SettingUp';
89 return 'Unknown-%s' % (eState,);
90
91
92class VirtualBoxWrapper(object): # pylint: disable=R0903
93 """
94 Wrapper around the IVirtualBox object that adds some (hopefully) useful
95 utility methods
96
97 The real object can be accessed thru the o member. That said, members can
98 be accessed directly as well.
99 """
100
101 def __init__(self, oVBox, oVBoxMgr, fpApiVer, oTstDrv):
102 self.o = oVBox;
103 self.oVBoxMgr = oVBoxMgr;
104 self.fpApiVer = fpApiVer;
105 self.oTstDrv = oTstDrv;
106
107 def __getattr__(self, sName):
108 # Try ourselves first.
109 try:
110 oAttr = self.__dict__[sName];
111 except:
112 #try:
113 # oAttr = dir(self)[sName];
114 #except AttributeError:
115 oAttr = getattr(self.o, sName);
116 return oAttr;
117
118 #
119 # Utilities.
120 #
121
122 def registerDerivedEventHandler(self, oSubClass, dArgs = None):
123 """
124 Create an instance of the given VirtualBoxEventHandlerBase sub-class
125 and register it.
126
127 The new instance is returned on success. None is returned on error.
128 """
129 dArgsCopy = dArgs.copy() if dArgs is not None else dict();
130 dArgsCopy['oVBox'] = self;
131 return oSubClass.registerDerivedEventHandler(self.oVBoxMgr, self.fpApiVer, oSubClass, dArgsCopy,
132 self.o, 'IVirtualBox', 'IVirtualBoxCallback');
133
134 def deleteHdByLocation(self, sHdLocation):
135 """
136 Deletes a disk image from the host, given it's location.
137 Returns True on success and False on failure. Error information is logged.
138 """
139 try:
140 oIMedium = self.o.findHardDisk(sHdLocation);
141 except:
142 try:
143 if self.fpApiVer >= 4.1:
144 oIMedium = self.o.openMedium(sHdLocation, vboxcon.DeviceType_HardDisk,
145 vboxcon.AccessMode_ReadWrite, False);
146 elif self.fpApiVer >= 4.0:
147 oIMedium = self.o.openMedium(sHdLocation, vboxcon.DeviceType_HardDisk,
148 vboxcon.AccessMode_ReadWrite);
149 else:
150 oIMedium = self.o.openHardDisk(sHdLocation, vboxcon.AccessMode_ReadOnly, False, "", False, "");
151 except:
152 return reporter.errorXcpt('failed to open hd "%s"' % (sHdLocation));
153 return self.deleteHdByMedium(oIMedium)
154
155 def deleteHdByMedium(self, oIMedium):
156 """
157 Deletes a disk image from the host, given an IMedium reference.
158 Returns True on success and False on failure. Error information is logged.
159 """
160 try: oProgressCom = oIMedium.deleteStorage();
161 except: return reporter.errorXcpt('deleteStorage() for disk %s failed' % (oIMedium,));
162 try: oProgress = ProgressWrapper(oProgressCom, self.oVBoxMgr, self.oTstDrv, 'delete disk %s' % (oIMedium.location));
163 except: return reporter.errorXcpt();
164 oProgress.wait();
165 oProgress.logResult();
166 return oProgress.isSuccess();
167
168
169
170class ProgressWrapper(TdTaskBase):
171 """
172 Wrapper around a progress object for making it a task and providing useful
173 utility methods.
174 The real progress object can be accessed thru the o member.
175 """
176
177 def __init__(self, oProgress, oVBoxMgr, oTstDrv, sName):
178 TdTaskBase.__init__(self, utils.getCallerName());
179 self.o = oProgress;
180 self.oVBoxMgr = oVBoxMgr;
181 self.oTstDrv = oTstDrv;
182 self.sName = sName;
183
184 def toString(self):
185 return '<%s sName=%s, oProgress=%s >' \
186 % (TdTaskBase.toString(self), self.sName, self.o);
187
188 #
189 # TdTaskBase overrides.
190 #
191
192 def pollTask(self, fLocked = False):
193 """
194 Overrides TdTaskBase.pollTask().
195
196 This method returns False until the progress object has completed.
197 """
198 self.doQuickApiTest();
199 try:
200 try:
201 if self.o.completed:
202 return True;
203 except:
204 pass;
205 finally:
206 self.oTstDrv.processPendingEvents();
207 return False;
208
209 def waitForTask(self, cMsTimeout = 0):
210 """
211 Overrides TdTaskBase.waitForTask().
212 Process XPCOM/COM events while waiting.
213 """
214 msStart = base.timestampMilli();
215 fState = self.pollTask(False);
216 while not fState:
217 cMsElapsed = base.timestampMilli() - msStart;
218 if cMsElapsed > cMsTimeout:
219 break;
220 cMsToWait = cMsTimeout - cMsElapsed;
221 if cMsToWait > 500:
222 cMsToWait = 500;
223 try:
224 self.o.waitForCompletion(cMsToWait);
225 except KeyboardInterrupt: raise;
226 except: pass;
227 reporter.doPollWork('ProgressWrapper.waitForTask');
228 fState = self.pollTask(False);
229 return fState;
230
231 #
232 # Utility methods.
233 #
234
235 def isSuccess(self):
236 """
237 Tests if the progress object completed successfully.
238 Returns True on success, False on failure or incomplete.
239 """
240 if not self.isCompleted():
241 return False;
242 return self.getResult() >= 0;
243
244 def isCompleted(self):
245 """
246 Wrapper around IProgress.completed.
247 """
248 return self.pollTask();
249
250 def isCancelable(self):
251 """
252 Wrapper around IProgress.cancelable.
253 """
254 try:
255 fRc = self.o.cancelable;
256 except:
257 reporter.logXcpt();
258 fRc = False;
259 return fRc;
260
261 def wasCanceled(self):
262 """
263 Wrapper around IProgress.canceled.
264 """
265 try:
266 fRc = self.o.canceled;
267 except:
268 reporter.logXcpt(self.sName);
269 fRc = False;
270 return fRc;
271
272 def cancel(self):
273 """
274 Wrapper around IProgress.cancel()
275 Returns True on success, False on failure (logged as error).
276 """
277 try:
278 self.o.cancel();
279 except:
280 reporter.errorXcpt(self.sName);
281 return False;
282 return True;
283
284 def getResult(self):
285 """
286 Wrapper around IProgress.resultCode.
287 """
288 try:
289 iRc = self.o.resultCode;
290 except:
291 reporter.logXcpt(self.sName);
292 iRc = -1;
293 return iRc;
294
295 def getErrInfoResultCode(self):
296 """
297 Wrapper around IProgress.errorInfo.resultCode.
298
299 Returns the string on success, -1 on bad objects (logged as error), and
300 -2 on missing errorInfo object.
301 """
302 iRc = -1;
303 try:
304 oErrInfo = self.o.errorInfo;
305 except:
306 reporter.errorXcpt(self.sName);
307 else:
308 if oErrInfo is None:
309 iRc = -2;
310 else:
311 try:
312 iRc = oErrInfo.resultCode;
313 except:
314 reporter.errorXcpt();
315 return iRc;
316
317 def getErrInfoText(self):
318 """
319 Wrapper around IProgress.errorInfo.text.
320
321 Returns the string on success, None on failure. Missing errorInfo is
322 not logged as an error, all other failures are.
323 """
324 sText = None;
325 try:
326 oErrInfo = self.o.errorInfo;
327 except:
328 reporter.log2Xcpt(self.sName);
329 else:
330 if oErrInfo is not None:
331 try:
332 sText = oErrInfo.text;
333 except:
334 reporter.errorXcpt();
335 return sText;
336
337 def stringifyErrorInfo(self):
338 """
339 Formats IProgress.errorInfo into a string.
340 """
341 try:
342 oErrInfo = self.o.errorInfo;
343 except:
344 reporter.logXcpt(self.sName);
345 sErr = 'no error info';
346 else:
347 sErr = vbox.stringifyErrorInfo(oErrInfo);
348 return sErr;
349
350 def stringifyResult(self):
351 """
352 Stringify the result.
353 """
354 if self.isCompleted():
355 if self.wasCanceled():
356 sRet = 'Progress %s: Canceled, hrc=%s' % (self.sName, vbox.ComError.toString(self.getResult()));
357 elif self.getResult() == 0:
358 sRet = 'Progress %s: Success' % (self.sName,);
359 elif self.getResult() > 0:
360 sRet = 'Progress %s: Success (hrc=%s)' % (self.sName, vbox.ComError.toString(self.getResult()));
361 else:
362 sRet = 'Progress %s: Failed! %s' % (self.sName, self.stringifyErrorInfo());
363 else:
364 sRet = 'Progress %s: Not completed yet...' % (self.sName);
365 return sRet;
366
367 def logResult(self, fIgnoreErrors = False):
368 """ Logs the result. """
369 sText = self.stringifyResult();
370 if self.isCompleted() and self.getResult() < 0 \
371 and fIgnoreErrors is False:
372 return reporter.error(sText);
373 return reporter.log(sText);
374
375 def waitOnProgress(self, cMsInterval = 1000):
376 """
377 See vbox.TestDriver.waitOnProgress.
378 """
379 self.doQuickApiTest();
380 return self.oTstDrv.waitOnProgress(self.o, cMsInterval);
381
382 def wait(self, cMsTimeout = 60000, fErrorOnTimeout = True, cMsInterval = 1000):
383 """
384 Wait on the progress object for a while.
385
386 Returns the resultCode of the progress object if completed.
387 Returns -1 on timeout, logged as error if fErrorOnTimeout is set.
388 Returns -2 is the progress object is invalid or waitForCompletion
389 fails (logged as errors).
390 """
391 msStart = base.timestampMilli();
392 while True:
393 self.oTstDrv.processPendingEvents();
394 self.doQuickApiTest();
395 try:
396 if self.o.completed:
397 break;
398 except:
399 reporter.errorXcpt(self.sName);
400 return -2;
401 self.oTstDrv.processPendingEvents();
402
403 cMsElapsed = base.timestampMilli() - msStart;
404 if cMsElapsed > cMsTimeout:
405 if fErrorOnTimeout:
406 reporter.error('Timing out after waiting for %u s on "%s"' % (cMsTimeout / 1000, self.sName))
407 return -1;
408
409 try:
410 self.o.waitForCompletion(cMsInterval);
411 except:
412 reporter.errorXcpt(self.sName);
413 return -2;
414 reporter.doPollWork('ProgressWrapper.wait');
415
416 try:
417 rc = self.o.resultCode;
418 except:
419 rc = -2;
420 reporter.errorXcpt(self.sName);
421 self.oTstDrv.processPendingEvents();
422 return rc;
423
424 def waitForOperation(self, iOperation, cMsTimeout = 60000, fErrorOnTimeout = True, cMsInterval = 1000, \
425 fIgnoreErrors = False):
426 """
427 Wait for the completion of a operation.
428
429 Negative iOperation values are relative to operationCount (this
430 property may changed at runtime).
431
432 Returns 0 if the operation completed normally.
433 Returns -1 on timeout, logged as error if fErrorOnTimeout is set.
434 Returns -2 is the progress object is invalid or waitForCompletion
435 fails (logged as errors).
436 Returns -3 if if the operation completed with an error, this is logged
437 as an error.
438 """
439 msStart = base.timestampMilli();
440 while True:
441 self.oTstDrv.processPendingEvents();
442 self.doQuickApiTest();
443 try:
444 iCurrentOperation = self.o.operation;
445 cOperations = self.o.operationCount;
446 if iOperation >= 0:
447 iRealOperation = iOperation;
448 else:
449 iRealOperation = cOperations + iOperation;
450
451 if iCurrentOperation > iRealOperation:
452 return 0;
453 if iCurrentOperation == iRealOperation \
454 and iRealOperation >= cOperations - 1 \
455 and self.o.completed:
456 if self.o.resultCode < 0:
457 self.logResult(fIgnoreErrors);
458 return -3;
459 return 0;
460 except:
461 if fIgnoreErrors:
462 reporter.logXcpt();
463 else:
464 reporter.errorXcpt();
465 return -2;
466 self.oTstDrv.processPendingEvents();
467
468 cMsElapsed = base.timestampMilli() - msStart;
469 if cMsElapsed > cMsTimeout:
470 if fErrorOnTimeout:
471 if fIgnoreErrors:
472 reporter.log('Timing out after waiting for %u s on "%s" operation %d' \
473 % (cMsTimeout / 1000, self.sName, iOperation))
474 else:
475 reporter.error('Timing out after waiting for %u s on "%s" operation %d' \
476 % (cMsTimeout / 1000, self.sName, iOperation))
477 return -1;
478
479 try:
480 self.o.waitForOperationCompletion(iRealOperation, cMsInterval);
481 except:
482 if fIgnoreErrors:
483 reporter.logXcpt(self.sName);
484 else:
485 reporter.errorXcpt(self.sName);
486 return -2;
487 reporter.doPollWork('ProgressWrapper.waitForOperation');
488 # Not reached.
489
490 def doQuickApiTest(self):
491 """
492 Queries everything that is stable and easy to get at and checks that
493 they don't throw errors.
494 """
495 if True is True:
496 try:
497 iPct = self.o.operationPercent;
498 sDesc = self.o.description;
499 fCancelable = self.o.cancelable;
500 cSecsRemain = self.o.timeRemaining;
501 fCanceled = self.o.canceled;
502 fCompleted = self.o.completed;
503 iOp = self.o.operation;
504 cOps = self.o.operationCount;
505 iOpPct = self.o.operationPercent;
506 sOpDesc = self.o.operationDescription;
507 except:
508 reporter.errorXcpt('%s: %s' % (self.sName, self.o,));
509 return False;
510 try:
511 # Very noisy -- only enable for debugging purposes.
512 #reporter.log2('%s: op=%u/%u/%s: %u%%; total=%u%% cancel=%s/%s compl=%s rem=%us; desc=%s' \
513 # % (self.sName, iOp, cOps, sOpDesc, iOpPct, iPct, fCanceled, fCancelable, fCompleted, \
514 # cSecsRemain, sDesc));
515 _ = iPct; _ = sDesc; _ = fCancelable; _ = cSecsRemain; _ = fCanceled; _ = fCompleted; _ = iOp;
516 _ = cOps; _ = iOpPct; _ = sOpDesc;
517 except:
518 reporter.errorXcpt();
519 return False;
520
521 return True;
522
523
524class SessionWrapper(TdTaskBase):
525 """
526 Wrapper around a machine session. The real session object can be accessed
527 thru the o member (short is good, right :-).
528 """
529
530 def __init__(self, oSession, oVM, oVBox, oVBoxMgr, oTstDrv, fRemoteSession, sFallbackName = None, sLogFile = None):
531 """
532 Initializes the session wrapper.
533 """
534 TdTaskBase.__init__(self, utils.getCallerName());
535 self.o = oSession;
536 self.oVBox = oVBox;
537 self.oVBoxMgr = oVBoxMgr;
538 self.oVM = oVM; # Not the session machine. Useful backdoor...
539 self.oTstDrv = oTstDrv;
540 self.fpApiVer = oTstDrv.fpApiVer;
541 self.fRemoteSession = fRemoteSession;
542 self.sLogFile = sLogFile;
543 self.oConsoleEventHandler = None;
544 self.uPid = None;
545 self.fHostMemoryLow = False; # see signalHostMemoryLow; read-only for outsiders.
546
547 try:
548 self.sName = oSession.machine.name;
549 except:
550 if sFallbackName is not None:
551 self.sName = sFallbackName;
552 else:
553 try: self.sName = str(oSession.machine);
554 except: self.sName = 'is-this-vm-already-off'
555
556 try:
557 self.sUuid = oSession.machine.id;
558 except:
559 self.sUuid = None;
560
561 # Try cache the SessionPID.
562 self.getPid();
563
564 def __del__(self):
565 """
566 Destructor that makes sure the callbacks are deregistered and
567 that the session is closed.
568 """
569 self.deregisterEventHandlerForTask();
570
571 if self.o is not None:
572 try:
573 self.close();
574 reporter.log('close session %s' % (self.o));
575 except:
576 pass;
577 self.o = None;
578
579 TdTaskBase.__del__(self);
580
581 def toString(self):
582 return '<%s: sUuid=%s, sName=%s, uPid=%s, sDbgCreated=%s, fRemoteSession=%s, oSession=%s,' \
583 ' oConsoleEventHandler=%s, oVM=%s >' \
584 % (type(self).__name__, self.sUuid, self.sName, self.uPid, self.sDbgCreated, self.fRemoteSession,
585 self.o, self.oConsoleEventHandler, self.oVM,);
586
587 def __str__(self):
588 return self.toString();
589
590 #
591 # TdTaskBase overrides.
592 #
593
594 def __pollTask(self):
595 """ Internal poller """
596 # Poll for events after doing the remote GetState call, otherwise we
597 # might end up sleepless because XPCOM queues a cleanup event.
598 try:
599 try:
600 eState = self.o.machine.state;
601 except Exception, oXcpt:
602 if vbox.ComError.notEqual(oXcpt, vbox.ComError.E_UNEXPECTED):
603 reporter.logXcpt();
604 return True;
605 finally:
606 self.oTstDrv.processPendingEvents();
607
608 # Switch
609 if eState == vboxcon.MachineState_Running:
610 return False;
611 if eState == vboxcon.MachineState_Paused:
612 return False;
613 if eState == vboxcon.MachineState_Teleporting:
614 return False;
615 if eState == vboxcon.MachineState_LiveSnapshotting:
616 return False;
617 if eState == vboxcon.MachineState_Starting:
618 return False;
619 if eState == vboxcon.MachineState_Stopping:
620 return False;
621 if eState == vboxcon.MachineState_Saving:
622 return False;
623 if eState == vboxcon.MachineState_Restoring:
624 return False;
625 if eState == vboxcon.MachineState_TeleportingPausedVM:
626 return False;
627 if eState == vboxcon.MachineState_TeleportingIn:
628 return False;
629
630 # *Beeep* fudge!
631 if self.fpApiVer < 3.2 \
632 and eState == vboxcon.MachineState_PoweredOff \
633 and self.getAgeAsMs() < 3000:
634 return False;
635
636 reporter.log('SessionWrapper::pollTask: eState=%s' % (eState));
637 return True;
638
639
640 def pollTask(self, fLocked = False):
641 """
642 Overrides TdTaskBase.pollTask().
643
644 This method returns False while the VM is online and running normally.
645 """
646
647 # Call super to check if the task was signalled by runtime error or similar,
648 # if not then check the VM state via __pollTask.
649 fRc = super(SessionWrapper, self).pollTask(fLocked);
650 if not fRc:
651 fRc = self.__pollTask();
652
653 # HACK ALERT: Lazily try registering the console event handler if
654 # we're not ready.
655 if not fRc and self.oConsoleEventHandler is None:
656 self.registerEventHandlerForTask();
657
658 # HACK ALERT: Lazily try get the PID and add it to the PID file.
659 if not fRc and self.uPid is None:
660 self.getPid();
661
662 return fRc;
663
664 def waitForTask(self, cMsTimeout = 0):
665 """
666 Overrides TdTaskBase.waitForTask().
667 Process XPCOM/COM events while waiting.
668 """
669 msStart = base.timestampMilli();
670 fState = self.pollTask(False);
671 while not fState:
672 cMsElapsed = base.timestampMilli() - msStart;
673 if cMsElapsed > cMsTimeout:
674 break;
675 cMsSleep = cMsTimeout - cMsElapsed;
676 if cMsSleep > 10000:
677 cMsSleep = 10000;
678 try: self.oVBoxMgr.waitForEvents(cMsSleep);
679 except KeyboardInterrupt: raise;
680 except: pass;
681 reporter.doPollWork('SessionWrapper.waitForTask');
682 fState = self.pollTask(False);
683 return fState;
684
685 def setTaskOwner(self, oOwner):
686 """
687 HACK ALERT!
688 Overrides TdTaskBase.setTaskOwner() so we can try call
689 registerEventHandlerForTask() again when when the testdriver calls
690 addTask() after VM has been spawned. Related to pollTask() above.
691
692 The testdriver must not add the task too early for this to work!
693 """
694 if oOwner is not None:
695 self.registerEventHandlerForTask()
696 return TdTaskBase.setTaskOwner(self, oOwner);
697
698
699 #
700 # Task helpers.
701 #
702
703 def registerEventHandlerForTask(self):
704 """
705 Registers the console event handlers for working the task state.
706 """
707 if self.oConsoleEventHandler is not None:
708 return True;
709 self.oConsoleEventHandler = self.registerDerivedEventHandler(vbox.SessionConsoleEventHandler, {}, False);
710 return self.oConsoleEventHandler is not None;
711
712 def deregisterEventHandlerForTask(self):
713 """
714 Deregisters the console event handlers.
715 """
716 if self.oConsoleEventHandler is not None:
717 self.oConsoleEventHandler.unregister();
718 self.oConsoleEventHandler = None;
719
720 def signalHostMemoryLow(self):
721 """
722 Used by a runtime error event handler to indicate that we're low on memory.
723 Signals the task.
724 """
725 self.fHostMemoryLow = True;
726 self.signalTask();
727 return True;
728
729 def needsPoweringOff(self):
730 """
731 Examins the machine state to see if the VM needs powering off.
732 """
733 try:
734 try:
735 eState = self.o.machine.state;
736 except Exception, oXcpt:
737 if vbox.ComError.notEqual(oXcpt, vbox.ComError.E_UNEXPECTED):
738 reporter.logXcpt();
739 return False;
740 finally:
741 self.oTstDrv.processPendingEvents();
742
743 # Switch
744 if eState == vboxcon.MachineState_Running:
745 return True;
746 if eState == vboxcon.MachineState_Paused:
747 return True;
748 if eState == vboxcon.MachineState_Stuck:
749 return True;
750 if eState == vboxcon.MachineState_Teleporting:
751 return True;
752 if eState == vboxcon.MachineState_LiveSnapshotting:
753 return True;
754 if eState == vboxcon.MachineState_Starting:
755 return True;
756 if eState == vboxcon.MachineState_Saving:
757 return True;
758 if eState == vboxcon.MachineState_Restoring:
759 return True;
760 if eState == vboxcon.MachineState_TeleportingPausedVM:
761 return True;
762 if eState == vboxcon.MachineState_TeleportingIn:
763 return True;
764 if eState == vboxcon.MachineState_FaultTolerantSyncing:
765 return True;
766 return False;
767
768 def assertPoweredOff(self):
769 """
770 Asserts that the VM is powered off, reporting an error if not.
771 Returns True if powered off, False + error msg if not.
772 """
773 try:
774 try:
775 eState = self.oVM.state;
776 except Exception:
777 reporter.errorXcpt();
778 return True;
779 finally:
780 self.oTstDrv.processPendingEvents();
781
782 if eState == vboxcon.MachineState_PoweredOff:
783 return True;
784 reporter.error('Expected machine state "PoweredOff", machine is in the "%s" state instead.'
785 % (_nameMachineState(eState),));
786 return False;
787
788 def getMachineStateWithName(self):
789 """
790 Gets the current machine state both as a constant number/whatever and
791 as a human readable string. On error, the constants will be set to
792 None and the string will be the error message.
793 """
794 try:
795 eState = self.oVM.state;
796 except:
797 return (None, '[error getting state: %s]' % (self.oVBoxMgr.xcptToString(),));
798 finally:
799 self.oTstDrv.processPendingEvents();
800 return (eState, _nameMachineState(eState));
801
802 def reportPrematureTermination(self, sPrefix = ''):
803 """
804 Reports a premature virtual machine termination.
805 Returns False to facilitate simpler error paths.
806 """
807
808 reporter.error(sPrefix + 'The virtual machine terminated prematurely!!');
809 (enmState, sStateNm) = self.getMachineStateWithName();
810 reporter.error(sPrefix + 'Machine state: %s' % (sStateNm,));
811
812 if enmState is not None \
813 and enmState == vboxcon.MachineState_Aborted \
814 and self.uPid is not None:
815 #
816 # Look for process crash info.
817 #
818 def addCrashFile(sLogFile, fBinary):
819 """ processCollectCrashInfo callback. """
820 reporter.addLogFile(sLogFile, 'crash/dump/vm' if fBinary else 'crash/report/vm');
821 utils.processCollectCrashInfo(self.uPid, reporter.log, addCrashFile);
822
823 return False;
824
825
826
827 #
828 # ISession / IMachine / ISomethingOrAnother wrappers.
829 #
830
831 def close(self):
832 """
833 Closes the session if it's open and removes it from the
834 vbox.TestDriver.aoRemoteSessions list.
835 Returns success indicator.
836 """
837 fRc = True;
838 if self.o is not None:
839 # Get the pid in case we need to kill the process later on.
840 self.getPid();
841
842 # Try close it.
843 try:
844 if self.fpApiVer < 3.3:
845 self.o.close();
846 else:
847 self.o.unlockMachine();
848 self.o = None;
849 except KeyboardInterrupt:
850 raise;
851 except:
852 # Kludge to ignore VBoxSVC's closing of our session when the
853 # direct session closes / VM process terminates. Fun!
854 try: fIgnore = self.o.state == vboxcon.SessionState_Unlocked;
855 except: fIgnore = False;
856 if not fIgnore:
857 reporter.errorXcpt('ISession::unlockMachine failed on %s' % (self.o));
858 fRc = False;
859
860 # Remove it from the remote session list if applicable (not 100% clean).
861 if fRc and self.fRemoteSession:
862 try:
863 if self in self.oTstDrv.aoRemoteSessions:
864 reporter.log2('SessionWrapper::close: Removing myself from oTstDrv.aoRemoteSessions');
865 self.oTstDrv.aoRemoteSessions.remove(self)
866 except:
867 reporter.logXcpt();
868
869 if self.uPid is not None:
870 self.oTstDrv.pidFileRemove(self.uPid);
871
872 self.oTstDrv.processPendingEvents();
873 return fRc;
874
875 def saveSettings(self, fClose = False):
876 """
877 Saves the settings and optionally closes the session.
878 Returns success indicator.
879 """
880 try:
881 try:
882 self.o.machine.saveSettings();
883 except:
884 reporter.errorXcpt('saveSettings failed on %s' % (self.o));
885 return False;
886 finally:
887 self.oTstDrv.processPendingEvents();
888 if fClose:
889 return self.close();
890 return True;
891
892 def discardSettings(self, fClose = False):
893 """
894 Discards the settings and optionally closes the session.
895 """
896 try:
897 try:
898 self.o.machine.discardSettings();
899 except:
900 reporter.errorXcpt('discardSettings failed on %s' % (self.o));
901 return False;
902 finally:
903 self.oTstDrv.processPendingEvents();
904 if fClose:
905 return self.close();
906 return True;
907
908 def enableVirtEx(self, fEnable):
909 """
910 Enables or disables AMD-V/VT-x.
911 Returns True on success and False on failure. Error information is logged.
912 """
913 # Enable/disable it.
914 fRc = True;
915 try:
916 self.o.machine.setHWVirtExProperty(vboxcon.HWVirtExPropertyType_Enabled, fEnable);
917 except:
918 reporter.errorXcpt('failed to set HWVirtExPropertyType_Enabled=%s for "%s"' % (fEnable, self.sName));
919 fRc = False;
920 else:
921 reporter.log('set HWVirtExPropertyType_Enabled=%s for "%s"' % (fEnable, self.sName));
922
923 # Force/unforce it.
924 if fRc and hasattr(vboxcon, 'HWVirtExPropertyType_Force'):
925 try:
926 self.o.machine.setHWVirtExProperty(vboxcon.HWVirtExPropertyType_Force, fEnable);
927 except:
928 reporter.errorXcpt('failed to set HWVirtExPropertyType_Force=%s for "%s"' % (fEnable, self.sName));
929 fRc = False;
930 else:
931 reporter.log('set HWVirtExPropertyType_Force=%s for "%s"' % (fEnable, self.sName));
932 else:
933 reporter.log('Warning! vboxcon has no HWVirtExPropertyType_Force attribute.');
934 ## @todo Modify CFGM to do the same for old VBox versions?
935
936 self.oTstDrv.processPendingEvents();
937 return fRc;
938
939 def enableNestedPaging(self, fEnable):
940 """
941 Enables or disables nested paging..
942 Returns True on success and False on failure. Error information is logged.
943 """
944 ## @todo Add/remove force CFGM thing, we don't want fallback logic when testing.
945 fRc = True;
946 try:
947 self.o.machine.setHWVirtExProperty(vboxcon.HWVirtExPropertyType_NestedPaging, fEnable);
948 except:
949 reporter.errorXcpt('failed to set HWVirtExPropertyType_NestedPaging=%s for "%s"' % (fEnable, self.sName));
950 fRc = False;
951 else:
952 reporter.log('set HWVirtExPropertyType_NestedPaging=%s for "%s"' % (fEnable, self.sName));
953 self.oTstDrv.processPendingEvents();
954 return fRc;
955
956 def enableLongMode(self, fEnable):
957 """
958 Enables or disables LongMode.
959 Returns True on success and False on failure. Error information is logged.
960 """
961 # Supported.
962 if self.fpApiVer < 4.2 or not hasattr(vboxcon, 'HWVirtExPropertyType_LongMode'):
963 return True;
964
965 # Enable/disable it.
966 fRc = True;
967 try:
968 self.o.machine.setCPUProperty(vboxcon.CPUPropertyType_LongMode, fEnable);
969 except:
970 reporter.errorXcpt('failed to set CPUPropertyType_LongMode=%s for "%s"' % (fEnable, self.sName));
971 fRc = False;
972 else:
973 reporter.log('set CPUPropertyType_LongMode=%s for "%s"' % (fEnable, self.sName));
974 self.oTstDrv.processPendingEvents();
975 return fRc;
976
977 def enablePae(self, fEnable):
978 """
979 Enables or disables PAE
980 Returns True on success and False on failure. Error information is logged.
981 """
982 fRc = True;
983 try:
984 if self.fpApiVer >= 3.2: # great, ain't it?
985 self.o.machine.setCPUProperty(vboxcon.CPUPropertyType_PAE, fEnable);
986 else:
987 self.o.machine.setCpuProperty(vboxcon.CpuPropertyType_PAE, fEnable);
988 except:
989 reporter.errorXcpt('failed to set CPUPropertyType_PAE=%s for "%s"' % (fEnable, self.sName));
990 fRc = False;
991 else:
992 reporter.log('set CPUPropertyType_PAE=%s for "%s"' % (fEnable, self.sName));
993 self.oTstDrv.processPendingEvents();
994 return fRc;
995
996 def enableIoApic(self, fEnable):
997 """
998 Enables or disables the IO-APIC
999 Returns True on success and False on failure. Error information is logged.
1000 """
1001 fRc = True;
1002 try:
1003 self.o.machine.BIOSSettings.IOAPICEnabled = fEnable;
1004 except:
1005 reporter.errorXcpt('failed to set BIOSSettings.IOAPICEnabled=%s for "%s"' % (fEnable, self.sName));
1006 fRc = False;
1007 else:
1008 reporter.log('set BIOSSettings.IOAPICEnabled=%s for "%s"' % (fEnable, self.sName));
1009 self.oTstDrv.processPendingEvents();
1010 return fRc;
1011
1012 def enableHpet(self, fEnable):
1013 """
1014 Enables or disables the HPET
1015 Returns True on success and False on failure. Error information is logged.
1016 """
1017 fRc = True;
1018 try:
1019 if self.fpApiVer >= 4.2:
1020 self.o.machine.HPETEnabled = fEnable;
1021 else:
1022 self.o.machine.hpetEnabled = fEnable;
1023 except:
1024 reporter.errorXcpt('failed to set HpetEnabled=%s for "%s"' % (fEnable, self.sName));
1025 fRc = False;
1026 else:
1027 reporter.log('set HpetEnabled=%s for "%s"' % (fEnable, self.sName));
1028 self.oTstDrv.processPendingEvents();
1029 return fRc;
1030
1031 def enableUsbHid(self, fEnable):
1032 """
1033 Enables or disables the USB HID
1034 Returns True on success and False on failure. Error information is logged.
1035 """
1036 fRc = True;
1037 try:
1038 if fEnable:
1039 if self.fpApiVer >= 4.3:
1040 cOhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_OHCI);
1041 if cOhciCtls == 0:
1042 self.o.machine.addUSBController('OHCI', vboxcon.USBControllerType_OHCI);
1043 else:
1044 self.o.machine.usbController.enabled = True;
1045
1046 if self.fpApiVer >= 4.2:
1047 self.o.machine.pointingHIDType = vboxcon.PointingHIDType_ComboMouse;
1048 self.o.machine.keyboardHIDType = vboxcon.KeyboardHIDType_ComboKeyboard;
1049 else:
1050 self.o.machine.pointingHidType = vboxcon.PointingHidType_ComboMouse;
1051 self.o.machine.keyboardHidType = vboxcon.KeyboardHidType_ComboKeyboard;
1052 else:
1053 if self.fpApiVer >= 4.2:
1054 self.o.machine.pointingHIDType = vboxcon.PointingHIDType_PS2Mouse;
1055 self.o.machine.keyboardHIDType = vboxcon.KeyboardHIDType_PS2Keyboard;
1056 else:
1057 self.o.machine.pointingHidType = vboxcon.PointingHidType_PS2Mouse;
1058 self.o.machine.keyboardHidType = vboxcon.KeyboardHidType_PS2Keyboard;
1059 except:
1060 reporter.errorXcpt('failed to change UsbHid to %s for "%s"' % (fEnable, self.sName));
1061 fRc = False;
1062 else:
1063 reporter.log('changed UsbHid to %s for "%s"' % (fEnable, self.sName));
1064 self.oTstDrv.processPendingEvents();
1065 return fRc;
1066
1067 def enableUsbOhci(self, fEnable):
1068 """
1069 Enables or disables the USB OHCI controller
1070 Returns True on success and False on failure. Error information is logged.
1071 """
1072 fRc = True;
1073 try:
1074 if fEnable:
1075 if self.fpApiVer >= 4.3:
1076 cOhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_OHCI);
1077 if cOhciCtls == 0:
1078 self.o.machine.addUSBController('OHCI', vboxcon.USBControllerType_OHCI);
1079 else:
1080 self.o.machine.usbController.enabled = True;
1081 else:
1082 if self.fpApiVer >= 4.3:
1083 cOhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_OHCI);
1084 if cOhciCtls == 1:
1085 self.o.machine.removeUSBController('OHCI');
1086 else:
1087 self.o.machine.usbController.enabled = False;
1088 except:
1089 reporter.errorXcpt('failed to change OHCI to %s for "%s"' % (fEnable, self.sName));
1090 fRc = False;
1091 else:
1092 reporter.log('changed OHCI to %s for "%s"' % (fEnable, self.sName));
1093 self.oTstDrv.processPendingEvents();
1094 return fRc;
1095
1096 def enableUsbEhci(self, fEnable):
1097 """
1098 Enables or disables the USB EHCI controller, enables also OHCI if it is still disabled.
1099 Returns True on success and False on failure. Error information is logged.
1100 """
1101 fRc = True;
1102 try:
1103 if fEnable:
1104 if self.fpApiVer >= 4.3:
1105 cOhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_OHCI);
1106 if cOhciCtls == 0:
1107 self.o.machine.addUSBController('OHCI', vboxcon.USBControllerType_OHCI);
1108
1109 cEhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_EHCI);
1110 if cEhciCtls == 0:
1111 self.o.machine.addUSBController('EHCI', vboxcon.USBControllerType_EHCI);
1112 else:
1113 self.o.machine.usbController.enabled = True;
1114 self.o.machine.usbController.enabledEHCI = True;
1115 else:
1116 if self.fpApiVer >= 4.3:
1117 cEhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_EHCI);
1118 if cEhciCtls == 1:
1119 self.o.machine.removeUSBController('EHCI');
1120 else:
1121 self.o.machine.usbController.enabledEHCI = False;
1122 except:
1123 reporter.errorXcpt('failed to change EHCI to %s for "%s"' % (fEnable, self.sName));
1124 fRc = False;
1125 else:
1126 reporter.log('changed EHCI to %s for "%s"' % (fEnable, self.sName));
1127 self.oTstDrv.processPendingEvents();
1128 return fRc;
1129
1130 def enableUsbXhci(self, fEnable):
1131 """
1132 Enables or disables the USB XHCI controller. Error information is logged.
1133 """
1134 fRc = True;
1135 try:
1136 if fEnable:
1137 cXhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_XHCI);
1138 if cXhciCtls == 0:
1139 self.o.machine.addUSBController('XHCI', vboxcon.USBControllerType_XHCI);
1140 else:
1141 cXhciCtls = self.o.machine.getUSBControllerCountByType(vboxcon.USBControllerType_XHCI);
1142 if cXhciCtls == 1:
1143 self.o.machine.removeUSBController('XHCI');
1144 except:
1145 reporter.errorXcpt('failed to change XHCI to %s for "%s"' % (fEnable, self.sName));
1146 fRc = False;
1147 else:
1148 reporter.log('changed XHCI to %s for "%s"' % (fEnable, self.sName));
1149 self.oTstDrv.processPendingEvents();
1150 return fRc;
1151
1152 def setFirmwareType(self, eType):
1153 """
1154 Sets the firmware type.
1155 Returns True on success and False on failure. Error information is logged.
1156 """
1157 fRc = True;
1158 try:
1159 self.o.machine.firmwareType = eType;
1160 except:
1161 reporter.errorXcpt('failed to set firmwareType=%s for "%s"' % (eType, self.sName));
1162 fRc = False;
1163 else:
1164 reporter.log('set firmwareType=%s for "%s"' % (eType, self.sName));
1165 self.oTstDrv.processPendingEvents();
1166 return fRc;
1167
1168 def setupBootLogo(self, fEnable, cMsLogoDisplay = 0):
1169 """
1170 Sets up the boot logo. fEnable toggles the fade and boot menu
1171 settings as well as the mode.
1172 """
1173 fRc = True;
1174 try:
1175 self.o.machine.BIOSSettings.logoFadeIn = not fEnable;
1176 self.o.machine.BIOSSettings.logoFadeOut = not fEnable;
1177 self.o.machine.BIOSSettings.logoDisplayTime = cMsLogoDisplay;
1178 if fEnable:
1179 self.o.machine.BIOSSettings.bootMenuMode = vboxcon.BIOSBootMenuMode_Disabled;
1180 else:
1181 self.o.machine.BIOSSettings.bootMenuMode = vboxcon.BIOSBootMenuMode_MessageAndMenu;
1182 except:
1183 reporter.errorXcpt('failed to set logoFadeIn/logoFadeOut/bootMenuMode=%s for "%s"' % (fEnable, self.sName));
1184 fRc = False;
1185 else:
1186 reporter.log('set logoFadeIn/logoFadeOut/bootMenuMode=%s for "%s"' % (fEnable, self.sName));
1187 self.oTstDrv.processPendingEvents();
1188 return fRc;
1189
1190 def setupVrdp(self, fEnable, uPort = None):
1191 """
1192 Configures VRDP.
1193 """
1194 fRc = True;
1195 try:
1196 if self.fpApiVer >= 4.0:
1197 self.o.machine.VRDEServer.enabled = fEnable;
1198 else:
1199 self.o.machine.VRDPServer.enabled = fEnable;
1200 except:
1201 reporter.errorXcpt('failed to set VRDEServer::enabled=%s for "%s"' % (fEnable, self.sName));
1202 fRc = False;
1203
1204 if uPort is not None and fRc:
1205 try:
1206 if self.fpApiVer >= 4.0:
1207 self.o.machine.VRDEServer.setVRDEProperty("TCP/Ports", str(uPort));
1208 else:
1209 self.o.machine.VRDPServer.ports = str(uPort);
1210 except:
1211 reporter.errorXcpt('failed to set VRDEServer::ports=%s for "%s"' % (uPort, self.sName));
1212 fRc = False;
1213 if fRc:
1214 reporter.log('set VRDEServer.enabled/ports=%s/%s for "%s"' % (fEnable, uPort, self.sName));
1215 self.oTstDrv.processPendingEvents();
1216 return fRc;
1217
1218 def getNicDriverNameFromType(self, eNicType):
1219 """
1220 Helper that translate the adapter type into a driver name.
1221 """
1222 if eNicType == vboxcon.NetworkAdapterType_Am79C970A \
1223 or eNicType == vboxcon.NetworkAdapterType_Am79C973:
1224 sName = 'pcnet';
1225 elif eNicType == vboxcon.NetworkAdapterType_I82540EM \
1226 or eNicType == vboxcon.NetworkAdapterType_I82543GC \
1227 or eNicType == vboxcon.NetworkAdapterType_I82545EM:
1228 sName = 'e1000';
1229 elif eNicType == vboxcon.NetworkAdapterType_Virtio:
1230 sName = 'virtio-net';
1231 else:
1232 reporter.error('Unknown adapter type "%s" (VM: "%s")' % (eNicType, self.sName));
1233 sName = 'pcnet';
1234 return sName;
1235
1236 def setupNatForwardingForTxs(self, iNic = 0, iHostPort = 5042):
1237 """
1238 Sets up NAT forwarding for port 5042 if applicable, cleans up if not.
1239 """
1240 try:
1241 oNic = self.o.machine.getNetworkAdapter(iNic);
1242 except:
1243 reporter.errorXcpt('getNetworkAdapter(%s) failed for "%s"' % (iNic, self.sName));
1244 return False;
1245
1246 # Nuke the old setup for all possible adapter types (in case we're
1247 # called after it changed).
1248 for sName in ('pcnet', 'e1000', 'virtio-net'):
1249 for sConfig in ('VBoxInternal/Devices/%s/%u/LUN#0/AttachedDriver/Config' % (sName, iNic), \
1250 'VBoxInternal/Devices/%s/%u/LUN#0/Config' % (sName, iNic)):
1251 try:
1252 self.o.machine.setExtraData('%s/txs/Protocol' % (sConfig), '');
1253 self.o.machine.setExtraData('%s/txs/HostPort' % (sConfig), '');
1254 self.o.machine.setExtraData('%s/txs/GuestPort' % (sConfig), '');
1255 except:
1256 reporter.errorXcpt();
1257
1258 # Set up port forwarding if NAT attachment.
1259 try:
1260 eAttType = oNic.attachmentType;
1261 except:
1262 reporter.errorXcpt('attachmentType on %s failed for "%s"' % (iNic, self.sName));
1263 return False;
1264 if eAttType != vboxcon.NetworkAttachmentType_NAT:
1265 return True;
1266
1267 try:
1268 eNicType = oNic.adapterType;
1269 fTraceEnabled = oNic.traceEnabled;
1270 except:
1271 reporter.errorXcpt('attachmentType/traceEnabled on %s failed for "%s"' % (iNic, self.sName));
1272 return False;
1273
1274 if self.fpApiVer >= 4.1:
1275 try:
1276 if self.fpApiVer >= 4.2:
1277 oNatEngine = oNic.NATEngine;
1278 else:
1279 oNatEngine = oNic.natDriver;
1280 except:
1281 reporter.errorXcpt('Failed to get INATEngine data on "%s"' % (self.sName));
1282 return False;
1283 try: oNatEngine.removeRedirect('txs');
1284 except: pass;
1285 try:
1286 oNatEngine.addRedirect('txs', vboxcon.NATProtocol_TCP, '127.0.0.1', '%s' % (iHostPort), '', '5042');
1287 except:
1288 reporter.errorXcpt('Failed to add a addRedirect redirect on "%s"' % (self.sName));
1289 return False;
1290
1291 else:
1292 sName = self.getNicDriverNameFromType(eNicType);
1293 if fTraceEnabled:
1294 sConfig = 'VBoxInternal/Devices/%s/%u/LUN#0/AttachedDriver/Config' % (sName, iNic)
1295 else:
1296 sConfig = 'VBoxInternal/Devices/%s/%u/LUN#0/Config' % (sName, iNic)
1297
1298 try:
1299 self.o.machine.setExtraData('%s/txs/Protocol' % (sConfig), 'TCP');
1300 self.o.machine.setExtraData('%s/txs/HostPort' % (sConfig), '%s' % (iHostPort));
1301 self.o.machine.setExtraData('%s/txs/GuestPort' % (sConfig), '5042');
1302 except:
1303 reporter.errorXcpt('Failed to set NAT extra data on "%s"' % (self.sName));
1304 return False;
1305 return True;
1306
1307 def setNicType(self, eType, iNic = 0):
1308 """
1309 Sets the NIC type of the specified NIC.
1310 Returns True on success and False on failure. Error information is logged.
1311 """
1312 try:
1313 try:
1314 oNic = self.o.machine.getNetworkAdapter(iNic);
1315 except:
1316 reporter.errorXcpt('getNetworkAdapter(%s) failed for "%s"' % (iNic, self.sName));
1317 return False;
1318 try:
1319 oNic.adapterType = eType;
1320 except:
1321 reporter.errorXcpt('failed to set NIC type on slot %s to %s for VM "%s"' % (iNic, eType, self.sName));
1322 return False;
1323 finally:
1324 self.oTstDrv.processPendingEvents();
1325
1326 if not self.setupNatForwardingForTxs(iNic):
1327 return False;
1328 reporter.log('set NIC type on slot %s to %s for VM "%s"' % (iNic, eType, self.sName));
1329 return True;
1330
1331 def setNicTraceEnabled(self, fTraceEnabled, sTraceFile, iNic = 0):
1332 """
1333 Sets the NIC trace enabled flag and file path.
1334 Returns True on success and False on failure. Error information is logged.
1335 """
1336 try:
1337 try:
1338 oNic = self.o.machine.getNetworkAdapter(iNic);
1339 except:
1340 reporter.errorXcpt('getNetworkAdapter(%s) failed for "%s"' % (iNic, self.sName));
1341 return False;
1342 try:
1343 oNic.traceEnabled = fTraceEnabled;
1344 oNic.traceFile = sTraceFile;
1345 except:
1346 reporter.errorXcpt('failed to set NIC trace flag on slot %s to %s for VM "%s"' \
1347 % (iNic, fTraceEnabled, self.sName));
1348 return False;
1349 finally:
1350 self.oTstDrv.processPendingEvents();
1351
1352 if not self.setupNatForwardingForTxs(iNic):
1353 return False;
1354 reporter.log('set NIC trace on slot %s to "%s" (path "%s") for VM "%s"' %
1355 (iNic, fTraceEnabled, sTraceFile, self.sName));
1356 return True;
1357
1358 def getDefaultNicName(self, eAttachmentType):
1359 """
1360 Return the default network / interface name for the NIC attachment type.
1361 """
1362 sRetName = '';
1363 if eAttachmentType == vboxcon.NetworkAttachmentType_Bridged:
1364 if self.oTstDrv.sDefBridgedNic is not None:
1365 sRetName = self.oTstDrv.sDefBridgedNic;
1366 else:
1367 sRetName = 'eth0';
1368 try:
1369 aoHostNics = self.oVBoxMgr.getArray(self.oVBox.host, 'networkInterfaces');
1370 for oHostNic in aoHostNics:
1371 if oHostNic.interfaceType == vboxcon.HostNetworkInterfaceType_Bridged \
1372 and oHostNic.status == vboxcon.HostNetworkInterfaceStatus_Up:
1373 sRetName = oHostNic.name;
1374 break;
1375 except:
1376 reporter.errorXcpt();
1377 elif eAttachmentType == vboxcon.NetworkAttachmentType_HostOnly:
1378 try:
1379 aoHostNics = self.oVBoxMgr.getArray(self.oVBox.host, 'networkInterfaces');
1380 for oHostNic in aoHostNics:
1381 if oHostNic.interfaceType == vboxcon.HostNetworkInterfaceType_HostOnly:
1382 if oHostNic.status == vboxcon.HostNetworkInterfaceStatus_Up:
1383 sRetName = oHostNic.name;
1384 break;
1385 if sRetName == '':
1386 sRetName = oHostNic.name;
1387 except:
1388 reporter.errorXcpt();
1389 if sRetName == '':
1390 sRetName = 'HostInterfaceNetwork-vboxnet0';
1391 elif eAttachmentType == vboxcon.NetworkAttachmentType_Internal:
1392 sRetName = 'VBoxTest';
1393 elif eAttachmentType == vboxcon.NetworkAttachmentType_NAT:
1394 sRetName = '';
1395 else:
1396 reporter.error('eAttachmentType=%s is not known' % (eAttachmentType));
1397 return sRetName;
1398
1399 def setNicAttachment(self, eAttachmentType, sName = None, iNic = 0):
1400 """
1401 Sets the attachment type of the specified NIC.
1402 Returns True on success and False on failure. Error information is logged.
1403 """
1404 try:
1405 oNic = self.o.machine.getNetworkAdapter(iNic);
1406 except:
1407 reporter.errorXcpt('getNetworkAdapter(%s) failed for "%s"' % (iNic, self.sName));
1408 return False;
1409
1410 try:
1411 if eAttachmentType is not None:
1412 try:
1413 if self.fpApiVer >= 4.1:
1414 oNic.attachmentType = eAttachmentType;
1415 else:
1416 if eAttachmentType == vboxcon.NetworkAttachmentType_NAT:
1417 oNic.attachToNAT();
1418 elif eAttachmentType == vboxcon.NetworkAttachmentType_Bridged:
1419 oNic.attachToBridgedInterface();
1420 elif eAttachmentType == vboxcon.NetworkAttachmentType_Internal:
1421 oNic.attachToInternalNetwork();
1422 elif eAttachmentType == vboxcon.NetworkAttachmentType_HostOnly:
1423 oNic.attachToHostOnlyInterface();
1424 else:
1425 raise base.GenError("eAttachmentType=%s is invalid" % (eAttachmentType));
1426 except:
1427 reporter.errorXcpt('failed to set the attachment type on slot %s to %s for VM "%s"' \
1428 % (iNic, eAttachmentType, self.sName));
1429 return False;
1430 else:
1431 try:
1432 eAttachmentType = oNic.attachmentType;
1433 except:
1434 reporter.errorXcpt('failed to get the attachment type on slot %s for VM "%s"' % (iNic, self.sName));
1435 return False;
1436 finally:
1437 self.oTstDrv.processPendingEvents();
1438
1439 if sName is not None:
1440 # Resolve the special 'default' name.
1441 if sName == 'default':
1442 sName = self.getDefaultNicName(eAttachmentType);
1443
1444 # The name translate to different attributes depending on the
1445 # attachment type.
1446 try:
1447 if eAttachmentType == vboxcon.NetworkAttachmentType_Bridged:
1448 ## @todo check this out on windows, may have to do a
1449 # translation of the name there or smth IIRC.
1450 try:
1451 if self.fpApiVer >= 4.1:
1452 oNic.bridgedInterface = sName;
1453 else:
1454 oNic.hostInterface = sName;
1455 except:
1456 reporter.errorXcpt('failed to set the hostInterface property on slot %s to "%s" for VM "%s"' \
1457 % (iNic, sName, self.sName));
1458 return False;
1459 elif eAttachmentType == vboxcon.NetworkAttachmentType_HostOnly:
1460 try:
1461 if self.fpApiVer >= 4.1:
1462 oNic.hostOnlyInterface = sName;
1463 else:
1464 oNic.hostInterface = sName;
1465 except:
1466 reporter.errorXcpt('failed to set the internalNetwork property on slot %s to "%s" for VM "%s"' \
1467 % (iNic, sName, self.sName));
1468 return False;
1469 elif eAttachmentType == vboxcon.NetworkAttachmentType_Internal:
1470 try:
1471 oNic.internalNetwork = sName;
1472 except:
1473 reporter.errorXcpt('failed to set the internalNetwork property on slot %s to "%s" for VM "%s"' \
1474 % (iNic, sName, self.sName));
1475 return False;
1476 elif eAttachmentType == vboxcon.NetworkAttachmentType_NAT:
1477 try:
1478 oNic.NATNetwork = sName;
1479 except:
1480 reporter.errorXcpt('failed to set the NATNetwork property on slot %s to "%s" for VM "%s"' \
1481 % (iNic, sName, self.sName));
1482 return False;
1483 finally:
1484 self.oTstDrv.processPendingEvents();
1485
1486 if not self.setupNatForwardingForTxs(iNic):
1487 return False;
1488 reporter.log('set NIC type on slot %s to %s for VM "%s"' % (iNic, eAttachmentType, self.sName));
1489 return True;
1490
1491 def setNicMacAddress(self, sMacAddr, iNic = 0):
1492 """
1493 Sets the MAC address of the specified NIC.
1494 Returns True on success and False on failure. Error information is logged.
1495 """
1496
1497 # Resolve missing MAC address prefix
1498 cchMacAddr = len(sMacAddr)
1499 if cchMacAddr > 0 and cchMacAddr < 12:
1500 sHostName = ''
1501 try:
1502 sHostName = socket.getfqdn()
1503 if '.' not in sHostName:
1504 # somewhat misconfigured system, needs expensive approach to guessing FQDN
1505 for aAI in socket.getaddrinfo(sHostName, None):
1506 sName, _ = socket.getnameinfo(aAI[4], 0)
1507 if '.' in sName and not set(sName).issubset(set('0123456789.')):
1508 sHostName = sName
1509 break
1510
1511 sHostIP = socket.gethostbyname(sHostName)
1512 abHostIP = socket.inet_aton(sHostIP)
1513 if ord(abHostIP[0]) == 127 \
1514 or (ord(abHostIP[0]) == 169 and ord(abHostIP[1]) == 254) \
1515 or (ord(abHostIP[0]) == 192 and ord(abHostIP[1]) == 168 and ord(abHostIP[2]) == 56):
1516 reporter.log('warning: host IP for "%s" is %s, most likely not unique.' % (sHostName, sHostIP))
1517 except:
1518 reporter.errorXcpt('failed to determine the host IP for "%s".' % (sHostName,))
1519 abHostIP = array.array('B', (0x80, 0x86, 0x00, 0x00)).tostring()
1520 sDefaultMac = '%02X%02X%02X%02X%02X%02X' \
1521 % (0x02, ord(abHostIP[0]), ord(abHostIP[1]), ord(abHostIP[2]), ord(abHostIP[3]), iNic)
1522 sMacAddr = sDefaultMac[0:(12 - cchMacAddr)] + sMacAddr
1523
1524 # Get the NIC object and try set it address.
1525 try:
1526 oNic = self.o.machine.getNetworkAdapter(iNic);
1527 except:
1528 reporter.errorXcpt('getNetworkAdapter(%s) failed for "%s"' % (iNic, self.sName));
1529 return False;
1530
1531 try:
1532 oNic.MACAddress = sMacAddr;
1533 except:
1534 reporter.errorXcpt('failed to set the MAC address on slot %s to "%s" for VM "%s"' \
1535 % (iNic, sMacAddr, self.sName));
1536 return False;
1537
1538 reporter.log('set MAC address on slot %s to %s for VM "%s"' % (iNic, sMacAddr, self.sName));
1539 return True;
1540
1541 def setRamSize(self, cMB):
1542 """
1543 Set the RAM size of the VM.
1544 Returns True on success and False on failure. Error information is logged.
1545 """
1546 fRc = True;
1547 try:
1548 self.o.machine.memorySize = cMB;
1549 except:
1550 reporter.errorXcpt('failed to set the RAM size of "%s" to %s' % (self.sName, cMB));
1551 fRc = False;
1552 else:
1553 reporter.log('set the RAM size of "%s" to %s' % (self.sName, cMB));
1554 self.oTstDrv.processPendingEvents();
1555 return fRc;
1556
1557 def setVRamSize(self, cMB):
1558 """
1559 Set the RAM size of the VM.
1560 Returns True on success and False on failure. Error information is logged.
1561 """
1562 fRc = True;
1563 try:
1564 self.o.machine.VRAMSize = cMB;
1565 except:
1566 reporter.errorXcpt('failed to set the VRAM size of "%s" to %s' % (self.sName, cMB));
1567 fRc = False;
1568 else:
1569 reporter.log('set the VRAM size of "%s" to %s' % (self.sName, cMB));
1570 self.oTstDrv.processPendingEvents();
1571 return fRc;
1572
1573 def setCpuCount(self, cCpus):
1574 """
1575 Set the number of CPUs.
1576 Returns True on success and False on failure. Error information is logged.
1577 """
1578 fRc = True;
1579 try:
1580 self.o.machine.CPUCount = cCpus;
1581 except:
1582 reporter.errorXcpt('failed to set the CPU count of "%s" to %s' % (self.sName, cCpus));
1583 fRc = False;
1584 else:
1585 reporter.log('set the CPU count of "%s" to %s' % (self.sName, cCpus));
1586 self.oTstDrv.processPendingEvents();
1587 return fRc;
1588
1589 def getCpuCount(self):
1590 """
1591 Returns the number of CPUs.
1592 Returns the number of CPUs on success and 0 on failure. Error information is logged.
1593 """
1594 cCpus = 0;
1595 try:
1596 cCpus = self.o.machine.CPUCount;
1597 except:
1598 reporter.errorXcpt('failed to get the CPU count of "%s"' % (self.sName,));
1599
1600 self.oTstDrv.processPendingEvents();
1601 return cCpus;
1602
1603 def ensureControllerAttached(self, sController):
1604 """
1605 Makes sure the specified controller is attached to the VM, attaching it
1606 if necessary.
1607 """
1608 try:
1609 try:
1610 self.o.machine.getStorageControllerByName(sController);
1611 except:
1612 iType = _ControllerNameToBus(sController);
1613 try:
1614 self.o.machine.addStorageController(sController, iType);
1615 reporter.log('added storage controller "%s" (type %s) to %s' % (sController, iType, self.sName));
1616 except:
1617 reporter.errorXcpt('addStorageController("%s",%s) failed on "%s"' % (sController, iType, self.sName) );
1618 return False;
1619 finally:
1620 self.oTstDrv.processPendingEvents();
1621 return True;
1622
1623 def setStorageControllerPortCount(self, sController, iPortCount):
1624 """
1625 Set maximum ports count for storage controller
1626 """
1627 try:
1628 oCtl = self.o.machine.getStorageControllerByName(sController)
1629 oCtl.portCount = iPortCount
1630 self.oTstDrv.processPendingEvents()
1631 reporter.log('set controller "%s" port count to value %d' % (sController, iPortCount))
1632 return True
1633 except:
1634 reporter.log('unable to set storage controller "%s" ports count to %d' % (sController, iPortCount))
1635
1636 return False
1637
1638 def setBootOrder(self, iPosition, eType):
1639 """
1640 Set guest boot order type
1641 @param iPosition boot order position
1642 @param eType device type (vboxcon.DeviceType_HardDisk,
1643 vboxcon.DeviceType_DVD, vboxcon.DeviceType_Floppy)
1644 """
1645 try:
1646 self.o.machine.setBootOrder(iPosition, eType)
1647 except:
1648 return reporter.errorXcpt('Unable to set boot order.')
1649
1650 reporter.log('Set boot order [%d] for device %s' % (iPosition, str(eType)))
1651 self.oTstDrv.processPendingEvents();
1652
1653 return True
1654
1655 def setStorageControllerType(self, eType, sController = "IDE Controller"):
1656 """
1657 Similar to ensureControllerAttached, except it will change the type.
1658 """
1659 try:
1660 oCtl = self.o.machine.getStorageControllerByName(sController);
1661 except:
1662 iType = _ControllerNameToBus(sController);
1663 try:
1664 oCtl = self.o.machine.addStorageController(sController, iType);
1665 reporter.log('added storage controller "%s" (type %s) to %s' % (sController, iType, self.sName));
1666 except:
1667 reporter.errorXcpt('addStorageController("%s",%s) failed on "%s"' % (sController, iType, self.sName) );
1668 return False;
1669 try:
1670 oCtl.controllerType = eType;
1671 except:
1672 reporter.errorXcpt('failed to set controller type of "%s" on "%s" to %s' % (sController, self.sName, eType) );
1673 return False;
1674 reporter.log('set controller type of "%s" on "%s" to %s' % (sController, self.sName, eType) );
1675 self.oTstDrv.processPendingEvents();
1676 return True;
1677
1678 def attachDvd(self, sImage = None, sController = "IDE Controller", iPort = 1, iDevice = 0):
1679 """
1680 Attaches a DVD drive to a VM, optionally with an ISO inserted.
1681 Returns True on success and False on failure. Error information is logged.
1682 """
1683 # Input validation.
1684 if sImage is not None and not self.oTstDrv.isResourceFile(sImage)\
1685 and not os.path.isabs(sImage): ## fixme - testsuite unzip ++
1686 reporter.fatal('"%s" is not in the resource set' % (sImage));
1687 return None;
1688
1689 if not self.ensureControllerAttached(sController):
1690 return False;
1691
1692 # Find/register the image if specified.
1693 oImage = None;
1694 sImageUuid = "";
1695 if sImage is not None:
1696 sFullName = self.oTstDrv.getFullResourceName(sImage)
1697 try:
1698 oImage = self.oVBox.findDVDImage(sFullName);
1699 except:
1700 try:
1701 if self.fpApiVer >= 4.1:
1702 oImage = self.oVBox.openMedium(sFullName, vboxcon.DeviceType_DVD, vboxcon.AccessMode_ReadOnly, False);
1703 elif self.fpApiVer >= 4.0:
1704 oImage = self.oVBox.openMedium(sFullName, vboxcon.DeviceType_DVD, vboxcon.AccessMode_ReadOnly);
1705 else:
1706 oImage = self.oVBox.openDVDImage(sFullName, "");
1707 except vbox.ComException, oXcpt:
1708 if oXcpt.errno != -1:
1709 reporter.errorXcpt('failed to open DVD image "%s" xxx' % (sFullName));
1710 else:
1711 reporter.errorXcpt('failed to open DVD image "%s" yyy' % (sFullName));
1712 return False;
1713 except:
1714 reporter.errorXcpt('failed to open DVD image "%s"' % (sFullName));
1715 return False;
1716 try:
1717 sImageUuid = oImage.id;
1718 except:
1719 reporter.errorXcpt('failed to get the UUID of "%s"' % (sFullName));
1720 return False;
1721
1722 # Attach the DVD.
1723 fRc = True;
1724 try:
1725 if self.fpApiVer >= 4.0:
1726 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_DVD, oImage);
1727 else:
1728 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_DVD, sImageUuid);
1729 except:
1730 reporter.errorXcpt('attachDevice("%s",%s,%s,HardDisk,"%s") failed on "%s"' \
1731 % (sController, iPort, iDevice, sImageUuid, self.sName) );
1732 fRc = False;
1733 else:
1734 reporter.log('attached DVD to %s, image="%s"' % (self.sName, sImage));
1735 self.oTstDrv.processPendingEvents();
1736 return fRc;
1737
1738 def attachHd(self, sHd, sController = "IDE Controller", iPort = 0, iDevice = 0, fImmutable = True, fForceResource = True):
1739 """
1740 Attaches a HD to a VM.
1741 Returns True on success and False on failure. Error information is logged.
1742 """
1743 # Input validation.
1744 if fForceResource and not self.oTstDrv.isResourceFile(sHd):
1745 reporter.fatal('"%s" is not in the resource set' % (sHd,));
1746 return None;
1747
1748 if not self.ensureControllerAttached(sController):
1749 return False;
1750
1751 # Find the HD, registering it if necessary (as immutable).
1752 if fForceResource:
1753 sFullName = self.oTstDrv.getFullResourceName(sHd);
1754 else:
1755 sFullName = sHd;
1756 try:
1757 oHd = self.oVBox.findHardDisk(sFullName);
1758 except:
1759 try:
1760 if self.fpApiVer >= 4.1:
1761 oHd = self.oVBox.openMedium(sFullName, vboxcon.DeviceType_HardDisk, vboxcon.AccessMode_ReadOnly, False);
1762 elif self.fpApiVer >= 4.0:
1763 oHd = self.oVBox.openMedium(sFullName, vboxcon.DeviceType_HardDisk, vboxcon.AccessMode_ReadOnly);
1764 else:
1765 oHd = self.oVBox.openHardDisk(sFullName, vboxcon.AccessMode_ReadOnly, False, "", False, "");
1766 except:
1767 reporter.errorXcpt('failed to open hd "%s"' % (sFullName));
1768 return False;
1769 try:
1770 if fImmutable:
1771 oHd.type = vboxcon.MediumType_Immutable;
1772 else:
1773 oHd.type = vboxcon.MediumType_Normal;
1774 except:
1775 if fImmutable:
1776 reporter.errorXcpt('failed to set hd "%s" immutable' % (sHd));
1777 else:
1778 reporter.errorXcpt('failed to set hd "%s" normal' % (sHd));
1779 return False;
1780
1781 # Attach it.
1782 fRc = True;
1783 try:
1784 if self.fpApiVer >= 4.0:
1785 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_HardDisk, oHd);
1786 else:
1787 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_HardDisk, oHd.id);
1788 except:
1789 reporter.errorXcpt('attachDevice("%s",%s,%s,HardDisk,"%s") failed on "%s"' \
1790 % (sController, iPort, iDevice, oHd.id, self.sName) );
1791 fRc = False;
1792 else:
1793 reporter.log('attached "%s" to %s' % (sHd, self.sName));
1794 self.oTstDrv.processPendingEvents();
1795 return fRc;
1796
1797 def createBaseHd(self, sHd, sFmt = "VDI", cb = 10*1024*1024*1024):
1798 """
1799 Creates a base HD.
1800 Returns Medium object on success and None on failure. Error information is logged.
1801 """
1802 try:
1803 if self.fpApiVer >= 5.0:
1804 oHd = self.oVBox.createMedium(sFmt, sHd, vboxcon.AccessMode_ReadWrite, vboxcon.DeviceType_HardDisk);
1805 else:
1806 oHd = self.oVBox.createHardDisk(sFmt, sHd);
1807 oProgressXpcom = oHd.createBaseStorage(cb, (vboxcon.MediumVariant_Standard, ))
1808 oProgress = ProgressWrapper(oProgressXpcom, self.oVBoxMgr, self.oTstDrv, 'create base disk %s' % (sHd));
1809 oProgress.wait();
1810 oProgress.logResult();
1811 except:
1812 reporter.errorXcpt('failed to create base hd "%s"' % (sHd));
1813 oHd = None
1814
1815 return oHd;
1816
1817 def createDiffHd(self, oParentHd, sHd, sFmt = "VDI"):
1818 """
1819 Creates a differencing HD.
1820 Returns Medium object on success and None on failure. Error information is logged.
1821 """
1822 try:
1823 if self.fpApiVer >= 5.0:
1824 oHd = self.oVBox.createMedium(sFmt, sHd, vboxcon.AccessMode_ReadWrite, vboxcon.DeviceType_HardDisk);
1825 else:
1826 oHd = self.oVBox.createHardDisk(sFmt, sHd);
1827 oProgressXpcom = oParentHd.createDiffStorage(oHd, (vboxcon.MediumVariant_Standard, ))
1828 oProgress = ProgressWrapper(oProgressXpcom, self.oVBoxMgr, self.oTstDrv, 'create diff disk %s' % (sHd));
1829 oProgress.wait();
1830 oProgress.logResult();
1831 except:
1832 reporter.errorXcpt('failed to create diff hd "%s"' % (sHd));
1833 oHd = None
1834
1835 return oHd;
1836
1837 def createAndAttachHd(self, sHd, sFmt = "VDI", sController = "IDE Controller", cb = 10*1024*1024*1024, \
1838 iPort = 0, iDevice = 0, fImmutable = True):
1839 """
1840 Creates and attaches a HD to a VM.
1841 Returns True on success and False on failure. Error information is logged.
1842 """
1843 if not self.ensureControllerAttached(sController):
1844 return False;
1845
1846 oHd = self.createBaseHd(sHd, sFmt, cb)
1847 if oHd is None:
1848 return False;
1849
1850 fRc = True;
1851 try:
1852 if fImmutable:
1853 oHd.type = vboxcon.MediumType_Immutable;
1854 else:
1855 oHd.type = vboxcon.MediumType_Normal;
1856 except:
1857 if fImmutable:
1858 reporter.errorXcpt('failed to set hd "%s" immutable' % (sHd));
1859 else:
1860 reporter.errorXcpt('failed to set hd "%s" normal' % (sHd));
1861 fRc = False;
1862
1863 # Attach it.
1864 if fRc is True:
1865 try:
1866 if self.fpApiVer >= 4.0:
1867 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_HardDisk, oHd);
1868 else:
1869 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_HardDisk, oHd.id);
1870 except:
1871 reporter.errorXcpt('attachDevice("%s",%s,%s,HardDisk,"%s") failed on "%s"' \
1872 % (sController, iPort, iDevice, oHd.id, self.sName) );
1873 fRc = False;
1874 else:
1875 reporter.log('attached "%s" to %s' % (sHd, self.sName));
1876
1877 # Delete disk in case of an error
1878 if fRc is False:
1879 try:
1880 oProgressCom = oHd.deleteStorage();
1881 except:
1882 reporter.errorXcpt('deleteStorage() for disk %s failed' % (sHd,));
1883 else:
1884 oProgress = ProgressWrapper(oProgressCom, self.oVBoxMgr, self.oTstDrv, 'delete disk %s' % (sHd));
1885 oProgress.wait();
1886 oProgress.logResult();
1887
1888 self.oTstDrv.processPendingEvents();
1889 return fRc;
1890
1891 def detachHd(self, sController = "IDE Controller", iPort = 0, iDevice = 0):
1892 """
1893 Detaches a HD, if attached, and returns a reference to it (IMedium).
1894
1895 In order to delete the detached medium, the caller must first save
1896 the changes made in this session.
1897
1898 Returns (fRc, oHd), where oHd is None unless fRc is True, and fRc is
1899 your standard success indicator. Error information is logged.
1900 """
1901
1902 # What's attached?
1903 try:
1904 oHd = self.o.machine.getMedium(sController, iPort, iDevice);
1905 except:
1906 if self.oVBoxMgr.xcptIsOurXcptKind() \
1907 and self.oVBoxMgr.xcptIsEqual(None, self.oVBoxMgr.constants.VBOX_E_OBJECT_NOT_FOUND):
1908 reporter.log('No HD attached (to %s %s:%s)' % (sController, iPort, iDevice));
1909 return (True, None);
1910 return (reporter.errorXcpt('Error getting media at port %s, device %s, on %s.'
1911 % (iPort, iDevice, sController)), None);
1912 # Detach it.
1913 try:
1914 self.o.machine.detachDevice(sController, iPort, iDevice);
1915 except:
1916 return (reporter.errorXcpt('detachDevice("%s",%s,%s) failed on "%s"' \
1917 % (sController, iPort, iDevice, self.sName) ), None);
1918 reporter.log('detached HD ("%s",%s,%s) from %s' % (sController, iPort, iDevice, self.sName));
1919 return (True, oHd);
1920
1921 def attachFloppy(self, sFloppy, sController = "Floppy Controller", iPort = 0, iDevice = 0):
1922 """
1923 Attaches a floppy image to a VM.
1924 Returns True on success and False on failure. Error information is logged.
1925 """
1926 # Input validation.
1927 ## @todo Fix this wrt to bootsector-xxx.img from the validationkit.zip.
1928 ##if not self.oTstDrv.isResourceFile(sFloppy):
1929 ## reporter.fatal('"%s" is not in the resource set' % (sFloppy));
1930 ## return None;
1931
1932 if not self.ensureControllerAttached(sController):
1933 return False;
1934
1935 # Find the floppy image, registering it if necessary (as immutable).
1936 sFullName = self.oTstDrv.getFullResourceName(sFloppy);
1937 try:
1938 oFloppy = self.oVBox.findFloppyImage(sFullName);
1939 except:
1940 try:
1941 if self.fpApiVer >= 4.1:
1942 oFloppy = self.oVBox.openMedium(sFullName, vboxcon.DeviceType_Floppy, vboxcon.AccessMode_ReadOnly, False);
1943 elif self.fpApiVer >= 4.0:
1944 oFloppy = self.oVBox.openMedium(sFullName, vboxcon.DeviceType_Floppy, vboxcon.AccessMode_ReadOnly);
1945 else:
1946 oFloppy = self.oVBox.openFloppyImage(sFullName, "");
1947 except:
1948 reporter.errorXcpt('failed to open floppy "%s"' % (sFullName));
1949 return False;
1950 ## @todo the following works but causes trouble below (asserts in main).
1951 #try:
1952 # oFloppy.type = vboxcon.MediumType_Immutable;
1953 #except:
1954 # reporter.errorXcpt('failed to make floppy "%s" immutable' % (sFullName));
1955 # return False;
1956
1957 # Attach it.
1958 fRc = True;
1959 try:
1960 if self.fpApiVer >= 4.0:
1961 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_Floppy, oFloppy);
1962 else:
1963 self.o.machine.attachDevice(sController, iPort, iDevice, vboxcon.DeviceType_Floppy, oFloppy.id);
1964 except:
1965 reporter.errorXcpt('attachDevice("%s",%s,%s,Floppy,"%s") failed on "%s"' \
1966 % (sController, iPort, iDevice, oFloppy.id, self.sName) );
1967 fRc = False;
1968 else:
1969 reporter.log('attached "%s" to %s' % (sFloppy, self.sName));
1970 self.oTstDrv.processPendingEvents();
1971 return fRc;
1972
1973 def setupNic(self, sType, sXXX):
1974 """
1975 Attaches a HD to a VM.
1976 Returns True on success and False on failure. Error information is logged.
1977 """
1978 if sType == "PCNet": enmType = vboxcon.NetworkAdapterType_Am79C973;
1979 elif sType == "PCNetOld": enmType = vboxcon.NetworkAdapterType_Am79C970A;
1980 elif sType == "E1000": enmType = vboxcon.NetworkAdapterType_I82545EM; # MT Server
1981 elif sType == "E1000Desk": enmType = vboxcon.NetworkAdapterType_I82540EM; # MT Desktop
1982 elif sType == "E1000Srv2": enmType = vboxcon.NetworkAdapterType_I82543GC; # T Server
1983 elif sType == "Virtio": enmType = vboxcon.NetworkAdapterType_Virtio;
1984 else:
1985 reporter.error('Invalid NIC type: "%s" (sXXX=%s)' % (sType, sXXX));
1986 return False;
1987 ## @todo Implement me!
1988 if enmType is not None: pass
1989 return True;
1990
1991 def setupAudio(self, eAudioCtlType):
1992 """
1993 Set guest audio controller type and host audio adapter to null
1994 @param eAudioCtlType device type (vboxcon.AudioControllerType_SB16,
1995 vboxcon.AudioControllerType_AC97, vboxcon.AudioControllerType_HDA)
1996 """
1997 try:
1998 oAudioAdapter = self.o.machine.audioAdapter;
1999
2000 oAudioAdapter.audioController = eAudioCtlType;
2001 oAudioAdapter.audioDriver = vboxcon.AudioDriverType_Null;
2002 # Disable by default
2003 oAudioAdapter.enabled = False;
2004 except:
2005 return reporter.errorXcpt('Unable to set audio adapter.')
2006
2007 reporter.log('set audio adapter type to %d' % (eAudioCtlType))
2008 self.oTstDrv.processPendingEvents();
2009
2010 return True
2011
2012 def setupPreferredConfig(self): # pylint: disable=R0914
2013 """
2014 Configures the VM according to the preferences of the guest type.
2015 """
2016 try:
2017 sOsTypeId = self.o.machine.OSTypeId;
2018 except:
2019 reporter.errorXcpt('failed to obtain the OSTypeId for "%s"' % (self.sName));
2020 return False;
2021
2022 try:
2023 oOsType = self.oVBox.getGuestOSType(sOsTypeId);
2024 except:
2025 reporter.errorXcpt('getGuestOSType("%s") failed for "%s"' % (sOsTypeId, self.sName));
2026 return False;
2027
2028 # get the attributes.
2029 try:
2030 #sFamilyId = oOsType.familyId;
2031 #f64Bit = oOsType.is64Bit;
2032 fIoApic = oOsType.recommendedIOAPIC;
2033 fVirtEx = oOsType.recommendedVirtEx;
2034 cMBRam = oOsType.recommendedRAM;
2035 cMBVRam = oOsType.recommendedVRAM;
2036 #cMBHdd = oOsType.recommendedHDD;
2037 eNicType = oOsType.adapterType;
2038 if self.fpApiVer >= 3.2:
2039 if self.fpApiVer >= 4.2:
2040 fPae = oOsType.recommendedPAE;
2041 fUsbHid = oOsType.recommendedUSBHID;
2042 fHpet = oOsType.recommendedHPET;
2043 eStorCtlType = oOsType.recommendedHDStorageController;
2044 else:
2045 fPae = oOsType.recommendedPae;
2046 fUsbHid = oOsType.recommendedUsbHid;
2047 fHpet = oOsType.recommendedHpet;
2048 eStorCtlType = oOsType.recommendedHdStorageController;
2049 eFirmwareType = oOsType.recommendedFirmware;
2050 else:
2051 fPae = False;
2052 fUsbHid = False;
2053 fHpet = False;
2054 eFirmwareType = -1;
2055 eStorCtlType = vboxcon.StorageControllerType_PIIX4;
2056 if self.fpApiVer >= 4.0:
2057 eAudioCtlType = oOsType.recommendedAudioController;
2058 except:
2059 reporter.errorXcpt('exception reading IGuestOSType(%s) attribute' % (sOsTypeId));
2060 self.oTstDrv.processPendingEvents();
2061 return False;
2062 self.oTstDrv.processPendingEvents();
2063
2064 # Do the setting. Continue applying settings on error in case the
2065 # caller ignores the return code
2066 fRc = True;
2067 if not self.enableIoApic(fIoApic): fRc = False;
2068 if not self.enableVirtEx(fVirtEx): fRc = False;
2069 if not self.enablePae(fPae): fRc = False;
2070 if not self.setRamSize(cMBRam): fRc = False;
2071 if not self.setVRamSize(cMBVRam): fRc = False;
2072 if not self.setNicType(eNicType, 0): fRc = False;
2073 if self.fpApiVer >= 3.2:
2074 if not self.setFirmwareType(eFirmwareType): fRc = False;
2075 if not self.enableUsbHid(fUsbHid): fRc = False;
2076 if not self.enableHpet(fHpet): fRc = False;
2077 if eStorCtlType == vboxcon.StorageControllerType_PIIX3 \
2078 or eStorCtlType == vboxcon.StorageControllerType_PIIX4 \
2079 or eStorCtlType == vboxcon.StorageControllerType_ICH6:
2080 if not self.setStorageControllerType(eStorCtlType, "IDE Controller"):
2081 fRc = False;
2082 if self.fpApiVer >= 4.0:
2083 if not self.setupAudio(eAudioCtlType): fRc = False;
2084
2085 return fRc;
2086
2087 def addUsbDeviceFilter(self, sName, sVendorId = None, sProductId = None, sRevision = None, # pylint: disable=R0913
2088 sManufacturer = None, sProduct = None, sSerialNumber = None,
2089 sPort = None, sRemote = None):
2090 """
2091 Creates a USB device filter and inserts it into the VM.
2092 Returns True on success.
2093 Returns False on failure (logged).
2094 """
2095 fRc = True;
2096
2097 try:
2098 oUsbDevFilter = self.o.machine.USBDeviceFilters.createDeviceFilter(sName);
2099 oUsbDevFilter.active = True;
2100 if sVendorId is not None:
2101 oUsbDevFilter.vendorId = sVendorId;
2102 if sProductId is not None:
2103 oUsbDevFilter.productId = sProductId;
2104 if sRevision is not None:
2105 oUsbDevFilter.revision = sRevision;
2106 if sManufacturer is not None:
2107 oUsbDevFilter.manufacturer = sManufacturer;
2108 if sProduct is not None:
2109 oUsbDevFilter.product = sProduct;
2110 if sSerialNumber is not None:
2111 oUsbDevFilter.serialnumber = sSerialNumber;
2112 if sPort is not None:
2113 oUsbDevFilter.port = sPort;
2114 if sRemote is not None:
2115 oUsbDevFilter.remote = sRemote;
2116 try:
2117 self.o.machine.USBDeviceFilters.insertDeviceFilter(0, oUsbDevFilter);
2118 except:
2119 reporter.errorXcpt('insertDeviceFilter(%s) failed on "%s"' \
2120 % (0, self.sName) );
2121 fRc = False;
2122 else:
2123 reporter.log('inserted USB device filter "%s" to %s' % (sName, self.sName));
2124 except:
2125 reporter.errorXcpt('createDeviceFilter("%s") failed on "%s"' \
2126 % (sName, self.sName) );
2127 fRc = False;
2128 return fRc;
2129
2130 def getGuestPropertyValue(self, sName):
2131 """
2132 Gets a guest property value.
2133 Returns the value on success, None on failure (logged).
2134 """
2135 try:
2136 sValue = self.o.machine.getGuestPropertyValue(sName);
2137 except:
2138 reporter.errorXcpt('IMachine::getGuestPropertyValue("%s") failed' % (sName));
2139 return None;
2140 return sValue;
2141
2142 def setGuestPropertyValue(self, sName, sValue):
2143 """
2144 Sets a guest property value.
2145 Returns the True on success, False on failure (logged).
2146 """
2147 try:
2148 self.o.machine.setGuestPropertyValue(sName, sValue);
2149 except:
2150 reporter.errorXcpt('IMachine::setGuestPropertyValue("%s","%s") failed' % (sName, sValue));
2151 return False;
2152 return True;
2153
2154 def delGuestPropertyValue(self, sName):
2155 """
2156 Deletes a guest property value.
2157 Returns the True on success, False on failure (logged).
2158 """
2159 try:
2160 oMachine = self.o.machine;
2161 if self.fpApiVer >= 4.2:
2162 oMachine.deleteGuestProperty(sName);
2163 else:
2164 oMachine.setGuestPropertyValue(sName, '');
2165 except:
2166 reporter.errorXcpt('Unable to delete guest property "%s"' % (sName,));
2167 return False;
2168 return True;
2169
2170 def setExtraData(self, sKey, sValue):
2171 """
2172 Sets extra data.
2173 Returns the True on success, False on failure (logged).
2174 """
2175 try:
2176 self.o.machine.setExtraData(sKey, sValue);
2177 except:
2178 reporter.errorXcpt('IMachine::setExtraData("%s","%s") failed' % (sKey, sValue));
2179 return False;
2180 return True;
2181
2182 def getExtraData(self, sKey):
2183 """
2184 Gets extra data.
2185 Returns value on success, None on failure.
2186 """
2187 try:
2188 sValue = self.o.machine.getExtraData(sKey)
2189 except:
2190 reporter.errorXcpt('IMachine::setExtraData("%s","%s") failed' % (sKey, sValue))
2191 return None
2192 return sValue
2193
2194 def setupTeleporter(self, fEnabled=True, uPort = 6500, sAddress = '', sPassword = ''):
2195 """
2196 Sets up the teleporter for the VM.
2197 Returns True on success, False on failure (logged).
2198 """
2199 try:
2200 self.o.machine.teleporterAddress = sAddress;
2201 self.o.machine.teleporterPort = uPort;
2202 self.o.machine.teleporterPassword = sPassword;
2203 self.o.machine.teleporterEnabled = fEnabled;
2204 except:
2205 reporter.errorXcpt('setupTeleporter(%s, %s, %s, %s)' % (fEnabled, sPassword, uPort, sAddress));
2206 return False;
2207 return True;
2208
2209 def enableTeleporter(self, fEnable=True):
2210 """
2211 Enables or disables the teleporter of the VM.
2212 Returns True on success, False on failure (logged).
2213 """
2214 try:
2215 self.o.machine.teleporterEnabled = fEnable;
2216 except:
2217 reporter.errorXcpt('IMachine::teleporterEnabled=%s failed' % (fEnable));
2218 return False;
2219 return True;
2220
2221 def teleport(self, sHostname = 'localhost', uPort = 6500, sPassword = 'password', cMsMaxDowntime = 250):
2222 """
2223 Wrapper around the IConsole::teleport() method.
2224 Returns a progress object on success, None on failure (logged).
2225 """
2226 reporter.log2('"%s"::teleport(%s,%s,%s,%s)...' % (self.sName, sHostname, uPort, sPassword, cMsMaxDowntime));
2227 try:
2228 oProgress = self.o.console.teleport(sHostname, uPort, sPassword, cMsMaxDowntime)
2229 except:
2230 reporter.errorXcpt('IConsole::teleport(%s,%s,%s,%s) failed' % (sHostname, uPort, sPassword, cMsMaxDowntime));
2231 return None;
2232 return ProgressWrapper(oProgress, self.oVBoxMgr, self.oTstDrv, 'teleport %s' % (self.sName,));
2233
2234 def getOsType(self):
2235 """
2236 Gets the IGuestOSType interface for the machine.
2237
2238 return IGuestOSType interface on success, None + errorXcpt on failure.
2239 No exceptions raised.
2240 """
2241 try:
2242 sOsTypeId = self.o.machine.OSTypeId;
2243 except:
2244 reporter.errorXcpt('failed to obtain the OSTypeId for "%s"' % (self.sName));
2245 return None;
2246
2247 try:
2248 oOsType = self.oVBox.getGuestOSType(sOsTypeId);
2249 except:
2250 reporter.errorXcpt('getGuestOSType("%s") failed for "%s"' % (sOsTypeId, self.sName));
2251 return None;
2252
2253 return oOsType;
2254
2255 def setOsType(self, sNewTypeId):
2256 """
2257 Changes the OS type.
2258
2259 returns True on success, False + errorXcpt on failure.
2260 No exceptions raised.
2261 """
2262 try:
2263 self.o.machine.OSTypeId = sNewTypeId;
2264 except:
2265 reporter.errorXcpt('failed to set the OSTypeId for "%s" to "%s"' % (self.sName, sNewTypeId));
2266 return False;
2267 return True;
2268
2269
2270 def setParavirtProvider(self, iProvider):
2271 """
2272 Sets a paravirtualisation provider.
2273 Returns the True on success, False on failure (logged).
2274 """
2275 try:
2276 self.o.machine.paravirtProvider = iProvider
2277 except:
2278 reporter.errorXcpt('Unable to set paravirtualisation provider "%s"' % (iProvider,))
2279 return False;
2280 return True;
2281
2282
2283
2284 #
2285 # IConsole wrappers.
2286 #
2287
2288 def powerOff(self, fFudgeOnFailure = True):
2289 """
2290 Powers off the VM.
2291
2292 Returns True on success.
2293 Returns False on IConsole::powerDown() failure.
2294 Returns None if the progress object returns failure.
2295 """
2296 #
2297 # Deregister event handler before we power off the VM, otherwise we're
2298 # racing for VM process termination and cause misleading spurious
2299 # error messages in the event handling code, because the event objects
2300 # disappear.
2301 #
2302 # Note! Doing this before powerDown to try prevent numerous smoketest
2303 # timeouts on XPCOM hosts.
2304 #
2305 self.deregisterEventHandlerForTask();
2306
2307
2308 # Try power if off.
2309 try:
2310 oProgress = self.o.console.powerDown();
2311 except:
2312 reporter.logXcpt('IConsole::powerDown failed on %s' % (self.sName));
2313 if fFudgeOnFailure:
2314 self.oTstDrv.waitOnDirectSessionClose(self.oVM, 5000); # fudge
2315 self.waitForTask(1000); # fudge
2316 return False;
2317
2318 # Wait on power off operation to complete.
2319 rc = self.oTstDrv.waitOnProgress(oProgress);
2320 if rc < 0:
2321 self.close();
2322 if fFudgeOnFailure:
2323 vbox.reportError(oProgress, 'powerDown for "%s" failed' % (self.sName));
2324 self.oTstDrv.waitOnDirectSessionClose(self.oVM, 5000); # fudge
2325 return None;
2326
2327 # Wait for the VM to really power off or we'll fail to open a new session to it.
2328 self.oTstDrv.waitOnDirectSessionClose(self.oVM, 5000); # fudge
2329 return self.waitForTask(30 * 1000); # fudge
2330
2331 def restoreSnapshot(self, oSnapshot, fFudgeOnFailure = True):
2332 """
2333 Restores the given snapshot.
2334
2335 Returns True on success.
2336 Returns False on IMachine::restoreSnapshot() failure.
2337 Returns None if the progress object returns failure.
2338 """
2339 try:
2340 if self.fpApiVer >= 5.0:
2341 oProgress = self.o.machine.restoreSnapshot(oSnapshot);
2342 else:
2343 oProgress = self.o.console.restoreSnapshot(oSnapshot);
2344 except:
2345 reporter.logXcpt('IMachine::restoreSnapshot failed on %s' % (self.sName));
2346 if fFudgeOnFailure:
2347 self.oTstDrv.waitOnDirectSessionClose(self.oVM, 5000); # fudge
2348 self.waitForTask(1000); # fudge
2349 return False;
2350
2351 rc = self.oTstDrv.waitOnProgress(oProgress);
2352 if rc < 0:
2353 self.close();
2354 if fFudgeOnFailure:
2355 vbox.reportError(oProgress, 'restoreSnapshot for "%s" failed' % (self.sName));
2356 return None;
2357
2358 return self.waitForTask(30 * 1000);
2359
2360 def deleteSnapshot(self, oSnapshot, fFudgeOnFailure = True, cMsTimeout = 30 * 1000):
2361 """
2362 Deletes the given snapshot merging the diff image into the base.
2363
2364 Returns True on success.
2365 Returns False on IMachine::deleteSnapshot() failure.
2366 """
2367 try:
2368 if self.fpApiVer >= 5.0:
2369 oProgressCom = self.o.machine.deleteSnapshot(oSnapshot);
2370 else:
2371 oProgressCom = self.o.console.deleteSnapshot(oSnapshot);
2372 oProgress = ProgressWrapper(oProgressCom, self.oVBoxMgr, self.oTstDrv, 'Delete Snapshot %s' % (oSnapshot));
2373 oProgress.wait(cMsTimeout);
2374 oProgress.logResult();
2375 except:
2376 reporter.logXcpt('IMachine::deleteSnapshot failed on %s' % (self.sName));
2377 if fFudgeOnFailure:
2378 self.oTstDrv.waitOnDirectSessionClose(self.oVM, 5000); # fudge
2379 self.waitForTask(1000); # fudge
2380 return False;
2381
2382 return True;
2383
2384 def takeSnapshot(self, sName, sDescription = '', fPause = True, fFudgeOnFailure = True, cMsTimeout = 30 * 1000):
2385 """
2386 Takes a snapshot with the given name
2387
2388 Returns True on success.
2389 Returns False on IMachine::takeSnapshot() or VM state change failure.
2390 """
2391 try:
2392 if fPause is True \
2393 and self.oVM.state is vboxcon.MachineState_Running:
2394 self.o.console.pause();
2395 if self.fpApiVer >= 5.0:
2396 (oProgressCom, _) = self.o.machine.takeSnapshot(sName, sDescription, True);
2397 else:
2398 oProgressCom = self.o.console.takeSnapshot(sName, sDescription);
2399 oProgress = ProgressWrapper(oProgressCom, self.oVBoxMgr, self.oTstDrv, 'Take Snapshot %s' % (sName));
2400 oProgress.wait(cMsTimeout);
2401 oProgress.logResult();
2402 except:
2403 reporter.logXcpt('IMachine::takeSnapshot failed on %s' % (self.sName));
2404 if fFudgeOnFailure:
2405 self.oTstDrv.waitOnDirectSessionClose(self.oVM, 5000); # fudge
2406 self.waitForTask(1000); # fudge
2407 return False;
2408
2409 if fPause is True \
2410 and self.oVM.state is vboxcon.MachineState_Paused:
2411 self.o.console.resume();
2412
2413 return True;
2414
2415 def findSnapshot(self, sName):
2416 """
2417 Returns the snapshot object with the given name
2418
2419 Returns snapshot object on success.
2420 Returns None if there is no snapshot with the given name.
2421 """
2422 return self.oVM.findSnapshot(sName);
2423
2424 def takeScreenshot(self, sFilename, iScreenId=0):
2425 """
2426 Take screenshot from the given display and save it to specified file.
2427
2428 Returns True on success
2429 Returns False on failure.
2430 """
2431 try:
2432 if self.fpApiVer >= 5.0:
2433 iWidth, iHeight, _, _, _, _ = self.o.console.display.getScreenResolution(iScreenId)
2434 aPngData = self.o.console.display.takeScreenShotToArray(iScreenId, iWidth, iHeight,
2435 vboxcon.BitmapFormat_PNG)
2436 else:
2437 iWidth, iHeight, _, _, _ = self.o.console.display.getScreenResolution(iScreenId)
2438 aPngData = self.o.console.display.takeScreenShotPNGToArray(iScreenId, iWidth, iHeight)
2439 except:
2440 reporter.logXcpt("Unable to take screenshot")
2441 return False
2442
2443 oFile = open(sFilename, 'wb')
2444 oFile.write(aPngData)
2445 oFile.close()
2446
2447 return True
2448
2449 #
2450 # IMachineDebugger wrappers.
2451 #
2452
2453 def queryOsKernelLog(self):
2454 """
2455 Tries to get the OS kernel log using the VM debugger interface.
2456
2457 Returns string containing the kernel log on success.
2458 Returns None on failure.
2459 """
2460 sOsKernelLog = None;
2461 try:
2462 self.o.console.debugger.loadPlugIn('all');
2463 except:
2464 reporter.logXcpt('Unable to load debugger plugins');
2465 else:
2466 try:
2467 sOsDetected = self.o.console.debugger.detectOS();
2468 except:
2469 reporter.logXcpt('Failed to detect the guest OS');
2470 else:
2471 try:
2472 sOsKernelLog = self.o.console.debugger.queryOSKernelLog(0);
2473 except:
2474 reporter.logXcpt('Unable to get the guest OS (%s) kernel log' % (sOsDetected,));
2475 return sOsKernelLog;
2476
2477 def queryDbgInfo(self, sItem, sArg = '', sDefault = None):
2478 """
2479 Simple wrapper around IMachineDebugger::info.
2480
2481 Returns string on success, sDefault on failure (logged).
2482 """
2483 try:
2484 return self.o.console.debugger.info(sItem, sArg);
2485 except:
2486 reporter.logXcpt('Unable to query "%s" with arg "%s"' % (sItem, sArg,));
2487 return sDefault;
2488
2489 def queryDbgInfoVgaText(self, sArg = 'all'):
2490 """
2491 Tries to get the 'info vgatext' output, provided we're in next mode.
2492
2493 Returns string containing text on success.
2494 Returns None on failure or not text mode.
2495 """
2496 sVgaText = None;
2497 try:
2498 sVgaText = self.o.console.debugger.info('vgatext', sArg);
2499 if sVgaText.startswith('Not in text mode!'):
2500 sVgaText = None;
2501 except:
2502 reporter.logXcpt('Unable to query vgatext with arg "%s"' % (sArg,));
2503 return sVgaText;
2504
2505 def queryDbgGuestStack(self, iCpu = 0):
2506 """
2507 Returns the guest stack for the given VCPU.
2508
2509 Returns string containing the guest stack for the selected VCPU on success.
2510 Returns None on failure.
2511 """
2512
2513 #
2514 # Load all plugins first and try to detect the OS so we can
2515 # get nicer stack traces.
2516 #
2517 try:
2518 self.o.console.debugger.loadPlugIn('all');
2519 except:
2520 reporter.logXcpt('Unable to load debugger plugins');
2521 else:
2522 try:
2523 sOsDetected = self.o.console.debugger.detectOS();
2524 _ = sOsDetected;
2525 except:
2526 reporter.logXcpt('Failed to detect the guest OS');
2527
2528 sGuestStack = None;
2529 try:
2530 sGuestStack = self.o.console.debugger.dumpGuestStack(iCpu);
2531 except:
2532 reporter.logXcpt('Unable to query guest stack for CPU %s' % (iCpu, ));
2533
2534 return sGuestStack;
2535
2536
2537 #
2538 # Other methods.
2539 #
2540
2541 def getPrimaryIp(self):
2542 """
2543 Tries to obtain the primary IP address of the guest via the guest
2544 properties.
2545
2546 Returns IP address on success.
2547 Returns empty string on failure.
2548 """
2549 sIpAddr = self.getGuestPropertyValue('/VirtualBox/GuestInfo/Net/0/V4/IP');
2550 if vbox.isIpAddrValid(sIpAddr):
2551 return sIpAddr;
2552 return '';
2553
2554 def getPid(self):
2555 """
2556 Gets the process ID for the direct session unless it's ourselves.
2557 """
2558 if self.uPid is None and self.o is not None and self.fRemoteSession:
2559 try:
2560 if self.fpApiVer >= 4.2:
2561 uPid = self.o.machine.sessionPID;
2562 else:
2563 uPid = self.o.machine.sessionPid;
2564 if uPid != os.getpid() and uPid != 0xffffffff:
2565 self.uPid = uPid;
2566 except Exception, oXcpt:
2567 if vbox.ComError.equal(oXcpt, vbox.ComError.E_UNEXPECTED):
2568 try:
2569 if self.fpApiVer >= 4.2:
2570 uPid = self.oVM.sessionPID;
2571 else:
2572 uPid = self.oVM.sessionPid;
2573 if uPid != os.getpid() and uPid != 0xffffffff:
2574 self.uPid = uPid;
2575 except:
2576 reporter.log2Xcpt();
2577 else:
2578 reporter.log2Xcpt();
2579 if self.uPid is not None:
2580 reporter.log2('getPid: %u' % (self.uPid,));
2581 self.oTstDrv.pidFileAdd(self.uPid);
2582 return self.uPid;
2583
2584 def addLogsToReport(self, cReleaseLogs = 1):
2585 """
2586 Retrieves and adds the release and debug logs to the test report.
2587 """
2588 fRc = True;
2589
2590 # Add each of the requested release logs to the report.
2591 for iLog in range(0, cReleaseLogs):
2592 try:
2593 if self.fpApiVer >= 3.2:
2594 sLogFile = self.oVM.queryLogFilename(iLog);
2595 elif iLog > 0:
2596 sLogFile = '%s/VBox.log' % (self.oVM.logFolder,);
2597 else:
2598 sLogFile = '%s/VBox.log.%u' % (self.oVM.logFolder, iLog);
2599 except:
2600 reporter.logXcpt('iLog=%s' % (iLog,));
2601 fRc = False;
2602 else:
2603 if sLogFile is not None and sLogFile != '': # the None bit is for a 3.2.0 bug.
2604 reporter.addLogFile(sLogFile, 'log/release/vm', '%s #%u' % (self.sName, iLog),
2605 sAltName = '%s-%s' % (self.sName, os.path.basename(sLogFile),));
2606
2607 # Now for the hardened windows startup log.
2608 try:
2609 sLogFile = os.path.join(self.oVM.logFolder, 'VBoxHardening.log');
2610 except:
2611 reporter.logXcpt();
2612 fRc = False;
2613 else:
2614 if os.path.isfile(sLogFile):
2615 reporter.addLogFile(sLogFile, 'log/release/vm', '%s hardening log' % (self.sName, ),
2616 sAltName = '%s-%s' % (self.sName, os.path.basename(sLogFile),));
2617
2618 # Now for the debug log.
2619 if self.sLogFile is not None and os.path.isfile(self.sLogFile):
2620 reporter.addLogFile(self.sLogFile, 'log/debug/vm', '%s debug' % (self.sName, ),
2621 sAltName = '%s-%s' % (self.sName, os.path.basename(self.sLogFile),));
2622
2623 return fRc;
2624
2625 def registerDerivedEventHandler(self, oSubClass, dArgs = None, fMustSucceed = True):
2626 """
2627 Create an instance of the given ConsoleEventHandlerBase sub-class and
2628 register it.
2629
2630 The new instance is returned on success. None is returned on error.
2631 """
2632
2633 # We need a console object.
2634 try:
2635 oConsole = self.o.console;
2636 except Exception, oXcpt:
2637 if fMustSucceed or vbox.ComError.notEqual(oXcpt, vbox.ComError.E_UNEXPECTED):
2638 reporter.errorXcpt('Failed to get ISession::console for "%s"' % (self.sName, ));
2639 return None;
2640
2641 # Add the base class arguments.
2642 dArgsCopy = dArgs.copy() if dArgs is not None else dict();
2643 dArgsCopy['oSession'] = self;
2644 dArgsCopy['oConsole'] = oConsole;
2645 sLogSuffix = 'on %s' % (self.sName,)
2646 return oSubClass.registerDerivedEventHandler(self.oVBoxMgr, self.fpApiVer, oSubClass, dArgsCopy,
2647 oConsole, 'IConsole', 'IConsoleCallback',
2648 fMustSucceed = fMustSucceed, sLogSuffix = sLogSuffix);
2649
2650 def enableVmmDevTestingPart(self, fEnabled, fEnableMMIO = False):
2651 """
2652 Enables the testing part of the VMMDev.
2653
2654 Returns True on success and False on failure. Error information is logged.
2655 """
2656 fRc = True;
2657 try:
2658 self.o.machine.setExtraData('VBoxInternal/Devices/VMMDev/0/Config/TestingEnabled',
2659 '1' if fEnabled else '');
2660 self.o.machine.setExtraData('VBoxInternal/Devices/VMMDev/0/Config/TestingMMIO',
2661 '1' if fEnableMMIO and fEnabled else '');
2662 except:
2663 reporter.errorXcpt('VM name "%s", fEnabled=%s' % (self.sName, fEnabled));
2664 fRc = False;
2665 else:
2666 reporter.log('set VMMDevTesting=%s for "%s"' % (fEnabled, self.sName));
2667 self.oTstDrv.processPendingEvents();
2668 return fRc;
2669
2670 #
2671 # Test eXecution Service methods.
2672 #
2673
2674 def txsConnectViaTcp(self, cMsTimeout = 10*60000, sIpAddr = None, sMacAddr = None, fNatForwardingForTxs = False):
2675 """
2676 Connects to the TXS using TCP/IP as transport. If no IP or MAC is
2677 addresses are specified, we'll get the IP from the guest additions.
2678
2679 Returns a TxsConnectTask object on success, None + log on failure.
2680 """
2681 # If the VM is configured with a NAT interface, connect to local host.
2682 fReversedSetup = False;
2683 fUseNatForTxs = False;
2684 if sIpAddr is None:
2685 try:
2686 oNic = self.oVM.getNetworkAdapter(0);
2687 if oNic.attachmentType == vboxcon.NetworkAttachmentType_NAT:
2688 fUseNatForTxs = True;
2689 except:
2690 reporter.errorXcpt();
2691 return None;
2692 if fUseNatForTxs:
2693 fReversedSetup = not fNatForwardingForTxs;
2694 sIpAddr = '127.0.0.1';
2695
2696 # Kick off the task.
2697 try:
2698 oTask = TxsConnectTask(self, cMsTimeout, sIpAddr, sMacAddr, fReversedSetup);
2699 except:
2700 reporter.errorXcpt();
2701 oTask = None;
2702 return oTask;
2703
2704 def txsTryConnectViaTcp(self, cMsTimeout, sHostname, fReversed = False):
2705 """
2706 Attempts to connect to a TXS instance.
2707
2708 Returns True if a connection was established, False if not (only grave
2709 failures are logged as errors).
2710
2711 Note! The timeout is more of a guideline...
2712 """
2713
2714 if sHostname is None or sHostname.strip() == '':
2715 raise base.GenError('Empty sHostname is not implemented yet');
2716
2717 oTxsSession = txsclient.tryOpenTcpSession(cMsTimeout, sHostname, fReversedSetup = fReversed,
2718 cMsIdleFudge = cMsTimeout / 2);
2719 if oTxsSession is None:
2720 return False;
2721
2722 # Wait for the connect task to time out.
2723 self.oTstDrv.addTask(oTxsSession);
2724 self.oTstDrv.processPendingEvents();
2725 oRc = self.oTstDrv.waitForTasks(cMsTimeout);
2726 self.oTstDrv.removeTask(oTxsSession);
2727 if oRc != oTxsSession:
2728 if oRc is not None:
2729 reporter.log('oRc=%s, expected %s' % (oRc, oTxsSession));
2730 self.oTstDrv.processPendingEvents();
2731 oTxsSession.cancelTask(); # this is synchronous
2732 return False;
2733
2734 # Check the status.
2735 reporter.log2('TxsSession is ready, isSuccess() -> %s.' % (oTxsSession.isSuccess(),));
2736 if not oTxsSession.isSuccess():
2737 return False;
2738
2739 reporter.log2('Disconnecting from TXS...');
2740 return oTxsSession.syncDisconnect();
2741
2742
2743
2744class TxsConnectTask(TdTaskBase):
2745 """
2746 Class that takes care of connecting to a VM.
2747 """
2748
2749 class TxsConnectTaskVBoxCallback(vbox.VirtualBoxEventHandlerBase):
2750 """ Class for looking for IPv4 address changes on interface 0."""
2751 def __init__(self, dArgs):
2752 vbox.VirtualBoxEventHandlerBase.__init__(self, dArgs); # pylint: disable=W0233
2753 self.oParentTask = dArgs['oParentTask'];
2754 self.sMachineId = dArgs['sMachineId'];
2755
2756 def onGuestPropertyChange(self, sMachineId, sName, sValue, sFlags):
2757 """Look for IP address."""
2758 reporter.log2('onGuestPropertyChange(,%s,%s,%s,%s)' % (sMachineId, sName, sValue, sFlags));
2759 if sMachineId == self.sMachineId \
2760 and sName == '/VirtualBox/GuestInfo/Net/0/V4/IP':
2761 self.oParentTask._setIp(sValue); # pylint: disable=W0212
2762
2763
2764 def __init__(self, oSession, cMsTimeout, sIpAddr, sMacAddr, fReversedSetup):
2765 TdTaskBase.__init__(self, utils.getCallerName());
2766 self.cMsTimeout = cMsTimeout;
2767 self.sIpAddr = None;
2768 self.sNextIpAddr = None;
2769 self.sMacAddr = sMacAddr;
2770 self.fReversedSetup = fReversedSetup;
2771 self.oVBox = oSession.oVBox;
2772 self.oVBoxEventHandler = None;
2773 self.oTxsSession = None;
2774 self.fpApiVer = oSession.fpApiVer;
2775
2776 # Skip things we don't implement.
2777 if sMacAddr is not None:
2778 reporter.error('TxsConnectTask does not implement sMacAddr yet');
2779 raise base.GenError();
2780
2781 reporter.log2('TxsConnectTask: sIpAddr=%s fReversedSetup=%s' % (sIpAddr, fReversedSetup))
2782 if fReversedSetup is True:
2783 self._openTcpSession(sIpAddr, fReversedSetup = True);
2784 elif sIpAddr is not None and sIpAddr.strip() != '':
2785 self._openTcpSession(sIpAddr, cMsIdleFudge = 5000);
2786 else:
2787 #
2788 # If we've got no IP address, register callbacks that listens for
2789 # the primary network adaptor of the VM to set a IPv4 guest prop.
2790 # Note! The order in which things are done here is kind of important.
2791 #
2792
2793 # 0. The caller zaps the property before starting the VM.
2794 #try:
2795 # oSession.delGuestPropertyValue('/VirtualBox/GuestInfo/Net/0/V4/IP');
2796 #except:
2797 # reporter.logXcpt();
2798
2799 # 1. Register the callback / event listener object.
2800 dArgs = {'oParentTask':self, 'sMachineId':oSession.o.machine.id};
2801 self.oVBoxEventHandler = self.oVBox.registerDerivedEventHandler(self.TxsConnectTaskVBoxCallback, dArgs);
2802
2803 # 2. Query the guest properties.
2804 try:
2805 sIpAddr = oSession.getGuestPropertyValue('/VirtualBox/GuestInfo/Net/0/V4/IP');
2806 except:
2807 reporter.errorXcpt('IMachine::getGuestPropertyValue("/VirtualBox/GuestInfo/Net/0/V4/IP") failed');
2808 self._deregisterEventHandler();
2809 raise;
2810 else:
2811 if sIpAddr is not None:
2812 self._setIp(sIpAddr);
2813 # end __init__
2814
2815 def __del__(self):
2816 """ Make sure we deregister the callback. """
2817 self._deregisterEventHandler();
2818 return TdTaskBase.__del__(self);
2819
2820 def toString(self):
2821 return '<%s cMsTimeout=%s, sIpAddr=%s, sNextIpAddr=%s, sMacAddr=%s, fReversedSetup=%s,' \
2822 ' oTxsSession=%s oVBoxEventHandler=%s, oVBox=%s>' \
2823 % (TdTaskBase.toString(self), self.cMsTimeout, self.sIpAddr, self.sNextIpAddr, self.sMacAddr, self.fReversedSetup,
2824 self.oTxsSession, self.oVBoxEventHandler, self.oVBox);
2825
2826 def _deregisterEventHandler(self):
2827 """Deregisters the event handler."""
2828 fRc = True;
2829 if self.oVBoxEventHandler is not None:
2830 fRc = self.oVBoxEventHandler.unregister();
2831 self.oVBoxEventHandler = None;
2832 return fRc;
2833
2834 def _setIp(self, sIpAddr, fInitCall = False):
2835 """Called when we get an IP. Will create a TXS session and signal the task."""
2836 sIpAddr = sIpAddr.strip();
2837
2838 if sIpAddr is not None \
2839 and sIpAddr != '':
2840 if vbox.isIpAddrValid(sIpAddr) or fInitCall:
2841 try:
2842 for s in sIpAddr.split('.'):
2843 i = int(s);
2844 if str(i) != s:
2845 raise Exception();
2846 except:
2847 reporter.fatalXcpt();
2848 else:
2849 self._openTcpSession(sIpAddr, cMsIdleFudge = 5000);
2850 return None;
2851
2852 reporter.log('TxsConnectTask: Ignoring Bad ip "%s"' % (sIpAddr));
2853 else:
2854 reporter.log2('TxsConnectTask: Ignoring empty ip "%s"' % (sIpAddr));
2855 return None;
2856
2857 def _openTcpSession(self, sIpAddr, uPort = None, fReversedSetup = False, cMsIdleFudge = 0):
2858 """
2859 Calls txsclient.openTcpSession and switches our task to reflect the
2860 state of the subtask.
2861 """
2862 self.oCv.acquire();
2863 if self.oTxsSession is None:
2864 reporter.log2('_openTcpSession: sIpAddr=%s, uPort=%d, fReversedSetup=%s' % \
2865 (sIpAddr, uPort if uPort is not None else 0, fReversedSetup));
2866 self.sIpAddr = sIpAddr;
2867 self.oTxsSession = txsclient.openTcpSession(self.cMsTimeout, sIpAddr, uPort, \
2868 fReversedSetup, cMsIdleFudge);
2869 self.oTxsSession.setTaskOwner(self);
2870 else:
2871 self.sNextIpAddr = sIpAddr;
2872 reporter.log2('_openTcpSession: sNextIpAddr=%s' % (sIpAddr,));
2873 self.oCv.release();
2874 return None;
2875
2876 def notifyAboutReadyTask(self, oTxsSession):
2877 """
2878 Called by the TXS session task when it's done.
2879
2880 We'll signal the task completed or retry depending on the result.
2881 """
2882
2883 self.oCv.acquire();
2884
2885 # Disassociate ourselves with the session (avoid cyclic ref)
2886 oTxsSession.setTaskOwner(None);
2887 fSuccess = oTxsSession.isSuccess();
2888 if self.oTxsSession is not None:
2889 if not fSuccess:
2890 self.oTxsSession = None;
2891 if fSuccess and self.fReversedSetup:
2892 self.sIpAddr = oTxsSession.oTransport.sHostname;
2893 else:
2894 fSuccess = False;
2895
2896 # Signal done, or retry?
2897 if fSuccess \
2898 or self.fReversedSetup \
2899 or self.getAgeAsMs() >= self.cMsTimeout:
2900 self.signalTaskLocked();
2901 else:
2902 sIpAddr = self.sNextIpAddr if self.sNextIpAddr is not None else self.sIpAddr;
2903 self._openTcpSession(sIpAddr, cMsIdleFudge = 5000);
2904
2905 self.oCv.release();
2906 return True;
2907
2908 #
2909 # Public methods
2910 #
2911
2912 def getResult(self):
2913 """
2914 Returns the connected TXS session object on success.
2915 Returns None on failure or if the task has not yet completed.
2916 """
2917 self.oCv.acquire();
2918 oTxsSession = self.oTxsSession;
2919 self.oCv.release();
2920
2921 if oTxsSession is not None and not oTxsSession.isSuccess():
2922 oTxsSession = None;
2923 return oTxsSession;
2924
2925 def cancelTask(self):
2926 """ Cancels the task. """
2927 self.oCv.acquire();
2928 if not self.fSignalled:
2929 oTxsSession = self.oTxsSession;
2930 if oTxsSession is not None:
2931 self.oCv.release();
2932 oTxsSession.setTaskOwner(None);
2933 oTxsSession.cancelTask();
2934 oTxsSession.waitForTask(1000);
2935 self.oCv.acquire();
2936 self.signalTaskLocked();
2937 self.oCv.release();
2938 return True;
2939
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