Home | History | Annotate | Download | only in site_utils
      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 """This module provides utility functions to help managing servers in server
      6 database (defined in global config section AUTOTEST_SERVER_DB).
      7 
      8 """
      9 
     10 import collections
     11 import json
     12 import socket
     13 import subprocess
     14 import sys
     15 
     16 import common
     17 
     18 import django.core.exceptions
     19 from autotest_lib.client.common_lib import utils
     20 from autotest_lib.client.common_lib.global_config import global_config
     21 from autotest_lib.frontend.server import models as server_models
     22 from autotest_lib.site_utils.lib import infra
     23 
     24 
     25 class ServerActionError(Exception):
     26     """Exception raised when action on server failed.
     27     """
     28 
     29 
     30 def use_server_db():
     31     """Check if use_server_db is enabled in configuration.
     32 
     33     @return: True if use_server_db is set to True in global config.
     34     """
     35     return global_config.get_config_value(
     36             'SERVER', 'use_server_db', default=False, type=bool)
     37 
     38 
     39 def warn_missing_role(role, exclude_server):
     40     """Post a warning if Autotest instance has no other primary server with
     41     given role.
     42 
     43     @param role: Name of the role.
     44     @param exclude_server: Server to be excluded from search for role.
     45     """
     46     servers = server_models.Server.objects.filter(
     47             roles__role=role,
     48             status=server_models.Server.STATUS.PRIMARY).exclude(
     49                     hostname=exclude_server.hostname)
     50     if not servers:
     51         message = ('WARNING! There will be no server with role %s after it\'s '
     52                    'removed from server %s. Autotest will not function '
     53                    'normally without any server in role %s.' %
     54                    (role, exclude_server.hostname, role))
     55         print >> sys.stderr, message
     56 
     57 
     58 def get_servers(hostname=None, role=None, status=None):
     59     """Find servers with given role and status.
     60 
     61     @param hostname: hostname of the server.
     62     @param role: Role of server, default to None.
     63     @param status: Status of server, default to None.
     64 
     65     @return: A list of server objects with given role and status.
     66     """
     67     filters = {}
     68     if hostname:
     69         filters['hostname'] = hostname
     70     if role:
     71         filters['roles__role'] = role
     72     if status:
     73         filters['status'] = status
     74     return list(server_models.Server.objects.filter(**filters))
     75 
     76 
     77 def format_servers(servers):
     78     """Format servers for printing.
     79 
     80     Example output:
     81 
     82         Hostname     : server2
     83         Status       : primary
     84         Roles        : drone
     85         Attributes   : {'max_processes':300}
     86         Date Created : 2014-11-25 12:00:00
     87         Date Modified: None
     88         Note         : Drone in lab1
     89 
     90     @param servers: Sequence of Server instances.
     91     @returns: Formatted output as string.
     92     """
     93     return '\n'.join(str(server) for server in servers)
     94 
     95 
     96 def format_servers_json(servers):
     97     """Format servers for printing as JSON.
     98 
     99     @param servers: Sequence of Server instances.
    100     @returns: String.
    101     """
    102     server_dicts = []
    103     for server in servers:
    104         if server.date_modified is None:
    105             date_modified = None
    106         else:
    107             date_modified = str(server.date_modified)
    108         attributes = {k: v for k, v in server.attributes.values_list(
    109                 'attribute', 'value')}
    110         server_dicts.append({'hostname': server.hostname,
    111                              'status': server.status,
    112                              'roles': server.get_role_names(),
    113                              'date_created': str(server.date_created),
    114                              'date_modified': date_modified,
    115                              'note': server.note,
    116                              'attributes': attributes})
    117     return json.dumps(server_dicts)
    118 
    119 
    120 _SERVER_TABLE_FORMAT = ('%(hostname)-30s | %(status)-7s | %(roles)-20s |'
    121                         ' %(date_created)-19s | %(date_modified)-19s |'
    122                         ' %(note)s')
    123 
    124 
    125 def format_servers_table(servers):
    126     """format servers for printing as a table.
    127 
    128     Example output:
    129 
    130         Hostname | Status  | Roles     | Date Created    | Date Modified | Note
    131         server1  | backup  | scheduler | 2014-11-25 23:45:19 |           |
    132         server2  | primary | drone     | 2014-11-25 12:00:00 |           | Drone
    133 
    134     @param servers: Sequence of Server instances.
    135     @returns: Formatted output as string.
    136     """
    137     result_lines = [(_SERVER_TABLE_FORMAT %
    138                      {'hostname': 'Hostname',
    139                       'status': 'Status',
    140                       'roles': 'Roles',
    141                       'date_created': 'Date Created',
    142                       'date_modified': 'Date Modified',
    143                       'note': 'Note'})]
    144     for server in servers:
    145         roles = ','.join(server.get_role_names())
    146         result_lines.append(_SERVER_TABLE_FORMAT %
    147                             {'hostname':server.hostname,
    148                              'status': server.status or '',
    149                              'roles': roles,
    150                              'date_created': server.date_created,
    151                              'date_modified': server.date_modified or '',
    152                              'note': server.note or ''})
    153     return '\n'.join(result_lines)
    154 
    155 
    156 def format_servers_summary(servers):
    157     """format servers for printing a summary.
    158 
    159     Example output:
    160 
    161         scheduler      : server1(backup), server3(primary),
    162         host_scheduler :
    163         drone          : server2(primary),
    164         devserver      :
    165         database       :
    166         crash_server   :
    167         No Role        :
    168 
    169     @param servers: Sequence of Server instances.
    170     @returns: Formatted output as string.
    171     """
    172     servers_by_role = _get_servers_by_role(servers)
    173     servers_with_roles = {server for role_servers in servers_by_role.itervalues()
    174                           for server in role_servers}
    175     servers_without_roles = [server for server in servers
    176                              if server not in servers_with_roles]
    177     result_lines = ['Roles and status of servers:', '']
    178     for role, role_servers in servers_by_role.iteritems():
    179         result_lines.append(_format_role_servers_summary(role, role_servers))
    180     if servers_without_roles:
    181         result_lines.append(
    182                 _format_role_servers_summary('No Role', servers_without_roles))
    183     return '\n'.join(result_lines)
    184 
    185 
    186 def format_servers_nameonly(servers):
    187     """format servers for printing names only
    188 
    189     @param servers: Sequence of Server instances.
    190     @returns: Formatted output as string.
    191     """
    192     return '\n'.join(s.hostname for s in servers)
    193 
    194 
    195 def _get_servers_by_role(servers):
    196     """Return a mapping from roles to servers.
    197 
    198     @param servers: Iterable of servers.
    199     @returns: Mapping of role strings to lists of servers.
    200     """
    201     roles = [role for role, _ in server_models.ServerRole.ROLE.choices()]
    202     servers_by_role = collections.defaultdict(list)
    203     for server in servers:
    204         for role in server.get_role_names():
    205             servers_by_role[role].append(server)
    206     return servers_by_role
    207 
    208 
    209 def _format_role_servers_summary(role, servers):
    210     """Format one line of servers for a role in a server list summary.
    211 
    212     @param role: Role string.
    213     @param servers: Iterable of Server instances.
    214     @returns: String.
    215     """
    216     servers_part = ', '.join(
    217             '%s(%s)' % (server.hostname, server.status)
    218             for server in servers)
    219     return '%-15s: %s' % (role, servers_part)
    220 
    221 
    222 def check_server(hostname, role):
    223     """Confirm server with given hostname is ready to be primary of given role.
    224 
    225     If the server is a backup and failed to be verified for the role, remove
    226     the role from its roles list. If it has no other role, set its status to
    227     repair_required.
    228 
    229     @param hostname: hostname of the server.
    230     @param role: Role to be checked.
    231     @return: True if server can be verified for the given role, otherwise
    232              return False.
    233     """
    234     # TODO(dshi): Add more logic to confirm server is ready for the role.
    235     # For now, the function just checks if server is ssh-able.
    236     try:
    237         infra.execute_command(hostname, 'true')
    238         return True
    239     except subprocess.CalledProcessError as e:
    240         print >> sys.stderr, ('Failed to check server %s, error: %s' %
    241                               (hostname, e))
    242         return False
    243 
    244 
    245 def verify_server(exist=True):
    246     """Decorator to check if server with given hostname exists in the database.
    247 
    248     @param exist: Set to True to confirm server exists in the database, raise
    249                   exception if not. If it's set to False, raise exception if
    250                   server exists in database. Default is True.
    251 
    252     @raise ServerActionError: If `exist` is True and server does not exist in
    253                               the database, or `exist` is False and server exists
    254                               in the database.
    255     """
    256     def deco_verify(func):
    257         """Wrapper for the decorator.
    258 
    259         @param func: Function to be called.
    260         """
    261         def func_verify(*args, **kwargs):
    262             """Decorator to check if server exists.
    263 
    264             If exist is set to True, raise ServerActionError is server with
    265             given hostname is not found in server database.
    266             If exist is set to False, raise ServerActionError is server with
    267             given hostname is found in server database.
    268 
    269             @param func: function to be called.
    270             @param args: arguments for function to be called.
    271             @param kwargs: keyword arguments for function to be called.
    272             """
    273             hostname = kwargs['hostname']
    274             try:
    275                 server = server_models.Server.objects.get(hostname=hostname)
    276             except django.core.exceptions.ObjectDoesNotExist:
    277                 server = None
    278 
    279             if not exist and server:
    280                 raise ServerActionError('Server %s already exists.' %
    281                                         hostname)
    282             if exist and not server:
    283                 raise ServerActionError('Server %s does not exist in the '
    284                                         'database.' % hostname)
    285             if server:
    286                 kwargs['server'] = server
    287             return func(*args, **kwargs)
    288         return func_verify
    289     return deco_verify
    290 
    291 
    292 def get_drones():
    293     """Get a list of drones in status primary.
    294 
    295     @return: A list of drones in status primary.
    296     """
    297     servers = get_servers(role=server_models.ServerRole.ROLE.DRONE,
    298                           status=server_models.Server.STATUS.PRIMARY)
    299     return [s.hostname for s in servers]
    300 
    301 
    302 def delete_attribute(server, attribute):
    303     """Delete the attribute from the host.
    304 
    305     @param server: An object of server_models.Server.
    306     @param attribute: Name of an attribute of the server.
    307     """
    308     attributes = server.attributes.filter(attribute=attribute)
    309     if not attributes:
    310         raise ServerActionError('Server %s does not have attribute %s' %
    311                                 (server.hostname, attribute))
    312     attributes[0].delete()
    313     print 'Attribute %s is deleted from server %s.' % (attribute,
    314                                                        server.hostname)
    315 
    316 
    317 def change_attribute(server, attribute, value):
    318     """Change the value of an attribute of the server.
    319 
    320     @param server: An object of server_models.Server.
    321     @param attribute: Name of an attribute of the server.
    322     @param value: Value of the attribute of the server.
    323 
    324     @raise ServerActionError: If the attribute already exists and has the
    325                               given value.
    326     """
    327     attributes = server_models.ServerAttribute.objects.filter(
    328             server=server, attribute=attribute)
    329     if attributes and attributes[0].value == value:
    330         raise ServerActionError('Attribute %s for Server %s already has '
    331                                 'value of %s.' %
    332                                 (attribute, server.hostname, value))
    333     if attributes:
    334         old_value = attributes[0].value
    335         attributes[0].value = value
    336         attributes[0].save()
    337         print ('Attribute `%s` of server %s is changed from %s to %s.' %
    338                      (attribute, server.hostname, old_value, value))
    339     else:
    340         server_models.ServerAttribute.objects.create(
    341                 server=server, attribute=attribute, value=value)
    342         print ('Attribute `%s` of server %s is set to %s.' %
    343                (attribute, server.hostname, value))
    344 
    345 
    346 def get_shards():
    347     """Get a list of shards in status primary.
    348 
    349     @return: A list of shards in status primary.
    350     """
    351     servers = get_servers(role=server_models.ServerRole.ROLE.SHARD,
    352                           status=server_models.Server.STATUS.PRIMARY)
    353     return [s.hostname for s in servers]
    354 
    355 
    356 def confirm_server_has_role(hostname, role):
    357     """Confirm a given server has the given role, and its status is primary.
    358 
    359     @param hostname: hostname of the server.
    360     @param role: Name of the role to be checked.
    361     @raise ServerActionError: If localhost does not have given role or it's
    362                               not in primary status.
    363     """
    364     if hostname.lower() in ['localhost', '127.0.0.1']:
    365         hostname = socket.gethostname()
    366     hostname = utils.normalize_hostname(hostname)
    367 
    368     servers = get_servers(role=role, status=server_models.Server.STATUS.PRIMARY)
    369     for server in servers:
    370         if hostname == utils.normalize_hostname(server.hostname):
    371             return True
    372     raise ServerActionError('Server %s does not have role of %s running in '
    373                             'status primary.' % (hostname, role))
    374