Home | History | Annotate | Download | only in cli
      1 # Copyright 2014 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 """
      6 The server module contains the objects and methods used to manage servers in
      7 Autotest.
      8 
      9 The valid actions are:
     10 list:      list all servers in the database
     11 create:    create a server
     12 delete:    deletes a server
     13 modify:    modify a server's role or status.
     14 
     15 The common options are:
     16 --role / -r:     role that's related to server actions.
     17 
     18 See topic_common.py for a High Level Design and Algorithm.
     19 """
     20 
     21 import common
     22 
     23 from autotest_lib.cli import action_common
     24 from autotest_lib.cli import topic_common
     25 from autotest_lib.client.common_lib import error
     26 # The django setup is moved here as test_that uses sqlite setup. If this line
     27 # is in server_manager, test_that unittest will fail.
     28 from autotest_lib.frontend import setup_django_environment
     29 from autotest_lib.site_utils import server_manager
     30 from autotest_lib.site_utils import server_manager_utils
     31 
     32 
     33 class server(topic_common.atest):
     34     """Server class
     35 
     36     atest server [list|create|delete|modify] <options>
     37     """
     38     usage_action = '[list|create|delete|modify]'
     39     topic = msg_topic = 'server'
     40     msg_items = '<server>'
     41 
     42     def __init__(self, hostname_required=True):
     43         """Add to the parser the options common to all the server actions.
     44 
     45         @param hostname_required: True to require the command has hostname
     46                                   specified. Default is True.
     47         """
     48         super(server, self).__init__()
     49 
     50         self.parser.add_option('-r', '--role',
     51                                help='Name of a role',
     52                                type='string',
     53                                default=None,
     54                                metavar='ROLE')
     55         self.parser.add_option('-x', '--action',
     56                                help=('Set to True to apply actions when role '
     57                                      'or status is changed, e.g., restart '
     58                                      'scheduler when a drone is removed.'),
     59                                action='store_true',
     60                                default=False,
     61                                metavar='ACTION')
     62 
     63         self.topic_parse_info = topic_common.item_parse_info(
     64                 attribute_name='hostname', use_leftover=True)
     65 
     66         self.hostname_required = hostname_required
     67 
     68 
     69     def parse(self):
     70         """Parse command arguments.
     71         """
     72         role_info = topic_common.item_parse_info(attribute_name='role')
     73         kwargs = {}
     74         if self.hostname_required:
     75             kwargs['req_items'] = 'hostname'
     76         (options, leftover) = super(server, self).parse([role_info], **kwargs)
     77         if options.web_server:
     78             self.invalid_syntax('Server actions will access server database '
     79                                 'defined in your local global config. It does '
     80                                 'not rely on RPC, no autotest server needs to '
     81                                 'be specified.')
     82 
     83         # self.hostname is a list. Action on server only needs one hostname at
     84         # most.
     85         if ((not self.hostname and self.hostname_required) or
     86             len(self.hostname) > 1):
     87             self.invalid_syntax('`server` topic can only manipulate 1 server. '
     88                                 'Use -h to see available options.')
     89         if self.hostname:
     90             # Override self.hostname with the first hostname in the list.
     91             self.hostname = self.hostname[0]
     92         self.role = options.role
     93         return (options, leftover)
     94 
     95 
     96     def output(self, results):
     97         """Display output.
     98 
     99         For most actions, the return is a string message, no formating needed.
    100 
    101         @param results: return of the execute call.
    102         """
    103         print results
    104 
    105 
    106 class server_help(server):
    107     """Just here to get the atest logic working. Usage is set by its parent.
    108     """
    109     pass
    110 
    111 
    112 class server_list(action_common.atest_list, server):
    113     """atest server list [--role <role>]"""
    114 
    115     def __init__(self):
    116         """Initializer.
    117         """
    118         super(server_list, self).__init__(hostname_required=False)
    119         self.parser.add_option('-t', '--table',
    120                                help=('List details of all servers in a table, '
    121                                      'e.g., \tHostname | Status  | Roles     | '
    122                                      'note\t\tserver1  | primary | scheduler | '
    123                                      'lab'),
    124                                action='store_true',
    125                                default=False,
    126                                metavar='TABLE')
    127         self.parser.add_option('-s', '--status',
    128                                help='Only show servers with given status',
    129                                type='string',
    130                                default=None,
    131                                metavar='STATUS')
    132         self.parser.add_option('-u', '--summary',
    133                                help=('Show the summary of roles and status '
    134                                      'only, e.g.,\tscheduler: server1(primary) '
    135                                      'server2(backup)\t\tdrone: server3(primary'
    136                                      ') server4(backup)'),
    137                                action='store_true',
    138                                default=False,
    139                                metavar='SUMMARY')
    140 
    141 
    142     def parse(self):
    143         """Parse command arguments.
    144         """
    145         (options, leftover) = super(server_list, self).parse()
    146         self.table = options.table
    147         self.status = options.status
    148         self.summary = options.summary
    149         if self.table and self.summary:
    150             self.invalid_syntax('Option --table and --summary cannot be both '
    151                                 'specified.')
    152         return (options, leftover)
    153 
    154 
    155     def execute(self):
    156         """Execute the command.
    157 
    158         @return: A list of servers matched given hostname and role.
    159         """
    160         try:
    161             return server_manager_utils.get_servers(hostname=self.hostname,
    162                                                     role=self.role,
    163                                                     status=self.status)
    164         except (server_manager_utils.ServerActionError,
    165                 error.InvalidDataError) as e:
    166             self.failure(e, what_failed='Failed to find servers',
    167                          item=self.hostname, fatal=True)
    168 
    169 
    170     def output(self, results):
    171         """Display output.
    172 
    173         @param results: return of the execute call, a list of server object that
    174                         contains server information.
    175         """
    176         if not results:
    177             self.failure('No server is found.',
    178                          what_failed='Failed to find servers',
    179                          item=self.hostname, fatal=True)
    180         else:
    181             print server_manager_utils.get_server_details(results, self.table,
    182                                                           self.summary)
    183 
    184 
    185 class server_create(server):
    186     """atest server create hostname --role <role> --note <note>
    187     """
    188 
    189     def __init__(self):
    190         """Initializer.
    191         """
    192         super(server_create, self).__init__()
    193         self.parser.add_option('-n', '--note',
    194                                help='note of the server',
    195                                type='string',
    196                                default=None,
    197                                metavar='NOTE')
    198 
    199 
    200     def parse(self):
    201         """Parse command arguments.
    202         """
    203         (options, leftover) = super(server_create, self).parse()
    204         self.note = options.note
    205 
    206         if not self.role:
    207             self.invalid_syntax('--role is required to create a server.')
    208 
    209         return (options, leftover)
    210 
    211 
    212     def execute(self):
    213         """Execute the command.
    214 
    215         @return: A Server object if it is created successfully.
    216         """
    217         try:
    218             return server_manager.create(hostname=self.hostname, role=self.role,
    219                                          note=self.note)
    220         except (server_manager_utils.ServerActionError,
    221                 error.InvalidDataError) as e:
    222             self.failure(e, what_failed='Failed to create server',
    223                          item=self.hostname, fatal=True)
    224 
    225 
    226     def output(self, results):
    227         """Display output.
    228 
    229         @param results: return of the execute call, a server object that
    230                         contains server information.
    231         """
    232         if results:
    233             print 'Server %s is added to server database:\n' % self.hostname
    234             print results
    235 
    236 
    237 class server_delete(server):
    238     """atest server delete hostname"""
    239 
    240     def execute(self):
    241         """Execute the command.
    242 
    243         @return: True if server is deleted successfully.
    244         """
    245         try:
    246             server_manager.delete(hostname=self.hostname)
    247             return True
    248         except (server_manager_utils.ServerActionError,
    249                 error.InvalidDataError) as e:
    250             self.failure(e, what_failed='Failed to delete server',
    251                          item=self.hostname, fatal=True)
    252 
    253 
    254     def output(self, results):
    255         """Display output.
    256 
    257         @param results: return of the execute call.
    258         """
    259         if results:
    260             print ('Server %s is deleted from server database successfully.' %
    261                    self.hostname)
    262 
    263 
    264 class server_modify(server):
    265     """atest server modify hostname
    266 
    267     modify action can only change one input at a time. Available inputs are:
    268     --status:       Status of the server.
    269     --note:         Note of the server.
    270     --role:         New role to be added to the server.
    271     --delete_role:  Existing role to be deleted from the server.
    272     """
    273 
    274     def __init__(self):
    275         """Initializer.
    276         """
    277         super(server_modify, self).__init__()
    278         self.parser.add_option('-s', '--status',
    279                                help='Status of the server',
    280                                type='string',
    281                                metavar='STATUS')
    282         self.parser.add_option('-n', '--note',
    283                                help='Note of the server',
    284                                type='string',
    285                                default=None,
    286                                metavar='NOTE')
    287         self.parser.add_option('-d', '--delete',
    288                                help=('Set to True to delete given role.'),
    289                                action='store_true',
    290                                default=False,
    291                                metavar='DELETE')
    292         self.parser.add_option('-a', '--attribute',
    293                                help='Name of the attribute of the server',
    294                                type='string',
    295                                default=None,
    296                                metavar='ATTRIBUTE')
    297         self.parser.add_option('-e', '--value',
    298                                help='Value for the attribute of the server',
    299                                type='string',
    300                                default=None,
    301                                metavar='VALUE')
    302 
    303 
    304     def parse(self):
    305         """Parse command arguments.
    306         """
    307         (options, leftover) = super(server_modify, self).parse()
    308         self.status = options.status
    309         self.note = options.note
    310         self.delete = options.delete
    311         self.attribute = options.attribute
    312         self.value = options.value
    313         self.action = options.action
    314 
    315         # modify supports various options. However, it's safer to limit one
    316         # option at a time so no complicated role-dependent logic is needed
    317         # to handle scenario that both role and status are changed.
    318         # self.parser is optparse, which does not have function in argparse like
    319         # add_mutually_exclusive_group. That's why the count is used here.
    320         flags = [self.status is not None, self.role is not None,
    321                  self.attribute is not None, self.note is not None]
    322         if flags.count(True) != 1:
    323             msg = ('Action modify only support one option at a time. You can '
    324                    'try one of following 5 options:\n'
    325                    '1. --status:                Change server\'s status.\n'
    326                    '2. --note:                  Change server\'s note.\n'
    327                    '3. --role with optional -d: Add/delete role from server.\n'
    328                    '4. --attribute --value:     Set/change the value of a '
    329                    'server\'s attribute.\n'
    330                    '5. --attribute -d:          Delete the attribute from the '
    331                    'server.\n'
    332                    '\nUse option -h to see a complete list of options.')
    333             self.invalid_syntax(msg)
    334         if (self.status != None or self.note != None) and self.delete:
    335             self.invalid_syntax('--delete does not apply to status or note.')
    336         if self.attribute != None and not self.delete and self.value == None:
    337             self.invalid_syntax('--attribute must be used with option --value '
    338                                 'or --delete.')
    339         return (options, leftover)
    340 
    341 
    342     def execute(self):
    343         """Execute the command.
    344 
    345         @return: The updated server object if it is modified successfully.
    346         """
    347         try:
    348             return server_manager.modify(hostname=self.hostname, role=self.role,
    349                                          status=self.status, delete=self.delete,
    350                                          note=self.note,
    351                                          attribute=self.attribute,
    352                                          value=self.value, action=self.action)
    353         except (server_manager_utils.ServerActionError,
    354                 error.InvalidDataError) as e:
    355             self.failure(e, what_failed='Failed to modify server',
    356                          item=self.hostname, fatal=True)
    357 
    358 
    359     def output(self, results):
    360         """Display output.
    361 
    362         @param results: return of the execute call, which is the updated server
    363                         object.
    364         """
    365         if results:
    366             print 'Server %s is modified successfully.' % self.hostname
    367             print results
    368