VirtualBox

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

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

(C) 2016

  • Property svn:eol-style set to LF
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
File size: 33.9 KB
Line 
1#!/usr/bin/env python
2# -*- coding: utf-8 -*-
3
4"""
5VirtualBox Installer Wrapper Driver.
6
7This installs VirtualBox, starts a sub driver which does the real testing,
8and then uninstall VirtualBox afterwards. This reduces the complexity of the
9other VBox test drivers.
10"""
11
12__copyright__ = \
13"""
14Copyright (C) 2010-2016 Oracle Corporation
15
16This file is part of VirtualBox Open Source Edition (OSE), as
17available from http://www.virtualbox.org. This file is free software;
18you can redistribute it and/or modify it under the terms of the GNU
19General Public License (GPL) as published by the Free Software
20Foundation, in version 2 as it comes in the "COPYING" file of the
21VirtualBox OSE distribution. VirtualBox OSE is distributed in the
22hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
23
24The contents of this file may alternatively be used under the terms
25of the Common Development and Distribution License Version 1.0
26(CDDL) only, as it comes in the "COPYING.CDDL" file of the
27VirtualBox OSE distribution, in which case the provisions of the
28CDDL are applicable instead of those of the GPL.
29
30You may elect to license modified versions of this file under the
31terms and conditions of either the GPL or the CDDL or both.
32"""
33__version__ = "$Revision: 62484 $"
34
35
36# Standard Python imports.
37import os
38import sys
39import re
40#import socket
41import tempfile
42import time
43
44# Only the main script needs to modify the path.
45try: __file__
46except: __file__ = sys.argv[0];
47g_ksValidationKitDir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)));
48sys.path.append(g_ksValidationKitDir);
49
50# Validation Kit imports.
51from common import utils, webutils;
52from common.constants import rtexitcode;
53from testdriver import reporter;
54from testdriver.base import TestDriverBase;
55
56
57
58class VBoxInstallerTestDriver(TestDriverBase):
59 """
60 Implementation of a top level test driver.
61 """
62
63
64 ## State file indicating that we've skipped installation.
65 ksVar_Skipped = 'vboxinstaller-skipped';
66
67
68 def __init__(self):
69 TestDriverBase.__init__(self);
70 self._asSubDriver = []; # The sub driver and it's arguments.
71 self._asBuildUrls = []; # The URLs passed us on the command line.
72 self._asBuildFiles = []; # The downloaded file names.
73 self._fAutoInstallPuelExtPack = True;
74
75 #
76 # Base method we override
77 #
78
79 def showUsage(self):
80 rc = TestDriverBase.showUsage(self);
81 # 0 1 2 3 4 5 6 7 8
82 # 012345678901234567890123456789012345678901234567890123456789012345678901234567890
83 reporter.log('');
84 reporter.log('vboxinstaller Options:');
85 reporter.log(' --vbox-build <url[,url2[,..]]>');
86 reporter.log(' Comma separated list of URL to file to download and install or/and');
87 reporter.log(' unpack. URLs without a schema are assumed to be files on the');
88 reporter.log(' build share and will be copied off it.');
89 reporter.log(' --no-puel-extpack');
90 reporter.log(' Indicates that the PUEL extension pack should not be installed if found.');
91 reporter.log(' The default is to install it if found in the vbox-build.');
92 reporter.log(' --');
93 reporter.log(' Indicates the end of our parameters and the start of the sub');
94 reporter.log(' testdriver and its arguments.');
95 return rc;
96
97 def parseOption(self, asArgs, iArg):
98 """
99 Parse our arguments.
100 """
101 if asArgs[iArg] == '--':
102 # End of our parameters and start of the sub driver invocation.
103 iArg = self.requireMoreArgs(1, asArgs, iArg);
104 assert len(self._asSubDriver) == 0;
105 self._asSubDriver = asArgs[iArg:];
106 self._asSubDriver[0] = self._asSubDriver[0].replace('/', os.path.sep);
107 iArg = len(asArgs) - 1;
108 elif asArgs[iArg] == '--vbox-build':
109 # List of files to copy/download and install.
110 iArg = self.requireMoreArgs(1, asArgs, iArg);
111 self._asBuildUrls = asArgs[iArg].split(',');
112 elif asArgs[iArg] == '--no-puel-extpack':
113 self._fAutoInstallPuelExtPack = False;
114 elif asArgs[iArg] == '--puel-extpack':
115 self._fAutoInstallPuelExtPack = True;
116 else:
117 return TestDriverBase.parseOption(self, asArgs, iArg);
118 return iArg + 1;
119
120 def completeOptions(self):
121 #
122 # Check that we've got what we need.
123 #
124 if len(self._asBuildUrls) == 0:
125 reporter.error('No build files specfiied ("--vbox-build file1[,file2[...]]")');
126 return False;
127 if len(self._asSubDriver) == 0:
128 reporter.error('No sub testdriver specified. (" -- test/stuff/tdStuff1.py args")');
129 return False;
130
131 #
132 # Construct _asBuildFiles as an array parallel to _asBuildUrls.
133 #
134 for sUrl in self._asBuildUrls:
135 sDstFile = os.path.join(self.sScratchPath, webutils.getFilename(sUrl));
136 self._asBuildFiles.append(sDstFile);
137
138 return TestDriverBase.completeOptions(self);
139
140 def actionExtract(self):
141 reporter.error('vboxinstall does not support extracting resources, you have to do that using the sub testdriver.');
142 return False;
143
144 def actionCleanupBefore(self):
145 """
146 Kills all VBox process we see.
147
148 This is only supposed to execute on a testbox so we don't need to go
149 all complicated wrt other users.
150 """
151 return self._killAllVBoxProcesses();
152
153 def actionConfig(self):
154 """
155 Install VBox and pass on the configure request to the sub testdriver.
156 """
157 fRc = self._installVBox();
158 if fRc is None: self._persistentVarSet(self.ksVar_Skipped, 'true');
159 else: self._persistentVarUnset(self.ksVar_Skipped);
160
161 ## @todo vbox.py still has bugs preventing us from invoking it seperately with each action.
162 if fRc is True and 'execute' not in self.asActions and 'all' not in self.asActions:
163 fRc = self._executeSubDriver([ 'verify', ]);
164 if fRc is True and 'execute' not in self.asActions and 'all' not in self.asActions:
165 fRc = self._executeSubDriver([ 'config', ]);
166 return fRc;
167
168 def actionExecute(self):
169 """
170 Execute the sub testdriver.
171 """
172 return self._executeSubDriver(self.asActions);
173
174 def actionCleanupAfter(self):
175 """
176 Forward this to the sub testdriver, then uninstall VBox.
177 """
178 fRc = True;
179 if 'execute' not in self.asActions and 'all' not in self.asActions:
180 fRc = self._executeSubDriver([ 'cleanup-after', ], fMaySkip = False);
181
182 if not self._killAllVBoxProcesses():
183 fRc = False;
184
185 if not self._uninstallVBox(self._persistentVarExists(self.ksVar_Skipped)):
186 fRc = False;
187
188 if utils.getHostOs() == 'darwin':
189 self._darwinUnmountDmg(fIgnoreError = True); # paranoia
190
191 if not TestDriverBase.actionCleanupAfter(self):
192 fRc = False;
193
194 return fRc;
195
196
197 def actionAbort(self):
198 """
199 Forward this to the sub testdriver first, then do the default pid file
200 based cleanup and finally swipe the scene with the heavy artillery.
201 """
202 fRc1 = self._executeSubDriver([ 'abort', ], fMaySkip = False);
203 fRc2 = TestDriverBase.actionAbort(self);
204 fRc3 = self._killAllVBoxProcesses();
205 return fRc1 and fRc2 and fRc3;
206
207
208 #
209 # Persistent variables.
210 #
211 ## @todo integrate into the base driver. Persisten accross scratch wipes?
212
213 def __persistentVarCalcName(self, sVar):
214 """Returns the (full) filename for the given persistent variable."""
215 assert re.match(r'^[a-zA-Z0-9_-]*$', sVar) is not None;
216 return os.path.join(self.sScratchPath, 'persistent-%s.var' % (sVar,));
217
218 def _persistentVarSet(self, sVar, sValue = ''):
219 """
220 Sets a persistent variable.
221
222 Returns True on success, False + reporter.error on failure.
223
224 May raise exception if the variable name is invalid or something
225 unexpected happens.
226 """
227 sFull = self.__persistentVarCalcName(sVar);
228 try:
229 oFile = open(sFull, 'w');
230 if len(sValue) > 0:
231 oFile.write(sValue.encode('utf-8'));
232 oFile.close();
233 except:
234 reporter.errorXcpt('Error creating "%s"' % (sFull,));
235 return False;
236 return True;
237
238 def _persistentVarUnset(self, sVar):
239 """
240 Unsets a persistent variable.
241
242 Returns True on success, False + reporter.error on failure.
243
244 May raise exception if the variable name is invalid or something
245 unexpected happens.
246 """
247 sFull = self.__persistentVarCalcName(sVar);
248 if os.path.exists(sFull):
249 try:
250 os.unlink(sFull);
251 except:
252 reporter.errorXcpt('Error unlinking "%s"' % (sFull,));
253 return False;
254 return True;
255
256 def _persistentVarExists(self, sVar):
257 """
258 Checks if a persistent variable exists.
259
260 Returns true/false.
261
262 May raise exception if the variable name is invalid or something
263 unexpected happens.
264 """
265 return os.path.exists(self.__persistentVarCalcName(sVar));
266
267 def _persistentVarGet(self, sVar):
268 """
269 Gets the value of a persistent variable.
270
271 Returns variable value on success.
272 Returns None if the variable doesn't exist or if an
273 error (reported) occured.
274
275 May raise exception if the variable name is invalid or something
276 unexpected happens.
277 """
278 sFull = self.__persistentVarCalcName(sVar);
279 if not os.path.exists(sFull):
280 return None;
281 try:
282 oFile = open(sFull, 'r');
283 sValue = oFile.read().decode('utf-8');
284 oFile.close();
285 except:
286 reporter.errorXcpt('Error creating "%s"' % (sFull,));
287 return None;
288 return sValue;
289
290
291 #
292 # Helpers.
293 #
294
295 def _killAllVBoxProcesses(self):
296 """
297 Kills all virtual box related processes we find in the system.
298 """
299
300 for iIteration in range(22):
301 # Gather processes to kill.
302 aoTodo = [];
303 for oProcess in utils.processListAll():
304 sBase = oProcess.getBaseImageNameNoExeSuff();
305 if sBase is None:
306 continue;
307 sBase = sBase.lower();
308 if sBase in [ 'vboxsvc', 'virtualbox', 'virtualboxvm', 'vboxheadless', 'vboxmanage', 'vboxsdl', 'vboxwebsrv',
309 'vboxautostart', 'vboxballoonctrl', 'vboxbfe', 'vboxextpackhelperapp', 'vboxnetdhcp',
310 'vboxnetadpctl', 'vboxtestogl', 'vboxtunctl', 'vboxvmmpreload', 'vboxxpcomipcd', 'vmCreator', ]:
311 aoTodo.append(oProcess);
312 if iIteration in [0, 21] and sBase in [ 'windbg', 'gdb', 'gdb-i386-apple-darwin', ]:
313 reporter.log('Warning: debugger running: %s (%s)' % (oProcess.iPid, sBase,));
314 if len(aoTodo) == 0:
315 return True;
316
317 # Kill.
318 for oProcess in aoTodo:
319 reporter.log('Loop #%d - Killing %s (%s)'
320 % (iIteration, oProcess.iPid, oProcess.sImage if oProcess.sName is None else oProcess.sName,));
321 utils.processKill(oProcess.iPid); # No mercy.
322
323 # Check if they're all dead like they should be.
324 time.sleep(0.1);
325 for oProcess in aoTodo:
326 if utils.processExists(oProcess.iPid):
327 time.sleep(2);
328 break;
329
330 return False;
331
332 def _executeSync(self, asArgs, fMaySkip = False):
333 """
334 Executes a child process synchronously.
335
336 Returns True if the process executed successfully and returned 0.
337 Returns None if fMaySkip is true and the child exits with RTEXITCODE_SKIPPED.
338 Returns False for all other cases.
339 """
340 reporter.log('Executing: %s' % (asArgs, ));
341 reporter.flushall();
342 try:
343 iRc = utils.processCall(asArgs, shell = False, close_fds = False);
344 except:
345 reporter.errorXcpt();
346 return False;
347 reporter.log('Exit code: %s (%s)' % (iRc, asArgs));
348 if fMaySkip and iRc == rtexitcode.RTEXITCODE_SKIPPED:
349 return None;
350 return iRc is 0;
351
352 def _sudoExecuteSync(self, asArgs):
353 """
354 Executes a sudo child process synchronously.
355 Returns a tuple [True, 0] if the process executed successfully
356 and returned 0, otherwise [False, rc] is returned.
357 """
358 reporter.log('Executing [sudo]: %s' % (asArgs, ));
359 reporter.flushall();
360 iRc = 0;
361 try:
362 iRc = utils.sudoProcessCall(asArgs, shell = False, close_fds = False);
363 except:
364 reporter.errorXcpt();
365 return (False, 0);
366 reporter.log('Exit code [sudo]: %s (%s)' % (iRc, asArgs));
367 return (iRc is 0, iRc);
368
369 def _executeSubDriver(self, asActions, fMaySkip = True):
370 """
371 Execute the sub testdriver with the specified action.
372 """
373 asArgs = list(self._asSubDriver)
374 asArgs.append('--no-wipe-clean');
375 asArgs.extend(asActions);
376 return self._executeSync(asArgs, fMaySkip = fMaySkip);
377
378 def _maybeUnpackArchive(self, sMaybeArchive, fNonFatal = False):
379 """
380 Attempts to unpack the given build file.
381 Updates _asBuildFiles.
382 Returns True/False. No exceptions.
383 """
384 def unpackFilter(sMember):
385 # type: (string) -> bool
386 """ Skips debug info. """
387 sLower = sMember.lower();
388 if sLower.endswith('.pdb'):
389 return False;
390 return True;
391
392 asMembers = utils.unpackFile(sMaybeArchive, self.sScratchPath, reporter.log,
393 reporter.log if fNonFatal else reporter.error,
394 fnFilter = unpackFilter);
395 if asMembers is None:
396 return False;
397 self._asBuildFiles.extend(asMembers);
398 return True;
399
400
401 def _installVBox(self):
402 """
403 Download / copy the build files into the scratch area and install them.
404 """
405 reporter.testStart('Installing VirtualBox');
406 reporter.log('CWD=%s' % (os.getcwd(),)); # curious
407
408 #
409 # Download the build files.
410 #
411 for i in range(len(self._asBuildUrls)):
412 if webutils.downloadFile(self._asBuildUrls[i], self._asBuildFiles[i],
413 self.sBuildPath, reporter.log, reporter.log) is not True:
414 reporter.testDone(fSkipped = True);
415 return None; # Failed to get binaries, probably deleted. Skip the test run.
416
417 #
418 # Unpack anything we know what is and append it to the build files
419 # list. This allows us to use VBoxAll*.tar.gz files.
420 #
421 for sFile in list(self._asBuildFiles):
422 if self._maybeUnpackArchive(sFile, fNonFatal = True) is not True:
423 reporter.testDone(fSkipped = True);
424 return None; # Failed to unpack. Probably local error, like busy
425 # DLLs on windows, no reason for failing the build.
426
427 #
428 # Go to system specific installation code.
429 #
430 sHost = utils.getHostOs()
431 if sHost == 'darwin': fRc = self._installVBoxOnDarwin();
432 elif sHost == 'linux': fRc = self._installVBoxOnLinux();
433 elif sHost == 'solaris': fRc = self._installVBoxOnSolaris();
434 elif sHost == 'win': fRc = self._installVBoxOnWindows();
435 else:
436 reporter.error('Unsupported host "%s".' % (sHost,));
437 if fRc is False:
438 reporter.testFailure('Installation error.');
439
440 #
441 # Install the extension pack.
442 #
443 if fRc is True and self._fAutoInstallPuelExtPack:
444 fRc = self._installExtPack();
445 if fRc is False:
446 reporter.testFailure('Extension pack installation error.');
447
448 # Some debugging...
449 try:
450 cMbFreeSpace = utils.getDiskUsage(self.sScratchPath);
451 reporter.log('Disk usage after VBox install: %d MB available at %s' % (cMbFreeSpace, self.sScratchPath,));
452 except:
453 reporter.logXcpt('Unable to get disk free space. Ignored. Continuing.');
454
455 reporter.testDone();
456 return fRc;
457
458 def _uninstallVBox(self, fIgnoreError = False):
459 """
460 Uninstall VirtualBox.
461 """
462 reporter.testStart('Uninstalling VirtualBox');
463
464 sHost = utils.getHostOs()
465 if sHost == 'darwin': fRc = self._uninstallVBoxOnDarwin();
466 elif sHost == 'linux': fRc = self._uninstallVBoxOnLinux();
467 elif sHost == 'solaris': fRc = self._uninstallVBoxOnSolaris();
468 elif sHost == 'win': fRc = self._uninstallVBoxOnWindows();
469 else:
470 reporter.error('Unsupported host "%s".' % (sHost,));
471 if fRc is False and not fIgnoreError:
472 reporter.testFailure('Uninstallation failed.');
473
474 fRc2 = self._uninstallAllExtPacks();
475 if not fRc2 and fRc:
476 fRc = fRc2;
477
478 reporter.testDone(fSkipped = (fRc is None));
479 return fRc;
480
481 def _findFile(self, sRegExp, fMandatory = False):
482 """
483 Returns the first build file that matches the given regular expression
484 (basename only).
485
486 Returns None if no match was found, logging it as an error if
487 fMandatory is set.
488 """
489 oRegExp = re.compile(sRegExp);
490
491 for sFile in self._asBuildFiles:
492 if oRegExp.match(os.path.basename(sFile)) and os.path.exists(sFile):
493 return sFile;
494
495 if fMandatory:
496 reporter.error('Failed to find a file matching "%s" in %s.' % (sRegExp, self._asBuildFiles,));
497 return None;
498
499 def _waitForTestManagerConnectivity(self, cSecTimeout):
500 """
501 Check and wait for network connectivity to the test manager.
502
503 This is used with the windows installation and uninstallation since
504 these usually disrupts network connectivity when installing the filter
505 driver. If we proceed to quickly, we might finish the test at a time
506 when we cannot report to the test manager and thus end up with an
507 abandonded test error.
508 """
509 cSecElapsed = 0;
510 secStart = utils.timestampSecond();
511 while reporter.checkTestManagerConnection() is False:
512 cSecElapsed = utils.timestampSecond() - secStart;
513 if cSecElapsed >= cSecTimeout:
514 reporter.log('_waitForTestManagerConnectivity: Giving up after %u secs.' % (cSecTimeout,));
515 return False;
516 time.sleep(2);
517
518 if cSecElapsed > 0:
519 reporter.log('_waitForTestManagerConnectivity: Waited %s secs.' % (cSecTimeout,));
520 return True;
521
522
523 #
524 # Darwin (Mac OS X).
525 #
526
527 def _darwinDmgPath(self):
528 """ Returns the path to the DMG mount."""
529 return os.path.join(self.sScratchPath, 'DmgMountPoint');
530
531 def _darwinUnmountDmg(self, fIgnoreError):
532 """
533 Umount any DMG on at the default mount point.
534 """
535 sMountPath = self._darwinDmgPath();
536 if not os.path.exists(sMountPath):
537 return True;
538
539 # Unmount.
540 fRc = self._executeSync(['hdiutil', 'detach', sMountPath ]);
541 if not fRc and not fIgnoreError:
542 reporter.error('Failed to unmount DMG at %s' % sMountPath);
543
544 # Remove dir.
545 try:
546 os.rmdir(sMountPath);
547 except:
548 if not fIgnoreError:
549 reporter.errorXcpt('Failed to remove directory %s' % sMountPath);
550 return fRc;
551
552 def _darwinMountDmg(self, sDmg):
553 """
554 Mount the DMG at the default mount point.
555 """
556 self._darwinUnmountDmg(fIgnoreError = True)
557
558 sMountPath = self._darwinDmgPath();
559 if not os.path.exists(sMountPath):
560 try:
561 os.mkdir(sMountPath, 0755);
562 except:
563 reporter.logXcpt();
564 return False;
565
566 return self._executeSync(['hdiutil', 'attach', '-readonly', '-mount', 'required', '-mountpoint', sMountPath, sDmg, ]);
567
568 def _installVBoxOnDarwin(self):
569 """ Installs VBox on Mac OS X."""
570 sDmg = self._findFile('^VirtualBox-.*\\.dmg$');
571 if sDmg is None:
572 return False;
573
574 # Mount the DMG.
575 fRc = self._darwinMountDmg(sDmg);
576 if fRc is not True:
577 return False;
578
579 # Uninstall any previous vbox version first.
580 sUninstaller = os.path.join(self._darwinDmgPath(), 'VirtualBox_Uninstall.tool');
581 fRc, _ = self._sudoExecuteSync([sUninstaller, '--unattended',]);
582 if fRc is True:
583
584 # Install the package.
585 sPkg = os.path.join(self._darwinDmgPath(), 'VirtualBox.pkg');
586 fRc, _ = self._sudoExecuteSync(['installer', '-verbose', '-dumplog', '-pkg', sPkg, '-target', '/']);
587
588 # Unmount the DMG and we're done.
589 if not self._darwinUnmountDmg(fIgnoreError = False):
590 fRc = False;
591 return fRc;
592
593 def _uninstallVBoxOnDarwin(self):
594 """ Uninstalls VBox on Mac OS X."""
595
596 # Is VirtualBox installed? If not, don't try uninstall it.
597 sVBox = self._getVBoxInstallPath(fFailIfNotFound = False);
598 if sVBox is None:
599 return True;
600
601 # Find the dmg.
602 sDmg = self._findFile('^VirtualBox-.*\\.dmg$');
603 if sDmg is None:
604 return False;
605 if not os.path.exists(sDmg):
606 return True;
607
608 # Mount the DMG.
609 fRc = self._darwinMountDmg(sDmg);
610 if fRc is not True:
611 return False;
612
613 # Execute the uninstaller.
614 sUninstaller = os.path.join(self._darwinDmgPath(), 'VirtualBox_Uninstall.tool');
615 fRc, _ = self._sudoExecuteSync([sUninstaller, '--unattended',]);
616
617 # Unmount the DMG and we're done.
618 if not self._darwinUnmountDmg(fIgnoreError = False):
619 fRc = False;
620 return fRc;
621
622 #
623 # GNU/Linux
624 #
625
626 def _installVBoxOnLinux(self):
627 """ Installs VBox on Linux."""
628 sRun = self._findFile('^VirtualBox-.*\\.run$');
629 if sRun is None:
630 return False;
631 utils.chmodPlusX(sRun);
632
633 # Install the new one.
634 fRc, _ = self._sudoExecuteSync([sRun,]);
635 return fRc;
636
637 def _uninstallVBoxOnLinux(self):
638 """ Uninstalls VBox on Linux."""
639
640 # Is VirtualBox installed? If not, don't try uninstall it.
641 sVBox = self._getVBoxInstallPath(fFailIfNotFound = False);
642 if sVBox is None:
643 return True;
644
645 # Find the .run file and use it.
646 sRun = self._findFile('^VirtualBox-.*\\.run$', fMandatory = False);
647 if sRun is not None:
648 utils.chmodPlusX(sRun);
649 fRc, _ = self._sudoExecuteSync([sRun, 'uninstall']);
650 return fRc;
651
652 # Try the installed uninstaller.
653 for sUninstaller in [os.path.join(sVBox, 'uninstall.sh'), '/opt/VirtualBox/uninstall.sh', ]:
654 if os.path.isfile(sUninstaller):
655 reporter.log('Invoking "%s"...' % (sUninstaller,));
656 fRc, _ = self._sudoExecuteSync([sUninstaller, 'uninstall']);
657 return fRc;
658
659 reporter.log('Did not find any VirtualBox install to uninstall.');
660 return True;
661
662
663 #
664 # Solaris
665 #
666
667 def _generateAutoResponseOnSolaris(self):
668 """
669 Generates an autoresponse file on solaris, returning the name.
670 None is return on failure.
671 """
672 sPath = os.path.join(self.sScratchPath, 'SolarisAutoResponse');
673 oFile = utils.openNoInherit(sPath, 'wt');
674 oFile.write('basedir=default\n'
675 'runlevel=nocheck\n'
676 'conflict=quit\n'
677 'setuid=nocheck\n'
678 'action=nocheck\n'
679 'partial=quit\n'
680 'instance=unique\n'
681 'idepend=quit\n'
682 'rdepend=quit\n'
683 'space=quit\n'
684 'mail=\n');
685 oFile.close();
686 return sPath;
687
688 def _installVBoxOnSolaris(self):
689 """ Installs VBox on Solaris."""
690 sPkg = self._findFile('^VirtualBox-.*\\.pkg$', fMandatory = False);
691 if sPkg is None:
692 sTar = self._findFile('^VirtualBox-.*-SunOS-.*\\.tar.gz$', fMandatory = False);
693 if sTar is not None:
694 if self._maybeUnpackArchive(sTar) is not True:
695 return False;
696 sPkg = self._findFile('^VirtualBox-.*\\.pkg$', fMandatory = True);
697 sRsp = self._findFile('^autoresponse$', fMandatory = True);
698 if sPkg is None or sRsp is None:
699 return False;
700
701 # Uninstall first (ignore result).
702 self._uninstallVBoxOnSolaris();
703
704 # Install the new one.
705 fRc, _ = self._sudoExecuteSync(['pkgadd', '-d', sPkg, '-n', '-a', sRsp, 'SUNWvbox']);
706 return fRc;
707
708 def _uninstallVBoxOnSolaris(self):
709 """ Uninstalls VBox on Solaris."""
710 reporter.flushall();
711 if utils.processCall(['pkginfo', '-q', 'SUNWvbox']) != 0:
712 return True;
713 sRsp = self._generateAutoResponseOnSolaris();
714 fRc, _ = self._sudoExecuteSync(['pkgrm', '-n', '-a', sRsp, 'SUNWvbox']);
715 return fRc;
716
717 #
718 # Windows
719 #
720
721 def _installVBoxOnWindows(self):
722 """ Installs VBox on Windows."""
723 sExe = self._findFile('^VirtualBox-.*-(MultiArch|Win).exe$');
724 if sExe is None:
725 return False;
726
727 # TEMPORARY HACK - START
728 # It seems that running the NDIS cleanup script upon uninstallation is not
729 # a good idea, so let's run it before installing VirtualBox.
730 #sHostName = socket.getfqdn();
731 #if not sHostName.startswith('testboxwin3') \
732 # and not sHostName.startswith('testboxharp2') \
733 # and not sHostName.startswith('wei01-b6ka-3') \
734 # and utils.getHostOsVersion() in ['8', '8.1', '9', '2008Server', '2008ServerR2', '2012Server']:
735 # reporter.log('Peforming extra NDIS cleanup...');
736 # sMagicScript = os.path.abspath(os.path.join(g_ksValidationKitDir, 'testdriver', 'win-vbox-net-uninstall.ps1'));
737 # fRc2, _ = self._sudoExecuteSync(['powershell.exe', '-Command', 'set-executionpolicy unrestricted']);
738 # if not fRc2:
739 # reporter.log('set-executionpolicy failed.');
740 # self._sudoExecuteSync(['powershell.exe', '-Command', 'get-executionpolicy']);
741 # fRc2, _ = self._sudoExecuteSync(['powershell.exe', '-File', sMagicScript]);
742 # if not fRc2:
743 # reporter.log('NDIS cleanup failed.');
744 # TEMPORARY HACK - END
745
746 # Uninstall any previous vbox version first.
747 fRc = self._uninstallVBoxOnWindows();
748 if fRc is not True:
749 return None; # There shouldn't be anything to uninstall, and if there is, it's not our fault.
750
751 # Install the new one.
752 asArgs = [sExe, '-vvvv', '--silent', '--logging'];
753 asArgs.extend(['--msiparams', 'REBOOT=ReallySuppress']);
754 sVBoxInstallPath = os.environ.get('VBOX_INSTALL_PATH', None);
755 if sVBoxInstallPath is not None:
756 asArgs.extend(['INSTALLDIR="%s"' % (sVBoxInstallPath,)]);
757 fRc2, iRc = self._sudoExecuteSync(asArgs);
758 if fRc2 is False:
759 if iRc == 3010: # ERROR_SUCCESS_REBOOT_REQUIRED
760 reporter.log('Note: Installer required a reboot to complete installation');
761 # Optional, don't fail.
762 else:
763 fRc = False;
764 sLogFile = os.path.join(tempfile.gettempdir(), 'VirtualBox', 'VBoxInstallLog.txt');
765 if sLogFile is not None \
766 and os.path.isfile(sLogFile):
767 reporter.addLogFile(sLogFile, 'log/installer', "Verbose MSI installation log file");
768 self._waitForTestManagerConnectivity(30);
769 return fRc;
770
771 def _uninstallVBoxOnWindows(self):
772 """
773 Uninstalls VBox on Windows, all installations we find to be on the safe side...
774 """
775
776 import win32com.client; # pylint: disable=F0401
777 win32com.client.gencache.EnsureModule('{000C1092-0000-0000-C000-000000000046}', 1033, 1, 0);
778 oInstaller = win32com.client.Dispatch('WindowsInstaller.Installer',
779 resultCLSID = '{000C1090-0000-0000-C000-000000000046}')
780
781 # Search installed products for VirtualBox.
782 asProdCodes = [];
783 for sProdCode in oInstaller.Products:
784 try:
785 sProdName = oInstaller.ProductInfo(sProdCode, "ProductName");
786 except:
787 reporter.logXcpt();
788 continue;
789 #reporter.log('Info: %s=%s' % (sProdCode, sProdName));
790 if sProdName.startswith('Oracle VM VirtualBox') \
791 or sProdName.startswith('Sun VirtualBox'):
792 asProdCodes.append([sProdCode, sProdName]);
793
794 # Before we start uninstalling anything, just ruthlessly kill any
795 # msiexec process we might find hanging around.
796 cKilled = 0;
797 for oProcess in utils.processListAll():
798 sBase = oProcess.getBaseImageNameNoExeSuff();
799 if sBase is not None and sBase.lower() in [ 'msiexec', ]:
800 reporter.log('Killing MSI process: %s (%s)' % (oProcess.iPid, sBase));
801 utils.processKill(oProcess.iPid);
802 cKilled += 1;
803 if cKilled > 0:
804 time.sleep(16); # fudge.
805
806 # Do the uninstalling.
807 fRc = True;
808 sLogFile = os.path.join(self.sScratchPath, 'VBoxUninstallLog.txt');
809 for sProdCode, sProdName in asProdCodes:
810 reporter.log('Uninstalling %s (%s)...' % (sProdName, sProdCode));
811 fRc2, iRc = self._sudoExecuteSync(['msiexec', '/uninstall', sProdCode, '/quiet', '/passive', '/norestart',
812 '/L*v', '%s' % (sLogFile), ]);
813 if fRc2 is False:
814 if iRc == 3010: # ERROR_SUCCESS_REBOOT_REQUIRED
815 reporter.log('Note: Uninstaller required a reboot to complete uninstallation');
816 # Optional, don't fail.
817 else:
818 fRc = False;
819 reporter.addLogFile(sLogFile, 'log/uninstaller', "Verbose MSI uninstallation log file");
820
821 self._waitForTestManagerConnectivity(30);
822 if fRc is False and os.path.isfile(sLogFile):
823 reporter.addLogFile(sLogFile, 'log/uninstaller');
824 return fRc;
825
826
827 #
828 # Extension pack.
829 #
830
831 def _getVBoxInstallPath(self, fFailIfNotFound):
832 """ Returns the default VBox installation path. """
833 sHost = utils.getHostOs();
834 if sHost == 'win':
835 sProgFiles = os.environ.get('ProgramFiles', 'C:\\Program Files');
836 asLocs = [
837 os.path.join(sProgFiles, 'Oracle', 'VirtualBox'),
838 os.path.join(sProgFiles, 'OracleVM', 'VirtualBox'),
839 os.path.join(sProgFiles, 'Sun', 'VirtualBox'),
840 ];
841 elif sHost == 'linux' or sHost == 'solaris':
842 asLocs = [ '/opt/VirtualBox', '/opt/VirtualBox-3.2', '/opt/VirtualBox-3.1', '/opt/VirtualBox-3.0'];
843 elif sHost == 'darwin':
844 asLocs = [ '/Applications/VirtualBox.app/Contents/MacOS' ];
845 else:
846 asLocs = [ '/opt/VirtualBox' ];
847 if 'VBOX_INSTALL_PATH' in os.environ:
848 asLocs.insert(0, os.environ.get('VBOX_INSTALL_PATH', None));
849
850 for sLoc in asLocs:
851 if os.path.isdir(sLoc):
852 return sLoc;
853 if fFailIfNotFound:
854 reporter.error('Failed to locate VirtualBox installation: %s' % (asLocs,));
855 else:
856 reporter.log2('Failed to locate VirtualBox installation: %s' % (asLocs,));
857 return None;
858
859 def _installExtPack(self):
860 """ Installs the extension pack. """
861 sVBox = self._getVBoxInstallPath(fFailIfNotFound = True);
862 if sVBox is None:
863 return False;
864 sExtPackDir = os.path.join(sVBox, 'ExtensionPacks');
865
866 if self._uninstallAllExtPacks() is not True:
867 return False;
868
869 sExtPack = self._findFile('Oracle_VM_VirtualBox_Extension_Pack.vbox-extpack');
870 if sExtPack is None:
871 sExtPack = self._findFile('Oracle_VM_VirtualBox_Extension_Pack.*.vbox-extpack');
872 if sExtPack is None:
873 return True;
874
875 sDstDir = os.path.join(sExtPackDir, 'Oracle_VM_VirtualBox_Extension_Pack');
876 reporter.log('Installing extension pack "%s" to "%s"...' % (sExtPack, sExtPackDir));
877 fRc, _ = self._sudoExecuteSync([ self.getBinTool('vts_tar'),
878 '--extract',
879 '--verbose',
880 '--gzip',
881 '--file', sExtPack,
882 '--directory', sDstDir,
883 '--file-mode-and-mask', '0644',
884 '--file-mode-or-mask', '0644',
885 '--dir-mode-and-mask', '0755',
886 '--dir-mode-or-mask', '0755',
887 '--owner', '0',
888 '--group', '0',
889 ]);
890 return fRc;
891
892 def _uninstallAllExtPacks(self):
893 """ Uninstalls all extension packs. """
894 sVBox = self._getVBoxInstallPath(fFailIfNotFound = False);
895 if sVBox is None:
896 return True;
897
898 sExtPackDir = os.path.join(sVBox, 'ExtensionPacks');
899 if not os.path.exists(sExtPackDir):
900 return True;
901
902 fRc, _ = self._sudoExecuteSync([self.getBinTool('vts_rm'), '-Rfv', '--', sExtPackDir]);
903 return fRc;
904
905
906
907if __name__ == '__main__':
908 sys.exit(VBoxInstallerTestDriver().main(sys.argv));
909
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