blob: 805857eb83fd082de36ad4f4ea76a885684d5b92 [file] [log] [blame]
J. Richard Barnette24adbf42012-04-11 15:04:53 -07001# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Dale Curtisaa5eedb2011-08-23 16:18:52 -07002# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
J. Richard Barnette1d78b012012-05-15 13:56:30 -07005import logging
Dan Shi0f466e82013-02-22 15:44:58 -08006import os
Simran Basid5e5e272012-09-24 15:23:59 -07007import re
Vincent Palatindf2372c2016-10-07 17:03:00 +02008import sys
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07009import time
10
mussa584b4462014-06-20 15:13:28 -070011import common
J. Richard Barnette45e93de2012-04-11 17:24:15 -070012from autotest_lib.client.bin import utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070013from autotest_lib.client.common_lib import autotemp
Richard Barnette0c73ffc2012-11-19 15:21:18 -080014from autotest_lib.client.common_lib import error
15from autotest_lib.client.common_lib import global_config
J. Richard Barnette91137f02016-03-10 16:52:26 -080016from autotest_lib.client.common_lib import hosts
Dan Shi549fb822015-03-24 18:01:11 -070017from autotest_lib.client.common_lib import lsbrelease_utils
Richard Barnette03a0c132012-11-05 12:40:35 -080018from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070019from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000020from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080021from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080022from autotest_lib.server import afe_utils
Anh Le3ad780d2019-05-28 10:54:51 -070023from autotest_lib.server import crashcollect
Dan Shia1ecd5c2013-06-06 11:21:31 -070024from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070025from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050026from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070027from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070028from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070029from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070030from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080031from autotest_lib.server.hosts import chameleon_host
Richard Barnetted31580e2018-05-14 19:58:00 +000032from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080033from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070034from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070035from autotest_lib.server.hosts import servo_host
Simran Basidcff4252012-11-20 16:13:20 -080036from autotest_lib.site_utils.rpm_control_system import rpm_client
Simran Basid5e5e272012-09-24 15:23:59 -070037
Simran Basi382506b2016-09-13 14:58:15 -070038# In case cros_host is being ran via SSP on an older Moblab version with an
39# older chromite version.
40try:
41 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080042except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080043 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080044
Simran Basid5e5e272012-09-24 15:23:59 -070045
Dan Shib8540a52015-07-16 14:18:23 -070046CONFIG = global_config.global_config
47
Dan Shid07ee2e2015-09-24 14:49:25 -070048
beepsc87ff602013-07-31 21:53:00 -070049class FactoryImageCheckerException(error.AutoservError):
50 """Exception raised when an image is a factory image."""
51 pass
52
53
Fang Deng0ca40e22013-08-27 17:47:44 -070054class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070055 """Chromium OS specific subclass of Host."""
56
Simran Basi5ace6f22016-01-06 17:30:44 -080057 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
58
Scott Zawalski62bacae2013-03-05 10:40:32 -050059 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070060
Richard Barnette03a0c132012-11-05 12:40:35 -080061 # Timeout values (in seconds) associated with various Chrome OS
62 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070063 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080064 # In general, a good rule of thumb is that the timeout can be up
65 # to twice the typical measured value on the slowest platform.
66 # The times here have not necessarily been empirically tested to
67 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070068 #
69 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080070 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
71 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080072 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070073 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080074 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070075 # screen delay, time to start the network on the DUT, and the
76 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070077 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080078 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080079 # network.
beepsf079cfb2013-09-18 17:49:51 -070080 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080081 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
82 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070083
84 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080085 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080086 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070087 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070088 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070089 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080090 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070091
Dan Shica503482015-03-30 17:23:25 -070092 # Minimum OS version that supports server side packaging. Older builds may
93 # not have server side package built or with Autotest code change to support
94 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070095 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -070096 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -070097
J. Richard Barnette84890bd2014-02-21 11:05:47 -080098 # REBOOT_TIMEOUT: How long to wait for a reboot.
99 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700100 # We have a long timeout to ensure we don't flakily fail due to other
101 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700102 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
103 # return from reboot' bug is solved.
104 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700105
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800106 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
107 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700108 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
109 # since changing servo role will reset USB state
110 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800111 _USB_POWER_TIMEOUT = 5
112 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700113 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800114
Fang Dengdeba14f2014-11-14 11:54:09 -0800115 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
116 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700117
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800118 # Constants used in ping_wait_up() and ping_wait_down().
119 #
120 # _PING_WAIT_COUNT is the approximate number of polling
121 # cycles to use when waiting for a host state change.
122 #
123 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
124 # for arguments to the internal _ping_wait_for_status()
125 # method.
126 _PING_WAIT_COUNT = 40
127 _PING_STATUS_DOWN = False
128 _PING_STATUS_UP = True
129
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800130 # Allowed values for the power_method argument.
131
Garry Wang5e5538a2019-04-08 15:36:18 -0700132 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
133 # DUTs except those with servo_v4 CCD.
134 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
135 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800136 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
137 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
138 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700139 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800140 POWER_CONTROL_SERVO = 'servoj10'
141 POWER_CONTROL_MANUAL = 'manual'
142
143 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700144 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800145 POWER_CONTROL_SERVO,
146 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800147
Garry Wang5e5538a2019-04-08 15:36:18 -0700148 # CCD_SERVO: The string of servo type to compare with the return value of
149 # self.servo.get_servo_version() to decide default power method.
150 CCD_SERVO = 'servo_v4_with_ccd_cr50'
151
Simran Basi5e6339a2013-03-21 11:34:32 -0700152 _RPM_OUTLET_CHANGED = 'outlet_changed'
153
Dan Shi9cb0eec2014-06-03 09:04:50 -0700154 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700155 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700156 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700157
Anh Le3ad780d2019-05-28 10:54:51 -0700158 # DUT_LOG_LOCATION: the directory in the DUT that the log is saved
159 # after re-imaging using chromeos-install.
160 # The location is specified in chromeos-install script.
161 DUT_LOG_LOCATION = '/mnt/stateful_partition/unencrypted/prior_logs'
J. Richard Barnette91137f02016-03-10 16:52:26 -0800162
J. Richard Barnette964fba02012-10-24 17:34:29 -0700163 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800164 def check_host(host, timeout=10):
165 """
166 Check if the given host is a chrome-os host.
167
168 @param host: An ssh host representing a device.
169 @param timeout: The timeout for the run command.
170
171 @return: True if the host device is chromeos.
172
beeps46dadc92013-11-07 14:07:10 -0800173 """
174 try:
Allen Liad719c12017-06-27 23:48:04 +0000175 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700176 'grep -q CHROMEOS /etc/lsb-release && '
177 '! test -f /mnt/stateful_partition/.android_tester && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700178 '! grep -q moblab /etc/lsb-release && '
179 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700180 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700181 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000182 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700183 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
184 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800185 return not (
186 lsbrelease_utils.is_jetstream(
187 lsb_release_content=lsb_release_content) or
188 lsbrelease_utils.is_gce_board(
189 lsb_release_content=lsb_release_content))
190
beeps46dadc92013-11-07 14:07:10 -0800191 except (error.AutoservRunError, error.AutoservSSHTimeout):
192 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700193
194 return False
beeps46dadc92013-11-07 14:07:10 -0800195
196
197 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800198 def get_chameleon_arguments(args_dict):
199 """Extract chameleon options from `args_dict` and return the result.
200
201 Recommended usage:
202 ~~~~~~~~
203 args_dict = utils.args_to_dict(args)
204 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
205 host = hosts.create_host(machine, chameleon_args=chameleon_args)
206 ~~~~~~~~
207
208 @param args_dict Dictionary from which to extract the chameleon
209 arguments.
210 """
howardchung83e55272019-08-08 14:08:05 +0800211 if 'chameleon_host_list' in args_dict:
212 result = []
213 for chameleon in args_dict['chameleon_host_list'].split(','):
214 result.append({key: value for key,value in
215 zip(('chameleon_host','chameleon_port'),
216 chameleon.split(':'))})
217
218 logging.info(result)
219 return result
220 else:
221 return {key: args_dict[key]
Allen Li083866b2016-08-18 10:07:10 -0700222 for key in ('chameleon_host', 'chameleon_port')
223 if key in args_dict}
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800224
225
226 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700227 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800228 """Extract chameleon options from `args_dict` and return the result.
229
230 Recommended usage:
231 ~~~~~~~~
232 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700233 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
234 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800235 ~~~~~~~~
236
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700237 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800238 arguments.
239 """
Allen Li083866b2016-08-18 10:07:10 -0700240 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700241 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700242 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800243
244
245 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800246 def get_servo_arguments(args_dict):
247 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800248
249 Recommended usage:
250 ~~~~~~~~
251 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700252 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800253 host = hosts.create_host(machine, servo_args=servo_args)
254 ~~~~~~~~
255
256 @param args_dict Dictionary from which to extract the servo
257 arguments.
258 """
Richard Barnettee519dcd2016-08-15 17:37:17 -0700259 servo_attrs = (servo_host.SERVO_HOST_ATTR,
260 servo_host.SERVO_PORT_ATTR,
Nick Sanders2f3c9852018-10-24 12:10:24 -0700261 servo_host.SERVO_BOARD_ATTR,
262 servo_host.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200263 servo_args = {key: args_dict[key]
264 for key in servo_attrs
265 if key in args_dict}
266 return (
267 None
268 if servo_host.SERVO_HOST_ATTR in servo_args
269 and not servo_args[servo_host.SERVO_HOST_ATTR]
270 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700271
J. Richard Barnette964fba02012-10-24 17:34:29 -0700272
J. Richard Barnette91137f02016-03-10 16:52:26 -0800273 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700274 pdtester_args=None, try_lab_servo=False,
Richard Barnette9a26ad62016-06-10 12:03:08 -0700275 try_servo_repair=False,
J. Richard Barnette91137f02016-03-10 16:52:26 -0800276 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700277 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800278 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700279
Fang Denge545abb2014-12-30 18:43:47 -0800280 This method will attempt to create the test-assistant object
281 (chameleon/servo) when it is needed by the test. Check
282 the docstring of chameleon_host.create_chameleon_host and
283 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700284
Fang Denge545abb2014-12-30 18:43:47 -0800285 @param hostname: Hostname of the dut.
286 @param chameleon_args: A dictionary that contains args for creating
287 a ChameleonHost. See chameleon_host for details.
288 @param servo_args: A dictionary that contains args for creating
289 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700290 @param try_lab_servo: When true, indicates that an attempt should
291 be made to create a ServoHost for a DUT in
292 the test lab, even if not required by
293 `servo_args`. See servo_host for details.
294 @param try_servo_repair: If a servo host is created, check it
295 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800296 See servo_host for details.
297 @param ssh_verbosity_flag: String, to pass to the ssh command to control
298 verbosity.
299 @param ssh_options: String, other ssh options to pass to the ssh
300 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700301 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700302 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700303 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800304 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Kevin Chenga2619dc2016-03-28 11:42:08 -0700305 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700306 # self.env is a dictionary of environment variable settings
307 # to be exported for commands run on the host.
308 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
309 # errors that might happen.
310 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700311 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700312 self._ssh_options = ssh_options
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700313 self.set_servo_host(
314 servo_host.create_servo_host(
Richard Barnetteea3e4602016-06-10 12:36:41 -0700315 dut=self, servo_args=servo_args,
Richard Barnette9a26ad62016-06-10 12:03:08 -0700316 try_lab_servo=try_lab_servo,
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700317 try_servo_repair=try_servo_repair))
Garry Wang5e5538a2019-04-08 15:36:18 -0700318 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700319
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800320 # TODO(waihong): Do the simplication on Chameleon too.
howardchung83e55272019-08-08 14:08:05 +0800321 if type(chameleon_args) is list:
322 self.multi_chameleon = True
323 chameleon_args_list = chameleon_args
324 else:
325 self.multi_chameleon = False
326 chameleon_args_list = [chameleon_args]
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800327
howardchung83e55272019-08-08 14:08:05 +0800328 self._chameleon_host_list = [
329 chameleon_host.create_chameleon_host(
330 dut=self.hostname, chameleon_args=_args)
331 for _args in chameleon_args_list]
332
333 self.chameleon_list = [_host.create_chameleon_board() for _host in
334 self._chameleon_host_list if _host is not None]
335 if len(self.chameleon_list) > 0:
336 self.chameleon = self.chameleon_list[0]
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800337 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800338 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700339
howardchung83e55272019-08-08 14:08:05 +0800340 # Add pdtester host if pdtester args were added on command line
341 self._pdtester_host = pdtester_host.create_pdtester_host(pdtester_args)
342
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700343 if self._pdtester_host:
344 self.pdtester_servo = self._pdtester_host.get_servo()
345 logging.info('pdtester_servo: %r', self.pdtester_servo)
346 # Create the pdtester object used to access the ec uart
347 self.pdtester = pdtester.PDTester(self.pdtester_servo,
348 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800349 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700350 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800351
Fang Deng5d518f42013-08-02 14:04:32 -0700352
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000353 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700354 """Get latest stable cros image name from AFE.
355
356 Use the board name from the info store. Should that fail, try to
357 retrieve the board name from the host's installed image itself.
358
359 @returns: current stable cros image name for this host.
360 """
361 board = self.host_info_store.get().board
362 if not board:
363 logging.warn('No board label value found. Trying to infer '
364 'from the host itself.')
365 try:
366 board = self.get_board().split(':')[1]
367 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
368 logging.error('Also failed to get the board name from the DUT '
369 'itself. %s.', str(e))
370 raise error.AutoservError('Cannot obtain repair image name.')
371 return afe_utils.get_stable_cros_image_name(board)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500372
373
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700374 def host_version_prefix(self, image):
375 """Return version label prefix.
376
377 In case the CrOS provisioning version is something other than the
378 standard CrOS version e.g. CrOS TH version, this function will
379 find the prefix from provision.py.
380
381 @param image: The image name to find its version prefix.
382 @returns: A prefix string for the image type.
383 """
384 return provision.get_version_label_prefix(image)
385
386
beepsdae65fd2013-07-26 16:24:41 -0700387 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700388 """
389 Make sure job_repo_url of this host is valid.
390
joychen03eaad92013-06-26 09:55:21 -0700391 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700392 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
393 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
394 download and extract it. If the devserver embedded in the url is
395 unresponsive, update the job_repo_url of the host after staging it on
396 another devserver.
397
398 @param job_repo_url: A url pointing to the devserver where the autotest
399 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700400 @param tag: The tag from the server job, in the format
401 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700402
403 @raises DevServerException: If we could not resolve a devserver.
404 @raises AutoservError: If we're unable to save the new job_repo_url as
405 a result of choosing a new devserver because the old one failed to
406 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700407 @raises urllib2.URLError: If the devserver embedded in job_repo_url
408 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700409 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800410 info = self.host_info_store.get()
411 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700412 if not job_repo_url:
413 logging.warning('No job repo url set on host %s', self.hostname)
414 return
415
416 logging.info('Verifying job repo url %s', job_repo_url)
417 devserver_url, image_name = tools.get_devserver_build_from_package_url(
418 job_repo_url)
419
beeps0c865032013-07-30 11:37:06 -0700420 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700421
422 logging.info('Staging autotest artifacts for %s on devserver %s',
423 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700424
425 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700426 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700427 stage_time = time.time() - start_time
428
429 # Record how much of the verification time comes from a devserver
430 # restage. If we're doing things right we should not see multiple
431 # devservers for a given board/build/branch path.
432 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800433 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700434 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800435 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700436 pass
437 else:
beeps0c865032013-07-30 11:37:06 -0700438 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700439 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700440 stats_key = {
441 'board': board,
442 'build_type': build_type,
443 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700444 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700445 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800446
447 monarch_fields = {
448 'board': board,
449 'build_type': build_type,
450 # TODO(akeshet): To be consistent with most other metrics,
451 # consider changing the following field to be named
452 # 'milestone'.
453 'branch': branch,
454 'dev_server': devserver,
455 }
456 metrics.Counter(
457 'chromeos/autotest/provision/verify_url'
458 ).increment(fields=monarch_fields)
459 metrics.SecondsDistribution(
460 'chromeos/autotest/provision/verify_url_duration'
461 ).add(stage_time, fields=monarch_fields)
462
463
Dan Shicf4d2032015-03-12 15:04:21 -0700464 def stage_server_side_package(self, image=None):
465 """Stage autotest server-side package on devserver.
466
467 @param image: Full path of an OS image to install or a build name.
468
469 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700470
471 @raise: error.AutoservError if fail to locate the build to test with, or
472 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700473 """
Dan Shid37736b2016-07-06 15:10:29 -0700474 # If enable_drone_in_restricted_subnet is False, do not set hostname
475 # in devserver.resolve call, so a devserver in non-restricted subnet
476 # is picked to stage autotest server package for drone to download.
477 hostname = self.hostname
478 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
479 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700480 if image:
481 image_name = tools.get_build_from_image(image)
482 if not image_name:
483 raise error.AutoservError(
484 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700485 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700486 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700487 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800488 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700489 if job_repo_url:
490 devserver_url, image_name = (
491 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700492 # If enable_drone_in_restricted_subnet is True, use the
493 # existing devserver. Otherwise, resolve a new one in
494 # non-restricted subnet.
495 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
496 ds = dev_server.ImageServer(devserver_url)
497 else:
498 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800499 elif info.build is not None:
500 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700501 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700502 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800503 raise error.AutoservError(
504 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700505 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700506
507 # Get the OS version of the build, for any build older than
508 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
509 match = re.match('.*/R\d+-(\d+)\.', image_name)
510 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700511 raise error.AutoservError(
512 'Build %s is older than %s. Server side packaging is '
513 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700514
Dan Shicf4d2032015-03-12 15:04:21 -0700515 ds.stage_artifacts(image_name, ['autotest_server_package'])
516 return '%s/static/%s/%s' % (ds.url(), image_name,
517 'autotest_server_package.tar.bz2')
518
519
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700520 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700521 """Stage a build on a devserver and return the update_url.
522
523 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700524 @param artifact: a string like 'test_image'. Requests
525 appropriate image to be staged.
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700526 @returns an update URL like:
527 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0
528 """
529 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000530 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700531 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800532 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700533 devserver.stage_artifacts(image_name, [artifact])
534 if artifact == 'test_image':
535 return devserver.get_test_image_url(image_name)
536 elif artifact == 'recovery_image':
537 return devserver.get_recovery_image_url(image_name)
538 else:
539 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700540
541
beepse539be02013-07-31 21:57:39 -0700542 def stage_factory_image_for_servo(self, image_name):
543 """Stage a build on a devserver and return the update_url.
544
545 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700546
beepse539be02013-07-31 21:57:39 -0700547 @return: An update URL, eg:
548 http://<devserver>/static/canary-channel/\
549 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700550
551 @raises: ValueError if the factory artifact name is missing from
552 the config.
553
beepse539be02013-07-31 21:57:39 -0700554 """
555 if not image_name:
556 logging.error('Need an image_name to stage a factory image.')
557 return
558
Dan Shib8540a52015-07-16 14:18:23 -0700559 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700560 'CROS', 'factory_artifact', type=str, default='')
561 if not factory_artifact:
562 raise ValueError('Cannot retrieve the factory artifact name from '
563 'autotest config, and hence cannot stage factory '
564 'artifacts.')
565
beepse539be02013-07-31 21:57:39 -0700566 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800567 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700568 devserver.stage_artifacts(
569 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700570 [factory_artifact],
571 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700572
573 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
574
575
Laurence Goodby778c9a42017-05-24 19:24:07 -0700576 def prepare_for_update(self):
577 """Prepares the DUT for an update.
578
579 Subclasses may override this to perform any special actions
580 required before updating.
581 """
Laurence Goodby468de252017-06-08 17:22:53 -0700582 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700583
584
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800585 def _clear_fw_version_labels(self, rw_only):
586 """Clear firmware version labels from the machine.
587
588 @param rw_only: True to only clear fwrw_version; otherewise, clear
589 both fwro_version and fwrw_version.
590 """
Dan Shi9cb0eec2014-06-03 09:04:50 -0700591 labels = self._AFE.get_labels(
Dan Shi0723bf52015-06-24 10:52:38 -0700592 name__startswith=provision.FW_RW_VERSION_PREFIX,
Dan Shi9cb0eec2014-06-03 09:04:50 -0700593 host__hostname=self.hostname)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800594 if not rw_only:
595 labels = labels + self._AFE.get_labels(
596 name__startswith=provision.FW_RO_VERSION_PREFIX,
597 host__hostname=self.hostname)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700598 for label in labels:
599 label.remove_hosts(hosts=[self.hostname])
600
601
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800602 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700603 """Add firmware version label to the machine.
604
605 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800606 @param rw_only: True to only add fwrw_version; otherwise, add both
607 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700608
609 """
Prathmesh Prabhu2c7471d2016-11-15 20:19:57 +0000610 fw_label = provision.fwrw_version_to_label(build)
MK Ryu73be9862015-07-06 12:25:00 -0700611 self._AFE.run('label_add_hosts', id=fw_label, hosts=[self.hostname])
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800612 if not rw_only:
Prathmesh Prabhu2c7471d2016-11-15 20:19:57 +0000613 fw_label = provision.fwro_version_to_label(build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800614 self._AFE.run('label_add_hosts', id=fw_label, hosts=[self.hostname])
Dan Shi9cb0eec2014-06-03 09:04:50 -0700615
616
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700617 def firmware_install(self, build=None, rw_only=False, dest=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700618 """Install firmware to the DUT.
619
620 Use stateful update if the DUT is already running the same build.
621 Stateful update does not update kernel and tends to run much faster
622 than a full reimage. If the DUT is running a different build, or it
623 failed to do a stateful update, full update, including kernel update,
624 will be applied to the DUT.
625
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800626 Once a host enters firmware_install its fw[ro|rw]_version label will
627 be removed. After the firmware is updated successfully, a new
628 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700629
630 @param build: The build version to which we want to provision the
631 firmware of the machine,
632 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800633 @param rw_only: True to only install firmware to its RW portions. Keep
634 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700635 @param dest: Directory to store the firmware in.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700636
637 TODO(dshi): After bug 381718 is fixed, update here with corresponding
638 exceptions that could be raised.
639
640 """
641 if not self.servo:
642 raise error.TestError('Host %s does not have servo.' %
643 self.hostname)
644
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700645 # Get the DUT board name from AFE.
646 info = self.host_info_store.get()
647 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700648 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800649
650 if board is None or board == '':
651 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700652
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700653 if model is None or model == '':
654 model = self.get_platform_from_fwid()
655
Chris Sosae92399e2015-04-24 11:32:59 -0700656 # If build is not set, try to install firmware from stable CrOS.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700657 if not build:
Richard Barnette260cbd02016-10-06 12:23:28 -0700658 build = afe_utils.get_stable_faft_version(board)
Dan Shi3d7a0e12015-10-12 11:55:45 -0700659 if not build:
660 raise error.TestError(
661 'Failed to find stable firmware build for %s.',
662 self.hostname)
663 logging.info('Will install firmware from build %s.', build)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700664
Dan Shi216389c2015-12-22 11:03:06 -0800665 ds = dev_server.ImageServer.resolve(build, self.hostname)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700666 ds.stage_artifacts(build, ['firmware'])
667
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700668 tmpd = None
669 if not dest:
670 tmpd = autotemp.tempdir(unique_id='fwimage')
671 dest = tmpd.name
Dan Shi9cb0eec2014-06-03 09:04:50 -0700672 try:
673 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700674 local_tarball = os.path.join(dest, os.path.basename(fwurl))
xixuan4e116822016-11-17 15:32:10 -0800675 ds.download_file(fwurl, local_tarball)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700676
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800677 self._clear_fw_version_labels(rw_only)
Shelley Chenac61d5a2019-06-24 15:35:46 -0700678 self.servo.program_firmware(board, model, local_tarball, rw_only)
Wai-Hong Tamb5f66ce2016-11-10 15:45:30 -0800679 if utils.host_is_in_lab_zone(self.hostname):
680 self._add_fw_version_label(build, rw_only)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700681 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700682 if tmpd:
683 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700684
685
beepsf079cfb2013-09-18 17:49:51 -0700686 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
687 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500688 """
689 Re-install the OS on the DUT by:
690 1) installing a test image on a USB storage device attached to the Servo
691 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800692 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700693 3) installing the image with chromeos-install.
694
Scott Zawalski62bacae2013-03-05 10:40:32 -0500695 @param image_url: If specified use as the url to install on the DUT.
696 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700697 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
698 Factory images need a longer usb_boot_timeout than regular
699 cros images.
700 @param install_timeout: The timeout to use when installing the chromeos
701 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800702
Scott Zawalski62bacae2013-03-05 10:40:32 -0500703 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700704
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800705 """
beepsf079cfb2013-09-18 17:49:51 -0700706 logging.info('Downloading image to USB, then booting from it. Usb boot '
707 'timeout = %s', usb_boot_timeout)
Allen Li48a13fe2016-11-22 14:10:40 -0800708 with metrics.SecondsTimer(
709 'chromeos/autotest/provision/servo_install/boot_duration'):
710 self.servo.install_recovery_image(image_url)
711 if not self.wait_up(timeout=usb_boot_timeout):
712 raise hosts.AutoservRepairError(
713 'DUT failed to boot from USB after %d seconds' %
Garry Wang954f8382019-01-23 13:49:29 -0800714 usb_boot_timeout, 'failed_to_reboot')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500715
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800716 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
717 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800718 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800719 try:
720 self.run('chromeos-tpm-recovery')
721 except error.AutoservRunError:
722 logging.warn('chromeos-tpm-recovery is too old.')
723
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800724
Allen Li48a13fe2016-11-22 14:10:40 -0800725 with metrics.SecondsTimer(
726 'chromeos/autotest/provision/servo_install/install_duration'):
727 logging.info('Installing image through chromeos-install.')
Anh Lee21e4032019-07-11 15:01:06 -0700728 try:
729 # Re-imaging the DUT with log collecting.
730 self.run(
731 'chromeos-install --yes '
732 '--lab_preserve_logs='
733 '"/usr/local/autotest/common_lib/logs_to_collect"',
734 timeout=install_timeout)
735 except Exception as e:
736 logging.exception(
737 'Fail to collect log from DUT.'
738 'Retry to fix DUT without collecting log.')
739 self.run(
740 'chromeos-install --yes',
741 timeout=install_timeout)
742
Allen Li48a13fe2016-11-22 14:10:40 -0800743 self.halt()
beepsf079cfb2013-09-18 17:49:51 -0700744
745 logging.info('Power cycling DUT through servo.')
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800746 self.servo.get_power_state_controller().power_off()
Fang Dengafb88142013-05-30 17:44:31 -0700747 self.servo.switch_usbkey('off')
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800748 # N.B. The Servo API requires that we use power_on() here
749 # for two reasons:
750 # 1) After turning on a DUT in recovery mode, you must turn
751 # it off and then on with power_on() once more to
752 # disable recovery mode (this is a Parrot specific
753 # requirement).
754 # 2) After power_off(), the only way to turn on is with
755 # power_on() (this is a Storm specific requirement).
J. Richard Barnettefbcc7122013-07-24 18:24:59 -0700756 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -0700757
758 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -0800759 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
760 raise error.AutoservError('DUT failed to reboot installed '
761 'test image after %d seconds' %
Scott Zawalski62bacae2013-03-05 10:40:32 -0500762 self.BOOT_TIMEOUT)
763
Anh Le3ad780d2019-05-28 10:54:51 -0700764 # The log saved after re-imaging process is transferred to shard
765 # result directory when the job instance exists.
766 # When we run repair manually, the result directory is created
767 # within local host.
768 try:
769 local_dir = crashcollect.get_crashinfo_dir(
770 self,
Anh Lee21e4032019-07-11 15:01:06 -0700771 'prior_log'
Anh Le3ad780d2019-05-28 10:54:51 -0700772 )
773
774 self.collect_logs(self.DUT_LOG_LOCATION, local_dir)
775 except OSError:
776 logging.exception('Fail to collect log. '
777 'The destination log directory does not exist')
778
Scott Zawalski62bacae2013-03-05 10:40:32 -0500779
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700780 def set_servo_host(self, host):
781 """Set our servo host member, and associated servo.
782
783 @param host Our new `ServoHost`.
784 """
785 self._servo_host = host
786 if self._servo_host is not None:
787 self.servo = self._servo_host.get_servo()
788 else:
789 self.servo = None
790
791
Richard Barnette9a26ad62016-06-10 12:03:08 -0700792 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -0700793 """
Richard Barnette9a26ad62016-06-10 12:03:08 -0700794 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -0700795
Richard Barnette9a26ad62016-06-10 12:03:08 -0700796 If the servo object is missing, attempt to repair the servo
797 host. Repair failures are passed back to the caller.
798
799 @raise AutoservError: If there is no servo host for this CrOS
800 host.
801 """
802 if self.servo:
803 return
804 if not self._servo_host:
805 raise error.AutoservError('No servo host for %s.' %
806 self.hostname)
807 self._servo_host.repair()
808 self.servo = self._servo_host.get_servo()
Dan Shi90466352015-09-22 15:01:05 -0700809
810
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -0800811 def repair(self):
812 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -0800813
814 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -0800815 not call back to the parent class, but instead relies on
816 `self._repair_strategy` to coordinate the verification and
817 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -0800818 """
Richard Barnetteabbdc252018-07-26 16:57:42 -0700819 message = 'Beginning repair for host %s board %s model %s'
820 info = self.host_info_store.get()
821 message %= (self.hostname, info.board, info.model)
822 self.record('INFO', None, None, message)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800823 self._repair_strategy.repair(self)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500824
Richard Barnette82c35912012-11-20 10:09:10 -0800825
J. Richard Barnette1d78b012012-05-15 13:56:30 -0700826 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -0800827 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -0700828 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +0800829
830 for chameleon_host in self._chameleon_host_list:
831 if chameleon_host:
832 chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -0800833
834 if self._servo_host:
835 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -0700836
837
Dan Shi49ca0932014-11-14 11:22:27 -0800838 def get_power_supply_info(self):
839 """Get the output of power_supply_info.
840
841 power_supply_info outputs the info of each power supply, e.g.,
842 Device: Line Power
843 online: no
844 type: Mains
845 voltage (V): 0
846 current (A): 0
847 Device: Battery
848 state: Discharging
849 percentage: 95.9276
850 technology: Li-ion
851
852 Above output shows two devices, Line Power and Battery, with details of
853 each device listed. This function parses the output into a dictionary,
854 with key being the device name, and value being a dictionary of details
855 of the device info.
856
857 @return: The dictionary of power_supply_info, e.g.,
858 {'Line Power': {'online': 'yes', 'type': 'main'},
859 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -0800860 @raise error.AutoservRunError if power_supply_info tool is not found in
861 the DUT. Caller should handle this error to avoid false failure
862 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -0800863 """
864 result = self.run('power_supply_info').stdout.strip()
865 info = {}
866 device_name = None
867 device_info = {}
868 for line in result.split('\n'):
869 pair = [v.strip() for v in line.split(':')]
870 if len(pair) != 2:
871 continue
872 if pair[0] == 'Device':
873 if device_name:
874 info[device_name] = device_info
875 device_name = pair[1]
876 device_info = {}
877 else:
878 device_info[pair[0]] = pair[1]
879 if device_name and not device_name in info:
880 info[device_name] = device_info
881 return info
882
883
884 def get_battery_percentage(self):
885 """Get the battery percentage.
886
887 @return: The percentage of battery level, value range from 0-100. Return
888 None if the battery info cannot be retrieved.
889 """
890 try:
891 info = self.get_power_supply_info()
892 logging.info(info)
893 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -0800894 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -0800895 return None
896
897
898 def is_ac_connected(self):
899 """Check if the dut has power adapter connected and charging.
900
901 @return: True if power adapter is connected and charging.
902 """
903 try:
904 info = self.get_power_supply_info()
905 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -0800906 except (KeyError, error.AutoservRunError):
907 return None
Dan Shi49ca0932014-11-14 11:22:27 -0800908
909
Simran Basi5e6339a2013-03-21 11:34:32 -0700910 def _cleanup_poweron(self):
911 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -0800912 info = self.host_info_store.get()
913 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -0700914 return
915 logging.debug('This host has recently interacted with the RPM'
916 ' Infrastructure. Ensuring power is on.')
917 try:
918 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -0800919 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -0700920 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -0700921 logging.error('Failed to turn Power On for this host after '
922 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -0800923
924 battery_percentage = self.get_battery_percentage()
Dan Shif01ebe22014-12-05 13:10:57 -0800925 if battery_percentage and battery_percentage < 50:
Dan Shi49ca0932014-11-14 11:22:27 -0800926 raise
927 elif self.is_ac_connected():
928 logging.info('The device has power adapter connected and '
929 'charging. No need to try to turn RPM on '
930 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -0800931 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -0800932 logging.info('Battery level is now at %s%%. The device may '
933 'still have enough power to run test, so no '
934 'exception will be raised.', battery_percentage)
935
Simran Basi5e6339a2013-03-21 11:34:32 -0700936
Garry Wangad4d4fd2019-01-30 17:00:38 -0800937 def _remove_rpm_changed_tag(self):
938 info = self.host_info_store.get()
939 del info.attributes[self._RPM_OUTLET_CHANGED]
940 self.host_info_store.commit(info)
941
942
943 def _add_rpm_changed_tag(self):
944 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -0800945 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -0800946 self.host_info_store.commit(info)
947
948
949
beepsc87ff602013-07-31 21:53:00 -0700950 def _is_factory_image(self):
951 """Checks if the image on the DUT is a factory image.
952
953 @return: True if the image on the DUT is a factory image.
954 False otherwise.
955 """
956 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
957 return result.exit_status == 0
958
959
960 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800961 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -0700962
963 @raises: FactoryImageCheckerException for factory images, since
964 we cannot attempt to restart ui on them.
965 error.AutoservRunError for any other type of error that
966 occurs while restarting ui.
967 """
968 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -0700969 raise FactoryImageCheckerException('Cannot restart ui on factory '
970 'images')
beepsc87ff602013-07-31 21:53:00 -0700971
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800972 # TODO(jrbarnette): The command to stop/start the ui job
973 # should live inside cros_ui, too. However that would seem
974 # to imply interface changes to the existing start()/restart()
975 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -0700976 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800977 self.run('stop ui; start ui')
978 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -0700979
980
Daniel Erat4b5f7e02018-07-04 22:05:30 -0700981 def _start_powerd_if_needed(self):
982 """Start powerd if it isn't already running."""
983 self.run('start powerd', ignore_status=True)
984
985
xixuana3bbc422017-05-04 15:57:21 -0700986 def _get_lsb_release_content(self):
987 """Return the content of lsb-release file of host."""
988 return self.run(
989 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
990
991
Dan Shi549fb822015-03-24 18:01:11 -0700992 def get_release_version(self):
993 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
994
995 @returns The version string in lsb-release, under attribute
996 CHROMEOS_RELEASE_VERSION.
997 """
Dan Shi549fb822015-03-24 18:01:11 -0700998 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -0700999 lsb_release_content=self._get_lsb_release_content())
1000
1001
Don Garrettb9f35802018-01-22 18:25:40 -08001002 def get_release_builder_path(self):
1003 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1004
1005 @returns The version string in lsb-release, under attribute
1006 CHROMEOS_RELEASE_BUILDER_PATH.
1007 """
1008 return lsbrelease_utils.get_chromeos_release_builder_path(
1009 lsb_release_content=self._get_lsb_release_content())
1010
1011
xixuana3bbc422017-05-04 15:57:21 -07001012 def get_chromeos_release_milestone(self):
1013 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1014 from lsb-release.
1015
1016 @returns The version string in lsb-release, under attribute
1017 CHROMEOS_RELEASE_BUILD_TYPE.
1018 """
1019 return lsbrelease_utils.get_chromeos_release_milestone(
1020 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001021
1022
1023 def verify_cros_version_label(self):
1024 """ Make sure host's cros-version label match the actual image in dut.
1025
1026 Remove any cros-version: label that doesn't match that installed in
1027 the dut.
1028
1029 @param raise_error: Set to True to raise exception if any mismatch found
1030
1031 @raise error.AutoservError: If any mismatch between cros-version label
1032 and the build installed in dut is found.
1033 """
1034 labels = self._AFE.get_labels(
1035 name__startswith=ds_constants.VERSION_PREFIX,
1036 host__hostname=self.hostname)
1037 mismatch_found = False
1038 if labels:
Richard Barnette147dda42018-02-15 10:54:02 -08001039 # Ask the DUT for its canonical image name. This will be in
1040 # a form like this: kevin-release/R66-10405.0.0
Don Garrettb9f35802018-01-22 18:25:40 -08001041 release_builder_path = self.get_release_builder_path()
Dan Shi549fb822015-03-24 18:01:11 -07001042 host_list = [self.hostname]
1043 for label in labels:
1044 # Remove any cros-version label that does not match
Richard Barnette147dda42018-02-15 10:54:02 -08001045 # the DUT's installed image.
1046 #
Richard Barnettec92805b2018-06-26 14:07:07 -07001047 # TODO(jrbarnette): We make exceptions for certain
1048 # known cases where the version label will not match the
1049 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1050 # * Tests for the `arc-presubmit` pool append
1051 # "-cheetsth" to the label.
1052 # * Moblab use cases based on `cros stage` store images
1053 # under a name with the string "-custom" embedded.
1054 # It's not reliable to match such an image name to the
1055 # label.
1056 label_version = label.name[len(ds_constants.VERSION_PREFIX):]
1057 if '-custom' in label_version:
1058 continue
1059 if label_version.endswith('-cheetsth'):
1060 label_version = label_version[:-len('-cheetsth')]
1061 if label_version != release_builder_path:
Don Garrettb9f35802018-01-22 18:25:40 -08001062 logging.warn(
Aviv Keshet3ccfc912019-04-10 16:50:49 -07001063 'version according to cros-version label "%s" does not '
1064 'match DUT-determined version %s. Removing the label.',
1065 label_version, release_builder_path)
Dan Shi549fb822015-03-24 18:01:11 -07001066 label.remove_hosts(hosts=host_list)
1067 mismatch_found = True
1068 if mismatch_found:
1069 raise error.AutoservError('The host has wrong cros-version label.')
1070
1071
Laurence Goodby778c9a42017-05-24 19:24:07 -07001072 def cleanup_services(self):
1073 """Reinitializes the device for cleanup.
1074
1075 Subclasses may override this to customize the cleanup method.
1076
1077 To indicate failure of the reset, the implementation may raise
1078 any of:
1079 error.AutoservRunError
1080 error.AutotestRunError
1081 FactoryImageCheckerException
1082
1083 @raises error.AutoservRunError
1084 @raises error.AutotestRunError
1085 @raises error.FactoryImageCheckerException
1086 """
1087 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001088 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001089
1090
beepsc87ff602013-07-31 21:53:00 -07001091 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001092 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001093 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001094 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001095 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001096 except (error.AutotestRunError, error.AutoservRunError,
1097 FactoryImageCheckerException):
Ilja H. Friedel04be2bd2014-05-07 21:29:59 -07001098 logging.warning('Unable to restart ui, rebooting device.')
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001099 # Since restarting the UI fails fall back to normal Autotest
1100 # cleanup routines, i.e. reboot the machine.
Fang Deng0ca40e22013-08-27 17:47:44 -07001101 super(CrosHost, self).cleanup()
Simran Basi5e6339a2013-03-21 11:34:32 -07001102 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001103 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001104 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001105 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001106
1107
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001108 def reboot(self, **dargs):
1109 """
1110 This function reboots the site host. The more generic
1111 RemoteHost.reboot() performs sync and sleeps for 5
1112 seconds. This is not necessary for Chrome OS devices as the
1113 sync should be finished in a short time during the reboot
1114 command.
1115 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001116 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001117 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001118 dargs['reboot_cmd'] = ('sleep 1; '
1119 'reboot & sleep %d; '
1120 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001121 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001122 if 'fastsync' not in dargs:
1123 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001124
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001125 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001126 # Record who called us
1127 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001128 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001129 'dut_host_name' : self.hostname,
1130 'success' : True}
1131 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001132 'caller' : "%s:%s" % (orig.co_filename,
1133 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001134 'success' : True,
1135 'error' : ''}
1136
Vincent Palatin80780b22016-07-27 16:02:37 +02001137 t0 = time.time()
1138 try:
1139 super(CrosHost, self).reboot(**dargs)
1140 except Exception as e:
1141 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001142 metric_debug_fields['success'] = False
1143 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001144 raise
1145 finally:
1146 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001147 metrics.Counter(
1148 'chromeos/autotest/autoserv/reboot_count').increment(
1149 fields=metric_fields)
1150 metrics.Counter(
1151 'chromeos/autotest/autoserv/reboot_debug').increment(
1152 fields=metric_debug_fields)
1153 metrics.SecondsDistribution(
1154 'chromeos/autotest/autoserv/reboot_duration').add(
1155 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001156
1157
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001158 def suspend(self, suspend_time=60,
1159 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001160 """
1161 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001162
1163 @param suspend_time: How long to suspend as integer seconds.
1164 @param suspend_cmd: Suspend command to execute.
1165 @param allow_early_resume: If False and if device resumes before
1166 |suspend_time|, throw an error.
1167
1168 @exception AutoservSuspendError Host resumed earlier than
1169 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001170 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001171
1172 if suspend_cmd is None:
1173 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001174 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001175 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
J. Richard Barnette9af19632015-09-25 12:18:03 -07001176 'powerd_dbus_suspend --delay=0'])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001177 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1178 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001179
1180
Simran Basiec564392014-08-25 16:48:09 -07001181 def upstart_status(self, service_name):
1182 """Check the status of an upstart init script.
1183
1184 @param service_name: Service to look up.
1185
1186 @returns True if the service is running, False otherwise.
1187 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001188 return 'start/running' in self.run('status %s' % service_name,
1189 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001190
Tom Hughese9552342018-12-18 14:29:25 -08001191 def upstart_stop(self, service_name):
1192 """Stops an upstart job if it's running.
1193
1194 @param service_name: Service to stop
1195
1196 @returns True if service has been stopped or was already stopped
1197 False otherwise.
1198 """
1199 if not self.upstart_status(service_name):
1200 return True
1201
1202 result = self.run('stop %s' % service_name, ignore_status=True)
1203 if result.exit_status != 0:
1204 return False
1205 return True
1206
1207 def upstart_restart(self, service_name):
1208 """Restarts (or starts) an upstart job.
1209
1210 @param service_name: Service to start/restart
1211
1212 @returns True if service has been started/restarted, False otherwise.
1213 """
1214 cmd = 'start'
1215 if self.upstart_status(service_name):
1216 cmd = 'restart'
1217 cmd = cmd + ' %s' % service_name
1218 result = self.run(cmd)
1219 if result.exit_status != 0:
1220 return False
1221 return True
Simran Basiec564392014-08-25 16:48:09 -07001222
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001223 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001224 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001225
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001226 Tests for the following conditions:
1227 1. All conditions tested by the parent version of this
1228 function.
1229 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001230 3. Sufficient space in /mnt/stateful_partition/encrypted.
1231 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001232
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001233 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001234 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001235 default_kilo_inodes_required = CONFIG.get_config_value(
1236 'SERVER', 'kilo_inodes_required', type=int, default=100)
1237 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1238 kilo_inodes_required = CONFIG.get_config_value(
1239 'SERVER', 'kilo_inodes_required_%s' % board,
1240 type=int, default=default_kilo_inodes_required)
1241 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001242 self.check_diskspace(
1243 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001244 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001245 'SERVER', 'gb_diskspace_required', type=float,
1246 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001247 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1248 # Not all targets build with encrypted stateful support.
1249 if self.path_exists(encrypted_stateful_path):
1250 self.check_diskspace(
1251 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001252 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001253 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1254 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001255
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001256 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001257
beepsc87ff602013-07-31 21:53:00 -07001258 # Factory images don't run update engine,
1259 # goofy controls dbus on these DUTs.
1260 if not self._is_factory_image():
1261 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001262
Dan Shi549fb822015-03-24 18:01:11 -07001263 self.verify_cros_version_label()
1264
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001265
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001266 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1267 def wait_for_system_services(self):
1268 """Waits for system-services to be running.
1269
1270 Sometimes, update_engine will take a while to update firmware, so we
1271 should give this some time to finish. See crbug.com/765686#c38 for
1272 details.
1273 """
1274 if not self.upstart_status('system-services'):
1275 raise error.AutoservError('Chrome failed to reach login. '
1276 'System services not running.')
1277
1278
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001279 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001280 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001281 message = 'Beginning verify for host %s board %s model %s'
1282 info = self.host_info_store.get()
1283 message %= (self.hostname, info.board, info.model)
1284 self.record('INFO', None, None, message)
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001285 self._repair_strategy.verify(self)
1286
1287
Fang Deng96667ca2013-08-01 17:46:18 -07001288 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001289 connect_timeout=None, alive_interval=None,
1290 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001291 """Override default make_ssh_command to use options tuned for Chrome OS.
1292
1293 Tuning changes:
1294 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1295 connection failure. Consistency with remote_access.sh.
1296
Samuel Tan2ce155b2015-06-23 18:24:38 -07001297 - ServerAliveInterval=900; which causes SSH to ping connection every
1298 900 seconds. In conjunction with ServerAliveCountMax ensures
1299 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001300 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001301 the test completed successfully. Later increased from 180 seconds to
1302 900 seconds to account for tests where the DUT is suspended for
1303 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001304
1305 - ServerAliveCountMax=3; consistency with remote_access.sh.
1306
1307 - ConnectAttempts=4; reduce flakiness in connection errors;
1308 consistency with remote_access.sh.
1309
1310 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1311 Host keys change with every new installation, don't waste
1312 memory/space saving them.
1313
1314 - SSH protocol forced to 2; needed for ServerAliveInterval.
1315
1316 @param user User name to use for the ssh connection.
1317 @param port Port on the target host to use for ssh connection.
1318 @param opts Additional options to the ssh command.
1319 @param hosts_file Ignored.
1320 @param connect_timeout Ignored.
1321 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001322 @param alive_count_max Ignored.
1323 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001324 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001325 options = ' '.join([opts, '-o Protocol=2'])
1326 return super(CrosHost, self).make_ssh_command(
1327 user=user, port=port, opts=options, hosts_file='/dev/null',
1328 connect_timeout=30, alive_interval=900, alive_count_max=3,
1329 connection_attempts=4)
1330
1331
Jason Abeleb6f924f2013-11-13 16:01:54 -08001332 def syslog(self, message, tag='autotest'):
1333 """Logs a message to syslog on host.
1334
1335 @param message String message to log into syslog
1336 @param tag String tag prefix for syslog
1337
1338 """
1339 self.run('logger -t "%s" "%s"' % (tag, message))
1340
1341
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001342 def _ping_check_status(self, status):
1343 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001344
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001345 @param status Check the ping status against this value.
1346 @return True iff `status` and the result of ping are the same
1347 (i.e. both True or both False).
1348
1349 """
1350 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1351 return not (status ^ (ping_val == 0))
1352
1353 def _ping_wait_for_status(self, status, timeout):
1354 """Wait for the host to have a given status (UP or DOWN).
1355
1356 Status is checked by polling. Polling will not last longer
1357 than the number of seconds in `timeout`. The polling
1358 interval will be long enough that only approximately
1359 _PING_WAIT_COUNT polling cycles will be executed, subject
1360 to a maximum interval of about one minute.
1361
1362 @param status Waiting will stop immediately if `ping` of the
1363 host returns this status.
1364 @param timeout Poll for at most this many seconds.
1365 @return True iff the host status from `ping` matched the
1366 requested status at the time of return.
1367
1368 """
1369 # _ping_check_status() takes about 1 second, hence the
1370 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001371 # FIXME: if the ping command errors then _ping_check_status()
1372 # returns instantly. If timeout is also smaller than twice
1373 # _PING_WAIT_COUNT then the while loop below forks many
1374 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1375 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1376 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001377 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1378 end_time = time.time() + timeout
1379 while time.time() <= end_time:
1380 if self._ping_check_status(status):
1381 return True
1382 if poll_interval > 0:
1383 time.sleep(poll_interval)
1384
1385 # The last thing we did was sleep(poll_interval), so it may
1386 # have been too long since the last `ping`. Check one more
1387 # time, just to be sure.
1388 return self._ping_check_status(status)
1389
1390 def ping_wait_up(self, timeout):
1391 """Wait for the host to respond to `ping`.
1392
1393 N.B. This method is not a reliable substitute for
1394 `wait_up()`, because a host that responds to ping will not
1395 necessarily respond to ssh. This method should only be used
1396 if the target DUT can be considered functional even if it
1397 can't be reached via ssh.
1398
1399 @param timeout Minimum time to allow before declaring the
1400 host to be non-responsive.
1401 @return True iff the host answered to ping before the timeout.
1402
1403 """
1404 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001405
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001406 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001407 """Wait until the host no longer responds to `ping`.
1408
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001409 This function can be used as a slightly faster version of
1410 `wait_down()`, by avoiding potentially long ssh timeouts.
1411
1412 @param timeout Minimum time to allow for the host to become
1413 non-responsive.
1414 @return True iff the host quit answering ping before the
1415 timeout.
1416
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001417 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001418 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001419
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001420 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001421 """Wait for the client to enter low-power sleep mode.
1422
1423 The test for "is asleep" can't distinguish a system that is
1424 powered off; to confirm that the unit was asleep, it is
1425 necessary to force resume, and then call
1426 `test_wait_for_resume()`.
1427
1428 This function is expected to be called from a test as part
1429 of a sequence like the following:
1430
1431 ~~~~~~~~
1432 boot_id = host.get_boot_id()
1433 # trigger sleep on the host
1434 host.test_wait_for_sleep()
1435 # trigger resume on the host
1436 host.test_wait_for_resume(boot_id)
1437 ~~~~~~~~
1438
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001439 @param sleep_timeout time limit in seconds to allow the host sleep.
1440
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001441 @exception TestFail The host did not go to sleep within
1442 the allowed time.
1443 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001444 if sleep_timeout is None:
1445 sleep_timeout = self.SLEEP_TIMEOUT
1446
1447 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001448 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001449 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001450
1451
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001452 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001453 """Wait for the client to resume from low-power sleep mode.
1454
1455 The `old_boot_id` parameter should be the value from
1456 `get_boot_id()` obtained prior to entering sleep mode. A
1457 `TestFail` exception is raised if the boot id changes.
1458
1459 See @ref test_wait_for_sleep for more on this function's
1460 usage.
1461
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001462 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001463 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001464 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001465
1466 @exception TestFail The host did not respond within the
1467 allowed time.
1468 @exception TestFail The host responded, but the boot id test
1469 indicated a reboot rather than a sleep
1470 cycle.
1471 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001472 if resume_timeout is None:
1473 resume_timeout = self.RESUME_TIMEOUT
1474
1475 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001476 raise error.TestFail(
1477 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001478 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001479 else:
1480 new_boot_id = self.get_boot_id()
1481 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001482 logging.error('client rebooted (old boot %s, new boot %s)',
1483 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001484 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001485 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001486
1487
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001488 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001489 """Wait for the client to shut down.
1490
1491 The test for "has shut down" can't distinguish a system that
1492 is merely asleep; to confirm that the unit was down, it is
1493 necessary to force boot, and then call test_wait_for_boot().
1494
1495 This function is expected to be called from a test as part
1496 of a sequence like the following:
1497
1498 ~~~~~~~~
1499 boot_id = host.get_boot_id()
1500 # trigger shutdown on the host
1501 host.test_wait_for_shutdown()
1502 # trigger boot on the host
1503 host.test_wait_for_boot(boot_id)
1504 ~~~~~~~~
1505
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001506 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001507 @exception TestFail The host did not shut down within the
1508 allowed time.
1509 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001510 if shutdown_timeout is None:
1511 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1512
1513 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001514 raise error.TestFail(
1515 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001516 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001517
1518
1519 def test_wait_for_boot(self, old_boot_id=None):
1520 """Wait for the client to boot from cold power.
1521
1522 The `old_boot_id` parameter should be the value from
1523 `get_boot_id()` obtained prior to shutting down. A
1524 `TestFail` exception is raised if the boot id does not
1525 change. The boot id test is omitted if `old_boot_id` is not
1526 specified.
1527
1528 See @ref test_wait_for_shutdown for more on this function's
1529 usage.
1530
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001531 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001532 shut down.
1533
1534 @exception TestFail The host did not respond within the
1535 allowed time.
1536 @exception TestFail The host responded, but the boot id test
1537 indicated that there was no reboot.
1538 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001539 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001540 raise error.TestFail(
1541 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001542 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001543 elif old_boot_id:
1544 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001545 logging.error('client not rebooted (boot %s)',
1546 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001547 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001548 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001549
1550
1551 @staticmethod
1552 def check_for_rpm_support(hostname):
1553 """For a given hostname, return whether or not it is powered by an RPM.
1554
Simran Basi1df55112013-09-06 11:25:09 -07001555 @param hostname: hostname to check for rpm support.
1556
Simran Basid5e5e272012-09-24 15:23:59 -07001557 @return None if this host does not follows the defined naming format
1558 for RPM powered DUT's in the lab. If it does follow the format,
1559 it returns a regular expression MatchObject instead.
1560 """
Fang Dengbaff9082015-01-06 13:46:15 -08001561 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001562
1563
1564 def has_power(self):
1565 """For this host, return whether or not it is powered by an RPM.
1566
1567 @return True if this host is in the CROS lab and follows the defined
1568 naming format.
1569 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001570 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001571
1572
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001573 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001574 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001575
1576 @param state Specifies which power state to set to DUT
1577 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001578 use. By default "RPM" or "CCD" will be used based
1579 on servo type. Valid values from
1580 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001581
1582 """
1583 ACCEPTABLE_STATES = ['ON', 'OFF']
1584
Garry Wang5e5538a2019-04-08 15:36:18 -07001585 if not power_method:
1586 power_method = self.get_default_power_method()
1587
1588 state = state.upper()
1589 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001590 raise error.TestError('State must be one of: %s.'
1591 % (ACCEPTABLE_STATES,))
1592
1593 if power_method == self.POWER_CONTROL_SERVO:
1594 logging.info('Setting servo port J10 to %s', state)
1595 self.servo.set('prtctl3_pwren', state.lower())
1596 time.sleep(self._USB_POWER_TIMEOUT)
1597 elif power_method == self.POWER_CONTROL_MANUAL:
1598 logging.info('You have %d seconds to set the AC power to %s.',
1599 self._POWER_CYCLE_TIMEOUT, state)
1600 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001601 elif power_method == self.POWER_CONTROL_CCD:
1602 servo_role = 'src' if state == 'ON' else 'snk'
1603 logging.info('servo ccd power pass through detected,'
1604 ' changing servo_role to %s.', servo_role)
1605 self.servo.set_servo_v4_role(servo_role)
1606 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001607 # Make sure we don't leave DUT with no power(servo_role=snk)
1608 # when DUT is not pingable, as we raise a exception here
1609 # that may break a power cycle in the middle.
1610 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001611 raise error.AutoservError(
1612 'DUT failed to regain network connection after %d seconds.'
1613 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001614 else:
1615 if not self.has_power():
1616 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001617 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07001618 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07001619
1620
Garry Wang5e5538a2019-04-08 15:36:18 -07001621 def power_off(self, power_method=None):
1622 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001623
1624 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001625 use. By default "RPM" or "CCD" will be used based
1626 on servo type. Valid values from
1627 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001628
1629 """
1630 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07001631
1632
Garry Wang5e5538a2019-04-08 15:36:18 -07001633 def power_on(self, power_method=None):
1634 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001635
1636 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001637 use. By default "RPM" or "CCD" will be used based
1638 on servo type. Valid values from
1639 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001640
1641 """
1642 self._set_power('ON', power_method)
1643
1644
Garry Wang5e5538a2019-04-08 15:36:18 -07001645 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001646 """Cycle power to this host by turning it OFF, then ON.
1647
1648 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001649 use. By default "RPM" or "CCD" will be used based
1650 on servo type. Valid values from
1651 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001652
1653 """
Garry Wang5e5538a2019-04-08 15:36:18 -07001654 if not power_method:
1655 power_method = self.get_default_power_method()
1656
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001657 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07001658 self.POWER_CONTROL_MANUAL,
1659 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001660 self.power_off(power_method=power_method)
1661 time.sleep(self._POWER_CYCLE_TIMEOUT)
1662 self.power_on(power_method=power_method)
1663 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08001664 self._add_rpm_changed_tag()
1665 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07001666
1667
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001668 def get_platform_from_fwid(self):
1669 """Determine the platform from the crossystem fwid.
1670
1671 @returns a string representing this host's platform.
1672 """
1673 # Look at the firmware for non-unibuild cases or if mosys fails.
1674 crossystem = utils.Crossystem(self)
1675 crossystem.init()
1676 # Extract fwid value and use the leading part as the platform id.
1677 # fwid generally follow the format of {platform}.{firmware version}
1678 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
1679 platform = crossystem.fwid().split('.')[0].lower()
1680 # Newer platforms start with 'Google_' while the older ones do not.
1681 return platform.replace('google_', '')
1682
Simran Basic6f1f7a2012-10-16 10:47:46 -07001683 def get_platform(self):
1684 """Determine the correct platform label for this host.
1685
1686 @returns a string representing this host's platform.
1687 """
C Shapiroed87c6f2018-04-19 09:13:58 -06001688 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1689 run_method=self.run)
1690 unibuild = release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1'
1691 platform = ''
1692 if unibuild:
1693 cmd = 'mosys platform model'
1694 result = self.run(command=cmd, ignore_status=True)
1695 if result.exit_status == 0:
1696 platform = result.stdout.strip()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001697 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07001698
1699
Hung-ying Tyanb1328032014-04-01 14:18:54 +08001700 def get_architecture(self):
1701 """Determine the correct architecture label for this host.
1702
1703 @returns a string representing this host's architecture.
1704 """
1705 crossystem = utils.Crossystem(self)
1706 crossystem.init()
1707 return crossystem.arch()
1708
1709
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001710 def get_chrome_version(self):
1711 """Gets the Chrome version number and milestone as strings.
1712
1713 Invokes "chrome --version" to get the version number and milestone.
1714
1715 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
1716 current Chrome version number as a string (in the form "W.X.Y.Z")
1717 and "milestone" is the first component of the version number
1718 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
1719 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
1720 of "chrome --version" and the milestone will be the empty string.
1721
1722 """
MK Ryu35d661e2014-09-25 17:44:10 -07001723 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001724 return utils.parse_chrome_version(version_string)
1725
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001726
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001727 def get_ec_version(self):
1728 """Get the ec version as strings.
1729
1730 @returns a string representing this host's ec version.
1731 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001732 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001733 result = self.run(command, ignore_status=True)
1734 if result.exit_status != 0:
1735 return ''
1736 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001737
1738
1739 def get_firmware_version(self):
1740 """Get the firmware version as strings.
1741
1742 @returns a string representing this host's firmware version.
1743 """
1744 crossystem = utils.Crossystem(self)
1745 crossystem.init()
1746 return crossystem.fwid()
1747
1748
1749 def get_hardware_revision(self):
1750 """Get the hardware revision as strings.
1751
1752 @returns a string representing this host's hardware revision.
1753 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001754 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001755 result = self.run(command, ignore_status=True)
1756 if result.exit_status != 0:
1757 return ''
1758 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001759
1760
1761 def get_kernel_version(self):
1762 """Get the kernel version as strings.
1763
1764 @returns a string representing this host's kernel version.
1765 """
1766 return self.run('uname -r').stdout.strip()
1767
1768
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08001769 def get_cpu_name(self):
1770 """Get the cpu name as strings.
1771
1772 @returns a string representing this host's cpu name.
1773 """
1774
1775 # Try get cpu name from device tree first
1776 if self.path_exists('/proc/device-tree/compatible'):
1777 command = ' | '.join(
1778 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
1779 'tail -1'])
1780 return self.run(command).stdout.strip().replace(',', ' ')
1781
1782 # Get cpu name from uname -p
1783 command = 'uname -p'
1784 ret = self.run(command).stdout.strip()
1785
1786 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
1787 # Try get cpu name from /proc/cpuinfo instead
1788 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
1789 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
1790 self = self.run(command).stdout.strip()
1791
1792 # Remove bloat from CPU name, for example
1793 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
1794 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
1795 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
1796 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
1797 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
1798 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
1799
1800
1801 def get_screen_resolution(self):
1802 """Get the screen(s) resolution as strings.
1803 In case of more than 1 monitor, return resolution for each monitor
1804 separate with plus sign.
1805
1806 @returns a string representing this host's screen(s) resolution.
1807 """
1808 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
1809 ret = self.run(command, ignore_status=True)
1810 # We might have Chromebox without a screen
1811 if ret.exit_status != 0:
1812 return ''
1813 return ret.stdout.strip().replace('\n', '+')
1814
1815
1816 def get_mem_total_gb(self):
1817 """Get total memory available in the system in GiB (2^20).
1818
1819 @returns an integer representing total memory
1820 """
1821 mem_total_kb = self.read_from_meminfo('MemTotal')
1822 kb_in_gb = float(2 ** 20)
1823 return int(round(mem_total_kb / kb_in_gb))
1824
1825
1826 def get_disk_size_gb(self):
1827 """Get size of disk in GB (10^9)
1828
1829 @returns an integer representing size of disk, 0 in Error Case
1830 """
1831 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
1832 result = self.run(command, ignore_status=True)
1833 if result.exit_status != 0:
1834 return 0
1835 _, _, block, _ = re.split(r' +', result.stdout.strip())
1836 byte_per_block = 1024.0
1837 disk_kb_in_gb = 1e9
1838 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
1839
1840
1841 def get_battery_size(self):
1842 """Get size of battery in Watt-hour via sysfs
1843
1844 This method assumes that battery support voltage_min_design and
1845 charge_full_design sysfs.
1846
1847 @returns a float representing Battery size, 0 if error.
1848 """
1849 # sysfs report data in micro scale
1850 battery_scale = 1e6
1851
1852 command = 'cat /sys/class/power_supply/*/voltage_min_design'
1853 result = self.run(command, ignore_status=True)
1854 if result.exit_status != 0:
1855 return 0
1856 voltage = float(result.stdout.strip()) / battery_scale
1857
1858 command = 'cat /sys/class/power_supply/*/charge_full_design'
1859 result = self.run(command, ignore_status=True)
1860 if result.exit_status != 0:
1861 return 0
1862 amphereHour = float(result.stdout.strip()) / battery_scale
1863
1864 return voltage * amphereHour
1865
1866
1867 def get_low_battery_shutdown_percent(self):
1868 """Get the percent-based low-battery shutdown threshold.
1869
1870 @returns a float representing low-battery shutdown percent, 0 if error.
1871 """
1872 ret = 0.0
1873 try:
1874 command = 'check_powerd_config --low_battery_shutdown_percent'
1875 ret = float(self.run(command).stdout)
1876 except error.CmdError:
1877 logging.debug("Can't run %s", command)
1878 except ValueError:
1879 logging.debug("Didn't get number from %s", command)
1880
1881 return ret
1882
1883
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07001884 def has_hammer(self):
1885 """Check whether DUT has hammer device or not.
1886
1887 @returns boolean whether device has hammer or not
1888 """
1889 command = 'grep Hammer /sys/bus/usb/devices/*/product'
1890 return self.run(command, ignore_status=True).exit_status == 0
1891
1892
Niranjan Kumar34618872017-05-31 12:57:09 -07001893 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07001894 """Returns True if the specified switch was provided to Chrome.
1895
1896 @param switch The chrome switch to search for.
1897 """
Niranjan Kumar34618872017-05-31 12:57:09 -07001898
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07001899 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
1900 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07001901
1902
1903 def oobe_triggers_update(self):
1904 """Returns True if this host has an OOBE flow during which
1905 it will perform an update check and perhaps an update.
1906 One example of such a flow is Hands-Off Zero-Touch Enrollment.
1907 As more such flows are developed, code handling them needs
1908 to be added here.
1909
1910 @return Boolean indicating whether this host's OOBE triggers an update.
1911 """
1912 return self.is_chrome_switch_present(
1913 '--enterprise-enable-zero-touch-enrollment=hands-off')
1914
1915
Kevin Chenga2619dc2016-03-28 11:42:08 -07001916 # TODO(kevcheng): change this to just return the board without the
1917 # 'board:' prefix and fix up all the callers. Also look into removing the
1918 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07001919 def get_board(self):
1920 """Determine the correct board label for this host.
1921
1922 @returns a string representing this host's board.
1923 """
1924 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1925 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001926 return (ds_constants.BOARD_PREFIX +
1927 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07001928
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07001929 def get_channel(self):
1930 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07001931
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07001932 @returns: a string represeting this host's build channel.
1933 (stable, dev, beta). None on fail.
1934 """
1935 return lsbrelease_utils.get_chromeos_channel(
1936 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07001937
Kevin Chenga328da62016-03-31 10:49:04 -07001938 def get_power_supply(self):
1939 """
1940 Determine what type of power supply the host has
1941
1942 @returns a string representing this host's power supply.
1943 'power:battery' when the device has a battery intended for
1944 extended use
1945 'power:AC_primary' when the device has a battery not intended
1946 for extended use (for moving the machine, etc)
1947 'power:AC_only' when the device has no battery at all.
1948 """
1949 psu = self.run(command='mosys psu type', ignore_status=True)
1950 if psu.exit_status:
1951 # The psu command for mosys is not included for all platforms. The
1952 # assumption is that the device will have a battery if the command
1953 # is not found.
1954 return 'power:battery'
1955
1956 psu_str = psu.stdout.strip()
1957 if psu_str == 'unknown':
1958 return None
1959
1960 return 'power:%s' % psu_str
1961
1962
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08001963 def has_battery(self):
1964 """Determine if DUT has a battery.
1965
1966 Returns:
1967 Boolean, False if known not to have battery, True otherwise.
1968 """
1969 rv = True
1970 power_supply = self.get_power_supply()
1971 if power_supply == 'power:battery':
1972 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
1973 board_type = self.get_board_type()
1974 if board_type in _NO_BATTERY_BOARD_TYPE:
1975 logging.warn('Do NOT believe type %s has battery. '
1976 'See debug for mosys details', board_type)
1977 psu = self.system_output('mosys -vvvv psu type',
1978 ignore_status=True)
1979 logging.debug(psu)
1980 rv = False
1981 elif power_supply == 'power:AC_only':
1982 rv = False
1983
1984 return rv
1985
1986
Kevin Chenga328da62016-03-31 10:49:04 -07001987 def get_servo(self):
1988 """Determine if the host has a servo attached.
1989
1990 If the host has a working servo attached, it should have a servo label.
1991
1992 @return: string 'servo' if the host has servo attached. Otherwise,
1993 returns None.
1994 """
1995 return 'servo' if self._servo_host else None
1996
1997
Kevin Chenga328da62016-03-31 10:49:04 -07001998 def has_internal_display(self):
1999 """Determine if the device under test is equipped with an internal
2000 display.
2001
2002 @return: 'internal_display' if one is present; None otherwise.
2003 """
2004 from autotest_lib.client.cros.graphics import graphics_utils
2005 from autotest_lib.client.common_lib import utils as common_utils
2006
2007 def __system_output(cmd):
2008 return self.run(cmd).stdout
2009
2010 def __read_file(remote_path):
2011 return self.run('cat %s' % remote_path).stdout
2012
2013 # Hijack the necessary client functions so that we can take advantage
2014 # of the client lib here.
2015 # FIXME: find a less hacky way than this
2016 original_system_output = utils.system_output
2017 original_read_file = common_utils.read_file
2018 utils.system_output = __system_output
2019 common_utils.read_file = __read_file
2020 try:
2021 return ('internal_display' if graphics_utils.has_internal_display()
2022 else None)
2023 finally:
2024 utils.system_output = original_system_output
2025 common_utils.read_file = original_read_file
2026
2027
Dan Shi85276d42014-04-08 22:11:45 -07002028 def is_boot_from_usb(self):
2029 """Check if DUT is boot from USB.
2030
2031 @return: True if DUT is boot from usb.
2032 """
2033 device = self.run('rootdev -s -d').stdout.strip()
2034 removable = int(self.run('cat /sys/block/%s/removable' %
2035 os.path.basename(device)).stdout.strip())
2036 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002037
2038
2039 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002040 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002041
2042 @param key: meminfo requested
2043
2044 @return the memory value as a string
2045
2046 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002047 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2048 logging.debug('%s', meminfo)
2049 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002050
2051
Rohit Makasana98e696f2016-06-03 18:48:10 -07002052 def get_cpu_arch(self):
2053 """Returns CPU arch of the device.
2054
2055 @return CPU architecture of the DUT.
2056 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002057 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002058 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2059 ignore_status=True).stdout:
2060 return 'x86_64'
2061 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2062 ignore_status=True).stdout:
2063 return 'arm'
2064 return 'i386'
2065
2066
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002067 def get_board_type(self):
2068 """
2069 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002070 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2071
2072 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002073 """
Danny Chan471a8d12015-08-18 14:57:41 -07002074 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2075 ignore_status=True).stdout
2076 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002077 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002078 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002079
2080
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002081 def get_arc_version(self):
2082 """Return ARC version installed on the DUT.
2083
2084 @returns ARC version as string if the CrOS build has ARC, else None.
2085 """
2086 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2087 ignore_status=True).stdout
2088 if arc_version:
2089 return arc_version.split('=')[-1].strip()
2090 return None
2091
2092
Gilad Arnolda76bef02015-09-29 13:55:15 -07002093 def get_os_type(self):
2094 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002095
2096
Kevin Chenga2619dc2016-03-28 11:42:08 -07002097 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002098 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002099 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002100
2101
2102 def get_default_power_method(self):
2103 """
2104 Get the default power method for power_on/off/cycle() methods.
2105 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2106 """
2107 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002108 self._default_power_method = self.POWER_CONTROL_RPM
Garry Wang5e5538a2019-04-08 15:36:18 -07002109 info = self.host_info_store.get()
2110 if info.attributes.get('servo_type') == self.CCD_SERVO:
Garry Wang1a004aa2019-05-16 22:56:51 -07002111 if self.servo:
2112 self._default_power_method = self.POWER_CONTROL_CCD
2113 else:
2114 logging.warn('CCD servo detected but failed to apply CCD'
2115 ' servo RPM method due to servo instance'
2116 ' was not initialized, fall back to normal'
2117 ' RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002118 return self._default_power_method