blob: 0f165f58024546b24991f68a71a4ed3df8e4128c [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
Otabek Kasimov6825b762020-06-23 23:42:44 -070018from autotest_lib.client.common_lib import utils as common_utils
Greg Edelstona7b05d12020-04-01 16:00:51 -060019from autotest_lib.client.common_lib.cros import cros_config
Richard Barnette03a0c132012-11-05 12:40:35 -080020from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070021from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000022from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080023from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080024from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070025from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070026from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050027from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070028from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070029from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070030from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070031from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080032from autotest_lib.server.hosts import chameleon_host
Otabek Kasimov832d9162020-07-27 19:24:57 -070033from autotest_lib.server.hosts import cros_constants
Richard Barnetted31580e2018-05-14 19:58:00 +000034from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080035from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070036from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070037from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070038from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080039from autotest_lib.site_utils.rpm_control_system import rpm_client
Otabek Kasimov808cd832020-05-28 18:27:46 -070040from autotest_lib.site_utils.admin_audit import constants as audit_const
Simran Basid5e5e272012-09-24 15:23:59 -070041
Simran Basi382506b2016-09-13 14:58:15 -070042# In case cros_host is being ran via SSP on an older Moblab version with an
43# older chromite version.
44try:
45 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080046except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080047 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080048
Simran Basid5e5e272012-09-24 15:23:59 -070049
Dan Shib8540a52015-07-16 14:18:23 -070050CONFIG = global_config.global_config
51
beepsc87ff602013-07-31 21:53:00 -070052class FactoryImageCheckerException(error.AutoservError):
53 """Exception raised when an image is a factory image."""
54 pass
55
56
Fang Deng0ca40e22013-08-27 17:47:44 -070057class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070058 """Chromium OS specific subclass of Host."""
59
Simran Basi5ace6f22016-01-06 17:30:44 -080060 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
61
Scott Zawalski62bacae2013-03-05 10:40:32 -050062 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070063
Richard Barnette03a0c132012-11-05 12:40:35 -080064 # Timeout values (in seconds) associated with various Chrome OS
65 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070066 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080067 # In general, a good rule of thumb is that the timeout can be up
68 # to twice the typical measured value on the slowest platform.
69 # The times here have not necessarily been empirically tested to
70 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070071 #
72 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080073 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
74 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080075 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070076 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080077 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070078 # screen delay, time to start the network on the DUT, and the
79 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070080 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080081 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080082 # network.
beepsf079cfb2013-09-18 17:49:51 -070083 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080084 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
85 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070086
87 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080088 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080089 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070090 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070091 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070092 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080093 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070094
Dan Shica503482015-03-30 17:23:25 -070095 # Minimum OS version that supports server side packaging. Older builds may
96 # not have server side package built or with Autotest code change to support
97 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070098 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -070099 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700100
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800101 # REBOOT_TIMEOUT: How long to wait for a reboot.
102 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700103 # We have a long timeout to ensure we don't flakily fail due to other
104 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700105 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
106 # return from reboot' bug is solved.
107 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700108
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800109 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
110 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700111 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
112 # since changing servo role will reset USB state
113 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800114 _USB_POWER_TIMEOUT = 5
115 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700116 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800117
Fang Dengdeba14f2014-11-14 11:54:09 -0800118 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
119 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700120
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800121 # Constants used in ping_wait_up() and ping_wait_down().
122 #
123 # _PING_WAIT_COUNT is the approximate number of polling
124 # cycles to use when waiting for a host state change.
125 #
126 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
127 # for arguments to the internal _ping_wait_for_status()
128 # method.
129 _PING_WAIT_COUNT = 40
130 _PING_STATUS_DOWN = False
131 _PING_STATUS_UP = True
132
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800133 # Allowed values for the power_method argument.
134
Garry Wang5e5538a2019-04-08 15:36:18 -0700135 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
136 # DUTs except those with servo_v4 CCD.
137 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
138 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800139 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
140 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
141 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700142 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800143 POWER_CONTROL_SERVO = 'servoj10'
144 POWER_CONTROL_MANUAL = 'manual'
145
146 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700147 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800148 POWER_CONTROL_SERVO,
149 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800150
Simran Basi5e6339a2013-03-21 11:34:32 -0700151 _RPM_OUTLET_CHANGED = 'outlet_changed'
152
Dan Shi9cb0eec2014-06-03 09:04:50 -0700153 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700154 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700155 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700156
Brent Peterson1cb623a2020-01-09 13:14:28 -0800157 # Regular expression for extracting EC version string
158 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
159
160 # Regular expression for extracting BIOS version string
161 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
162
Brent Petersonc70a1832020-01-24 15:54:35 -0800163 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700164 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800165
J. Richard Barnette964fba02012-10-24 17:34:29 -0700166 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800167 def check_host(host, timeout=10):
168 """
169 Check if the given host is a chrome-os host.
170
171 @param host: An ssh host representing a device.
172 @param timeout: The timeout for the run command.
173
174 @return: True if the host device is chromeos.
175
beeps46dadc92013-11-07 14:07:10 -0800176 """
177 try:
Allen Liad719c12017-06-27 23:48:04 +0000178 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700179 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700180 '! grep -q moblab /etc/lsb-release && '
181 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700182 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700183 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000184 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700185 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
186 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800187 return not (
188 lsbrelease_utils.is_jetstream(
189 lsb_release_content=lsb_release_content) or
190 lsbrelease_utils.is_gce_board(
191 lsb_release_content=lsb_release_content))
192
beeps46dadc92013-11-07 14:07:10 -0800193 except (error.AutoservRunError, error.AutoservSSHTimeout):
194 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700195
196 return False
beeps46dadc92013-11-07 14:07:10 -0800197
198
199 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800200 def get_chameleon_arguments(args_dict):
201 """Extract chameleon options from `args_dict` and return the result.
202
203 Recommended usage:
204 ~~~~~~~~
205 args_dict = utils.args_to_dict(args)
206 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
207 host = hosts.create_host(machine, chameleon_args=chameleon_args)
208 ~~~~~~~~
209
210 @param args_dict Dictionary from which to extract the chameleon
211 arguments.
212 """
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800213 return {key: args_dict[key]
Allen Li083866b2016-08-18 10:07:10 -0700214 for key in ('chameleon_host', 'chameleon_port')
215 if key in args_dict}
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800216
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800217 @staticmethod
218 def get_btpeer_arguments(args_dict):
219 """Extract btpeer options from `args_dict` and return the result.
220
221 This is used to parse details of Bluetooth peer.
222 Recommended usage:
223 ~~~~~~~~
224 args_dict = utils.args_to_dict(args)
225 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700226 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800227 ~~~~~~~~
228
229 @param args_dict: Dictionary from which to extract the btpeer
230 arguments.
231 """
232 if 'btpeer_host_list' in args_dict:
233 result = []
234 for btpeer in args_dict['btpeer_host_list'].split(','):
235 result.append({key: value for key,value in
236 zip(('btpeer_host','btpeer_port'),
237 btpeer.split(':'))})
238 return result
239 else:
240 return {key: args_dict[key]
241 for key in ('btpeer_host', 'btpeer_port')
242 if key in args_dict}
243
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800244
245 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700246 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800247 """Extract chameleon options from `args_dict` and return the result.
248
249 Recommended usage:
250 ~~~~~~~~
251 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700252 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
253 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800254 ~~~~~~~~
255
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700256 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800257 arguments.
258 """
Allen Li083866b2016-08-18 10:07:10 -0700259 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700260 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700261 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800262
263
264 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800265 def get_servo_arguments(args_dict):
266 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800267
268 Recommended usage:
269 ~~~~~~~~
270 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700271 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800272 host = hosts.create_host(machine, servo_args=servo_args)
273 ~~~~~~~~
274
275 @param args_dict Dictionary from which to extract the servo
276 arguments.
277 """
Garry Wang11b5e872020-03-11 15:14:08 -0700278 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
279 servo_constants.SERVO_PORT_ATTR,
280 servo_constants.SERVO_BOARD_ATTR,
281 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200282 servo_args = {key: args_dict[key]
283 for key in servo_attrs
284 if key in args_dict}
285 return (
286 None
Garry Wang11b5e872020-03-11 15:14:08 -0700287 if servo_constants.SERVO_HOST_ATTR in servo_args
288 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200289 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700290
J. Richard Barnette964fba02012-10-24 17:34:29 -0700291
J. Richard Barnette91137f02016-03-10 16:52:26 -0800292 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700293 pdtester_args=None, try_lab_servo=False,
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700294 try_servo_repair=False, btpeer_args=[],
J. Richard Barnette91137f02016-03-10 16:52:26 -0800295 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700296 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800297 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700298
Fang Denge545abb2014-12-30 18:43:47 -0800299 This method will attempt to create the test-assistant object
300 (chameleon/servo) when it is needed by the test. Check
301 the docstring of chameleon_host.create_chameleon_host and
302 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700303
Fang Denge545abb2014-12-30 18:43:47 -0800304 @param hostname: Hostname of the dut.
305 @param chameleon_args: A dictionary that contains args for creating
306 a ChameleonHost. See chameleon_host for details.
307 @param servo_args: A dictionary that contains args for creating
308 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700309 @param try_lab_servo: When true, indicates that an attempt should
310 be made to create a ServoHost for a DUT in
311 the test lab, even if not required by
312 `servo_args`. See servo_host for details.
313 @param try_servo_repair: If a servo host is created, check it
314 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800315 See servo_host for details.
316 @param ssh_verbosity_flag: String, to pass to the ssh command to control
317 verbosity.
318 @param ssh_options: String, other ssh options to pass to the ssh
319 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700320 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700321 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700322 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800323 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700324 # hold special dut_state for repair process
325 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700326 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700327 # self.env is a dictionary of environment variable settings
328 # to be exported for commands run on the host.
329 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
330 # errors that might happen.
331 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700332 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700333 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700334 _servo_host, servo_state = servo_host.create_servo_host(
335 dut=self,
336 servo_args=servo_args,
337 try_lab_servo=try_lab_servo,
338 try_servo_repair=try_servo_repair,
339 dut_host_info=self.host_info_store.get())
340 self.set_servo_host(_servo_host, servo_state)
Garry Wang5e5538a2019-04-08 15:36:18 -0700341 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700342
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800343 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800344 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700345 dut=self.hostname,
346 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800347 if self._chameleon_host:
348 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800349 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800350 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700351
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700352 # Initialize Bluetooth peers.
353 try:
354 self.initialize_btpeer(btpeer_args)
355 except Exception as e:
356 logging.error('Exception %s in initialize_btpeer', str(e))
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800357
howardchung83e55272019-08-08 14:08:05 +0800358 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700359 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700360 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800361
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700362 if self._pdtester_host:
363 self.pdtester_servo = self._pdtester_host.get_servo()
364 logging.info('pdtester_servo: %r', self.pdtester_servo)
365 # Create the pdtester object used to access the ec uart
366 self.pdtester = pdtester.PDTester(self.pdtester_servo,
367 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800368 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700369 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800370
Fang Deng5d518f42013-08-02 14:04:32 -0700371
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800372 def initialize_btpeer(self, btpeer_args):
373 """ Initialize the Bluetooth peers
374
375 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
376 is chameleon host on Raspberry Pi.
377 @param btpeer_args: A dictionary that contains args for creating
378 a ChameleonHost. See chameleon_host for details.
379
380 """
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700381 #TODO (b:142486063) Remove the try..except
382 try:
383 self._btpeer_host_list = []
384 self.btpeer_list = []
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800385 self.btpeer = None
386
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700387 if type(btpeer_args) is list:
388 btpeer_args_list = btpeer_args
389 else:
390 btpeer_args_list = [btpeer_args]
391
392 self._btpeer_host_list = chameleon_host.create_btpeer_host(
393 dut=self.hostname, btpeer_args_list=btpeer_args_list)
394 logging.debug('Bluetooth peer hosts are %s',
395 self._btpeer_host_list)
396 self.btpeer_list = [_host.create_chameleon_board() for _host in
397 self._btpeer_host_list if _host is not None]
398
399 if len(self.btpeer_list) > 0:
400 self.btpeer = self.btpeer_list[0]
401
402 logging.debug('After initialize_btpeer btpeer_list %s '
403 'btpeer_host_list is %s and btpeer is %s',
404 self.btpeer_list, self._btpeer_host_list,
405 self.btpeer)
406 except Exception as e:
407 logging.error('Exception %s in initialize_btpeer', str(e))
408
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800409
410
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000411 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700412 """Get latest stable cros image name from AFE.
413
414 Use the board name from the info store. Should that fail, try to
415 retrieve the board name from the host's installed image itself.
416
417 @returns: current stable cros image name for this host.
418 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700419 info = self.host_info_store.get()
420 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700421 logging.warn('No board label value found. Trying to infer '
422 'from the host itself.')
423 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700424 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700425 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
426 logging.error('Also failed to get the board name from the DUT '
427 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700428 raise error.AutoservError('Cannot determine board of the DUT'
429 ' while getting repair image name.')
430 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500431
432
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700433 def host_version_prefix(self, image):
434 """Return version label prefix.
435
436 In case the CrOS provisioning version is something other than the
437 standard CrOS version e.g. CrOS TH version, this function will
438 find the prefix from provision.py.
439
440 @param image: The image name to find its version prefix.
441 @returns: A prefix string for the image type.
442 """
443 return provision.get_version_label_prefix(image)
444
Andrew Luo3332ab22020-04-28 16:42:03 -0700445 def stage_build_to_usb(self, build):
446 """Stage the current ChromeOS image on the USB stick connected to the
447 servo.
448
449 @param build: The build to download and send to USB.
450 """
451 if not self.servo:
452 raise error.TestError('Host %s does not have servo.' %
453 self.hostname)
454
455 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700456
457 try:
458 self.servo.image_to_servo_usb(update_url)
459 finally:
460 # servo.image_to_servo_usb turned the DUT off, so turn it back on
461 logging.debug('Turn DUT power back on.')
462 self.servo.get_power_state_controller().power_on()
463
Andrew Luo3332ab22020-04-28 16:42:03 -0700464 logging.debug('ChromeOS image %s is staged on the USB stick.',
465 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700466
beepsdae65fd2013-07-26 16:24:41 -0700467 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700468 """
469 Make sure job_repo_url of this host is valid.
470
joychen03eaad92013-06-26 09:55:21 -0700471 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700472 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
473 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
474 download and extract it. If the devserver embedded in the url is
475 unresponsive, update the job_repo_url of the host after staging it on
476 another devserver.
477
478 @param job_repo_url: A url pointing to the devserver where the autotest
479 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700480 @param tag: The tag from the server job, in the format
481 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700482
483 @raises DevServerException: If we could not resolve a devserver.
484 @raises AutoservError: If we're unable to save the new job_repo_url as
485 a result of choosing a new devserver because the old one failed to
486 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700487 @raises urllib2.URLError: If the devserver embedded in job_repo_url
488 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700489 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800490 info = self.host_info_store.get()
491 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700492 if not job_repo_url:
493 logging.warning('No job repo url set on host %s', self.hostname)
494 return
495
496 logging.info('Verifying job repo url %s', job_repo_url)
497 devserver_url, image_name = tools.get_devserver_build_from_package_url(
498 job_repo_url)
499
beeps0c865032013-07-30 11:37:06 -0700500 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700501
502 logging.info('Staging autotest artifacts for %s on devserver %s',
503 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700504
505 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700506 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700507 stage_time = time.time() - start_time
508
509 # Record how much of the verification time comes from a devserver
510 # restage. If we're doing things right we should not see multiple
511 # devservers for a given board/build/branch path.
512 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800513 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700514 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800515 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700516 pass
517 else:
beeps0c865032013-07-30 11:37:06 -0700518 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700519 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700520 stats_key = {
521 'board': board,
522 'build_type': build_type,
523 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700524 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700525 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800526
527 monarch_fields = {
528 'board': board,
529 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800530 'branch': branch,
531 'dev_server': devserver,
532 }
533 metrics.Counter(
534 'chromeos/autotest/provision/verify_url'
535 ).increment(fields=monarch_fields)
536 metrics.SecondsDistribution(
537 'chromeos/autotest/provision/verify_url_duration'
538 ).add(stage_time, fields=monarch_fields)
539
540
Dan Shicf4d2032015-03-12 15:04:21 -0700541 def stage_server_side_package(self, image=None):
542 """Stage autotest server-side package on devserver.
543
544 @param image: Full path of an OS image to install or a build name.
545
546 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700547
548 @raise: error.AutoservError if fail to locate the build to test with, or
549 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700550 """
Dan Shid37736b2016-07-06 15:10:29 -0700551 # If enable_drone_in_restricted_subnet is False, do not set hostname
552 # in devserver.resolve call, so a devserver in non-restricted subnet
553 # is picked to stage autotest server package for drone to download.
554 hostname = self.hostname
555 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
556 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700557 if image:
558 image_name = tools.get_build_from_image(image)
559 if not image_name:
560 raise error.AutoservError(
561 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700562 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700563 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700564 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800565 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700566 if job_repo_url:
567 devserver_url, image_name = (
568 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700569 # If enable_drone_in_restricted_subnet is True, use the
570 # existing devserver. Otherwise, resolve a new one in
571 # non-restricted subnet.
572 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
573 ds = dev_server.ImageServer(devserver_url)
574 else:
575 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800576 elif info.build is not None:
577 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700578 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700579 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800580 raise error.AutoservError(
581 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700582 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700583
584 # Get the OS version of the build, for any build older than
585 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
586 match = re.match('.*/R\d+-(\d+)\.', image_name)
587 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700588 raise error.AutoservError(
589 'Build %s is older than %s. Server side packaging is '
590 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700591
Dan Shicf4d2032015-03-12 15:04:21 -0700592 ds.stage_artifacts(image_name, ['autotest_server_package'])
593 return '%s/static/%s/%s' % (ds.url(), image_name,
594 'autotest_server_package.tar.bz2')
595
596
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700597 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700598 """Stage a build on a devserver and return the update_url.
599
600 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700601 @param artifact: a string like 'test_image'. Requests
602 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700603 @returns a tuple of (image_name, URL) like
604 (lumpy-release/R27-3837.0.0,
605 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700606 """
607 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000608 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700609 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800610 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700611 devserver.stage_artifacts(image_name, [artifact])
612 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700613 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700614 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700615 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700616 else:
617 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700618
619
beepse539be02013-07-31 21:57:39 -0700620 def stage_factory_image_for_servo(self, image_name):
621 """Stage a build on a devserver and return the update_url.
622
623 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700624
beepse539be02013-07-31 21:57:39 -0700625 @return: An update URL, eg:
626 http://<devserver>/static/canary-channel/\
627 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700628
629 @raises: ValueError if the factory artifact name is missing from
630 the config.
631
beepse539be02013-07-31 21:57:39 -0700632 """
633 if not image_name:
634 logging.error('Need an image_name to stage a factory image.')
635 return
636
Dan Shib8540a52015-07-16 14:18:23 -0700637 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700638 'CROS', 'factory_artifact', type=str, default='')
639 if not factory_artifact:
640 raise ValueError('Cannot retrieve the factory artifact name from '
641 'autotest config, and hence cannot stage factory '
642 'artifacts.')
643
beepse539be02013-07-31 21:57:39 -0700644 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800645 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700646 devserver.stage_artifacts(
647 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700648 [factory_artifact],
649 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700650
651 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
652
653
Laurence Goodby778c9a42017-05-24 19:24:07 -0700654 def prepare_for_update(self):
655 """Prepares the DUT for an update.
656
657 Subclasses may override this to perform any special actions
658 required before updating.
659 """
Laurence Goodby468de252017-06-08 17:22:53 -0700660 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700661
662
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800663 def _clear_fw_version_labels(self, rw_only):
664 """Clear firmware version labels from the machine.
665
666 @param rw_only: True to only clear fwrw_version; otherewise, clear
667 both fwro_version and fwrw_version.
668 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700669 info = self.host_info_store.get()
670 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800671 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700672 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
673 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700674
675
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800676 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700677 """Add firmware version label to the machine.
678
679 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800680 @param rw_only: True to only add fwrw_version; otherwise, add both
681 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700682
683 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700684 info = self.host_info_store.get()
685 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800686 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700687 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
688 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700689
690
Namyoon Woo33f38852020-04-13 17:26:58 -0700691 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800692 """Search for the latest package release version from the image archive,
693 and return it.
694
Namyoon Woo33f38852020-04-13 17:26:58 -0700695 @param platform: platform name, a.k.a. board or model
696 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800697
Namyoon Woo33f38852020-04-13 17:26:58 -0700698 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
699 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800700 or None if LATEST release file does not exist.
701 """
702
Namyoon Woo33f38852020-04-13 17:26:58 -0700703 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800704
Namyoon Woo33f38852020-04-13 17:26:58 -0700705 # Search the image path in reference board archive as well.
706 # For example, bob has its binary image under its reference board (gru)
707 # image archive.
708 if ref_board:
709 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800710
Namyoon Woo33f38852020-04-13 17:26:58 -0700711 for board in platforms:
712 # Read 'LATEST-1.0.0' file
713 branch_dir = provision.FW_BRANCH_GLOB % board
714 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
715 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800716
Namyoon Woo33f38852020-04-13 17:26:58 -0700717 try:
718 # The result could be one or more.
719 result = utils.system_output('gsutil ls -d ' + latest_file)
720
721 candidates = re.findall('gs://.*', result)
722
723 # Found the directory candidates. No need to check the other
724 # board name cadidates. Let's break the loop.
725 break
726 except error.CmdError:
727 # It doesn't exist. Let's move on to the next item.
728 pass
729 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800730 logging.error('No LATEST release info is available.')
731 return None
732
Namyoon Woo406c7d42020-01-24 15:57:11 -0800733 for cand_dir in candidates:
734 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800735
Namyoon Woo406c7d42020-01-24 15:57:11 -0800736 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700737 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800738 try:
739 # Check if release_path does exist.
740 release = utils.system_output('gsutil ls -d ' + release_path)
741 # Now 'release' has a full directory path: e.g.
742 # gs://chromeos-image-archive/firmware-octopus-11297.B-
743 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
744
745 # Remove "gs://chromeos-image-archive".
746 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
747
748 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
749 return release.strip('/')
750 except error.CmdError:
751 # The directory might not exist. Let's try next candidate.
752 pass
753 else:
754 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800755
Brent Peterson1cb623a2020-01-09 13:14:28 -0800756 @staticmethod
757 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800758 """Get version string from binary image using regular expression.
759
760 @param image: Binary image to search
761 @param version_regex: Regular expression to search for
762
763 @return Version string
764
765 @raises TestFail if no version string is found in image
766 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800767 with open(image, 'rb') as f:
768 image_data = f.read()
769 match = re.findall(version_regex, image_data)
770 if match:
771 return match[0]
772 else:
773 raise error.TestFail('Failed to read version from %s.' % image)
774
775
Garry Wangad2a1712020-03-26 15:06:43 -0700776 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800777 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700778 try_scp=False, install_ec=True, install_bios=True,
779 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700780 """Install firmware to the DUT.
781
782 Use stateful update if the DUT is already running the same build.
783 Stateful update does not update kernel and tends to run much faster
784 than a full reimage. If the DUT is running a different build, or it
785 failed to do a stateful update, full update, including kernel update,
786 will be applied to the DUT.
787
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800788 Once a host enters firmware_install its fw[ro|rw]_version label will
789 be removed. After the firmware is updated successfully, a new
790 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700791
792 @param build: The build version to which we want to provision the
793 firmware of the machine,
794 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800795 @param rw_only: True to only install firmware to its RW portions. Keep
796 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700797 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800798 @param local_tarball: Path to local firmware image for installing
799 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800800 @param verify_version: True to verify EC and BIOS versions after
801 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800802 @param try_scp: False to always program using servo, true to try copying
803 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700804 @param install_ec: True to install EC FW, and False to skip it.
805 @param install_bios: True to install BIOS, and False to skip it.
806 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700807
808 TODO(dshi): After bug 381718 is fixed, update here with corresponding
809 exceptions that could be raised.
810
811 """
812 if not self.servo:
813 raise error.TestError('Host %s does not have servo.' %
814 self.hostname)
815
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700816 # Get the DUT board name from AFE.
817 info = self.host_info_store.get()
818 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700819 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800820
821 if board is None or board == '':
822 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700823
Namyoon Woo382e5892020-05-20 16:48:40 -0700824 # if board_as argument is passed, then use it instead of the original
825 # board name.
826 if board_as:
827 board = board_as
828
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700829 if model is None or model == '':
830 model = self.get_platform_from_fwid()
831
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800832 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700833 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800834 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700835 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800836
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700837 try:
838 ds = dev_server.ImageServer.resolve(build, self.hostname)
839 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800840
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700841 if not dest:
842 tmpd = autotemp.tempdir(unique_id='fwimage')
843 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800844
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700845 # Download firmware image
846 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
847 local_tarball = os.path.join(dest, os.path.basename(fwurl))
848 ds.download_file(fwurl, local_tarball)
849 except Exception as e:
850 raise error.TestError('Failed to download firmware package: %s'
851 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700852
Namyoon Woo382e5892020-05-20 16:48:40 -0700853 ec_image = None
854 if install_ec:
855 # Extract EC image from tarball
856 logging.info('Extracting EC image.')
857 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800858
Namyoon Woo382e5892020-05-20 16:48:40 -0700859 bios_image = None
860 if install_bios:
861 # Extract BIOS image from tarball
862 logging.info('Extracting BIOS image.')
863 bios_image = self.servo.extract_bios_image(board, model,
864 local_tarball)
865
866 if not bios_image and not ec_image:
867 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800868
Brent Petersonc70a1832020-01-24 15:54:35 -0800869 # Clear firmware version labels
870 self._clear_fw_version_labels(rw_only)
871
872 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800873 try:
Brent Petersonc70a1832020-01-24 15:54:35 -0800874 # Check if DUT is available and copying to DUT is enabled
875 if self.is_up() and try_scp:
876 # DUT is available, make temp firmware directory to store images
877 logging.info('Making temp folder.')
878 dest_folder = '/tmp/firmware'
879 self.run('mkdir -p ' + dest_folder)
880
Namyoon Woo68b68082020-06-02 13:13:14 -0700881 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800882
Namyoon Woo382e5892020-05-20 16:48:40 -0700883 if bios_image:
884 # Send BIOS firmware image to DUT
885 logging.info('Sending BIOS firmware.')
886 dest_bios_path = os.path.join(dest_folder,
887 os.path.basename(bios_image))
888 self.send_file(bios_image, dest_bios_path)
889
890 # Initialize firmware update command for BIOS image
891 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800892
893 # Send EC firmware image to DUT when EC image was found
894 if ec_image:
895 logging.info('Sending EC firmware.')
896 dest_ec_path = os.path.join(dest_folder,
897 os.path.basename(ec_image))
898 self.send_file(ec_image, dest_ec_path)
899
900 # Add EC image to firmware update command
901 fw_cmd += ' -e %s' % dest_ec_path
902
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700903 # Make sure command is allowed to finish even if ssh fails.
904 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
905
Brent Peterson669edf42020-02-07 15:07:54 -0800906 # Update firmware on DUT
907 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700908 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700909 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700910 except error.AutoservRunError as e:
911 if e.result_obj.exit_status != 255:
912 raise
913 elif ec_image:
914 logging.warn("DUT network dropped during update"
915 " (often caused by EC resetting USB)")
916 else:
917 logging.error("DUT network dropped during update"
918 " (unexpected, since no EC image)")
919 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800920 else:
921 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800922 if ec_image:
923 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700924 if bios_image:
925 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800926 if utils.host_is_in_lab_zone(self.hostname):
927 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800928
929 # Reboot and wait for DUT after installing firmware
930 logging.info('Rebooting DUT.')
931 self.servo.get_power_state_controller().reset()
932 time.sleep(self.servo.BOOT_DELAY)
933 self.test_wait_for_boot()
934
935 # When enabled verify EC and BIOS firmware version after programming
936 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800937 # Check programmed EC firmware when EC image was found
938 if ec_image:
939 logging.info('Checking EC firmware version.')
940 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800941 ec_version_prefix = dest_ec_version.split('_', 1)[0]
942 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800943 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800944 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800945 if dest_ec_version != image_ec_version:
946 raise error.TestFail(
947 'Failed to update EC RO, version %s (expected %s)' %
948 (dest_ec_version, image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800949
Namyoon Woo382e5892020-05-20 16:48:40 -0700950 if bios_image:
951 # Check programmed BIOS firmware against expected version
952 logging.info('Checking BIOS firmware version.')
953 dest_bios_version = self.get_firmware_version()
954 bios_version_prefix = dest_bios_version.split('.', 1)[0]
955 bios_regex = self._BIOS_REGEX % bios_version_prefix
956 image_bios_version = self.get_version_from_image(bios_image,
957 bios_regex)
958 if dest_bios_version != image_bios_version:
959 raise error.TestFail(
960 'Failed to update BIOS RO, version %s '
961 '(expected %s)' % (dest_bios_version,
962 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700963 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700964 if tmpd:
965 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700966
967
beepsf079cfb2013-09-18 17:49:51 -0700968 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
969 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500970 """
971 Re-install the OS on the DUT by:
972 1) installing a test image on a USB storage device attached to the Servo
973 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800974 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700975 3) installing the image with chromeos-install.
976
Scott Zawalski62bacae2013-03-05 10:40:32 -0500977 @param image_url: If specified use as the url to install on the DUT.
978 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700979 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
980 Factory images need a longer usb_boot_timeout than regular
981 cros images.
982 @param install_timeout: The timeout to use when installing the chromeos
983 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800984
Scott Zawalski62bacae2013-03-05 10:40:32 -0500985 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700986
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800987 """
Garry Wang7b0e1b72020-03-25 19:08:59 -0700988 if image_url:
989 logging.info('Downloading image to USB, then booting from it.'
990 ' Usb boot timeout = %s', usb_boot_timeout)
991 else:
992 logging.info('Booting from USB directly. Usb boot timeout = %s',
993 usb_boot_timeout)
994
995 metrics_field = {'download': bool(image_url)}
996 metrics.Counter(
997 'chromeos/autotest/provision/servo_install/download_image'
998 ).increment(fields=metrics_field)
999
Allen Li48a13fe2016-11-22 14:10:40 -08001000 with metrics.SecondsTimer(
1001 'chromeos/autotest/provision/servo_install/boot_duration'):
1002 self.servo.install_recovery_image(image_url)
1003 if not self.wait_up(timeout=usb_boot_timeout):
1004 raise hosts.AutoservRepairError(
1005 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001006 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001007
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001008 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1009 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001010 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001011 try:
1012 self.run('chromeos-tpm-recovery')
1013 except error.AutoservRunError:
1014 logging.warn('chromeos-tpm-recovery is too old.')
1015
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001016
Allen Li48a13fe2016-11-22 14:10:40 -08001017 with metrics.SecondsTimer(
1018 'chromeos/autotest/provision/servo_install/install_duration'):
1019 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001020 try:
1021 self.run('chromeos-install --yes',timeout=install_timeout)
1022 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001023 except Exception as e:
1024 storage_errors = [
1025 'No space left on device',
1026 'I/O error when trying to write primary GPT',
1027 'Input/output error while writing out',
1028 'cannot read GPT header',
1029 'can not determine destination device'
1030 ]
1031 has_error = [msg for msg in storage_errors if(msg in str(e))]
1032 if has_error:
1033 info = self.host_info_store.get()
1034 info.set_version_label(
1035 audit_const.DUT_STORAGE_STATE_PREFIX,
1036 audit_const.HW_STATE_NEED_REPLACEMENT)
1037 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001038 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001039 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001040 logging.debug(
1041 'Fail install image from USB; Storage error; %s', e)
1042 raise error.AutoservError(
1043 'Failed to install image from USB due to a suspect '
1044 'disk failure, DUT storage state changed to '
1045 'need_replacement, please check debug log '
1046 'for details.')
1047 else:
1048 logging.debug('Fail install image from USB; %s', e)
1049 raise error.AutoservError(
1050 'Failed to install image from USB due to unexpected '
1051 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001052 finally:
1053 # We need reset the DUT no matter re-install success or not,
1054 # as we don't want leave the DUT in boot from usb state.
1055 logging.info('Power cycling DUT through servo.')
1056 self.servo.get_power_state_controller().power_off()
1057 self.servo.switch_usbkey('off')
1058 # N.B. The Servo API requires that we use power_on() here
1059 # for two reasons:
1060 # 1) After turning on a DUT in recovery mode, you must turn
1061 # it off and then on with power_on() once more to
1062 # disable recovery mode (this is a Parrot specific
1063 # requirement).
1064 # 2) After power_off(), the only way to turn on is with
1065 # power_on() (this is a Storm specific requirement).
1066 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001067
1068 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001069 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001070 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1071 'test image after %d seconds' %
1072 self.BOOT_TIMEOUT,
1073 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001074
1075
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001076 def set_servo_host(self, host, servo_state = None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001077 """Set our servo host member, and associated servo.
1078
1079 @param host Our new `ServoHost`.
1080 """
1081 self._servo_host = host
1082 if self._servo_host is not None:
1083 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001084 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001085 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001086 else:
1087 self.servo = None
Otabek Kasimov41301a22020-05-10 15:28:21 -07001088 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001089 self.set_servo_state(servo_state)
1090
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001091
Richard Barnette9a26ad62016-06-10 12:03:08 -07001092 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001093 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001094 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001095
Richard Barnette9a26ad62016-06-10 12:03:08 -07001096 If the servo object is missing, attempt to repair the servo
1097 host. Repair failures are passed back to the caller.
1098
1099 @raise AutoservError: If there is no servo host for this CrOS
1100 host.
1101 """
1102 if self.servo:
1103 return
1104 if not self._servo_host:
1105 raise error.AutoservError('No servo host for %s.' %
1106 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001107 try:
1108 self._servo_host.repair()
1109 except:
1110 raise
1111 finally:
1112 self.set_servo_host(self._servo_host)
1113
1114
Otabek Kasimov41301a22020-05-10 15:28:21 -07001115 def set_servo_type(self):
1116 """Set servo info labels to dut host_info"""
1117 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001118 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001119 return
1120 servo_type = self.servo.get_servo_type()
1121 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001122 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001123 ' by `dut-control servo_type`! Please file a bug'
1124 ' and inform infra team as we are not expected '
1125 ' to reach this point.')
1126 return
1127 host_info = self.host_info_store.get()
1128 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1129 old_type = host_info.get_label_value(prefix)
1130 if old_type == servo_type:
1131 # do not need update
1132 return
1133 host_info.set_version_label(prefix, servo_type)
1134 self.host_info_store.commit(host_info)
1135 logging.info('ServoHost: servo_type updated to %s '
1136 '(previous: %s)', servo_type, old_type)
1137
1138
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001139 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001140 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001141 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001142 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001143 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001144 old_state = host_info.get_label_value(servo_state_prefix)
1145 if old_state == servo_state:
1146 # do not need update
1147 return
1148 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001149 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001150 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1151 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001152
1153
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001154 def get_servo_state(self):
1155 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001156 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001157 return host_info.get_label_value(servo_state_prefix)
1158
Otabek Kasimov41301a22020-05-10 15:28:21 -07001159
Garry Wang000c6c02020-05-11 21:27:23 -07001160 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1161 if smart_usbhub_detected is None:
1162 # skip the label update here as this indicate we wasn't able
1163 # to confirm usbhub type.
1164 return
1165 host_info = self.host_info_store.get()
1166 if (smart_usbhub_detected ==
1167 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1168 # skip label update if current label match the truth.
1169 return
1170 if smart_usbhub_detected:
1171 logging.info('Adding %s label to host %s',
1172 servo_constants.SMART_USBHUB_LABEL,
1173 self.hostname)
1174 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1175 else:
1176 logging.info('Removing %s label from host %s',
1177 servo_constants.SMART_USBHUB_LABEL,
1178 self.hostname)
1179 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1180 self.host_info_store.commit(host_info)
1181
1182
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001183 def repair(self):
1184 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001185
1186 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001187 not call back to the parent class, but instead relies on
1188 `self._repair_strategy` to coordinate the verification and
1189 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001190 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001191 message = 'Beginning repair for host %s board %s model %s'
1192 info = self.host_info_store.get()
1193 message %= (self.hostname, info.board, info.model)
1194 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001195 try:
1196 self._repair_strategy.repair(self)
1197 except hosts.AutoservVerifyDependencyError as e:
1198 # We don't want flag a DUT as failed if only non-critical
1199 # verifier(s) failed during the repair.
1200 if e.is_critical():
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001201 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001202 raise
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001203
Richard Barnette82c35912012-11-20 10:09:10 -08001204
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001205 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001206 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001207 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001208
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001209 if self._chameleon_host:
1210 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001211
1212 if self._servo_host:
1213 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001214
1215
Dan Shi49ca0932014-11-14 11:22:27 -08001216 def get_power_supply_info(self):
1217 """Get the output of power_supply_info.
1218
1219 power_supply_info outputs the info of each power supply, e.g.,
1220 Device: Line Power
1221 online: no
1222 type: Mains
1223 voltage (V): 0
1224 current (A): 0
1225 Device: Battery
1226 state: Discharging
1227 percentage: 95.9276
1228 technology: Li-ion
1229
1230 Above output shows two devices, Line Power and Battery, with details of
1231 each device listed. This function parses the output into a dictionary,
1232 with key being the device name, and value being a dictionary of details
1233 of the device info.
1234
1235 @return: The dictionary of power_supply_info, e.g.,
1236 {'Line Power': {'online': 'yes', 'type': 'main'},
1237 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001238 @raise error.AutoservRunError if power_supply_info tool is not found in
1239 the DUT. Caller should handle this error to avoid false failure
1240 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001241 """
1242 result = self.run('power_supply_info').stdout.strip()
1243 info = {}
1244 device_name = None
1245 device_info = {}
1246 for line in result.split('\n'):
1247 pair = [v.strip() for v in line.split(':')]
1248 if len(pair) != 2:
1249 continue
1250 if pair[0] == 'Device':
1251 if device_name:
1252 info[device_name] = device_info
1253 device_name = pair[1]
1254 device_info = {}
1255 else:
1256 device_info[pair[0]] = pair[1]
1257 if device_name and not device_name in info:
1258 info[device_name] = device_info
1259 return info
1260
1261
1262 def get_battery_percentage(self):
1263 """Get the battery percentage.
1264
1265 @return: The percentage of battery level, value range from 0-100. Return
1266 None if the battery info cannot be retrieved.
1267 """
1268 try:
1269 info = self.get_power_supply_info()
1270 logging.info(info)
1271 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001272 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001273 return None
1274
1275
Philip Chenaf69ead2020-03-27 13:06:42 -07001276 def get_battery_state(self):
1277 """Get the battery charging state.
1278
1279 @return: A string representing the battery charging state. It can be
1280 'Charging', 'Fully charged', or 'Discharging'.
1281 """
1282 try:
1283 info = self.get_power_supply_info()
1284 logging.info(info)
1285 return info['Battery']['state']
1286 except (KeyError, ValueError, error.AutoservRunError):
1287 return None
1288
1289
Daniel Campello8ca25c22019-12-13 16:48:26 -07001290 def get_battery_display_percentage(self):
1291 """Get the battery display percentage.
1292
1293 @return: The display percentage of battery level, value range from
1294 0-100. Return None if the battery info cannot be retrieved.
1295 """
1296 try:
1297 info = self.get_power_supply_info()
1298 logging.info(info)
1299 return float(info['Battery']['display percentage'])
1300 except (KeyError, ValueError, error.AutoservRunError):
1301 return None
1302
1303
Dan Shi49ca0932014-11-14 11:22:27 -08001304 def is_ac_connected(self):
1305 """Check if the dut has power adapter connected and charging.
1306
1307 @return: True if power adapter is connected and charging.
1308 """
1309 try:
1310 info = self.get_power_supply_info()
1311 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001312 except (KeyError, error.AutoservRunError):
1313 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001314
1315
Simran Basi5e6339a2013-03-21 11:34:32 -07001316 def _cleanup_poweron(self):
1317 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001318 info = self.host_info_store.get()
1319 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001320 return
1321 logging.debug('This host has recently interacted with the RPM'
1322 ' Infrastructure. Ensuring power is on.')
1323 try:
1324 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001325 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001326 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001327 logging.error('Failed to turn Power On for this host after '
1328 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001329
1330 battery_percentage = self.get_battery_percentage()
Gregory Nisbet02273172020-07-13 09:26:17 -07001331 if (battery_percentage and
1332 battery_percentage < cros_repair.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001333 raise
1334 elif self.is_ac_connected():
1335 logging.info('The device has power adapter connected and '
1336 'charging. No need to try to turn RPM on '
1337 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001338 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001339 logging.info('Battery level is now at %s%%. The device may '
1340 'still have enough power to run test, so no '
1341 'exception will be raised.', battery_percentage)
1342
Simran Basi5e6339a2013-03-21 11:34:32 -07001343
Garry Wangad4d4fd2019-01-30 17:00:38 -08001344 def _remove_rpm_changed_tag(self):
1345 info = self.host_info_store.get()
1346 del info.attributes[self._RPM_OUTLET_CHANGED]
1347 self.host_info_store.commit(info)
1348
1349
1350 def _add_rpm_changed_tag(self):
1351 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001352 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001353 self.host_info_store.commit(info)
1354
1355
1356
beepsc87ff602013-07-31 21:53:00 -07001357 def _is_factory_image(self):
1358 """Checks if the image on the DUT is a factory image.
1359
1360 @return: True if the image on the DUT is a factory image.
1361 False otherwise.
1362 """
1363 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1364 return result.exit_status == 0
1365
1366
1367 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001368 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001369
1370 @raises: FactoryImageCheckerException for factory images, since
1371 we cannot attempt to restart ui on them.
1372 error.AutoservRunError for any other type of error that
1373 occurs while restarting ui.
1374 """
1375 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001376 raise FactoryImageCheckerException('Cannot restart ui on factory '
1377 'images')
beepsc87ff602013-07-31 21:53:00 -07001378
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001379 # TODO(jrbarnette): The command to stop/start the ui job
1380 # should live inside cros_ui, too. However that would seem
1381 # to imply interface changes to the existing start()/restart()
1382 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001383 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001384 self.run('stop ui; start ui')
1385 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001386
1387
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001388 def _start_powerd_if_needed(self):
1389 """Start powerd if it isn't already running."""
1390 self.run('start powerd', ignore_status=True)
1391
1392
xixuana3bbc422017-05-04 15:57:21 -07001393 def _get_lsb_release_content(self):
1394 """Return the content of lsb-release file of host."""
1395 return self.run(
1396 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1397
1398
Dan Shi549fb822015-03-24 18:01:11 -07001399 def get_release_version(self):
1400 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1401
1402 @returns The version string in lsb-release, under attribute
1403 CHROMEOS_RELEASE_VERSION.
1404 """
Dan Shi549fb822015-03-24 18:01:11 -07001405 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001406 lsb_release_content=self._get_lsb_release_content())
1407
1408
Don Garrettb9f35802018-01-22 18:25:40 -08001409 def get_release_builder_path(self):
1410 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1411
1412 @returns The version string in lsb-release, under attribute
1413 CHROMEOS_RELEASE_BUILDER_PATH.
1414 """
1415 return lsbrelease_utils.get_chromeos_release_builder_path(
1416 lsb_release_content=self._get_lsb_release_content())
1417
1418
xixuana3bbc422017-05-04 15:57:21 -07001419 def get_chromeos_release_milestone(self):
1420 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1421 from lsb-release.
1422
1423 @returns The version string in lsb-release, under attribute
1424 CHROMEOS_RELEASE_BUILD_TYPE.
1425 """
1426 return lsbrelease_utils.get_chromeos_release_milestone(
1427 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001428
1429
1430 def verify_cros_version_label(self):
1431 """ Make sure host's cros-version label match the actual image in dut.
1432
1433 Remove any cros-version: label that doesn't match that installed in
1434 the dut.
1435
1436 @param raise_error: Set to True to raise exception if any mismatch found
1437
1438 @raise error.AutoservError: If any mismatch between cros-version label
1439 and the build installed in dut is found.
1440 """
Prathmesh Prabhuce2da3a2019-10-04 11:54:51 -07001441 # crbug.com/1007333: This check is being removed.
1442 return True
Dan Shi549fb822015-03-24 18:01:11 -07001443
1444
Laurence Goodby778c9a42017-05-24 19:24:07 -07001445 def cleanup_services(self):
1446 """Reinitializes the device for cleanup.
1447
1448 Subclasses may override this to customize the cleanup method.
1449
1450 To indicate failure of the reset, the implementation may raise
1451 any of:
1452 error.AutoservRunError
1453 error.AutotestRunError
1454 FactoryImageCheckerException
1455
1456 @raises error.AutoservRunError
1457 @raises error.AutotestRunError
1458 @raises error.FactoryImageCheckerException
1459 """
1460 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001461 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001462
1463
beepsc87ff602013-07-31 21:53:00 -07001464 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001465 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001466 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001467 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001468 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001469 except (error.AutotestRunError, error.AutoservRunError,
1470 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001471 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001472
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001473 # cleanup routines, i.e. reboot the machine.
1474 super(CrosHost, self).cleanup()
1475
Simran Basi5e6339a2013-03-21 11:34:32 -07001476 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001477 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001478 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001479 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001480
1481
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001482 def reboot(self, **dargs):
1483 """
1484 This function reboots the site host. The more generic
1485 RemoteHost.reboot() performs sync and sleeps for 5
1486 seconds. This is not necessary for Chrome OS devices as the
1487 sync should be finished in a short time during the reboot
1488 command.
1489 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001490 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001491 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001492 dargs['reboot_cmd'] = ('sleep 1; '
1493 'reboot & sleep %d; '
1494 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001495 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001496 if 'fastsync' not in dargs:
1497 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001498
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001499 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001500 # Record who called us
1501 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001502 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001503 'dut_host_name' : self.hostname,
1504 'success' : True}
1505 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001506 'caller' : "%s:%s" % (orig.co_filename,
1507 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001508 'success' : True,
1509 'error' : ''}
1510
Vincent Palatin80780b22016-07-27 16:02:37 +02001511 t0 = time.time()
1512 try:
1513 super(CrosHost, self).reboot(**dargs)
1514 except Exception as e:
1515 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001516 metric_debug_fields['success'] = False
1517 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001518 raise
1519 finally:
1520 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001521 metrics.Counter(
1522 'chromeos/autotest/autoserv/reboot_count').increment(
1523 fields=metric_fields)
1524 metrics.Counter(
1525 'chromeos/autotest/autoserv/reboot_debug').increment(
1526 fields=metric_debug_fields)
1527 metrics.SecondsDistribution(
1528 'chromeos/autotest/autoserv/reboot_duration').add(
1529 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001530
1531
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001532 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001533 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001534 """
1535 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001536
1537 @param suspend_time: How long to suspend as integer seconds.
1538 @param suspend_cmd: Suspend command to execute.
1539 @param allow_early_resume: If False and if device resumes before
1540 |suspend_time|, throw an error.
1541
1542 @exception AutoservSuspendError Host resumed earlier than
1543 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001544 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001545
1546 if suspend_cmd is None:
1547 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001548 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001549 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001550 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001551 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1552 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001553
1554
Simran Basiec564392014-08-25 16:48:09 -07001555 def upstart_status(self, service_name):
1556 """Check the status of an upstart init script.
1557
1558 @param service_name: Service to look up.
1559
1560 @returns True if the service is running, False otherwise.
1561 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001562 return 'start/running' in self.run('status %s' % service_name,
1563 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001564
Tom Hughese9552342018-12-18 14:29:25 -08001565 def upstart_stop(self, service_name):
1566 """Stops an upstart job if it's running.
1567
1568 @param service_name: Service to stop
1569
1570 @returns True if service has been stopped or was already stopped
1571 False otherwise.
1572 """
1573 if not self.upstart_status(service_name):
1574 return True
1575
1576 result = self.run('stop %s' % service_name, ignore_status=True)
1577 if result.exit_status != 0:
1578 return False
1579 return True
1580
1581 def upstart_restart(self, service_name):
1582 """Restarts (or starts) an upstart job.
1583
1584 @param service_name: Service to start/restart
1585
1586 @returns True if service has been started/restarted, False otherwise.
1587 """
1588 cmd = 'start'
1589 if self.upstart_status(service_name):
1590 cmd = 'restart'
1591 cmd = cmd + ' %s' % service_name
1592 result = self.run(cmd)
1593 if result.exit_status != 0:
1594 return False
1595 return True
Simran Basiec564392014-08-25 16:48:09 -07001596
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001597 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001598 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001599
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001600 Tests for the following conditions:
1601 1. All conditions tested by the parent version of this
1602 function.
1603 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001604 3. Sufficient space in /mnt/stateful_partition/encrypted.
1605 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001606
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001607 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001608 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001609 default_kilo_inodes_required = CONFIG.get_config_value(
1610 'SERVER', 'kilo_inodes_required', type=int, default=100)
1611 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1612 kilo_inodes_required = CONFIG.get_config_value(
1613 'SERVER', 'kilo_inodes_required_%s' % board,
1614 type=int, default=default_kilo_inodes_required)
1615 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001616 self.check_diskspace(
1617 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001618 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001619 'SERVER', 'gb_diskspace_required', type=float,
1620 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001621 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1622 # Not all targets build with encrypted stateful support.
1623 if self.path_exists(encrypted_stateful_path):
1624 self.check_diskspace(
1625 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001626 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001627 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1628 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001629
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001630 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001631
beepsc87ff602013-07-31 21:53:00 -07001632 # Factory images don't run update engine,
1633 # goofy controls dbus on these DUTs.
1634 if not self._is_factory_image():
1635 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001636
Dan Shi549fb822015-03-24 18:01:11 -07001637 self.verify_cros_version_label()
1638
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001639
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001640 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1641 def wait_for_system_services(self):
1642 """Waits for system-services to be running.
1643
1644 Sometimes, update_engine will take a while to update firmware, so we
1645 should give this some time to finish. See crbug.com/765686#c38 for
1646 details.
1647 """
1648 if not self.upstart_status('system-services'):
1649 raise error.AutoservError('Chrome failed to reach login. '
1650 'System services not running.')
1651
1652
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001653 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001654 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001655 message = 'Beginning verify for host %s board %s model %s'
1656 info = self.host_info_store.get()
1657 message %= (self.hostname, info.board, info.model)
1658 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001659 try:
1660 self._repair_strategy.verify(self)
1661 except hosts.AutoservVerifyDependencyError as e:
1662 # We don't want flag a DUT as failed if only non-critical
1663 # verifier(s) failed during the repair.
1664 if e.is_critical():
1665 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001666
1667
Fang Deng96667ca2013-08-01 17:46:18 -07001668 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001669 connect_timeout=None, alive_interval=None,
1670 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001671 """Override default make_ssh_command to use options tuned for Chrome OS.
1672
1673 Tuning changes:
1674 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1675 connection failure. Consistency with remote_access.sh.
1676
Samuel Tan2ce155b2015-06-23 18:24:38 -07001677 - ServerAliveInterval=900; which causes SSH to ping connection every
1678 900 seconds. In conjunction with ServerAliveCountMax ensures
1679 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001680 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001681 the test completed successfully. Later increased from 180 seconds to
1682 900 seconds to account for tests where the DUT is suspended for
1683 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001684
1685 - ServerAliveCountMax=3; consistency with remote_access.sh.
1686
1687 - ConnectAttempts=4; reduce flakiness in connection errors;
1688 consistency with remote_access.sh.
1689
1690 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1691 Host keys change with every new installation, don't waste
1692 memory/space saving them.
1693
1694 - SSH protocol forced to 2; needed for ServerAliveInterval.
1695
1696 @param user User name to use for the ssh connection.
1697 @param port Port on the target host to use for ssh connection.
1698 @param opts Additional options to the ssh command.
1699 @param hosts_file Ignored.
1700 @param connect_timeout Ignored.
1701 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001702 @param alive_count_max Ignored.
1703 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001704 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001705 options = ' '.join([opts, '-o Protocol=2'])
1706 return super(CrosHost, self).make_ssh_command(
1707 user=user, port=port, opts=options, hosts_file='/dev/null',
1708 connect_timeout=30, alive_interval=900, alive_count_max=3,
1709 connection_attempts=4)
1710
1711
Jason Abeleb6f924f2013-11-13 16:01:54 -08001712 def syslog(self, message, tag='autotest'):
1713 """Logs a message to syslog on host.
1714
1715 @param message String message to log into syslog
1716 @param tag String tag prefix for syslog
1717
1718 """
1719 self.run('logger -t "%s" "%s"' % (tag, message))
1720
1721
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001722 def _ping_check_status(self, status):
1723 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001724
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001725 @param status Check the ping status against this value.
1726 @return True iff `status` and the result of ping are the same
1727 (i.e. both True or both False).
1728
1729 """
1730 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1731 return not (status ^ (ping_val == 0))
1732
1733 def _ping_wait_for_status(self, status, timeout):
1734 """Wait for the host to have a given status (UP or DOWN).
1735
1736 Status is checked by polling. Polling will not last longer
1737 than the number of seconds in `timeout`. The polling
1738 interval will be long enough that only approximately
1739 _PING_WAIT_COUNT polling cycles will be executed, subject
1740 to a maximum interval of about one minute.
1741
1742 @param status Waiting will stop immediately if `ping` of the
1743 host returns this status.
1744 @param timeout Poll for at most this many seconds.
1745 @return True iff the host status from `ping` matched the
1746 requested status at the time of return.
1747
1748 """
1749 # _ping_check_status() takes about 1 second, hence the
1750 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001751 # FIXME: if the ping command errors then _ping_check_status()
1752 # returns instantly. If timeout is also smaller than twice
1753 # _PING_WAIT_COUNT then the while loop below forks many
1754 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1755 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1756 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001757 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1758 end_time = time.time() + timeout
1759 while time.time() <= end_time:
1760 if self._ping_check_status(status):
1761 return True
1762 if poll_interval > 0:
1763 time.sleep(poll_interval)
1764
1765 # The last thing we did was sleep(poll_interval), so it may
1766 # have been too long since the last `ping`. Check one more
1767 # time, just to be sure.
1768 return self._ping_check_status(status)
1769
1770 def ping_wait_up(self, timeout):
1771 """Wait for the host to respond to `ping`.
1772
1773 N.B. This method is not a reliable substitute for
1774 `wait_up()`, because a host that responds to ping will not
1775 necessarily respond to ssh. This method should only be used
1776 if the target DUT can be considered functional even if it
1777 can't be reached via ssh.
1778
1779 @param timeout Minimum time to allow before declaring the
1780 host to be non-responsive.
1781 @return True iff the host answered to ping before the timeout.
1782
1783 """
1784 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001785
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001786 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001787 """Wait until the host no longer responds to `ping`.
1788
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001789 This function can be used as a slightly faster version of
1790 `wait_down()`, by avoiding potentially long ssh timeouts.
1791
1792 @param timeout Minimum time to allow for the host to become
1793 non-responsive.
1794 @return True iff the host quit answering ping before the
1795 timeout.
1796
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001797 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001798 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001799
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001800 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001801 """Wait for the client to enter low-power sleep mode.
1802
1803 The test for "is asleep" can't distinguish a system that is
1804 powered off; to confirm that the unit was asleep, it is
1805 necessary to force resume, and then call
1806 `test_wait_for_resume()`.
1807
1808 This function is expected to be called from a test as part
1809 of a sequence like the following:
1810
1811 ~~~~~~~~
1812 boot_id = host.get_boot_id()
1813 # trigger sleep on the host
1814 host.test_wait_for_sleep()
1815 # trigger resume on the host
1816 host.test_wait_for_resume(boot_id)
1817 ~~~~~~~~
1818
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001819 @param sleep_timeout time limit in seconds to allow the host sleep.
1820
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001821 @exception TestFail The host did not go to sleep within
1822 the allowed time.
1823 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001824 if sleep_timeout is None:
1825 sleep_timeout = self.SLEEP_TIMEOUT
1826
1827 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001828 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001829 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001830
1831
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001832 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001833 """Wait for the client to resume from low-power sleep mode.
1834
1835 The `old_boot_id` parameter should be the value from
1836 `get_boot_id()` obtained prior to entering sleep mode. A
1837 `TestFail` exception is raised if the boot id changes.
1838
1839 See @ref test_wait_for_sleep for more on this function's
1840 usage.
1841
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001842 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001843 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001844 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001845
1846 @exception TestFail The host did not respond within the
1847 allowed time.
1848 @exception TestFail The host responded, but the boot id test
1849 indicated a reboot rather than a sleep
1850 cycle.
1851 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001852 if resume_timeout is None:
1853 resume_timeout = self.RESUME_TIMEOUT
1854
1855 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001856 raise error.TestFail(
1857 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001858 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001859 else:
1860 new_boot_id = self.get_boot_id()
1861 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001862 logging.error('client rebooted (old boot %s, new boot %s)',
1863 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001864 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001865 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001866
1867
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001868 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001869 """Wait for the client to shut down.
1870
1871 The test for "has shut down" can't distinguish a system that
1872 is merely asleep; to confirm that the unit was down, it is
1873 necessary to force boot, and then call test_wait_for_boot().
1874
1875 This function is expected to be called from a test as part
1876 of a sequence like the following:
1877
1878 ~~~~~~~~
1879 boot_id = host.get_boot_id()
1880 # trigger shutdown on the host
1881 host.test_wait_for_shutdown()
1882 # trigger boot on the host
1883 host.test_wait_for_boot(boot_id)
1884 ~~~~~~~~
1885
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001886 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001887 @exception TestFail The host did not shut down within the
1888 allowed time.
1889 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001890 if shutdown_timeout is None:
1891 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1892
1893 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001894 raise error.TestFail(
1895 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001896 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001897
1898
1899 def test_wait_for_boot(self, old_boot_id=None):
1900 """Wait for the client to boot from cold power.
1901
1902 The `old_boot_id` parameter should be the value from
1903 `get_boot_id()` obtained prior to shutting down. A
1904 `TestFail` exception is raised if the boot id does not
1905 change. The boot id test is omitted if `old_boot_id` is not
1906 specified.
1907
1908 See @ref test_wait_for_shutdown for more on this function's
1909 usage.
1910
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001911 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001912 shut down.
1913
1914 @exception TestFail The host did not respond within the
1915 allowed time.
1916 @exception TestFail The host responded, but the boot id test
1917 indicated that there was no reboot.
1918 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001919 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001920 raise error.TestFail(
1921 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001922 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001923 elif old_boot_id:
1924 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001925 logging.error('client not rebooted (boot %s)',
1926 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001927 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001928 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001929
1930
1931 @staticmethod
1932 def check_for_rpm_support(hostname):
1933 """For a given hostname, return whether or not it is powered by an RPM.
1934
Simran Basi1df55112013-09-06 11:25:09 -07001935 @param hostname: hostname to check for rpm support.
1936
Simran Basid5e5e272012-09-24 15:23:59 -07001937 @return None if this host does not follows the defined naming format
1938 for RPM powered DUT's in the lab. If it does follow the format,
1939 it returns a regular expression MatchObject instead.
1940 """
Fang Dengbaff9082015-01-06 13:46:15 -08001941 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001942
1943
1944 def has_power(self):
1945 """For this host, return whether or not it is powered by an RPM.
1946
1947 @return True if this host is in the CROS lab and follows the defined
1948 naming format.
1949 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001950 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001951
1952
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001953 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001954 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001955
1956 @param state Specifies which power state to set to DUT
1957 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001958 use. By default "RPM" or "CCD" will be used based
1959 on servo type. Valid values from
1960 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001961
1962 """
1963 ACCEPTABLE_STATES = ['ON', 'OFF']
1964
Garry Wang5e5538a2019-04-08 15:36:18 -07001965 if not power_method:
1966 power_method = self.get_default_power_method()
1967
1968 state = state.upper()
1969 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001970 raise error.TestError('State must be one of: %s.'
1971 % (ACCEPTABLE_STATES,))
1972
1973 if power_method == self.POWER_CONTROL_SERVO:
1974 logging.info('Setting servo port J10 to %s', state)
1975 self.servo.set('prtctl3_pwren', state.lower())
1976 time.sleep(self._USB_POWER_TIMEOUT)
1977 elif power_method == self.POWER_CONTROL_MANUAL:
1978 logging.info('You have %d seconds to set the AC power to %s.',
1979 self._POWER_CYCLE_TIMEOUT, state)
1980 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001981 elif power_method == self.POWER_CONTROL_CCD:
1982 servo_role = 'src' if state == 'ON' else 'snk'
1983 logging.info('servo ccd power pass through detected,'
1984 ' changing servo_role to %s.', servo_role)
1985 self.servo.set_servo_v4_role(servo_role)
1986 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001987 # Make sure we don't leave DUT with no power(servo_role=snk)
1988 # when DUT is not pingable, as we raise a exception here
1989 # that may break a power cycle in the middle.
1990 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001991 raise error.AutoservError(
1992 'DUT failed to regain network connection after %d seconds.'
1993 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001994 else:
1995 if not self.has_power():
1996 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001997 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07001998 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07001999
2000
Garry Wang5e5538a2019-04-08 15:36:18 -07002001 def power_off(self, power_method=None):
2002 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002003
2004 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002005 use. By default "RPM" or "CCD" will be used based
2006 on servo type. Valid values from
2007 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002008
2009 """
2010 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002011
2012
Garry Wang5e5538a2019-04-08 15:36:18 -07002013 def power_on(self, power_method=None):
2014 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002015
2016 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002017 use. By default "RPM" or "CCD" will be used based
2018 on servo type. Valid values from
2019 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002020
2021 """
2022 self._set_power('ON', power_method)
2023
2024
Garry Wang5e5538a2019-04-08 15:36:18 -07002025 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002026 """Cycle power to this host by turning it OFF, then ON.
2027
2028 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002029 use. By default "RPM" or "CCD" will be used based
2030 on servo type. Valid values from
2031 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002032
2033 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002034 if not power_method:
2035 power_method = self.get_default_power_method()
2036
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002037 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002038 self.POWER_CONTROL_MANUAL,
2039 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002040 self.power_off(power_method=power_method)
2041 time.sleep(self._POWER_CYCLE_TIMEOUT)
2042 self.power_on(power_method=power_method)
2043 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002044 self._add_rpm_changed_tag()
2045 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002046
2047
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002048 def get_platform_from_fwid(self):
2049 """Determine the platform from the crossystem fwid.
2050
2051 @returns a string representing this host's platform.
2052 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002053 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002054 crossystem = utils.Crossystem(self)
2055 crossystem.init()
2056 # Extract fwid value and use the leading part as the platform id.
2057 # fwid generally follow the format of {platform}.{firmware version}
2058 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2059 platform = crossystem.fwid().split('.')[0].lower()
2060 # Newer platforms start with 'Google_' while the older ones do not.
2061 return platform.replace('google_', '')
2062
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002063
Simran Basic6f1f7a2012-10-16 10:47:46 -07002064 def get_platform(self):
2065 """Determine the correct platform label for this host.
2066
2067 @returns a string representing this host's platform.
2068 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002069 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2070 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002071 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002072 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002073 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002074 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002075
2076
Greg Edelstona7b05d12020-04-01 16:00:51 -06002077 def get_model_from_cros_config(self):
2078 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002079
Greg Edelstona7b05d12020-04-01 16:00:51 -06002080 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002081 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002082 return cros_config.call_cros_config_get_output('/ name',
2083 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002084
2085
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002086 def get_architecture(self):
2087 """Determine the correct architecture label for this host.
2088
2089 @returns a string representing this host's architecture.
2090 """
2091 crossystem = utils.Crossystem(self)
2092 crossystem.init()
2093 return crossystem.arch()
2094
2095
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002096 def get_chrome_version(self):
2097 """Gets the Chrome version number and milestone as strings.
2098
2099 Invokes "chrome --version" to get the version number and milestone.
2100
2101 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2102 current Chrome version number as a string (in the form "W.X.Y.Z")
2103 and "milestone" is the first component of the version number
2104 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2105 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2106 of "chrome --version" and the milestone will be the empty string.
2107
2108 """
MK Ryu35d661e2014-09-25 17:44:10 -07002109 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002110 return utils.parse_chrome_version(version_string)
2111
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002112
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002113 def get_ec_version(self):
2114 """Get the ec version as strings.
2115
2116 @returns a string representing this host's ec version.
2117 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002118 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002119 result = self.run(command, ignore_status=True)
2120 if result.exit_status != 0:
2121 return ''
2122 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002123
2124
2125 def get_firmware_version(self):
2126 """Get the firmware version as strings.
2127
2128 @returns a string representing this host's firmware version.
2129 """
2130 crossystem = utils.Crossystem(self)
2131 crossystem.init()
2132 return crossystem.fwid()
2133
2134
2135 def get_hardware_revision(self):
2136 """Get the hardware revision as strings.
2137
2138 @returns a string representing this host's hardware revision.
2139 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002140 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002141 result = self.run(command, ignore_status=True)
2142 if result.exit_status != 0:
2143 return ''
2144 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002145
2146
2147 def get_kernel_version(self):
2148 """Get the kernel version as strings.
2149
2150 @returns a string representing this host's kernel version.
2151 """
2152 return self.run('uname -r').stdout.strip()
2153
2154
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002155 def get_cpu_name(self):
2156 """Get the cpu name as strings.
2157
2158 @returns a string representing this host's cpu name.
2159 """
2160
2161 # Try get cpu name from device tree first
2162 if self.path_exists('/proc/device-tree/compatible'):
2163 command = ' | '.join(
2164 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2165 'tail -1'])
2166 return self.run(command).stdout.strip().replace(',', ' ')
2167
2168 # Get cpu name from uname -p
2169 command = 'uname -p'
2170 ret = self.run(command).stdout.strip()
2171
2172 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2173 # Try get cpu name from /proc/cpuinfo instead
2174 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2175 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2176 self = self.run(command).stdout.strip()
2177
2178 # Remove bloat from CPU name, for example
2179 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2180 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2181 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2182 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2183 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2184 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2185
2186
2187 def get_screen_resolution(self):
2188 """Get the screen(s) resolution as strings.
2189 In case of more than 1 monitor, return resolution for each monitor
2190 separate with plus sign.
2191
2192 @returns a string representing this host's screen(s) resolution.
2193 """
2194 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2195 ret = self.run(command, ignore_status=True)
2196 # We might have Chromebox without a screen
2197 if ret.exit_status != 0:
2198 return ''
2199 return ret.stdout.strip().replace('\n', '+')
2200
2201
2202 def get_mem_total_gb(self):
2203 """Get total memory available in the system in GiB (2^20).
2204
2205 @returns an integer representing total memory
2206 """
2207 mem_total_kb = self.read_from_meminfo('MemTotal')
2208 kb_in_gb = float(2 ** 20)
2209 return int(round(mem_total_kb / kb_in_gb))
2210
2211
2212 def get_disk_size_gb(self):
2213 """Get size of disk in GB (10^9)
2214
2215 @returns an integer representing size of disk, 0 in Error Case
2216 """
2217 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2218 result = self.run(command, ignore_status=True)
2219 if result.exit_status != 0:
2220 return 0
2221 _, _, block, _ = re.split(r' +', result.stdout.strip())
2222 byte_per_block = 1024.0
2223 disk_kb_in_gb = 1e9
2224 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2225
2226
2227 def get_battery_size(self):
2228 """Get size of battery in Watt-hour via sysfs
2229
2230 This method assumes that battery support voltage_min_design and
2231 charge_full_design sysfs.
2232
2233 @returns a float representing Battery size, 0 if error.
2234 """
2235 # sysfs report data in micro scale
2236 battery_scale = 1e6
2237
2238 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2239 result = self.run(command, ignore_status=True)
2240 if result.exit_status != 0:
2241 return 0
2242 voltage = float(result.stdout.strip()) / battery_scale
2243
2244 command = 'cat /sys/class/power_supply/*/charge_full_design'
2245 result = self.run(command, ignore_status=True)
2246 if result.exit_status != 0:
2247 return 0
2248 amphereHour = float(result.stdout.strip()) / battery_scale
2249
2250 return voltage * amphereHour
2251
2252
2253 def get_low_battery_shutdown_percent(self):
2254 """Get the percent-based low-battery shutdown threshold.
2255
2256 @returns a float representing low-battery shutdown percent, 0 if error.
2257 """
2258 ret = 0.0
2259 try:
2260 command = 'check_powerd_config --low_battery_shutdown_percent'
2261 ret = float(self.run(command).stdout)
2262 except error.CmdError:
2263 logging.debug("Can't run %s", command)
2264 except ValueError:
2265 logging.debug("Didn't get number from %s", command)
2266
2267 return ret
2268
2269
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002270 def has_hammer(self):
2271 """Check whether DUT has hammer device or not.
2272
2273 @returns boolean whether device has hammer or not
2274 """
2275 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2276 return self.run(command, ignore_status=True).exit_status == 0
2277
2278
Niranjan Kumar34618872017-05-31 12:57:09 -07002279 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002280 """Returns True if the specified switch was provided to Chrome.
2281
2282 @param switch The chrome switch to search for.
2283 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002284
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002285 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2286 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002287
2288
2289 def oobe_triggers_update(self):
2290 """Returns True if this host has an OOBE flow during which
2291 it will perform an update check and perhaps an update.
2292 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2293 As more such flows are developed, code handling them needs
2294 to be added here.
2295
2296 @return Boolean indicating whether this host's OOBE triggers an update.
2297 """
2298 return self.is_chrome_switch_present(
2299 '--enterprise-enable-zero-touch-enrollment=hands-off')
2300
2301
Kevin Chenga2619dc2016-03-28 11:42:08 -07002302 # TODO(kevcheng): change this to just return the board without the
2303 # 'board:' prefix and fix up all the callers. Also look into removing the
2304 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002305 def get_board(self):
2306 """Determine the correct board label for this host.
2307
2308 @returns a string representing this host's board.
2309 """
2310 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2311 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002312 return (ds_constants.BOARD_PREFIX +
2313 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002314
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002315 def get_channel(self):
2316 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002317
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002318 @returns: a string represeting this host's build channel.
2319 (stable, dev, beta). None on fail.
2320 """
2321 return lsbrelease_utils.get_chromeos_channel(
2322 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002323
Kevin Chenga328da62016-03-31 10:49:04 -07002324 def get_power_supply(self):
2325 """
2326 Determine what type of power supply the host has
2327
2328 @returns a string representing this host's power supply.
2329 'power:battery' when the device has a battery intended for
2330 extended use
2331 'power:AC_primary' when the device has a battery not intended
2332 for extended use (for moving the machine, etc)
2333 'power:AC_only' when the device has no battery at all.
2334 """
2335 psu = self.run(command='mosys psu type', ignore_status=True)
2336 if psu.exit_status:
2337 # The psu command for mosys is not included for all platforms. The
2338 # assumption is that the device will have a battery if the command
2339 # is not found.
2340 return 'power:battery'
2341
2342 psu_str = psu.stdout.strip()
2343 if psu_str == 'unknown':
2344 return None
2345
2346 return 'power:%s' % psu_str
2347
2348
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002349 def has_battery(self):
2350 """Determine if DUT has a battery.
2351
2352 Returns:
2353 Boolean, False if known not to have battery, True otherwise.
2354 """
2355 rv = True
2356 power_supply = self.get_power_supply()
2357 if power_supply == 'power:battery':
2358 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2359 board_type = self.get_board_type()
2360 if board_type in _NO_BATTERY_BOARD_TYPE:
2361 logging.warn('Do NOT believe type %s has battery. '
2362 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002363 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002364 ignore_status=True)
2365 logging.debug(psu)
2366 rv = False
2367 elif power_supply == 'power:AC_only':
2368 rv = False
2369
2370 return rv
2371
2372
Kevin Chenga328da62016-03-31 10:49:04 -07002373 def get_servo(self):
2374 """Determine if the host has a servo attached.
2375
2376 If the host has a working servo attached, it should have a servo label.
2377
2378 @return: string 'servo' if the host has servo attached. Otherwise,
2379 returns None.
2380 """
2381 return 'servo' if self._servo_host else None
2382
2383
Kevin Chenga328da62016-03-31 10:49:04 -07002384 def has_internal_display(self):
2385 """Determine if the device under test is equipped with an internal
2386 display.
2387
2388 @return: 'internal_display' if one is present; None otherwise.
2389 """
2390 from autotest_lib.client.cros.graphics import graphics_utils
2391 from autotest_lib.client.common_lib import utils as common_utils
2392
2393 def __system_output(cmd):
2394 return self.run(cmd).stdout
2395
2396 def __read_file(remote_path):
2397 return self.run('cat %s' % remote_path).stdout
2398
2399 # Hijack the necessary client functions so that we can take advantage
2400 # of the client lib here.
2401 # FIXME: find a less hacky way than this
2402 original_system_output = utils.system_output
2403 original_read_file = common_utils.read_file
2404 utils.system_output = __system_output
2405 common_utils.read_file = __read_file
2406 try:
2407 return ('internal_display' if graphics_utils.has_internal_display()
2408 else None)
2409 finally:
2410 utils.system_output = original_system_output
2411 common_utils.read_file = original_read_file
2412
2413
Dan Shi85276d42014-04-08 22:11:45 -07002414 def is_boot_from_usb(self):
2415 """Check if DUT is boot from USB.
2416
2417 @return: True if DUT is boot from usb.
2418 """
2419 device = self.run('rootdev -s -d').stdout.strip()
2420 removable = int(self.run('cat /sys/block/%s/removable' %
2421 os.path.basename(device)).stdout.strip())
2422 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002423
2424
2425 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002426 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002427
2428 @param key: meminfo requested
2429
2430 @return the memory value as a string
2431
2432 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002433 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2434 logging.debug('%s', meminfo)
2435 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002436
2437
Rohit Makasana98e696f2016-06-03 18:48:10 -07002438 def get_cpu_arch(self):
2439 """Returns CPU arch of the device.
2440
2441 @return CPU architecture of the DUT.
2442 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002443 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002444 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2445 ignore_status=True).stdout:
2446 return 'x86_64'
2447 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2448 ignore_status=True).stdout:
2449 return 'arm'
2450 return 'i386'
2451
2452
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002453 def get_board_type(self):
2454 """
2455 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002456 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2457
2458 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002459 """
Danny Chan471a8d12015-08-18 14:57:41 -07002460 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2461 ignore_status=True).stdout
2462 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002463 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002464 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002465
2466
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002467 def get_arc_version(self):
2468 """Return ARC version installed on the DUT.
2469
2470 @returns ARC version as string if the CrOS build has ARC, else None.
2471 """
2472 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2473 ignore_status=True).stdout
2474 if arc_version:
2475 return arc_version.split('=')[-1].strip()
2476 return None
2477
2478
Gilad Arnolda76bef02015-09-29 13:55:15 -07002479 def get_os_type(self):
2480 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002481
2482
Kevin Chenga2619dc2016-03-28 11:42:08 -07002483 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002484 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002485 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002486
2487
2488 def get_default_power_method(self):
2489 """
2490 Get the default power method for power_on/off/cycle() methods.
2491 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2492 """
2493 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002494 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002495 if self.servo and self.servo.supports_built_in_pd_control():
2496 self._default_power_method = self.POWER_CONTROL_CCD
2497 else:
2498 logging.debug('Either servo is unitialized or the servo '
2499 'setup does not support pd controls. Falling '
2500 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002501 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002502
2503
2504 def find_usb_devices(self, idVendor, idProduct):
2505 """
2506 Get usb device sysfs name for specific device.
2507
2508 @param idVendor Vendor ID to search in sysfs directory.
2509 @param idProduct Product ID to search in sysfs directory.
2510
2511 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2512 """
2513 # Look for matching file and cut at position 7 to get dir name.
2514 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2515
2516 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2517 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2518
2519 # Use uniq -d to print duplicate line from both command
2520 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2521
2522 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2523
2524
2525 def bind_usb_device(self, usb_node):
2526 """
2527 Bind usb device
2528
2529 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2530 """
2531 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2532 self.run(cmd, ignore_status=True)
2533
2534
2535 def unbind_usb_device(self, usb_node):
2536 """
2537 Unbind usb device
2538
2539 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2540 """
2541 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2542 self.run(cmd, ignore_status=True)
2543
2544
2545 def get_wlan_ip(self):
2546 """
2547 Get ip address of wlan interface.
2548
2549 @return ip address of wlan or empty string if wlan is not connected.
2550 """
2551 cmds = [
2552 'iw dev', # List wlan physical device
2553 'grep Interface', # Grep only interface name
2554 'cut -f 2 -d" "', # Cut the name part
2555 'xargs ifconfig', # Feed it to ifconfig to get ip
2556 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2557 'cut -f 2 -d " "' # Cut the ip part
2558 ]
2559 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002560
2561 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2562 """
2563 Connect to wifi network
2564
2565 @param ssid SSID of the wifi network.
2566 @param passphrase Passphrase of the wifi network. None if not existed.
2567 @param security Security of the wifi network. Default to "psk" if
2568 passphase is given without security. Possible values
2569 are "none", "psk", "802_1x".
2570
2571 @return True if succeed, False if not.
2572 """
2573 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2574 if passphrase:
2575 cmd += ' ' + passphrase
2576 if security:
2577 cmd += ' ' + security
2578 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002579
2580 def get_device_repair_state(self):
2581 """Get device repair state"""
2582 return self._device_repair_state
2583
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002584 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002585 """Set device repair state.
2586
2587 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002588 file in result directory. The file will be read by Lucifer. The
2589 file will not be created if result directory not specified.
2590
2591 @params state: The new state for the device.
2592 @params resultdir: The path to result directory. If path not provided
2593 will be attempt to get retrieve it from job
2594 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002595 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002596 resultdir = resultdir or getattr(self.job, 'resultdir', '')
2597 if resultdir:
2598 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07002599 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002600 logging.info('Set device state as %s. '
2601 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07002602 else:
2603 logging.debug('Cannot write the device state due missing info '
2604 'about result dir.')
2605 self._device_repair_state = state
2606
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002607 def set_device_needs_replacement(self, resultdir=None):
2608 """Set device as required replacement.
2609
2610 @params resultdir: The path to result directory. If path not provided
2611 will be attempt to get retrieve it from job
2612 if present.
2613 """
2614 self.set_device_repair_state(
2615 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
2616 resultdir=resultdir)
2617
2618 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002619 """Check if device require manual attention to be fixed.
2620
2621 The state 'needs_manual_repair' can be set when auto repair cannot
2622 fix the device due hardware or cable issues.
2623 """
2624 # ignore the logic if state present
2625 # state can be set by any cros repair actions
2626 if self.get_device_repair_state():
2627 return
2628
2629 # set need manual attention if servo has hardware issue
2630 servo_state_required_manual_fix = [
2631 servo_constants.SERVO_STATE_NOT_CONNECTED,
2632 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
2633 servo_constants.SERVO_STATE_LID_OPEN_FAILED,
2634 servo_constants.SERVO_STATE_BAD_RIBBON_CABLE,
2635 servo_constants.SERVO_STATE_EC_BROKEN,
2636 ]
2637 if self.get_servo_state() in servo_state_required_manual_fix:
Otabek Kasimovde8eea32020-07-01 12:12:22 -07002638 data = {'host': self.hostname,
Otabek Kasimov832d9162020-07-27 19:24:57 -07002639 'state': cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR}
Otabek Kasimov6825b762020-06-23 23:42:44 -07002640 metrics.Counter(
2641 'chromeos/autotest/repair/special_dut_state'
2642 ).increment(fields=data)
2643 # TODO (otabek) unblock when be sure that we do not have flakiness
Otabek Kasimov832d9162020-07-27 19:24:57 -07002644 # self.set_device_repair_state(
2645 # cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)