1# Copyright 2015 The Chromium OS Authors. All rights reserved.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
5"""Argument processing for the DUT deployment tool.
6
7The argument processing is mostly a conventional client of
8`argparse`, except that if the command is invoked without required
9arguments, code here will start a line-oriented text dialog with the
10user to get the arguments.
11
12These are the arguments:
13  * (required) Board of the DUTs to be deployed.
14  * (required) Hostnames of the DUTs to be deployed.
15  * (optional) Version of the test image to be made the stable
16    repair image for the board to be deployed.  If omitted, the
17    existing setting is retained.
18
19The interactive dialog is invoked if the board and hostnames
20are omitted from the command line.
21
22"""
23
24import argparse
25import collections
26import csv
27import datetime
28import os
29import re
30import subprocess
31import sys
32
33import dateutil.tz
34
35import common
36from autotest_lib.server.hosts import servo_host
37
38# _BUILD_URI_FORMAT
39# A format template for a Google storage URI that designates
40# one build.  The template is to be filled in with a board
41# name and build version number.
42
43_BUILD_URI_FORMAT = 'gs://chromeos-image-archive/%s-release/%s'
44
45
46# _BUILD_PATTERNS
47# For user convenience, argument parsing allows various formats
48# for build version strings.  The function _normalize_build_name()
49# is used to convert the recognized syntaxes into the name as
50# it appears in Google storage.
51#
52# _BUILD_PATTERNS describe the recognized syntaxes for user-supplied
53# build versions, and information about how to convert them.  See the
54# normalize function for details.
55#
56# For user-supplied build versions, the following forms are supported:
57#   ####        - Indicates a canary; equivalent to ####.0.0.
58#   ####.#.#    - A full build version without the leading R##- prefix.
59#   R##-###.#.# - Canonical form of a build version.
60
61_BUILD_PATTERNS = [
62    (re.compile(r'^R\d+-\d+\.\d+\.\d+$'),   None),
63    (re.compile(r'^\d+\.\d+\.\d+$'),        'LATEST-%s'),
64    (re.compile(r'^\d+$'),                  'LATEST-%s.0.0'),
65]
66
67
68# _VALID_HOSTNAME_PATTERNS
69# A list of REs describing patterns that are acceptable as names
70# for DUTs in the test lab.  Names that don't match one of the
71# patterns will be rejected as invalid.
72
73_VALID_HOSTNAME_PATTERNS = [
74    re.compile(r'chromeos\d+-row\d+-rack\d+-host\d+')
75]
76
77
78# _EXPECTED_NUMBER_OF_HOST_INFO
79# The number of items per line when parsing the hostname_file csv file.
80_EXPECTED_NUMBER_OF_HOST_INFO = 8
81
82# HostInfo
83# Namedtuple to store host info for processing when creating host in the afe.
84HostInfo = collections.namedtuple('HostInfo', ['hostname', 'host_attr_dict'])
85
86
87def _build_path_exists(board, buildpath):
88    """Return whether a given build file exists in Google storage.
89
90    The `buildpath` refers to a specific file associated with
91    release builds for `board`.  The path may be one of the "LATEST"
92    files (e.g. "LATEST-7356.0.0"), or it could refer to a build
93    artifact (e.g. "R46-7356.0.0/image.zip").
94
95    The function constructs the full GS URI from the arguments, and
96    then tests for its existence with `gsutil ls`.
97
98    @param board        Board to be tested.
99    @param buildpath    Partial path of a file in Google storage.
100
101    @return Return a true value iff the designated file exists.
102    """
103    try:
104        gsutil_cmd = [
105                'gsutil', 'ls',
106                _BUILD_URI_FORMAT % (board, buildpath)
107        ]
108        status = subprocess.call(gsutil_cmd,
109                                 stdout=open('/dev/null', 'w'),
110                                 stderr=subprocess.STDOUT)
111        return status == 0
112    except:
113        return False
114
115
116def _normalize_build_name(board, build):
117    """Convert a user-supplied build version to canonical form.
118
119    Canonical form looks like  R##-####.#.#, e.g. R46-7356.0.0.
120    Acceptable user-supplied forms are describe under
121    _BUILD_PATTERNS, above.  The returned value will be the name of
122    a directory containing build artifacts from a release builder
123    for the board.
124
125    Walk through `_BUILD_PATTERNS`, trying to convert a user
126    supplied build version name into a directory name for valid
127    build artifacts.  Searching stops at the first pattern matched,
128    regardless of whether the designated build actually exists.
129
130    `_BUILD_PATTERNS` is a list of tuples.  The first element of the
131    tuple is an RE describing a valid user input.  The second
132    element of the tuple is a format pattern for a "LATEST" filename
133    in storage that can be used to obtain the full build version
134    associated with the user supplied version.  If the second element
135    is `None`, the user supplied build version is already in canonical
136    form.
137
138    @param board    Board to be tested.
139    @param build    User supplied version name.
140
141    @return Return the name of a directory in canonical form, or
142            `None` if the build doesn't exist.
143    """
144    for regex, fmt in _BUILD_PATTERNS:
145        if not regex.match(build):
146            continue
147        if fmt is not None:
148            try:
149                gsutil_cmd = [
150                    'gsutil', 'cat',
151                    _BUILD_URI_FORMAT % (board, fmt % build)
152                ]
153                return subprocess.check_output(
154                        gsutil_cmd, stderr=open('/dev/null', 'w'))
155            except:
156                return None
157        elif _build_path_exists(board, '%s/image.zip' % build):
158            return build
159        else:
160            return None
161    return None
162
163
164def _validate_board(board):
165    """Return whether a given board exists in Google storage.
166
167    For purposes of this function, a board exists if it has a
168    "LATEST-master" file in its release builder's directory.
169
170    N.B. For convenience, this function prints an error message
171    on stderr in certain failure cases.  This is currently useful
172    for argument processing, but isn't really ideal if the callers
173    were to get more complicated.
174
175    @param board    The board to be tested for existence.
176    @return Return a true value iff the board exists.
177    """
178    # In this case, the board doesn't exist, but we don't want
179    # an error message.
180    if board is None:
181        return False
182    # Check Google storage; report failures on stderr.
183    if _build_path_exists(board, 'LATEST-master'):
184        return True
185    else:
186        sys.stderr.write('Board %s doesn\'t exist.\n' % board)
187        return False
188
189
190def _validate_build(board, build):
191    """Return whether a given build exists in Google storage.
192
193    N.B. For convenience, this function prints an error message
194    on stderr in certain failure cases.  This is currently useful
195    for argument processing, but isn't really ideal if the callers
196    were to get more complicated.
197
198    @param board    The board to be tested for a build
199    @param build    The version of the build to be tested for.  This
200                    build may be in a user-specified (non-canonical)
201                    form.
202    @return If the given board+build exists, return its canonical
203            (normalized) version string.  If the build doesn't
204            exist, return a false value.
205    """
206    canonical_build = _normalize_build_name(board, build)
207    if not canonical_build:
208        sys.stderr.write(
209                'Build %s is not a valid build version for %s.\n' %
210                (build, board))
211    return canonical_build
212
213
214def _validate_hostname(hostname):
215    """Return whether a given hostname is valid for the test lab.
216
217    This is a sanity check meant to guarantee that host names follow
218    naming requirements for the test lab.
219
220    N.B. For convenience, this function prints an error message
221    on stderr in certain failure cases.  This is currently useful
222    for argument processing, but isn't really ideal if the callers
223    were to get more complicated.
224
225    @param hostname The host name to be checked.
226    @return Return a true value iff the hostname is valid.
227    """
228    for p in _VALID_HOSTNAME_PATTERNS:
229        if p.match(hostname):
230            return True
231    sys.stderr.write(
232            'Hostname %s doesn\'t match a valid location name.\n' %
233                hostname)
234    return False
235
236
237def _is_hostname_file_valid(hostname_file):
238    """Check that the hostname file is valid.
239
240    The hostname file is deemed valid if:
241     - the file exists.
242     - the file is non-empty.
243
244    @param hostname_file  Filename of the hostname file to check.
245
246    @return `True` if the hostname file is valid, False otherse.
247    """
248    return os.path.exists(hostname_file) and os.path.getsize(hostname_file) > 0
249
250
251def _validate_arguments(arguments):
252    """Check command line arguments, and account for defaults.
253
254    Check that all command-line argument constraints are satisfied.
255    If errors are found, they are reported on `sys.stderr`.
256
257    If there are any fields with defined defaults that couldn't be
258    calculated when we constructed the argument parser, calculate
259    them now.
260
261    @param arguments  Parsed results from
262                      `ArgumentParser.parse_args()`.
263    @return Return `True` if there are no errors to report, or
264            `False` if there are.
265    """
266    # If both hostnames and hostname_file are specified, complain about that.
267    if arguments.hostnames and arguments.hostname_file:
268        sys.stderr.write(
269                'DUT hostnames and hostname file both specified, only '
270                'specify one or the other.\n')
271        return False
272    if (arguments.hostname_file and
273        not _is_hostname_file_valid(arguments.hostname_file)):
274        sys.stderr.write(
275                'Specified hostname file must exist and be non-empty.\n')
276        return False
277    if (not arguments.hostnames and not arguments.hostname_file and
278            (arguments.board or arguments.build)):
279        sys.stderr.write(
280                'DUT hostnames are required with board or build.\n')
281        return False
282    if arguments.board is not None:
283        if not _validate_board(arguments.board):
284            return False
285        if (arguments.build is not None and
286                not _validate_build(arguments.board, arguments.build)):
287            return False
288    return True
289
290
291def _read_with_prompt(input, prompt):
292    """Print a prompt and then read a line of text.
293
294    @param input File-like object from which to read the line.
295    @param prompt String to print to stderr prior to reading.
296    @return Returns a string, stripped of whitespace.
297    """
298    full_prompt = '%s> ' % prompt
299    sys.stderr.write(full_prompt)
300    return input.readline().strip()
301
302
303def _read_board(input, default_board):
304    """Read a valid board name from user input.
305
306    Prompt the user to supply a board name, and read one line.  If
307    the line names a valid board, return the board name.  If the
308    line is blank and `default_board` is a non-empty string, returns
309    `default_board`.  Retry until a valid input is obtained.
310
311    `default_board` isn't checked; the caller is responsible for
312    ensuring its validity.
313
314    @param input          File-like object from which to read the
315                          board.
316    @param default_board  Value to return if the user enters a
317                          blank line.
318    @return Returns `default_board` or a validated board name.
319    """
320    if default_board:
321        board_prompt = 'board name [%s]' % default_board
322    else:
323        board_prompt = 'board name'
324    new_board = None
325    while not _validate_board(new_board):
326        new_board = _read_with_prompt(input, board_prompt).lower()
327        if new_board:
328            sys.stderr.write('Checking for valid board.\n')
329        elif default_board:
330            return default_board
331    return new_board
332
333
334def _read_build(input, board):
335    """Read a valid build version from user input.
336
337    Prompt the user to supply a build version, and read one line.
338    If the line names an existing version for the given board,
339    return the canonical build version.  If the line is blank,
340    return `None` (indicating the build shouldn't change).
341
342    @param input    File-like object from which to read the build.
343    @param board    Board for the build.
344    @return Returns canonical build version, or `None`.
345    """
346    build = False
347    prompt = 'build version (optional)'
348    while not build:
349        build = _read_with_prompt(input, prompt)
350        if not build:
351            return None
352        sys.stderr.write('Checking for valid build.\n')
353        build = _validate_build(board, build)
354    return build
355
356
357def _read_hostnames(input):
358    """Read a list of host names from user input.
359
360    Prompt the user to supply a list of host names.  Any number of
361    lines are allowed; input is terminated at the first blank line.
362    Any number of hosts names are allowed on one line.  Names are
363    separated by whitespace.
364
365    Only valid host names are accepted.  Invalid host names are
366    ignored, and a warning is printed.
367
368    @param input    File-like object from which to read the names.
369    @return Returns a list of validated host names.
370    """
371    hostnames = []
372    y_n = 'yes'
373    while not 'no'.startswith(y_n):
374        sys.stderr.write('enter hosts (blank line to end):\n')
375        while True:
376            new_hosts = input.readline().strip().split()
377            if not new_hosts:
378                break
379            for h in new_hosts:
380                if _validate_hostname(h):
381                    hostnames.append(h)
382        if not hostnames:
383            sys.stderr.write('Must provide at least one hostname.\n')
384            continue
385        prompt = 'More hosts? [y/N]'
386        y_n = _read_with_prompt(input, prompt).lower() or 'no'
387    return hostnames
388
389
390def _read_arguments(input, arguments):
391    """Dialog to read all needed arguments from the user.
392
393    The user is prompted in turn for a board, a build, and
394    hostnames.  Responses are stored in `arguments`.  The user is
395    given opportunity to accept or reject the responses before
396    continuing.
397
398    @param input      File-like object from which to read user
399                      responses.
400    @param arguments  Namespace object returned from
401                      `ArgumentParser.parse_args()`.  Results are
402                      stored here.
403    """
404    y_n = 'no'
405    while not 'yes'.startswith(y_n):
406        arguments.board = _read_board(input, arguments.board)
407        arguments.build = _read_build(input, arguments.board)
408        prompt = '%s build %s? [Y/n]' % (
409                arguments.board, arguments.build)
410        y_n = _read_with_prompt(input, prompt).lower() or 'yes'
411    arguments.hostnames = _read_hostnames(input)
412
413
414def get_default_logdir_name(arguments):
415    """Get default log directory name.
416
417    @param arguments  Namespace object returned from argument parsing.
418    @return  A filename as a string.
419    """
420    return '{time}-{board}'.format(
421        time=arguments.start_time.isoformat(),
422        board=arguments.board)
423
424
425class _ArgumentParser(argparse.ArgumentParser):
426    """ArgumentParser extended with boolean option pairs."""
427
428    # Arguments required when adding an option pair.
429    _REQUIRED_PAIR_ARGS = {'dest', 'default'}
430
431    def add_argument_pair(self, yes_flags, no_flags, **kwargs):
432        """Add a pair of argument flags for a boolean option.
433
434        @param yes_flags  Iterable of flags to turn option on.
435                          May also be a single string.
436        @param no_flags   Iterable of flags to turn option off.
437                          May also be a single string.
438        @param *kwargs    Other arguments to pass to add_argument()
439        """
440        missing_args = self._REQUIRED_PAIR_ARGS - set(kwargs)
441        if missing_args:
442            raise ValueError("Argument pair must have explicit %s"
443                             % (', '.join(missing_args),))
444
445        if isinstance(yes_flags, (str, unicode)):
446            yes_flags = [yes_flags]
447        if isinstance(no_flags, (str, unicode)):
448            no_flags = [no_flags]
449
450        self.add_argument(*yes_flags, action='store_true', **kwargs)
451        self.add_argument(*no_flags, action='store_false', **kwargs)
452
453
454def _make_common_parser(command_name):
455    """Create argument parser for common arguments.
456
457    @param command_name The command name.
458    @return ArgumentParser instance.
459    """
460    parser = _ArgumentParser(
461            prog=command_name,
462            description='Install a test image on newly deployed DUTs')
463    # frontend.AFE(server=None) will use the default web server,
464    # so default for --web is `None`.
465    parser.add_argument('-w', '--web', metavar='SERVER', default=None,
466                        help='specify web server')
467    parser.add_argument('-d', '--dir', dest='logdir',
468                        help='directory for logs')
469    parser.add_argument('-i', '--build',
470                        help='select stable test build version')
471    parser.add_argument('-n', '--noinstall', action='store_true',
472                        help='skip install (for script testing)')
473    parser.add_argument('-s', '--nostage', action='store_true',
474                        help='skip staging test image (for script testing)')
475    parser.add_argument('-t', '--nostable', action='store_true',
476                        help='skip changing stable test image '
477                             '(for script testing)')
478    parser.add_argument('-f', '--hostname_file',
479                        help='CSV file that contains a list of hostnames and '
480                             'their details to install with.')
481    parser.add_argument('board', nargs='?', metavar='BOARD',
482                        help='board for DUTs to be installed')
483    parser.add_argument('hostnames', nargs='*', metavar='HOSTNAME',
484                        help='host names of DUTs to be installed')
485    return parser
486
487
488def _add_upload_argument_pair(parser, default):
489    """Add option pair for uploading logs.
490
491    @param parser   _ArgumentParser instance.
492    @param default  Default option value.
493    """
494    parser.add_argument_pair('--upload', '--noupload', dest='upload',
495                             default=default,
496                             help='upload logs to GS bucket',)
497
498
499def _parse_hostname_file_line(hostname_file_row):
500    """
501    Parse a line from the hostname_file and return a dict of the info.
502
503    @param hostname_file_row: List of strings from each line in the hostname
504                              file.
505
506    @returns a NamedTuple of (hostname, host_attr_dict).  host_attr_dict is a
507             dict of host attributes for the host.
508    """
509    if len(hostname_file_row) != _EXPECTED_NUMBER_OF_HOST_INFO:
510        raise Exception('hostname_file line has unexpected number of items '
511                        '%d (expect %d): %s' %
512                        (len(hostname_file_row), _EXPECTED_NUMBER_OF_HOST_INFO,
513                         hostname_file_row))
514    # The file will have the info in the following order:
515    # 0: board
516    # 1: dut hostname
517    # 2: dut/v4 mac address
518    # 3: dut ip
519    # 4: labstation hostname
520    # 5: servo serial
521    # 6: servo mac address
522    # 7: servo ip
523    return HostInfo(
524            hostname=hostname_file_row[1],
525            host_attr_dict={servo_host.SERVO_HOST_ATTR: hostname_file_row[4],
526                            servo_host.SERVO_SERIAL_ATTR: hostname_file_row[5]})
527
528
529def parse_hostname_file(hostname_file):
530    """
531    Parse the hostname_file and return a list of dicts for each line.
532
533    @param hostname_file:  CSV file that contains all the goodies.
534
535    @returns a list of dicts where each line is broken down into a dict.
536    """
537    host_info_list = []
538    # First line will be the header, no need to parse that.
539    first_line_skipped = False
540    with open(hostname_file) as f:
541        hostname_file_reader = csv.reader(f)
542        for row in hostname_file_reader:
543            if not first_line_skipped:
544                first_line_skipped = True
545                continue
546            host_info_list.append(_parse_hostname_file_line(row))
547
548    return host_info_list
549
550def parse_command(argv, full_deploy):
551    """Get arguments for install from `argv` or the user.
552
553    Create an argument parser for this command's syntax, parse the
554    command line, and return the result of the ArgumentParser
555    parse_args() method.
556
557    If mandatory arguments are missing, execute a dialog with the
558    user to read the arguments from `sys.stdin`.  Fill in the
559    return value with the values read prior to returning.
560
561    @param argv         Standard command line argument vector;
562                        argv[0] is assumed to be the command name.
563    @param full_deploy  Whether this is for full deployment or
564                        repair.
565
566    @return Result, as returned by ArgumentParser.parse_args().
567    """
568    command_name = os.path.basename(argv[0])
569    parser = _make_common_parser(command_name)
570    _add_upload_argument_pair(parser, default=full_deploy)
571
572    arguments = parser.parse_args(argv[1:])
573    arguments.full_deploy = full_deploy
574    if arguments.board is None:
575        _read_arguments(sys.stdin, arguments)
576    elif not _validate_arguments(arguments):
577        return None
578
579    arguments.start_time = datetime.datetime.now(dateutil.tz.tzlocal())
580    if not arguments.logdir:
581        basename = get_default_logdir_name(arguments)
582        arguments.logdir = os.path.join(os.environ['HOME'],
583                                     'Documents', basename)
584        os.makedirs(arguments.logdir)
585    elif not os.path.isdir(arguments.logdir):
586        os.mkdir(arguments.logdir)
587
588    if arguments.hostname_file:
589        # Populate arguments.hostnames with the hostnames from the file.
590        hostname_file_info_list = parse_hostname_file(arguments.hostname_file)
591        arguments.hostnames = [host_info.hostname
592                               for host_info in hostname_file_info_list]
593        arguments.host_info_list = hostname_file_info_list
594    else:
595        arguments.host_info_list = []
596    return arguments
597