|  | #!/usr/bin/env python | 
|  | # Copyright 2015 The Chromium Authors. All rights reserved. | 
|  | # Use of this source code is governed by a BSD-style license that can be | 
|  | # found in the LICENSE file. | 
|  |  | 
|  | """The test controller for the chromoting localhost browser_tests. | 
|  |  | 
|  | This test uses the legion framework to setup this controller which will run | 
|  | the chromoting_browser_tests on a task machine. This is intended to | 
|  | be an example Legion-based test for the chromoting team. | 
|  |  | 
|  | The controller will start a task machine to run browser_tests_launcher on. The | 
|  | output of these tests are streamed back to the test controller to be output | 
|  | on the controller's stdout and stderr channels. The final test output is then | 
|  | read and becomes the final output of the controller, mirroring the test's | 
|  | pass/fail result. | 
|  | """ | 
|  |  | 
|  | import argparse | 
|  | import logging | 
|  | import os | 
|  | import sys | 
|  | import time | 
|  |  | 
|  | # Map the legion directory so we can import the host controller. | 
|  | SRC_DIR = os.path.join('..', '..', '..') | 
|  | sys.path.append(os.path.join(SRC_DIR, 'testing')) | 
|  | from legion import test_controller | 
|  |  | 
|  |  | 
|  | class ExampleController(test_controller.TestController): | 
|  | """The test controller for the Chromoting browser_tests.""" | 
|  |  | 
|  | def __init__(self): | 
|  | super(ExampleController, self).__init__() | 
|  | self.task = None | 
|  | self.args = None | 
|  |  | 
|  | def RunTest(self): | 
|  | """Main method to run the test code.""" | 
|  | self.ParseArgs() | 
|  | self.CreateTask() | 
|  | self.TestIntegrationTests() | 
|  |  | 
|  | def CreateBrowserTestsLauncherCommand(self): | 
|  | return [ | 
|  | 'python', | 
|  | self.TaskAbsPath('../browser_tests_launcher.py'), | 
|  | '--commands_file', self.TaskAbsPath(self.args.commands_file), | 
|  | '--prod_dir', self.TaskAbsPath(self.args.prod_dir), | 
|  | '--cfg_file', self.TaskAbsPath(self.args.cfg_file), | 
|  | '--me2me_manifest_file', self.TaskAbsPath( | 
|  | self.args.me2me_manifest_file), | 
|  | '--it2me_manifest_file', self.TaskAbsPath( | 
|  | self.args.it2me_manifest_file), | 
|  | '--user_profile_dir', self.args.user_profile_dir, | 
|  | ] | 
|  |  | 
|  | def TaskAbsPath(self, path): | 
|  | """Returns the absolute path to the resource on the task machine. | 
|  |  | 
|  | Args: | 
|  | path: The relative path to the resource. | 
|  |  | 
|  | Since the test controller and the task machines run in different tmp dirs | 
|  | on different machines the absolute path cannot be calculated correctly on | 
|  | this machine. This function maps the relative path (from this directory) | 
|  | to an absolute path on the task machine. | 
|  | """ | 
|  | return self.task.rpc.AbsPath(path) | 
|  |  | 
|  | def CreateTask(self): | 
|  | """Creates a task object and sets the proper values.""" | 
|  | self.task = self.CreateNewTask( | 
|  | isolated_hash=self.args.task_machine, | 
|  | dimensions={'os': 'Ubuntu-14.04', 'pool': 'Chromoting'}) | 
|  | self.task.Create() | 
|  | self.task.WaitForConnection() | 
|  |  | 
|  | def ParseArgs(self): | 
|  | """Gets the command line args.""" | 
|  | parser = argparse.ArgumentParser() | 
|  | parser.add_argument('--task_machine', | 
|  | help='isolated hash of the task machine.') | 
|  | # The rest of the args are taken from | 
|  | # testing/chromoting/browser_tests_launcher.py. | 
|  | parser.add_argument('-f', '--commands_file', | 
|  | help='path to file listing commands to be launched.') | 
|  | parser.add_argument('-p', '--prod_dir', | 
|  | help='path to folder having product and test binaries.') | 
|  | parser.add_argument('-c', '--cfg_file', | 
|  | help='path to test host config file.') | 
|  | parser.add_argument('--me2me_manifest_file', | 
|  | help='path to me2me host manifest file.') | 
|  | parser.add_argument('--it2me_manifest_file', | 
|  | help='path to it2me host manifest file.') | 
|  | parser.add_argument( | 
|  | '-u', '--user_profile_dir', | 
|  | help='path to user-profile-dir, used by connect-to-host tests.') | 
|  | self.args, _ = parser.parse_known_args() | 
|  |  | 
|  | def TestIntegrationTests(self): | 
|  | """Runs the integration tests via browser_tests_launcher.py.""" | 
|  | # Create a process object, configure it, and start it. | 
|  | # All interactions with the process are based on this "proc" key. | 
|  | proc = self.task.rpc.subprocess.Process( | 
|  | self.CreateBrowserTestsLauncherCommand()) | 
|  | # Set the cwd to browser_tests_launcher relative to this directory. | 
|  | # This allows browser_test_launcher to use relative paths. | 
|  | self.task.rpc.subprocess.SetCwd(proc, '../') | 
|  | # Set the task verbosity to true to allow stdout/stderr to be echo'ed to | 
|  | # run_task's stdout/stderr on the task machine. This can assist in | 
|  | # debugging. | 
|  | self.task.rpc.subprocess.SetVerbose(proc) | 
|  | # Set the process as detached to create it in a new process group. | 
|  | self.task.rpc.subprocess.SetDetached(proc) | 
|  | # Start the actual process on the task machine. | 
|  | self.task.rpc.subprocess.Start(proc) | 
|  |  | 
|  | # Collect the stdout/stderr and emit it from this controller while the | 
|  | # process is running. | 
|  | while self.task.rpc.subprocess.Poll(proc) is None: | 
|  | # Output the test's stdout and stderr in semi-realtime. | 
|  | # This is not true realtime due to the RPC calls and the 1s sleep. | 
|  | stdout, stderr = self.task.rpc.subprocess.ReadOutput(proc) | 
|  | if stdout: | 
|  | sys.stdout.write(stdout) | 
|  | if stderr: | 
|  | sys.stderr.write(stderr) | 
|  | time.sleep(1) | 
|  |  | 
|  | # Get the return code, clean up the process object. | 
|  | returncode = self.task.rpc.subprocess.GetReturncode(proc) | 
|  | self.task.rpc.subprocess.Delete(proc) | 
|  |  | 
|  | # Pass or fail depending on the return code from the browser_tests_launcher. | 
|  | if returncode != 0: | 
|  | raise AssertionError('browser_tests_launcher failed with return code ' | 
|  | '%i' % returncode) | 
|  |  | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | ExampleController().RunController() |