1"""This class defines the Remote host class.""" 2 3import os, logging, urllib, time 4import re 5from autotest_lib.client.common_lib import error 6from autotest_lib.server import utils 7from autotest_lib.server.hosts import base_classes 8 9 10class RemoteHost(base_classes.Host): 11 """ 12 This class represents a remote machine on which you can run 13 programs. 14 15 It may be accessed through a network, a serial line, ... 16 It is not the machine autoserv is running on. 17 18 Implementation details: 19 This is an abstract class, leaf subclasses must implement the methods 20 listed here and in parent classes which have no implementation. They 21 may reimplement methods which already have an implementation. You 22 must not instantiate this class but should instantiate one of those 23 leaf subclasses. 24 """ 25 26 DEFAULT_REBOOT_TIMEOUT = base_classes.Host.DEFAULT_REBOOT_TIMEOUT 27 DEFAULT_HALT_TIMEOUT = 2 * 60 28 _LABEL_FUNCTIONS = [] 29 _DETECTABLE_LABELS = [] 30 31 VAR_LOG_MESSAGES_COPY_PATH = "/var/tmp/messages.autotest_start" 32 TMP_DIR_TEMPLATE = '/usr/local/tmp/autoserv-XXXXXX' 33 34 35 def _initialize(self, hostname, autodir=None, *args, **dargs): 36 super(RemoteHost, self)._initialize(*args, **dargs) 37 38 self.hostname = hostname 39 self.autodir = autodir 40 self.tmp_dirs = [] 41 42 43 def __repr__(self): 44 return "<remote host: %s>" % self.hostname 45 46 47 def close(self): 48 # pylint: disable=missing-docstring 49 super(RemoteHost, self).close() 50 self.stop_loggers() 51 52 if hasattr(self, 'tmp_dirs'): 53 for dir in self.tmp_dirs: 54 try: 55 self.run('rm -rf "%s"' % (utils.sh_escape(dir))) 56 except error.AutoservRunError: 57 pass 58 59 60 def job_start(self): 61 """ 62 Abstract method, called the first time a remote host object 63 is created for a specific host after a job starts. 64 65 This method depends on the create_host factory being used to 66 construct your host object. If you directly construct host objects 67 you will need to call this method yourself (and enforce the 68 single-call rule). 69 """ 70 try: 71 cmd = ('test ! -e /var/log/messages || cp -f /var/log/messages ' 72 '%s') % self.VAR_LOG_MESSAGES_COPY_PATH 73 self.run(cmd) 74 except Exception, e: 75 # Non-fatal error 76 logging.info('Failed to copy /var/log/messages at startup: %s', e) 77 78 79 def get_autodir(self): 80 return self.autodir 81 82 83 def set_autodir(self, autodir): 84 """ 85 This method is called to make the host object aware of the 86 where autotest is installed. Called in server/autotest.py 87 after a successful install 88 """ 89 self.autodir = autodir 90 91 92 def sysrq_reboot(self): 93 # pylint: disable=missing-docstring 94 self.run_background('echo b > /proc/sysrq-trigger') 95 96 97 def halt(self, timeout=DEFAULT_HALT_TIMEOUT, wait=True): 98 """ 99 Shut down the remote host. 100 101 N.B. This method makes no provision to bring the target back 102 up. The target will be offline indefinitely if there's no 103 independent hardware (servo, RPM, etc.) to force the target to 104 power on. 105 106 @param timeout Maximum time to wait for host down, in seconds. 107 @param wait Whether to wait for the host to go offline. 108 """ 109 self.run_background('sleep 1 ; halt') 110 if wait: 111 self.wait_down(timeout=timeout) 112 113 114 def reboot(self, timeout=DEFAULT_REBOOT_TIMEOUT, wait=True, 115 fastsync=False, reboot_cmd=None, **dargs): 116 """ 117 Reboot the remote host. 118 119 Args: 120 timeout - How long to wait for the reboot. 121 wait - Should we wait to see if the machine comes back up. 122 If this is set to True, ignores reboot_cmd's error 123 even if occurs. 124 fastsync - Don't wait for the sync to complete, just start one 125 and move on. This is for cases where rebooting prompty 126 is more important than data integrity and/or the 127 machine may have disks that cause sync to never return. 128 reboot_cmd - Reboot command to execute. 129 """ 130 self.reboot_setup(**dargs) 131 if not reboot_cmd: 132 reboot_cmd = ('sync & sleep 5; ' 133 'reboot & sleep 60; ' 134 'reboot -f & sleep 10; ' 135 'reboot -nf & sleep 10; ' 136 'telinit 6') 137 138 def reboot(): 139 # pylint: disable=missing-docstring 140 self.record("GOOD", None, "reboot.start") 141 current_boot_id = None 142 try: 143 current_boot_id = self.get_boot_id() 144 145 # sync before starting the reboot, so that a long sync during 146 # shutdown isn't timed out by wait_down's short timeout 147 if not fastsync: 148 self.run('sync; sync', timeout=timeout, ignore_status=True) 149 150 self.run_background(reboot_cmd) 151 except error.AutoservRunError: 152 # If wait is set, ignore the error here, and rely on the 153 # wait_for_restart() for stability, instead. 154 # reboot_cmd sometimes causes an error even if reboot is 155 # successfully in progress. This is difficult to be avoided, 156 # because we have no much control on remote machine after 157 # "reboot" starts. 158 if not wait or current_boot_id is None: 159 # TODO(b/37652392): Revisit no-wait case, later. 160 self.record("ABORT", None, "reboot.start", 161 "reboot command failed") 162 raise 163 if wait: 164 self.wait_for_restart(timeout, old_boot_id=current_boot_id, 165 **dargs) 166 167 # if this is a full reboot-and-wait, run the reboot inside a group 168 if wait: 169 self.log_op(self.OP_REBOOT, reboot) 170 else: 171 reboot() 172 173 def suspend(self, timeout, suspend_cmd, 174 allow_early_resume=False): 175 """ 176 Suspend the remote host. 177 178 Args: 179 timeout - How long to wait for the suspend in integer seconds. 180 suspend_cmd - suspend command to execute. 181 allow_early_resume - Boolean that indicate whether resume 182 before |timeout| is ok. 183 Raises: 184 error.AutoservSuspendError - If |allow_early_resume| is False 185 and if device resumes before 186 |timeout|. 187 """ 188 # define a function for the supend and run it in a group 189 def suspend(): 190 # pylint: disable=missing-docstring 191 self.record("GOOD", None, "suspend.start for %d seconds" % (timeout)) 192 try: 193 self.run_background(suspend_cmd) 194 except error.AutoservRunError: 195 self.record("ABORT", None, "suspend.start", 196 "suspend command failed") 197 raise error.AutoservSuspendError("suspend command failed") 198 199 # Wait for some time, to ensure the machine is going to sleep. 200 # Not too long to check if the machine really suspended. 201 time_slice = min(timeout / 2, 300) 202 time.sleep(time_slice) 203 time_counter = time_slice 204 while time_counter < timeout + 60: 205 # Check if the machine is back. We check regularely to 206 # ensure the machine was suspended long enough. 207 if utils.ping(self.hostname, tries=1, deadline=1) == 0: 208 return 209 else: 210 if time_counter > timeout - 10: 211 time_slice = 5 212 time.sleep(time_slice) 213 time_counter += time_slice 214 215 if utils.ping(self.hostname, tries=1, deadline=1) != 0: 216 raise error.AutoservSuspendError( 217 "DUT is not responding after %d seconds" % (time_counter)) 218 219 start_time = time.time() 220 self.log_op(self.OP_SUSPEND, suspend) 221 lasted = time.time() - start_time 222 logging.info("Device resumed after %d secs", lasted) 223 if (lasted < timeout and not allow_early_resume): 224 raise error.AutoservSuspendError( 225 "Suspend did not last long enough: %d instead of %d" % ( 226 lasted, timeout)) 227 228 def reboot_followup(self, *args, **dargs): 229 # pylint: disable=missing-docstring 230 super(RemoteHost, self).reboot_followup(*args, **dargs) 231 if self.job: 232 self.job.profilers.handle_reboot(self) 233 234 235 def wait_for_restart(self, timeout=DEFAULT_REBOOT_TIMEOUT, **dargs): 236 """ 237 Wait for the host to come back from a reboot. This wraps the 238 generic wait_for_restart implementation in a reboot group. 239 """ 240 def op_func(): 241 # pylint: disable=missing-docstring 242 super(RemoteHost, self).wait_for_restart(timeout=timeout, **dargs) 243 self.log_op(self.OP_REBOOT, op_func) 244 245 246 def cleanup(self): 247 # pylint: disable=missing-docstring 248 super(RemoteHost, self).cleanup() 249 self.reboot() 250 251 252 def get_tmp_dir(self, parent='/tmp'): 253 """ 254 Return the pathname of a directory on the host suitable 255 for temporary file storage. 256 257 The directory and its content will be deleted automatically 258 on the destruction of the Host object that was used to obtain 259 it. 260 """ 261 self.run("mkdir -p %s" % parent) 262 template = os.path.join(parent, self.TMP_DIR_TEMPLATE) 263 dir_name = self.run("mktemp -d %s" % template).stdout.rstrip() 264 self.tmp_dirs.append(dir_name) 265 return dir_name 266 267 268 def get_platform_label(self): 269 """ 270 Return the platform label, or None if platform label is not set. 271 """ 272 273 if self.job: 274 keyval_path = os.path.join(self.job.resultdir, 'host_keyvals', 275 self.hostname) 276 keyvals = utils.read_keyval(keyval_path) 277 return keyvals.get('platform', None) 278 else: 279 return None 280 281 282 def get_all_labels(self): 283 """ 284 Return all labels, or empty list if label is not set. 285 """ 286 if self.job: 287 keyval_path = os.path.join(self.job.resultdir, 'host_keyvals', 288 self.hostname) 289 keyvals = utils.read_keyval(keyval_path) 290 all_labels = keyvals.get('labels', '') 291 if all_labels: 292 all_labels = all_labels.split(',') 293 return [urllib.unquote(label) for label in all_labels] 294 return [] 295 296 297 def delete_tmp_dir(self, tmpdir): 298 """ 299 Delete the given temporary directory on the remote machine. 300 301 @param tmpdir The directory to delete. 302 """ 303 self.run('rm -rf "%s"' % utils.sh_escape(tmpdir), ignore_status=True) 304 self.tmp_dirs.remove(tmpdir) 305 306 307 def delete_all_tmp_dirs(self, parent='/tmp'): 308 """ 309 Delete all directories in parent that were created by get_tmp_dir 310 311 Note that this may involve deleting directories created by calls to 312 get_tmp_dir on a different RemoteHost instance than the one running this 313 method. Only perform this operation when certain that this will not 314 cause unexpected behavior. 315 """ 316 # follow mktemp's behavior of only expanding 3 or more consecutive Xs 317 base_template = re.sub('XXXX*', '*', self.TMP_DIR_TEMPLATE) 318 # distinguish between non-wildcard asterisks in parent directory name 319 # and wildcards inserted from the template 320 base = '*'.join(map(lambda x: '"%s"' % utils.sh_escape(x), 321 base_template.split('*'))) 322 path = '"%s' % os.path.join(utils.sh_escape(parent), base[1:]) 323 self.run('rm -rf %s' % path, ignore_status=True) 324 # remove deleted directories from tmp_dirs 325 regex = os.path.join(parent, re.sub('(XXXX*)', 326 lambda match: '[a-zA-Z0-9]{%d}' % len(match.group(1)), 327 self.TMP_DIR_TEMPLATE)) 328 regex += '(/|$)' # remove if matches, or is within a dir that matches 329 self.tmp_dirs = filter(lambda x: not re.match(regex, x), self.tmp_dirs) 330 331 332 def check_uptime(self): 333 """ 334 Check that uptime is available and monotonically increasing. 335 """ 336 if not self.is_up(): 337 raise error.AutoservHostError('Client does not appear to be up') 338 result = self.run("/bin/cat /proc/uptime", 30) 339 return result.stdout.strip().split()[0] 340 341 342 def check_for_lkdtm(self): 343 """ 344 Check for kernel dump test module. return True if exist. 345 """ 346 cmd = 'ls /sys/kernel/debug/provoke-crash/DIRECT' 347 return self.run(cmd, ignore_status=True).exit_status == 0 348 349 350 def are_wait_up_processes_up(self): 351 """ 352 Checks if any HOSTS waitup processes are running yet on the 353 remote host. 354 355 Returns True if any the waitup processes are running, False 356 otherwise. 357 """ 358 processes = self.get_wait_up_processes() 359 if len(processes) == 0: 360 return True # wait up processes aren't being used 361 for procname in processes: 362 exit_status = self.run("{ ps -e || ps; } | grep '%s'" % procname, 363 ignore_status=True).exit_status 364 if exit_status == 0: 365 return True 366 return False 367 368 369 def get_labels(self): 370 """Return a list of labels for this given host. 371 372 This is the main way to retrieve all the automatic labels for a host 373 as it will run through all the currently implemented label functions. 374 """ 375 labels = [] 376 for label_function in self._LABEL_FUNCTIONS: 377 try: 378 label = label_function(self) 379 except Exception: 380 logging.exception('Label function %s failed; ignoring it.', 381 label_function.__name__) 382 label = None 383 if label: 384 if type(label) is str: 385 labels.append(label) 386 elif type(label) is list: 387 labels.extend(label) 388 return labels 389