1"""\ 2Logic for control file generation. 3""" 4 5__author__ = 'showard@google.com (Steve Howard)' 6 7import re, os 8 9import common 10from autotest_lib.client.common_lib import error 11from autotest_lib.client.common_lib.cros import dev_server 12from autotest_lib.frontend.afe import model_logic 13from autotest_lib.server.cros.dynamic_suite import control_file_getter 14from autotest_lib.server.cros.dynamic_suite import suite_common 15import frontend.settings 16 17AUTOTEST_DIR = os.path.abspath(os.path.join( 18 os.path.dirname(frontend.settings.__file__), '..')) 19 20EMPTY_TEMPLATE = 'def step_init():\n' 21 22CLIENT_STEP_TEMPLATE = " job.next_step('step%d')\n" 23SERVER_STEP_TEMPLATE = ' step%d()\n' 24 25 26def _read_control_file(test): 27 """Reads the test control file from local disk. 28 29 @param test The test name. 30 31 @return The test control file string. 32 """ 33 control_file = open(os.path.join(AUTOTEST_DIR, test.path)) 34 control_contents = control_file.read() 35 control_file.close() 36 return control_contents 37 38 39def _add_boilerplate_to_nested_steps(lines): 40 """Adds boilerplate magic. 41 42 @param lines The string of lines. 43 44 @returns The string lines. 45 """ 46 # Look for a line that begins with 'def step_init():' while 47 # being flexible on spacing. If it's found, this will be 48 # a nested set of steps, so add magic to make it work. 49 # See client/bin/job.py's step_engine for more info. 50 if re.search(r'^(.*\n)*def\s+step_init\s*\(\s*\)\s*:', lines): 51 lines += '\nreturn locals() ' 52 lines += '# Boilerplate magic for nested sets of steps' 53 return lines 54 55 56def _format_step(item, lines): 57 """Format a line item. 58 @param item The item number. 59 @param lines The string of lines. 60 61 @returns The string lines. 62 """ 63 lines = _indent_text(lines, ' ') 64 lines = 'def step%d():\n%s' % (item, lines) 65 return lines 66 67 68def _get_tests_stanza(tests, is_server, prepend=None, append=None, 69 client_control_file='', test_source_build=None): 70 """ Constructs the control file test step code from a list of tests. 71 72 @param tests A sequence of test control files to run. 73 @param is_server bool, Is this a server side test? 74 @param prepend A list of steps to prepend to each client test. 75 Defaults to []. 76 @param append A list of steps to append to each client test. 77 Defaults to []. 78 @param client_control_file If specified, use this text as the body of a 79 final client control file to run after tests. is_server must be False. 80 @param test_source_build: Build to be used to retrieve test code. Default 81 to None. 82 83 @returns The control file test code to be run. 84 """ 85 assert not (client_control_file and is_server) 86 if not prepend: 87 prepend = [] 88 if not append: 89 append = [] 90 if test_source_build: 91 raw_control_files = _get_test_control_files_by_build( 92 tests, test_source_build) 93 else: 94 raw_control_files = [_read_control_file(test) for test in tests] 95 if client_control_file: 96 # 'return locals()' is always appended in case the user forgot, it 97 # is necessary to allow for nested step engine execution to work. 98 raw_control_files.append(client_control_file + '\nreturn locals()') 99 raw_steps = prepend + [_add_boilerplate_to_nested_steps(step) 100 for step in raw_control_files] + append 101 steps = [_format_step(index, step) 102 for index, step in enumerate(raw_steps)] 103 if is_server: 104 step_template = SERVER_STEP_TEMPLATE 105 footer = '\n\nstep_init()\n' 106 else: 107 step_template = CLIENT_STEP_TEMPLATE 108 footer = '' 109 110 header = ''.join(step_template % i for i in xrange(len(steps))) 111 return header + '\n' + '\n\n'.join(steps) + footer 112 113 114def _indent_text(text, indent): 115 """Indent given lines of python code avoiding indenting multiline 116 quoted content (only for triple " and ' quoting for now). 117 118 @param text The string of lines. 119 @param indent The indent string. 120 121 @return The indented string. 122 """ 123 regex = re.compile('(\\\\*)("""|\'\'\')') 124 125 res = [] 126 in_quote = None 127 for line in text.splitlines(): 128 # if not within a multinline quote indent the line contents 129 if in_quote: 130 res.append(line) 131 else: 132 res.append(indent + line) 133 134 while line: 135 match = regex.search(line) 136 if match: 137 # for an even number of backslashes before the triple quote 138 if len(match.group(1)) % 2 == 0: 139 if not in_quote: 140 in_quote = match.group(2)[0] 141 elif in_quote == match.group(2)[0]: 142 # if we found a matching end triple quote 143 in_quote = None 144 line = line[match.end():] 145 else: 146 break 147 148 return '\n'.join(res) 149 150 151def _get_profiler_commands(profilers, is_server, profile_only): 152 prepend, append = [], [] 153 if profile_only is not None: 154 prepend.append("job.default_profile_only = %r" % profile_only) 155 for profiler in profilers: 156 prepend.append("job.profilers.add('%s')" % profiler.name) 157 append.append("job.profilers.delete('%s')" % profiler.name) 158 return prepend, append 159 160 161def _sanity_check_generate_control(is_server, client_control_file): 162 """ 163 Sanity check some of the parameters to generate_control(). 164 165 This exists as its own function so that site_control_file may call it as 166 well from its own generate_control(). 167 168 @raises ValidationError if any of the parameters do not make sense. 169 """ 170 if is_server and client_control_file: 171 raise model_logic.ValidationError( 172 {'tests' : 'You cannot run server tests at the same time ' 173 'as directly supplying a client-side control file.'}) 174 175 176def generate_control(tests, is_server=False, profilers=(), 177 client_control_file='', profile_only=None, 178 test_source_build=None): 179 """ 180 Generate a control file for a sequence of tests. 181 182 @param tests A sequence of test control files to run. 183 @param is_server bool, Is this a server control file rather than a client? 184 @param profilers A list of profiler objects to enable during the tests. 185 @param client_control_file Contents of a client control file to run as the 186 last test after everything in tests. Requires is_server=False. 187 @param profile_only bool, should this control file run all tests in 188 profile_only mode by default 189 @param test_source_build: Build to be used to retrieve test code. Default 190 to None. 191 192 @returns The control file text as a string. 193 """ 194 _sanity_check_generate_control(is_server=is_server, 195 client_control_file=client_control_file) 196 control_file_text = EMPTY_TEMPLATE 197 prepend, append = _get_profiler_commands(profilers, is_server, profile_only) 198 control_file_text += _get_tests_stanza(tests, is_server, prepend, append, 199 client_control_file, 200 test_source_build) 201 return control_file_text 202 203 204def _get_test_control_files_by_build(tests, build, ignore_invalid_tests=False): 205 """Get the test control files that are available for the specified build. 206 207 @param tests A sequence of test objects to run. 208 @param build: unique name by which to refer to the image. 209 @param ignore_invalid_tests: flag on if unparsable tests are ignored. 210 211 @return: A sorted list of all tests that are in the build specified. 212 """ 213 raw_control_files = [] 214 # shortcut to avoid staging the image. 215 if not tests: 216 return raw_control_files 217 218 cfile_getter = _initialize_control_file_getter(build) 219 if suite_common.ENABLE_CONTROLS_IN_BATCH: 220 control_file_info_list = cfile_getter.get_suite_info() 221 222 for test in tests: 223 # Read and parse the control file 224 if suite_common.ENABLE_CONTROLS_IN_BATCH: 225 control_file = control_file_info_list[test.path] 226 else: 227 control_file = cfile_getter.get_control_file_contents( 228 test.path) 229 raw_control_files.append(control_file) 230 return raw_control_files 231 232 233def _initialize_control_file_getter(build): 234 """Get the remote control file getter. 235 236 @param build: unique name by which to refer to a remote build image. 237 238 @return: A control file getter object. 239 """ 240 # Stage the test artifacts. 241 try: 242 ds = dev_server.ImageServer.resolve(build) 243 ds_name = ds.hostname 244 build = ds.translate(build) 245 except dev_server.DevServerException as e: 246 raise ValueError('Could not resolve build %s: %s' % 247 (build, e)) 248 249 try: 250 ds.stage_artifacts(image=build, artifacts=['test_suites']) 251 except dev_server.DevServerException as e: 252 raise error.StageControlFileFailure( 253 'Failed to stage %s on %s: %s' % (build, ds_name, e)) 254 255 # Collect the control files specified in this build 256 return control_file_getter.DevServerGetter.create(build, ds) 257