1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
|
#!/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):
"""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()
|