1# Copyright 2009 Google Inc. Released under the GPL v2 2 3""" 4This module defines the base classes for the Host hierarchy. 5 6Implementation details: 7You should import the "hosts" package instead of importing each type of host. 8 9 Host: a machine on which you can run programs 10""" 11 12__author__ = """ 13mbligh@google.com (Martin J. Bligh), 14poirier@google.com (Benjamin Poirier), 15stutsman@google.com (Ryan Stutsman) 16""" 17 18import json, logging, os, re, time 19 20from autotest_lib.client.common_lib import global_config, error, utils 21from autotest_lib.client.common_lib.cros import path_utils 22 23 24class Host(object): 25 """ 26 This class represents a machine on which you can run programs. 27 28 It may be a local machine, the one autoserv is running on, a remote 29 machine or a virtual machine. 30 31 Implementation details: 32 This is an abstract class, leaf subclasses must implement the methods 33 listed here. You must not instantiate this class but should 34 instantiate one of those leaf subclasses. 35 36 When overriding methods that raise NotImplementedError, the leaf class 37 is fully responsible for the implementation and should not chain calls 38 to super. When overriding methods that are a NOP in Host, the subclass 39 should chain calls to super(). The criteria for fitting a new method into 40 one category or the other should be: 41 1. If two separate generic implementations could reasonably be 42 concatenated, then the abstract implementation should pass and 43 subclasses should chain calls to super. 44 2. If only one class could reasonably perform the stated function 45 (e.g. two separate run() implementations cannot both be executed) 46 then the method should raise NotImplementedError in Host, and 47 the implementor should NOT chain calls to super, to ensure that 48 only one implementation ever gets executed. 49 """ 50 51 job = None 52 DEFAULT_REBOOT_TIMEOUT = global_config.global_config.get_config_value( 53 "HOSTS", "default_reboot_timeout", type=int, default=1800) 54 WAIT_DOWN_REBOOT_TIMEOUT = global_config.global_config.get_config_value( 55 "HOSTS", "wait_down_reboot_timeout", type=int, default=840) 56 WAIT_DOWN_REBOOT_WARNING = global_config.global_config.get_config_value( 57 "HOSTS", "wait_down_reboot_warning", type=int, default=540) 58 HOURS_TO_WAIT_FOR_RECOVERY = global_config.global_config.get_config_value( 59 "HOSTS", "hours_to_wait_for_recovery", type=float, default=2.5) 60 # the number of hardware repair requests that need to happen before we 61 # actually send machines to hardware repair 62 HARDWARE_REPAIR_REQUEST_THRESHOLD = 4 63 OP_REBOOT = 'reboot' 64 OP_SUSPEND = 'suspend' 65 PWR_OPERATION = [OP_REBOOT, OP_SUSPEND] 66 67 68 def __init__(self, *args, **dargs): 69 self._initialize(*args, **dargs) 70 71 72 def _initialize(self, *args, **dargs): 73 pass 74 75 76 @property 77 def job_repo_url_attribute(self): 78 """Get the host attribute name for job_repo_url. 79 """ 80 return 'job_repo_url' 81 82 83 def close(self): 84 """Close the connection to the host. 85 """ 86 pass 87 88 89 def setup(self): 90 """Setup the host object. 91 """ 92 pass 93 94 95 def run(self, command, timeout=3600, ignore_status=False, 96 stdout_tee=utils.TEE_TO_LOGS, stderr_tee=utils.TEE_TO_LOGS, 97 stdin=None, args=()): 98 """ 99 Run a command on this host. 100 101 @param command: the command line string 102 @param timeout: time limit in seconds before attempting to 103 kill the running process. The run() function 104 will take a few seconds longer than 'timeout' 105 to complete if it has to kill the process. 106 @param ignore_status: do not raise an exception, no matter 107 what the exit code of the command is. 108 @param stdout_tee: where to tee the stdout 109 @param stderr_tee: where to tee the stderr 110 @param stdin: stdin to pass (a string) to the executed command 111 @param args: sequence of strings to pass as arguments to command by 112 quoting them in " and escaping their contents if necessary 113 114 @return a utils.CmdResult object 115 116 @raises AutotestHostRunError: the exit code of the command execution 117 was not 0 and ignore_status was not enabled 118 """ 119 raise NotImplementedError('Run not implemented!') 120 121 122 def run_output(self, command, *args, **dargs): 123 """Run and retrieve the value of stdout stripped of whitespace. 124 125 @param command: Command to execute. 126 @param *args: Extra arguments to run. 127 @param **dargs: Extra keyword arguments to run. 128 129 @return: String value of stdout. 130 """ 131 return self.run(command, *args, **dargs).stdout.rstrip() 132 133 134 def reboot(self): 135 """Reboot the host. 136 """ 137 raise NotImplementedError('Reboot not implemented!') 138 139 140 def suspend(self): 141 """Suspend the host. 142 """ 143 raise NotImplementedError('Suspend not implemented!') 144 145 146 def sysrq_reboot(self): 147 """Execute host reboot via SysRq key. 148 """ 149 raise NotImplementedError('Sysrq reboot not implemented!') 150 151 152 def reboot_setup(self, *args, **dargs): 153 """Prepare for reboot. 154 155 This doesn't appear to be implemented by any current hosts. 156 157 @param *args: Extra arguments to ?. 158 @param **dargs: Extra keyword arguments to ?. 159 """ 160 pass 161 162 163 def reboot_followup(self, *args, **dargs): 164 """Post reboot work. 165 166 This doesn't appear to be implemented by any current hosts. 167 168 @param *args: Extra arguments to ?. 169 @param **dargs: Extra keyword arguments to ?. 170 """ 171 pass 172 173 174 def get_file(self, source, dest, delete_dest=False): 175 """Retrieve a file from the host. 176 177 @param source: Remote file path (directory, file or list). 178 @param dest: Local file path (directory, file or list). 179 @param delete_dest: Delete files in remote path that are not in local 180 path. 181 """ 182 raise NotImplementedError('Get file not implemented!') 183 184 185 def send_file(self, source, dest, delete_dest=False, excludes=None): 186 """Send a file to the host. 187 188 @param source: Local file path (directory, file or list). 189 @param dest: Remote file path (directory, file or list). 190 @param delete_dest: Delete files in remote path that are not in local 191 path. 192 @param excludes: A list of file pattern that matches files not to be 193 sent. `send_file` will fail if exclude is not 194 supported. 195 """ 196 raise NotImplementedError('Send file not implemented!') 197 198 199 def get_tmp_dir(self): 200 """Create a temporary directory on the host. 201 """ 202 raise NotImplementedError('Get temp dir not implemented!') 203 204 205 def is_up(self): 206 """Confirm the host is online. 207 """ 208 raise NotImplementedError('Is up not implemented!') 209 210 211 def is_shutting_down(self): 212 """ Indicates is a machine is currently shutting down. """ 213 return False 214 215 216 def get_wait_up_processes(self): 217 """ Gets the list of local processes to wait for in wait_up. """ 218 get_config = global_config.global_config.get_config_value 219 proc_list = get_config("HOSTS", "wait_up_processes", 220 default="").strip() 221 processes = set(p.strip() for p in proc_list.split(",")) 222 processes.discard("") 223 return processes 224 225 226 def get_boot_id(self, timeout=60): 227 """ Get a unique ID associated with the current boot. 228 229 Should return a string with the semantics such that two separate 230 calls to Host.get_boot_id() return the same string if the host did 231 not reboot between the two calls, and two different strings if it 232 has rebooted at least once between the two calls. 233 234 @param timeout The number of seconds to wait before timing out. 235 236 @return A string unique to this boot or None if not available.""" 237 BOOT_ID_FILE = '/proc/sys/kernel/random/boot_id' 238 NO_ID_MSG = 'no boot_id available' 239 cmd = 'if [ -f %r ]; then cat %r; else echo %r; fi' % ( 240 BOOT_ID_FILE, BOOT_ID_FILE, NO_ID_MSG) 241 boot_id = self.run(cmd, timeout=timeout).stdout.strip() 242 if boot_id == NO_ID_MSG: 243 return None 244 return boot_id 245 246 247 def wait_up(self, timeout=None): 248 """Wait for the host to come up. 249 250 @param timeout: Max seconds to wait. 251 """ 252 raise NotImplementedError('Wait up not implemented!') 253 254 255 def wait_down(self, timeout=None, warning_timer=None, old_boot_id=None): 256 """Wait for the host to go down. 257 258 @param timeout: Max seconds to wait before returning. 259 @param warning_timer: Seconds before warning host is not down. 260 @param old_boot_id: Result of self.get_boot_id() before shutdown. 261 """ 262 raise NotImplementedError('Wait down not implemented!') 263 264 265 def _construct_host_metadata(self, type_str): 266 """Returns dict of metadata with type_str, hostname, time_recorded. 267 268 @param type_str: String representing _type field in es db. 269 For example: type_str='reboot_total'. 270 """ 271 metadata = { 272 'hostname': self.hostname, 273 'time_recorded': time.time(), 274 '_type': type_str, 275 } 276 return metadata 277 278 279 def wait_for_restart(self, timeout=DEFAULT_REBOOT_TIMEOUT, 280 down_timeout=WAIT_DOWN_REBOOT_TIMEOUT, 281 down_warning=WAIT_DOWN_REBOOT_WARNING, 282 log_failure=True, old_boot_id=None, **dargs): 283 """Wait for the host to come back from a reboot. 284 285 This is a generic implementation based entirely on wait_up and 286 wait_down. 287 288 @param timeout: Max seconds to wait for reboot to start. 289 @param down_timeout: Max seconds to wait for host to go down. 290 @param down_warning: Seconds to wait before warning host hasn't gone 291 down. 292 @param log_failure: bool(Log when host does not go down.) 293 @param old_boot_id: Result of self.get_boot_id() before restart. 294 @param **dargs: Extra arguments to reboot_followup. 295 296 @raises AutoservRebootError if host does not come back up. 297 """ 298 if not self.wait_down(timeout=down_timeout, 299 warning_timer=down_warning, 300 old_boot_id=old_boot_id): 301 if log_failure: 302 self.record("ABORT", None, "reboot.verify", "shut down failed") 303 raise error.AutoservShutdownError("Host did not shut down") 304 if self.wait_up(timeout): 305 self.record("GOOD", None, "reboot.verify") 306 self.reboot_followup(**dargs) 307 else: 308 self.record("ABORT", None, "reboot.verify", 309 "Host did not return from reboot") 310 raise error.AutoservRebootError("Host did not return from reboot") 311 312 313 def verify(self): 314 """Check if host is in good state. 315 """ 316 self.verify_hardware() 317 self.verify_connectivity() 318 self.verify_software() 319 320 321 def verify_hardware(self): 322 """Check host hardware. 323 """ 324 pass 325 326 327 def verify_connectivity(self): 328 """Check host network connectivity. 329 """ 330 pass 331 332 333 def verify_software(self): 334 """Check host software. 335 """ 336 pass 337 338 339 def check_diskspace(self, path, gb): 340 """Raises an error if path does not have at least gb GB free. 341 342 @param path The path to check for free disk space. 343 @param gb A floating point number to compare with a granularity 344 of 1 MB. 345 346 1000 based SI units are used. 347 348 @raises AutoservDiskFullHostError if path has less than gb GB free. 349 @raises AutoservDirectoryNotFoundError if path is not a valid directory. 350 @raises AutoservDiskSizeUnknownError the return from du is not parsed 351 correctly. 352 """ 353 one_mb = 10 ** 6 # Bytes (SI unit). 354 mb_per_gb = 1000.0 355 logging.info('Checking for >= %s GB of space under %s on machine %s', 356 gb, path, self.hostname) 357 358 if not self.path_exists(path): 359 msg = 'Path does not exist on host: %s' % path 360 logging.warning(msg) 361 raise error.AutoservDirectoryNotFoundError(msg) 362 363 cmd = 'df -PB %d %s | tail -1' % (one_mb, path) 364 df = self.run(cmd).stdout.split() 365 try: 366 free_space_gb = int(df[3]) / mb_per_gb 367 except (IndexError, ValueError): 368 msg = ('Could not determine the size of %s. ' 369 'Output from df: %s') % (path, df) 370 logging.error(msg) 371 raise error.AutoservDiskSizeUnknownError(msg) 372 373 if free_space_gb < gb: 374 raise error.AutoservDiskFullHostError(path, gb, free_space_gb) 375 else: 376 logging.info('Found %s GB >= %s GB of space under %s on machine %s', 377 free_space_gb, gb, path, self.hostname) 378 379 380 def check_inodes(self, path, min_kilo_inodes): 381 """Raises an error if a file system is short on i-nodes. 382 383 @param path The path to check for free i-nodes. 384 @param min_kilo_inodes Minimum number of i-nodes required, 385 in units of 1000 i-nodes. 386 387 @raises AutoservNoFreeInodesError If the minimum required 388 i-node count isn't available. 389 """ 390 min_inodes = 1000 * min_kilo_inodes 391 logging.info('Checking for >= %d i-nodes under %s ' 392 'on machine %s', min_inodes, path, self.hostname) 393 df = self.run('df -Pi %s | tail -1' % path).stdout.split() 394 free_inodes = int(df[3]) 395 if free_inodes < min_inodes: 396 raise error.AutoservNoFreeInodesError(path, min_inodes, 397 free_inodes) 398 else: 399 logging.info('Found %d >= %d i-nodes under %s on ' 400 'machine %s', free_inodes, min_inodes, 401 path, self.hostname) 402 403 404 def erase_dir_contents(self, path, ignore_status=True, timeout=3600): 405 """Empty a given directory path contents. 406 407 @param path: Path to empty. 408 @param ignore_status: Ignore the exit status from run. 409 @param timeout: Max seconds to allow command to complete. 410 """ 411 rm_cmd = 'find "%s" -mindepth 1 -maxdepth 1 -print0 | xargs -0 rm -rf' 412 self.run(rm_cmd % path, ignore_status=ignore_status, timeout=timeout) 413 414 415 def repair(self): 416 """Try and get the host to pass `self.verify()`.""" 417 self.verify() 418 419 420 def disable_ipfilters(self): 421 """Allow all network packets in and out of the host.""" 422 self.run('iptables-save > /tmp/iptable-rules') 423 self.run('iptables -P INPUT ACCEPT') 424 self.run('iptables -P FORWARD ACCEPT') 425 self.run('iptables -P OUTPUT ACCEPT') 426 427 428 def enable_ipfilters(self): 429 """Re-enable the IP filters disabled from disable_ipfilters()""" 430 if self.path_exists('/tmp/iptable-rules'): 431 self.run('iptables-restore < /tmp/iptable-rules') 432 433 434 def cleanup(self): 435 """Restore host to clean state. 436 """ 437 pass 438 439 440 def install(self, installableObject): 441 """Call install on a thing. 442 443 @param installableObject: Thing with install method that will accept our 444 self. 445 """ 446 installableObject.install(self) 447 448 449 def get_autodir(self): 450 raise NotImplementedError('Get autodir not implemented!') 451 452 453 def set_autodir(self): 454 raise NotImplementedError('Set autodir not implemented!') 455 456 457 def start_loggers(self): 458 """ Called to start continuous host logging. """ 459 pass 460 461 462 def stop_loggers(self): 463 """ Called to stop continuous host logging. """ 464 pass 465 466 467 # some extra methods simplify the retrieval of information about the 468 # Host machine, with generic implementations based on run(). subclasses 469 # should feel free to override these if they can provide better 470 # implementations for their specific Host types 471 472 def get_num_cpu(self): 473 """ Get the number of CPUs in the host according to /proc/cpuinfo. """ 474 proc_cpuinfo = self.run('cat /proc/cpuinfo', 475 stdout_tee=open(os.devnull, 'w')).stdout 476 cpus = 0 477 for line in proc_cpuinfo.splitlines(): 478 if line.startswith('processor'): 479 cpus += 1 480 return cpus 481 482 483 def get_arch(self): 484 """ Get the hardware architecture of the remote machine. """ 485 cmd_uname = path_utils.must_be_installed('/bin/uname', host=self) 486 arch = self.run('%s -m' % cmd_uname).stdout.rstrip() 487 if re.match(r'i\d86$', arch): 488 arch = 'i386' 489 return arch 490 491 492 def get_kernel_ver(self): 493 """ Get the kernel version of the remote machine. """ 494 cmd_uname = path_utils.must_be_installed('/bin/uname', host=self) 495 return self.run('%s -r' % cmd_uname).stdout.rstrip() 496 497 498 def get_cmdline(self): 499 """ Get the kernel command line of the remote machine. """ 500 return self.run('cat /proc/cmdline').stdout.rstrip() 501 502 503 def get_meminfo(self): 504 """ Get the kernel memory info (/proc/meminfo) of the remote machine 505 and return a dictionary mapping the various statistics. """ 506 meminfo_dict = {} 507 meminfo = self.run('cat /proc/meminfo').stdout.splitlines() 508 for key, val in (line.split(':', 1) for line in meminfo): 509 meminfo_dict[key.strip()] = val.strip() 510 return meminfo_dict 511 512 513 def path_exists(self, path): 514 """Determine if path exists on the remote machine. 515 516 @param path: path to check 517 518 @return: bool(path exists)""" 519 result = self.run('test -e "%s"' % utils.sh_escape(path), 520 ignore_status=True) 521 return result.exit_status == 0 522 523 524 # some extra helpers for doing job-related operations 525 526 def record(self, *args, **dargs): 527 """ Helper method for recording status logs against Host.job that 528 silently becomes a NOP if Host.job is not available. The args and 529 dargs are passed on to Host.job.record unchanged. """ 530 if self.job: 531 self.job.record(*args, **dargs) 532 533 534 def log_kernel(self): 535 """ Helper method for logging kernel information into the status logs. 536 Intended for cases where the "current" kernel is not really defined 537 and we want to explicitly log it. Does nothing if this host isn't 538 actually associated with a job. """ 539 if self.job: 540 kernel = self.get_kernel_ver() 541 self.job.record("INFO", None, None, 542 optional_fields={"kernel": kernel}) 543 544 545 def log_op(self, op, op_func): 546 """ Decorator for wrapping a management operaiton in a group for status 547 logging purposes. 548 549 @param op: name of the operation. 550 @param op_func: a function that carries out the operation 551 (reboot, suspend) 552 """ 553 if self.job and not hasattr(self, "RUNNING_LOG_OP"): 554 self.RUNNING_LOG_OP = True 555 try: 556 self.job.run_op(op, op_func, self.get_kernel_ver) 557 finally: 558 del self.RUNNING_LOG_OP 559 else: 560 op_func() 561 562 563 def list_files_glob(self, glob): 564 """Get a list of files on a remote host given a glob pattern path. 565 566 @param glob: pattern 567 568 @return: list of files 569 """ 570 SCRIPT = ("python -c 'import json, glob, sys;" 571 "json.dump(glob.glob(sys.argv[1]), sys.stdout)'") 572 output = self.run(SCRIPT, args=(glob,), stdout_tee=None, 573 timeout=60).stdout 574 return json.loads(output) 575 576 577 def symlink_closure(self, paths): 578 """ 579 Given a sequence of path strings, return the set of all paths that 580 can be reached from the initial set by following symlinks. 581 582 @param paths: sequence of path strings. 583 @return: a sequence of path strings that are all the unique paths that 584 can be reached from the given ones after following symlinks. 585 """ 586 SCRIPT = ("python -c 'import json, os, sys\n" 587 "paths = json.load(sys.stdin)\n" 588 "closure = {}\n" 589 "while paths:\n" 590 " path = next(iter(paths))\n" 591 " del paths[path]\n" 592 " if not os.path.exists(path):\n" 593 " continue\n" 594 " closure[path] = None\n" 595 " if os.path.islink(path):\n" 596 " link_to = os.path.join(os.path.dirname(path),\n" 597 " os.readlink(path))\n" 598 " if link_to not in closure:\n" 599 " paths[link_to] = None\n" 600 "json.dump(closure.keys(), sys.stdout, 0)'") 601 input_data = json.dumps(dict((path, None) for path in paths), 0) 602 output = self.run(SCRIPT, stdout_tee=None, stdin=input_data, 603 timeout=60).stdout 604 return json.loads(output) 605 606 607 def cleanup_kernels(self, boot_dir='/boot'): 608 """ 609 Remove any kernel image and associated files (vmlinux, system.map, 610 modules) for any image found in the boot directory that is not 611 referenced by entries in the bootloader configuration. 612 613 @param boot_dir: boot directory path string, default '/boot' 614 """ 615 # find all the vmlinuz images referenced by the bootloader 616 vmlinuz_prefix = os.path.join(boot_dir, 'vmlinuz-') 617 boot_info = self.bootloader.get_entries() 618 used_kernver = [boot['kernel'][len(vmlinuz_prefix):] 619 for boot in boot_info.itervalues()] 620 621 # find all the unused vmlinuz images in /boot 622 all_vmlinuz = self.list_files_glob(vmlinuz_prefix + '*') 623 used_vmlinuz = self.symlink_closure(vmlinuz_prefix + kernver 624 for kernver in used_kernver) 625 unused_vmlinuz = set(all_vmlinuz) - set(used_vmlinuz) 626 627 # find all the unused vmlinux images in /boot 628 vmlinux_prefix = os.path.join(boot_dir, 'vmlinux-') 629 all_vmlinux = self.list_files_glob(vmlinux_prefix + '*') 630 used_vmlinux = self.symlink_closure(vmlinux_prefix + kernver 631 for kernver in used_kernver) 632 unused_vmlinux = set(all_vmlinux) - set(used_vmlinux) 633 634 # find all the unused System.map files in /boot 635 systemmap_prefix = os.path.join(boot_dir, 'System.map-') 636 all_system_map = self.list_files_glob(systemmap_prefix + '*') 637 used_system_map = self.symlink_closure( 638 systemmap_prefix + kernver for kernver in used_kernver) 639 unused_system_map = set(all_system_map) - set(used_system_map) 640 641 # find all the module directories associated with unused kernels 642 modules_prefix = '/lib/modules/' 643 all_moddirs = [dir for dir in self.list_files_glob(modules_prefix + '*') 644 if re.match(modules_prefix + r'\d+\.\d+\.\d+.*', dir)] 645 used_moddirs = self.symlink_closure(modules_prefix + kernver 646 for kernver in used_kernver) 647 unused_moddirs = set(all_moddirs) - set(used_moddirs) 648 649 # remove all the vmlinuz files we don't use 650 # TODO: if needed this should become package manager agnostic 651 for vmlinuz in unused_vmlinuz: 652 # try and get an rpm package name 653 rpm = self.run('rpm -qf', args=(vmlinuz,), 654 ignore_status=True, timeout=120) 655 if rpm.exit_status == 0: 656 packages = set(line.strip() for line in 657 rpm.stdout.splitlines()) 658 # if we found some package names, try to remove them 659 for package in packages: 660 self.run('rpm -e', args=(package,), 661 ignore_status=True, timeout=120) 662 # remove the image files anyway, even if rpm didn't 663 self.run('rm -f', args=(vmlinuz,), 664 ignore_status=True, timeout=120) 665 666 # remove all the vmlinux and System.map files left over 667 for f in (unused_vmlinux | unused_system_map): 668 self.run('rm -f', args=(f,), 669 ignore_status=True, timeout=120) 670 671 # remove all unused module directories 672 # the regex match should keep us safe from removing the wrong files 673 for moddir in unused_moddirs: 674 self.run('rm -fr', args=(moddir,), ignore_status=True) 675 676 677 def get_attributes_to_clear_before_provision(self): 678 """Get a list of attributes to be cleared before machine_install starts. 679 680 If provision runs in a lab environment, it is necessary to clear certain 681 host attributes for the host in afe_host_attributes table. For example, 682 `job_repo_url` is a devserver url pointed to autotest packages for 683 CrosHost, it needs to be removed before provision starts for tests to 684 run reliably. 685 """ 686 return ['job_repo_url'] 687 688 689 def get_platform(self): 690 """Determine the correct platform label for this host. 691 692 @return: A string representing this host's platform. 693 """ 694 raise NotImplementedError("Get platform not implemented!") 695 696 697 def get_labels(self): 698 """Return a list of the labels gathered from the devices connected. 699 700 @return: A list of strings that denote the labels from all the devices 701 connected. 702 """ 703 raise NotImplementedError("Get labels not implemented!") 704 705 706 def check_cached_up_status(self, expiration_seconds): 707 """Check if the DUT responded to ping in the past `expiration_seconds`. 708 709 @param expiration_seconds: The number of seconds to keep the cached 710 status of whether the DUT responded to ping. 711 @return: True if the DUT has responded to ping during the past 712 `expiration_seconds`. 713 """ 714 raise NotImplementedError("check_cached_up_status not implemented!") 715