Chromium Code Reviews| Index: testing/chromoting/multi_machine_example/example_test_controller.py |
| diff --git a/testing/chromoting/multi_machine_example/example_test_controller.py b/testing/chromoting/multi_machine_example/example_test_controller.py |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..2f03f347f16ccff6e309e3469c5e3f0e5e0fed10 |
| --- /dev/null |
| +++ b/testing/chromoting/multi_machine_example/example_test_controller.py |
| @@ -0,0 +1,139 @@ |
| +#!/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_integration_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): |
|
anandc
2015/03/16 23:24:46
Add an Args section for 'path'?
Mike Meade
2015/03/16 23:35:50
Done.
|
| + """Returns the absolute path to the resource on the task machine. |
| + |
| + 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.""" |
|
anandc
2015/03/16 23:24:46
Missing ), or remove (
Mike Meade
2015/03/16 23:35:50
Good catch.
|
| + # 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. |
|
anandc
2015/03/16 23:24:46
It would help to explain a bit why it is not fully
Mike Meade
2015/03/16 23:35:50
I added it. There are 2 RPC calls (Poll and ReadOu
|
| + 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() |