| import argparse |
| import ast |
| import logging |
| import os |
| import shlex |
| import sys |
| |
| |
| class autoserv_parser(object): |
| """Custom command-line options parser for autoserv. |
| |
| We can't use the general getopt methods here, as there will be unknown |
| extra arguments that we pass down into the control file instead. |
| Thus we process the arguments by hand, for which we are duly repentant. |
| Making a single function here just makes it harder to read. Suck it up. |
| """ |
| def __init__(self): |
| self.args = sys.argv[1:] |
| self.parser = argparse.ArgumentParser( |
| usage='%(prog)s [options] [control-file]') |
| self.setup_options() |
| |
| # parse an empty list of arguments in order to set self.options |
| # to default values so that codepaths that assume they are always |
| # reached from an autoserv process (when they actually are not) |
| # will still work |
| self.options = self.parser.parse_args(args=[]) |
| |
| |
| def setup_options(self): |
| """Setup options to call autoserv command. |
| """ |
| self.parser.add_argument('-m', action='store', type=str, |
| dest='machines', |
| help='list of machines') |
| self.parser.add_argument('-M', action='store', type=str, |
| dest='machines_file', |
| help='list of machines from file') |
| self.parser.add_argument('-c', action='store_true', |
| dest='client', default=False, |
| help='control file is client side') |
| self.parser.add_argument('-s', action='store_true', |
| dest='server', default=False, |
| help='control file is server side') |
| self.parser.add_argument('-r', action='store', type=str, |
| dest='results', default=None, |
| help='specify results directory') |
| self.parser.add_argument('-l', action='store', type=str, |
| dest='label', default='', |
| help='label for the job') |
| self.parser.add_argument('-G', action='store', type=str, |
| dest='group_name', default='', |
| help='The host_group_name to store in keyvals') |
| self.parser.add_argument('-u', action='store', type=str, |
| dest='user', |
| default=os.environ.get('USER'), |
| help='username for the job') |
| self.parser.add_argument('-P', action='store', type=str, |
| dest='parse_job', |
| default='', |
| help=('DEPRECATED.' |
| ' Use --execution-tag instead.')) |
| self.parser.add_argument('--execution-tag', action='store', |
| type=str, dest='execution_tag', |
| default='', |
| help=('Accessible in control files as job.tag;' |
| ' Defaults to the value passed to -P.')) |
| self.parser.add_argument('-v', action='store_true', |
| dest='verify', default=False, |
| help='verify the machines only') |
| self.parser.add_argument('-R', action='store_true', |
| dest='repair', default=False, |
| help='repair the machines') |
| self.parser.add_argument('-C', '--cleanup', action='store_true', |
| default=False, |
| help='cleanup all machines after the job') |
| self.parser.add_argument('--provision', action='store_true', |
| default=False, |
| help='Provision the machine.') |
| self.parser.add_argument('--job-labels', action='store', |
| help='Comma seperated job labels.') |
| self.parser.add_argument('-T', '--reset', action='store_true', |
| default=False, |
| help=('Reset (cleanup and verify) all machines' |
| ' after the job')) |
| self.parser.add_argument('-n', action='store_true', |
| dest='no_tee', default=False, |
| help='no teeing the status to stdout/err') |
| self.parser.add_argument('-N', action='store_true', |
| dest='no_logging', default=False, |
| help='no logging') |
| self.parser.add_argument('--verbose', action='store_true', |
| help=('Include DEBUG messages in console ' |
| 'output')) |
| self.parser.add_argument('--no_console_prefix', action='store_true', |
| help=('Disable the logging prefix on console ' |
| 'output')) |
| self.parser.add_argument('-p', '--write-pidfile', action='store_true', |
| dest='write_pidfile', default=False, |
| help=('write pidfile (pidfile name is ' |
| 'determined by --pidfile-label')) |
| self.parser.add_argument('--pidfile-label', action='store', |
| default='autoserv', |
| help=('Determines filename to use as pidfile ' |
| '(if -p is specified). Pidfile will be ' |
| '.<label>_execute. Default to ' |
| 'autoserv.')) |
| self.parser.add_argument('--use-existing-results', action='store_true', |
| help=('Indicates that autoserv is working with' |
| ' an existing results directory')) |
| self.parser.add_argument('-a', '--args', dest='args', |
| help='additional args to pass to control file') |
| self.parser.add_argument('--ssh-user', action='store', |
| type=str, dest='ssh_user', default='root', |
| help='specify the user for ssh connections') |
| self.parser.add_argument('--ssh-port', action='store', |
| type=int, dest='ssh_port', default=22, |
| help=('specify the port to use for ssh ' |
| 'connections')) |
| self.parser.add_argument('--ssh-pass', action='store', |
| type=str, dest='ssh_pass', |
| default='', |
| help=('specify the password to use for ssh ' |
| 'connections')) |
| self.parser.add_argument('--install-in-tmpdir', action='store_true', |
| dest='install_in_tmpdir', default=False, |
| help=('by default install autotest clients in ' |
| 'a temporary directory')) |
| self.parser.add_argument('--collect-crashinfo', action='store_true', |
| dest='collect_crashinfo', default=False, |
| help='just run crashinfo collection') |
| self.parser.add_argument('--control-filename', action='store', |
| type=str, default=None, |
| help=('filename to use for the server control ' |
| 'file in the results directory')) |
| self.parser.add_argument('--verify_job_repo_url', action='store_true', |
| dest='verify_job_repo_url', default=False, |
| help=('Verify that the job_repo_url of the ' |
| 'host has staged packages for the job.')) |
| self.parser.add_argument('--no_collect_crashinfo', action='store_true', |
| dest='skip_crash_collection', default=False, |
| help=('Turns off crash collection to shave ' |
| 'time off test runs.')) |
| self.parser.add_argument('--disable_sysinfo', action='store_true', |
| dest='disable_sysinfo', default=False, |
| help=('Turns off sysinfo collection to shave ' |
| 'time off test runs.')) |
| self.parser.add_argument('--ssh_verbosity', action='store', |
| dest='ssh_verbosity', default=0, |
| type=str, choices=['0', '1', '2', '3'], |
| help=('Verbosity level for ssh, between 0 ' |
| 'and 3 inclusive. ' |
| '[default: %(default)s]')) |
| self.parser.add_argument('--ssh_options', action='store', |
| dest='ssh_options', default='', |
| help=('A string giving command line flags ' |
| 'that will be included in ssh commands')) |
| self.parser.add_argument('--require-ssp', action='store_true', |
| dest='require_ssp', default=False, |
| help=('Force the autoserv process to run with ' |
| 'server-side packaging')) |
| self.parser.add_argument('--no_use_packaging', action='store_true', |
| dest='no_use_packaging', default=False, |
| help=('Disable install modes that use the ' |
| 'packaging system.')) |
| self.parser.add_argument('--source_isolate', action='store', |
| type=str, default='', |
| dest='isolate', |
| help=('Hash for isolate containing build ' |
| 'contents needed for server-side ' |
| 'packaging. Takes precedence over ' |
| 'test_source_build, if present.')) |
| self.parser.add_argument('--test_source_build', action='store', |
| type=str, default='', |
| dest='test_source_build', |
| help=('Alternative build that contains the ' |
| 'test code for server-side packaging. ' |
| 'Default is to use the build on the ' |
| 'target DUT.')) |
| self.parser.add_argument('--parent_job_id', action='store', |
| type=str, default=None, |
| dest='parent_job_id', |
| help=('ID of the parent job. Default to None ' |
| 'if the job does not have a parent job')) |
| self.parser.add_argument('--host_attributes', action='store', |
| dest='host_attributes', default='{}', |
| help=('Host attribute to be applied to all ' |
| 'machines/hosts for this autoserv run. ' |
| 'Must be a string-encoded dict. ' |
| 'Example: {"key1":"value1", "key2":' |
| '"value2"}')) |
| self.parser.add_argument('--lab', action='store', type=str, |
| dest='lab', default='', |
| help=argparse.SUPPRESS) |
| self.parser.add_argument('--cloud_trace_context', type=str, default='', |
| action='store', dest='cloud_trace_context', |
| help=('Global trace context to configure ' |
| 'emission of data to Cloud Trace.')) |
| self.parser.add_argument('--cloud_trace_context_enabled', type=str, |
| default='False', action='store', |
| dest='cloud_trace_context_enabled', |
| help=('Global trace context to configure ' |
| 'emission of data to Cloud Trace.')) |
| self.parser.add_argument( |
| '--host-info-subdir', |
| default='host_info_store', |
| help='Optional path to a directory containing host ' |
| 'information for the machines. The path is relative to ' |
| 'the results directory (see -r) and must be inside ' |
| 'the directory.' |
| ) |
| self.parser.add_argument( |
| '--local-only-host-info', |
| default='False', |
| help='By default, host status are immediately reported back to ' |
| 'the AFE, shadowing the updates to disk. This flag ' |
| 'disables the AFE interaction completely, and assumes ' |
| 'that initial host information is supplied to autoserv. ' |
| 'See also: --host-info-subdir', |
| ) |
| self.parser.add_argument( |
| '--control-name', |
| action='store', |
| help='NAME attribute of the control file to stage and run. ' |
| 'This overrides the control file provided via the ' |
| 'positional args.', |
| ) |
| self.parser.add_argument( |
| '--sync-offload-dir', action='store', type=str, default='', |
| help='Relative path from results directory to the sub-directory ' |
| 'which should be offloaded synchronously', |
| ) |
| self.parser.add_argument( |
| '--ssp-base-image-name', |
| action='store', |
| help='Name of the base container image to use for' |
| ' Server Side Packaging (SSP). Only meaningful when SSP is' |
| ' enabled. The default value is provided via the global' |
| ' config setting for AUTOSERV/container_base_name.' |
| ) |
| |
| # |
| # Warning! Please read before adding any new arguments! |
| # |
| # New arguments will be ignored if a test runs with server-side |
| # packaging and if the test source build does not have the new |
| # arguments. |
| # |
| # New arguments should NOT set action to `store_true`. A workaround is |
| # to use string value of `True` or `False`, then convert them to boolean |
| # in code. |
| # The reason is that parse_args will always ignore the argument name and |
| # value. An unknown argument without a value will lead to positional |
| # argument being removed unexpectedly. |
| # |
| |
| |
| def parse_args(self): |
| """Parse and process command line arguments. |
| """ |
| # Positional arguments from the end of the command line will be included |
| # in the list of unknown_args. |
| self.options, unknown_args = self.parser.parse_known_args() |
| # Filter out none-positional arguments |
| removed_args = [] |
| while unknown_args and unknown_args[0] and unknown_args[0][0] == '-': |
| removed_args.append(unknown_args.pop(0)) |
| # Always assume the argument has a value. |
| if unknown_args: |
| removed_args.append(unknown_args.pop(0)) |
| if removed_args: |
| logging.warn('Unknown arguments are removed from the options: %s', |
| removed_args) |
| |
| self.args = unknown_args + shlex.split(self.options.args or '') |
| |
| self.options.host_attributes = ast.literal_eval( |
| self.options.host_attributes) |
| if self.options.lab and self.options.host_attributes: |
| logging.warn( |
| '--lab and --host-attributes are mutually exclusive. ' |
| 'Ignoring custom host attributes: %s', |
| str(self.options.host_attributes)) |
| self.options.host_attributes = [] |
| |
| self.options.local_only_host_info = _interpret_bool_arg( |
| self.options.local_only_host_info) |
| if not self.options.execution_tag: |
| self.options.execution_tag = self.options.parse_job |
| |
| |
| def _interpret_bool_arg(value): |
| return value.lower() in {'yes', 'true'} |
| |
| |
| # create the one and only one instance of autoserv_parser |
| autoserv_parser = autoserv_parser() |