| # -*- coding: utf-8 -*- |
| # Copyright (c) 2011 The Chromium OS Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| """Module containing implementation of an au_worker for virtual machines.""" |
| |
| from __future__ import print_function |
| |
| import os |
| import shutil |
| import tempfile |
| |
| from chromite.cbuildbot import constants as buildbot_constants |
| from chromite.lib import constants as chromite_constants |
| from chromite.lib import cros_build_lib |
| from chromite.lib import cros_logging as logging |
| from crostestutils.lib import constants |
| from crostestutils.au_test_harness import au_worker |
| from crostestutils.au_test_harness import update_exception |
| |
| |
| class VMAUWorker(au_worker.AUWorker): |
| """Test harness for updating virtual machines.""" |
| |
| def __init__(self, options, test_results_root): |
| """Processes vm-specific options.""" |
| super(VMAUWorker, self).__init__(options, test_results_root) |
| self.verbose = options.verbose |
| self.graphics_flag = '--no-display' if options.no_graphics else '' |
| if not self.board: |
| cros_build_lib.Die('Need board to convert base image to vm.') |
| self.whitelist_chrome_crashes = options.whitelist_chrome_crashes |
| |
| def _StopVM(self): |
| """Stops an existing VM.""" |
| cros_build_lib.RunCommand(['./cros_vm', '--stop', |
| '--ssh-port=%s' % self._ssh_port], |
| print_cmd=False, error_code_ok=True, |
| cwd=chromite_constants.CHROMITE_BIN_DIR) |
| |
| def CleanUp(self): |
| """Stop the vm after a test.""" |
| self._StopVM() |
| |
| def PrepareBase(self, image_path, signed_base=False): |
| """Creates an update-able VM based on base image.""" |
| original_image_path = self.PrepareVMBase(image_path, signed_base) |
| # This worker may be running in parallel with other VMAUWorkers, as |
| # well as the archive stage of cbuildbot. Make a private copy of |
| # the VM image, to avoid any conflict. |
| _, private_image_path = tempfile.mkstemp( |
| prefix="%s." % buildbot_constants.VM_DISK_PREFIX) |
| shutil.copy(self.vm_image_path, private_image_path) |
| self.TestInfo('Copied shared disk image %s to %s.' % |
| (self.vm_image_path, private_image_path)) |
| self.vm_image_path = private_image_path |
| # Although we will run the VM with |private_image_path|, we return |
| # |original_image_path|, because our return value is used to find the |
| # update files. And the update files are shared across the tests |
| # that share the original image. |
| return original_image_path |
| |
| def _HandleFail(self, log_directory, fail_directory): |
| parent_dir = os.path.dirname(fail_directory) |
| if not os.path.isdir(parent_dir): |
| os.makedirs(parent_dir) |
| |
| # Copy logs. Must be done before moving image, as this creates |
| # |fail_directory|. |
| try: |
| shutil.copytree(log_directory, fail_directory) |
| except shutil.Error as e: |
| logging.warning('Ignoring errors while copying logs: %s', e) |
| |
| # Copy VM disk image. |
| try: |
| self._StopVM() |
| shutil.move(self.vm_image_path, fail_directory) |
| except shutil.Error as e: |
| logging.warning('Ignoring errors while copying VM files: %s', e) |
| |
| def _UpdateCmd(self, log_directory, stateful_change): |
| """Update command to use.""" |
| cmd = ['%s/cros_run_test' % chromite_constants.CHROMITE_BIN_DIR, |
| '--board=%s' % self.board, |
| '--image-path=%s' % self.vm_image_path, |
| '--ssh-port=%s' % self._ssh_port, |
| self.graphics_flag, |
| '--host-cmd', '--', |
| os.path.join(constants.CROSTESTUTILS_DIR, 'image_to_live.sh'), |
| '--ssh_port=%s' % self._ssh_port, |
| '--noupdate_hostkey', |
| '--update_log=%s' % os.path.join(log_directory, 'update_engine.log'), |
| '--verify', |
| '--remote=127.0.0.1', |
| self.GetStatefulChangeFlag(stateful_change)] |
| if self.ssh_private_key: |
| cmd.append('--private_key=%s' % self.ssh_private_key) |
| return cmd |
| |
| def UpdateImage(self, image_path, src_image_path='', stateful_change='old', |
| proxy_port='', payload_signing_key=None): |
| """Updates VM image with image_path.""" |
| log_directory, fail_directory = self.GetNextResultsPath('update') |
| cmd = self._UpdateCmd(log_directory, stateful_change) |
| self.AppendUpdateFlags(cmd, image_path, src_image_path, proxy_port, |
| payload_signing_key) |
| self.TestInfo(self.GetUpdateMessage(image_path, src_image_path, True, |
| proxy_port)) |
| try: |
| self.RunUpdateCmd(cmd, log_directory) |
| except update_exception.UpdateException: |
| self._HandleFail(log_directory, fail_directory) |
| raise |
| |
| def UpdateUsingPayload(self, update_path, stateful_change='old', |
| proxy_port=None): |
| """Updates a vm image using image_to_live.sh.""" |
| log_directory, fail_directory = self.GetNextResultsPath('update') |
| cmd = self._UpdateCmd(log_directory, stateful_change) + [ |
| '--payload=%s' % update_path |
| ] |
| if proxy_port: |
| cmd.append('--proxy_port=%s' % proxy_port) |
| self.TestInfo(self.GetUpdateMessage(update_path, None, True, proxy_port)) |
| try: |
| self.RunUpdateCmd(cmd, log_directory) |
| except update_exception.UpdateException: |
| self._HandleFail(log_directory, fail_directory) |
| raise |
| |
| def AppendUpdateFlags(self, cmd, image_path, src_image_path, proxy_port, |
| payload_signing_key, for_vm=False): |
| """Appends common args to an update cmd defined by an array. |
| |
| Calls super function with for_vm set to True. See base class for |
| descriptions for arguments. |
| """ |
| super(VMAUWorker, self).AppendUpdateFlags( |
| cmd, image_path, src_image_path, proxy_port, payload_signing_key, |
| for_vm=True) |
| |
| def VerifyImage(self, unittest, percent_required_to_pass=100, test=''): |
| # VMAUWorker disregards |unittest| and |percent_required_to_pass|. |
| return self._VerifyImage(test) |
| |
| # pylint: disable=W0221 |
| def _VerifyImage(self, test=''): |
| """Runs vm smoke suite or any single test to verify image. |
| |
| Returns True upon success. Prints test output and returns False otherwise. |
| """ |
| log_directory, fail_directory = self.GetNextResultsPath('autotest_tests') |
| # image_to_live already verifies lsb-release matching. This is just |
| # for additional steps. |
| if not test: |
| test = self.verify_suite |
| command = ['./cros_run_test', |
| '--board=%s' % self.board, |
| '--image-path=%s' % self.vm_image_path, |
| '--ssh-port=%s' % self._ssh_port, |
| '--results-dir=%s' % log_directory, |
| '--autotest=%s' % test, |
| ] |
| if self.graphics_flag: |
| command.append(self.graphics_flag) |
| if self.whitelist_chrome_crashes: |
| command.append('--test_that-args=--whitelist-chrome-crashes') |
| if self.ssh_private_key: |
| command.append('--private-key=%s' % self.ssh_private_key) |
| |
| self.TestInfo('Running %s suite to verify image.' % test) |
| |
| result = cros_build_lib.RunCommand( |
| command, print_cmd=self.verbose, combine_stdout_stderr=True, |
| cwd=chromite_constants.CHROMITE_BIN_DIR, error_code_ok=True, |
| capture_output=not self.verbose) |
| # If the command failed or printed warnings, print the output. |
| # TODO(pwang): result.output should contain the output of the command even |
| # though verbose is set to print to stdout. |
| if result.output is None: |
| result.output = '' |
| if result.returncode != 0 or '@@@STEP_WARNINGS@@@' in result.output: |
| print(result.output) |
| self._HandleFail(log_directory, fail_directory) |
| |
| return result.returncode == 0 |