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