#!/usr/bin/python # Copyright 2018 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. """Fake implementation of tast executable used by tast_unittest.py. In unit tests, this file is executed instead of the real tast executable by the tast.py server test. It: - reads a config.json file installed alongside this script, - parses command-line arguments passed by tast.py, - checks that the arguments included all required args specified by the config file for the given command (e.g. 'list', 'run' etc.), and - performs actions specified by the config file. """ import argparse import json import os import sys def main(): # pylint: disable=missing-docstring # The config file is written by TastTest._run_test in tast_unittest.py and # consists of a dict from command names (e.g. 'list', 'run', etc.) to # command definition dicts (see TastCommand in tast_unittest.py). path = os.path.join(os.path.dirname(os.path.realpath(__file__)), 'config.json') with open(path) as f: cfg = json.load(f) args = parse_args() cmd = cfg.get(args.command) if not cmd: raise RuntimeError('Unexpected command "%s"' % args.command) for arg in cmd.get('required_args', []): name, expected_value = arg.split('=') # argparse puts the repeated "pattern" args into a list of lists # instead of a single list. Pull the args back out in this case. val = getattr(args, name) if isinstance(val, list) and len(val) == 1 and isinstance(val[0], list): val = val[0] actual_value = str(val) if actual_value != expected_value: raise RuntimeError('Got arg %s with value "%s"; want "%s"' % (name, actual_value, expected_value)) if cmd.get('stdout'): sys.stdout.write(cmd['stdout']) if cmd.get('stderr'): sys.stderr.write(cmd['stderr']) if cmd.get('files_to_write'): for path, data in cmd['files_to_write'].iteritems(): dirname = os.path.dirname(path) if not os.path.exists(dirname): os.makedirs(dirname, 0755) with open(path, 'w') as f: f.write(data) sys.exit(cmd.get('status')) def parse_args(): """Parses command-line arguments. @returns: argparse.Namespace object containing parsed attributes. """ def to_bool(v): """Converts a boolean flag's value to a Python bool value.""" if v == 'true' or v == '': return True if v == 'false': return False raise argparse.ArgumentTypeError('"true" or "false" expected') parser = argparse.ArgumentParser() parser.add_argument('-logtime', type=to_bool, default=False, nargs='?') parser.add_argument('-verbose', type=to_bool, default=False, nargs='?') parser.add_argument('-version', action='version', version='1.0') subparsers = parser.add_subparsers(dest='command') def add_common_args(subparser): """Adds arguments shared between tast's 'list' and 'run' subcommands.""" subparser.add_argument('-build', type=to_bool, default=True, nargs='?') subparser.add_argument('-downloadprivatebundles', type=to_bool, default=False, nargs='?') subparser.add_argument('-devservers') subparser.add_argument('-remotebundledir') subparser.add_argument('-remotedatadir') subparser.add_argument('-remoterunner') subparser.add_argument('-sshretries') subparser.add_argument('target') subparser.add_argument('patterns', action='append', nargs='*') list_parser = subparsers.add_parser('list') add_common_args(list_parser) list_parser.add_argument('-json', type=to_bool, default=False, nargs='?') run_parser = subparsers.add_parser('run') add_common_args(run_parser) run_parser.add_argument('-resultsdir') run_parser.add_argument('-waituntilready') run_parser.add_argument('-timeout') return parser.parse_args() if __name__ == '__main__': main()