blob: 1de782416e0d4bdff182517d67f647744b0c1ac4 [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
Richard Barnetted31580e2018-05-14 19:58:00 +000033from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080034from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070035from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070036from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070037from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080038from autotest_lib.site_utils.rpm_control_system import rpm_client
Otabek Kasimov808cd832020-05-28 18:27:46 -070039from autotest_lib.site_utils.admin_audit import constants as audit_const
Simran Basid5e5e272012-09-24 15:23:59 -070040
Simran Basi382506b2016-09-13 14:58:15 -070041# In case cros_host is being ran via SSP on an older Moblab version with an
42# older chromite version.
43try:
44 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080045except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080046 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080047
Simran Basid5e5e272012-09-24 15:23:59 -070048
Dan Shib8540a52015-07-16 14:18:23 -070049CONFIG = global_config.global_config
50
Otabek Kasimov6825b762020-06-23 23:42:44 -070051# Device is not fixable due issues with hardware and has to be replaced
52DEVICE_STATE_NEEDS_REPLACEMENT = 'needs_replacement'
53# Device required manual attention to be fixed
54DEVICE_STATE_NEEDS_MANUAL_REPAIR = 'needs_manual_repair'
55
Dan Shid07ee2e2015-09-24 14:49:25 -070056
beepsc87ff602013-07-31 21:53:00 -070057class FactoryImageCheckerException(error.AutoservError):
58 """Exception raised when an image is a factory image."""
59 pass
60
61
Fang Deng0ca40e22013-08-27 17:47:44 -070062class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070063 """Chromium OS specific subclass of Host."""
64
Simran Basi5ace6f22016-01-06 17:30:44 -080065 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
66
Scott Zawalski62bacae2013-03-05 10:40:32 -050067 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070068
Richard Barnette03a0c132012-11-05 12:40:35 -080069 # Timeout values (in seconds) associated with various Chrome OS
70 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070071 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080072 # In general, a good rule of thumb is that the timeout can be up
73 # to twice the typical measured value on the slowest platform.
74 # The times here have not necessarily been empirically tested to
75 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070076 #
77 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080078 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
79 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080080 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070081 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080082 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070083 # screen delay, time to start the network on the DUT, and the
84 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070085 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080086 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080087 # network.
beepsf079cfb2013-09-18 17:49:51 -070088 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080089 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
90 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070091
92 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080093 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080094 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070095 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070096 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070097 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080098 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070099
Dan Shica503482015-03-30 17:23:25 -0700100 # Minimum OS version that supports server side packaging. Older builds may
101 # not have server side package built or with Autotest code change to support
102 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -0700103 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -0700104 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700105
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800106 # REBOOT_TIMEOUT: How long to wait for a reboot.
107 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700108 # We have a long timeout to ensure we don't flakily fail due to other
109 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700110 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
111 # return from reboot' bug is solved.
112 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700113
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800114 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
115 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700116 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
117 # since changing servo role will reset USB state
118 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800119 _USB_POWER_TIMEOUT = 5
120 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700121 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800122
Fang Dengdeba14f2014-11-14 11:54:09 -0800123 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
124 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700125
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800126 # Constants used in ping_wait_up() and ping_wait_down().
127 #
128 # _PING_WAIT_COUNT is the approximate number of polling
129 # cycles to use when waiting for a host state change.
130 #
131 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
132 # for arguments to the internal _ping_wait_for_status()
133 # method.
134 _PING_WAIT_COUNT = 40
135 _PING_STATUS_DOWN = False
136 _PING_STATUS_UP = True
137
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800138 # Allowed values for the power_method argument.
139
Garry Wang5e5538a2019-04-08 15:36:18 -0700140 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
141 # DUTs except those with servo_v4 CCD.
142 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
143 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800144 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
145 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
146 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700147 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800148 POWER_CONTROL_SERVO = 'servoj10'
149 POWER_CONTROL_MANUAL = 'manual'
150
151 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700152 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800153 POWER_CONTROL_SERVO,
154 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800155
Simran Basi5e6339a2013-03-21 11:34:32 -0700156 _RPM_OUTLET_CHANGED = 'outlet_changed'
157
Dan Shi9cb0eec2014-06-03 09:04:50 -0700158 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700159 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700160 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700161
Brent Peterson1cb623a2020-01-09 13:14:28 -0800162 # Regular expression for extracting EC version string
163 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
164
165 # Regular expression for extracting BIOS version string
166 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
167
Brent Petersonc70a1832020-01-24 15:54:35 -0800168 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700169 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800170
J. Richard Barnette964fba02012-10-24 17:34:29 -0700171 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800172 def check_host(host, timeout=10):
173 """
174 Check if the given host is a chrome-os host.
175
176 @param host: An ssh host representing a device.
177 @param timeout: The timeout for the run command.
178
179 @return: True if the host device is chromeos.
180
beeps46dadc92013-11-07 14:07:10 -0800181 """
182 try:
Allen Liad719c12017-06-27 23:48:04 +0000183 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700184 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700185 '! grep -q moblab /etc/lsb-release && '
186 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700187 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700188 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000189 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700190 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
191 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800192 return not (
193 lsbrelease_utils.is_jetstream(
194 lsb_release_content=lsb_release_content) or
195 lsbrelease_utils.is_gce_board(
196 lsb_release_content=lsb_release_content))
197
beeps46dadc92013-11-07 14:07:10 -0800198 except (error.AutoservRunError, error.AutoservSSHTimeout):
199 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700200
201 return False
beeps46dadc92013-11-07 14:07:10 -0800202
203
204 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800205 def get_chameleon_arguments(args_dict):
206 """Extract chameleon options from `args_dict` and return the result.
207
208 Recommended usage:
209 ~~~~~~~~
210 args_dict = utils.args_to_dict(args)
211 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
212 host = hosts.create_host(machine, chameleon_args=chameleon_args)
213 ~~~~~~~~
214
215 @param args_dict Dictionary from which to extract the chameleon
216 arguments.
217 """
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800218 return {key: args_dict[key]
Allen Li083866b2016-08-18 10:07:10 -0700219 for key in ('chameleon_host', 'chameleon_port')
220 if key in args_dict}
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800221
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800222 @staticmethod
223 def get_btpeer_arguments(args_dict):
224 """Extract btpeer options from `args_dict` and return the result.
225
226 This is used to parse details of Bluetooth peer.
227 Recommended usage:
228 ~~~~~~~~
229 args_dict = utils.args_to_dict(args)
230 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700231 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800232 ~~~~~~~~
233
234 @param args_dict: Dictionary from which to extract the btpeer
235 arguments.
236 """
237 if 'btpeer_host_list' in args_dict:
238 result = []
239 for btpeer in args_dict['btpeer_host_list'].split(','):
240 result.append({key: value for key,value in
241 zip(('btpeer_host','btpeer_port'),
242 btpeer.split(':'))})
243 return result
244 else:
245 return {key: args_dict[key]
246 for key in ('btpeer_host', 'btpeer_port')
247 if key in args_dict}
248
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800249
250 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700251 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800252 """Extract chameleon options from `args_dict` and return the result.
253
254 Recommended usage:
255 ~~~~~~~~
256 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700257 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
258 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800259 ~~~~~~~~
260
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700261 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800262 arguments.
263 """
Allen Li083866b2016-08-18 10:07:10 -0700264 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700265 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700266 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800267
268
269 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800270 def get_servo_arguments(args_dict):
271 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800272
273 Recommended usage:
274 ~~~~~~~~
275 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700276 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800277 host = hosts.create_host(machine, servo_args=servo_args)
278 ~~~~~~~~
279
280 @param args_dict Dictionary from which to extract the servo
281 arguments.
282 """
Garry Wang11b5e872020-03-11 15:14:08 -0700283 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
284 servo_constants.SERVO_PORT_ATTR,
285 servo_constants.SERVO_BOARD_ATTR,
286 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200287 servo_args = {key: args_dict[key]
288 for key in servo_attrs
289 if key in args_dict}
290 return (
291 None
Garry Wang11b5e872020-03-11 15:14:08 -0700292 if servo_constants.SERVO_HOST_ATTR in servo_args
293 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200294 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700295
J. Richard Barnette964fba02012-10-24 17:34:29 -0700296
J. Richard Barnette91137f02016-03-10 16:52:26 -0800297 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700298 pdtester_args=None, try_lab_servo=False,
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700299 try_servo_repair=False, btpeer_args=[],
J. Richard Barnette91137f02016-03-10 16:52:26 -0800300 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700301 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800302 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700303
Fang Denge545abb2014-12-30 18:43:47 -0800304 This method will attempt to create the test-assistant object
305 (chameleon/servo) when it is needed by the test. Check
306 the docstring of chameleon_host.create_chameleon_host and
307 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700308
Fang Denge545abb2014-12-30 18:43:47 -0800309 @param hostname: Hostname of the dut.
310 @param chameleon_args: A dictionary that contains args for creating
311 a ChameleonHost. See chameleon_host for details.
312 @param servo_args: A dictionary that contains args for creating
313 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700314 @param try_lab_servo: When true, indicates that an attempt should
315 be made to create a ServoHost for a DUT in
316 the test lab, even if not required by
317 `servo_args`. See servo_host for details.
318 @param try_servo_repair: If a servo host is created, check it
319 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800320 See servo_host for details.
321 @param ssh_verbosity_flag: String, to pass to the ssh command to control
322 verbosity.
323 @param ssh_options: String, other ssh options to pass to the ssh
324 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700325 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700326 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700327 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800328 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700329 # hold special dut_state for repair process
330 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700331 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700332 # self.env is a dictionary of environment variable settings
333 # to be exported for commands run on the host.
334 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
335 # errors that might happen.
336 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700337 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700338 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700339 _servo_host, servo_state = servo_host.create_servo_host(
340 dut=self,
341 servo_args=servo_args,
342 try_lab_servo=try_lab_servo,
343 try_servo_repair=try_servo_repair,
344 dut_host_info=self.host_info_store.get())
345 self.set_servo_host(_servo_host, servo_state)
Garry Wang5e5538a2019-04-08 15:36:18 -0700346 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700347
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800348 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800349 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700350 dut=self.hostname,
351 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800352 if self._chameleon_host:
353 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800354 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800355 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700356
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700357 # Initialize Bluetooth peers.
358 try:
359 self.initialize_btpeer(btpeer_args)
360 except Exception as e:
361 logging.error('Exception %s in initialize_btpeer', str(e))
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800362
howardchung83e55272019-08-08 14:08:05 +0800363 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700364 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700365 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800366
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700367 if self._pdtester_host:
368 self.pdtester_servo = self._pdtester_host.get_servo()
369 logging.info('pdtester_servo: %r', self.pdtester_servo)
370 # Create the pdtester object used to access the ec uart
371 self.pdtester = pdtester.PDTester(self.pdtester_servo,
372 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800373 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700374 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800375
Fang Deng5d518f42013-08-02 14:04:32 -0700376
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800377 def initialize_btpeer(self, btpeer_args):
378 """ Initialize the Bluetooth peers
379
380 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
381 is chameleon host on Raspberry Pi.
382 @param btpeer_args: A dictionary that contains args for creating
383 a ChameleonHost. See chameleon_host for details.
384
385 """
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700386 #TODO (b:142486063) Remove the try..except
387 try:
388 self._btpeer_host_list = []
389 self.btpeer_list = []
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800390 self.btpeer = None
391
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700392 if type(btpeer_args) is list:
393 btpeer_args_list = btpeer_args
394 else:
395 btpeer_args_list = [btpeer_args]
396
397 self._btpeer_host_list = chameleon_host.create_btpeer_host(
398 dut=self.hostname, btpeer_args_list=btpeer_args_list)
399 logging.debug('Bluetooth peer hosts are %s',
400 self._btpeer_host_list)
401 self.btpeer_list = [_host.create_chameleon_board() for _host in
402 self._btpeer_host_list if _host is not None]
403
404 if len(self.btpeer_list) > 0:
405 self.btpeer = self.btpeer_list[0]
406
407 logging.debug('After initialize_btpeer btpeer_list %s '
408 'btpeer_host_list is %s and btpeer is %s',
409 self.btpeer_list, self._btpeer_host_list,
410 self.btpeer)
411 except Exception as e:
412 logging.error('Exception %s in initialize_btpeer', str(e))
413
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800414
415
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000416 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700417 """Get latest stable cros image name from AFE.
418
419 Use the board name from the info store. Should that fail, try to
420 retrieve the board name from the host's installed image itself.
421
422 @returns: current stable cros image name for this host.
423 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700424 info = self.host_info_store.get()
425 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700426 logging.warn('No board label value found. Trying to infer '
427 'from the host itself.')
428 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700429 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700430 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
431 logging.error('Also failed to get the board name from the DUT '
432 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700433 raise error.AutoservError('Cannot determine board of the DUT'
434 ' while getting repair image name.')
435 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500436
437
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700438 def host_version_prefix(self, image):
439 """Return version label prefix.
440
441 In case the CrOS provisioning version is something other than the
442 standard CrOS version e.g. CrOS TH version, this function will
443 find the prefix from provision.py.
444
445 @param image: The image name to find its version prefix.
446 @returns: A prefix string for the image type.
447 """
448 return provision.get_version_label_prefix(image)
449
Andrew Luo3332ab22020-04-28 16:42:03 -0700450 def stage_build_to_usb(self, build):
451 """Stage the current ChromeOS image on the USB stick connected to the
452 servo.
453
454 @param build: The build to download and send to USB.
455 """
456 if not self.servo:
457 raise error.TestError('Host %s does not have servo.' %
458 self.hostname)
459
460 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700461
462 try:
463 self.servo.image_to_servo_usb(update_url)
464 finally:
465 # servo.image_to_servo_usb turned the DUT off, so turn it back on
466 logging.debug('Turn DUT power back on.')
467 self.servo.get_power_state_controller().power_on()
468
Andrew Luo3332ab22020-04-28 16:42:03 -0700469 logging.debug('ChromeOS image %s is staged on the USB stick.',
470 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700471
beepsdae65fd2013-07-26 16:24:41 -0700472 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700473 """
474 Make sure job_repo_url of this host is valid.
475
joychen03eaad92013-06-26 09:55:21 -0700476 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700477 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
478 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
479 download and extract it. If the devserver embedded in the url is
480 unresponsive, update the job_repo_url of the host after staging it on
481 another devserver.
482
483 @param job_repo_url: A url pointing to the devserver where the autotest
484 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700485 @param tag: The tag from the server job, in the format
486 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700487
488 @raises DevServerException: If we could not resolve a devserver.
489 @raises AutoservError: If we're unable to save the new job_repo_url as
490 a result of choosing a new devserver because the old one failed to
491 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700492 @raises urllib2.URLError: If the devserver embedded in job_repo_url
493 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700494 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800495 info = self.host_info_store.get()
496 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700497 if not job_repo_url:
498 logging.warning('No job repo url set on host %s', self.hostname)
499 return
500
501 logging.info('Verifying job repo url %s', job_repo_url)
502 devserver_url, image_name = tools.get_devserver_build_from_package_url(
503 job_repo_url)
504
beeps0c865032013-07-30 11:37:06 -0700505 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700506
507 logging.info('Staging autotest artifacts for %s on devserver %s',
508 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700509
510 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700511 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700512 stage_time = time.time() - start_time
513
514 # Record how much of the verification time comes from a devserver
515 # restage. If we're doing things right we should not see multiple
516 # devservers for a given board/build/branch path.
517 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800518 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700519 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800520 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700521 pass
522 else:
beeps0c865032013-07-30 11:37:06 -0700523 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700524 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700525 stats_key = {
526 'board': board,
527 'build_type': build_type,
528 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700529 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700530 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800531
532 monarch_fields = {
533 'board': board,
534 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800535 'branch': branch,
536 'dev_server': devserver,
537 }
538 metrics.Counter(
539 'chromeos/autotest/provision/verify_url'
540 ).increment(fields=monarch_fields)
541 metrics.SecondsDistribution(
542 'chromeos/autotest/provision/verify_url_duration'
543 ).add(stage_time, fields=monarch_fields)
544
545
Dan Shicf4d2032015-03-12 15:04:21 -0700546 def stage_server_side_package(self, image=None):
547 """Stage autotest server-side package on devserver.
548
549 @param image: Full path of an OS image to install or a build name.
550
551 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700552
553 @raise: error.AutoservError if fail to locate the build to test with, or
554 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700555 """
Dan Shid37736b2016-07-06 15:10:29 -0700556 # If enable_drone_in_restricted_subnet is False, do not set hostname
557 # in devserver.resolve call, so a devserver in non-restricted subnet
558 # is picked to stage autotest server package for drone to download.
559 hostname = self.hostname
560 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
561 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700562 if image:
563 image_name = tools.get_build_from_image(image)
564 if not image_name:
565 raise error.AutoservError(
566 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700567 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700568 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700569 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800570 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700571 if job_repo_url:
572 devserver_url, image_name = (
573 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700574 # If enable_drone_in_restricted_subnet is True, use the
575 # existing devserver. Otherwise, resolve a new one in
576 # non-restricted subnet.
577 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
578 ds = dev_server.ImageServer(devserver_url)
579 else:
580 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800581 elif info.build is not None:
582 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700583 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700584 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800585 raise error.AutoservError(
586 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700587 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700588
589 # Get the OS version of the build, for any build older than
590 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
591 match = re.match('.*/R\d+-(\d+)\.', image_name)
592 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700593 raise error.AutoservError(
594 'Build %s is older than %s. Server side packaging is '
595 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700596
Dan Shicf4d2032015-03-12 15:04:21 -0700597 ds.stage_artifacts(image_name, ['autotest_server_package'])
598 return '%s/static/%s/%s' % (ds.url(), image_name,
599 'autotest_server_package.tar.bz2')
600
601
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700602 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700603 """Stage a build on a devserver and return the update_url.
604
605 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700606 @param artifact: a string like 'test_image'. Requests
607 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700608 @returns a tuple of (image_name, URL) like
609 (lumpy-release/R27-3837.0.0,
610 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700611 """
612 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000613 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700614 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800615 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700616 devserver.stage_artifacts(image_name, [artifact])
617 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700618 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700619 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700620 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700621 else:
622 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700623
624
beepse539be02013-07-31 21:57:39 -0700625 def stage_factory_image_for_servo(self, image_name):
626 """Stage a build on a devserver and return the update_url.
627
628 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700629
beepse539be02013-07-31 21:57:39 -0700630 @return: An update URL, eg:
631 http://<devserver>/static/canary-channel/\
632 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700633
634 @raises: ValueError if the factory artifact name is missing from
635 the config.
636
beepse539be02013-07-31 21:57:39 -0700637 """
638 if not image_name:
639 logging.error('Need an image_name to stage a factory image.')
640 return
641
Dan Shib8540a52015-07-16 14:18:23 -0700642 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700643 'CROS', 'factory_artifact', type=str, default='')
644 if not factory_artifact:
645 raise ValueError('Cannot retrieve the factory artifact name from '
646 'autotest config, and hence cannot stage factory '
647 'artifacts.')
648
beepse539be02013-07-31 21:57:39 -0700649 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800650 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700651 devserver.stage_artifacts(
652 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700653 [factory_artifact],
654 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700655
656 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
657
658
Laurence Goodby778c9a42017-05-24 19:24:07 -0700659 def prepare_for_update(self):
660 """Prepares the DUT for an update.
661
662 Subclasses may override this to perform any special actions
663 required before updating.
664 """
Laurence Goodby468de252017-06-08 17:22:53 -0700665 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700666
667
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800668 def _clear_fw_version_labels(self, rw_only):
669 """Clear firmware version labels from the machine.
670
671 @param rw_only: True to only clear fwrw_version; otherewise, clear
672 both fwro_version and fwrw_version.
673 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700674 info = self.host_info_store.get()
675 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800676 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700677 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
678 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700679
680
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800681 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700682 """Add firmware version label to the machine.
683
684 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800685 @param rw_only: True to only add fwrw_version; otherwise, add both
686 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700687
688 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700689 info = self.host_info_store.get()
690 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800691 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700692 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
693 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700694
695
Namyoon Woo33f38852020-04-13 17:26:58 -0700696 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800697 """Search for the latest package release version from the image archive,
698 and return it.
699
Namyoon Woo33f38852020-04-13 17:26:58 -0700700 @param platform: platform name, a.k.a. board or model
701 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800702
Namyoon Woo33f38852020-04-13 17:26:58 -0700703 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
704 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800705 or None if LATEST release file does not exist.
706 """
707
Namyoon Woo33f38852020-04-13 17:26:58 -0700708 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800709
Namyoon Woo33f38852020-04-13 17:26:58 -0700710 # Search the image path in reference board archive as well.
711 # For example, bob has its binary image under its reference board (gru)
712 # image archive.
713 if ref_board:
714 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800715
Namyoon Woo33f38852020-04-13 17:26:58 -0700716 for board in platforms:
717 # Read 'LATEST-1.0.0' file
718 branch_dir = provision.FW_BRANCH_GLOB % board
719 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
720 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800721
Namyoon Woo33f38852020-04-13 17:26:58 -0700722 try:
723 # The result could be one or more.
724 result = utils.system_output('gsutil ls -d ' + latest_file)
725
726 candidates = re.findall('gs://.*', result)
727
728 # Found the directory candidates. No need to check the other
729 # board name cadidates. Let's break the loop.
730 break
731 except error.CmdError:
732 # It doesn't exist. Let's move on to the next item.
733 pass
734 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800735 logging.error('No LATEST release info is available.')
736 return None
737
Namyoon Woo406c7d42020-01-24 15:57:11 -0800738 for cand_dir in candidates:
739 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800740
Namyoon Woo406c7d42020-01-24 15:57:11 -0800741 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700742 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800743 try:
744 # Check if release_path does exist.
745 release = utils.system_output('gsutil ls -d ' + release_path)
746 # Now 'release' has a full directory path: e.g.
747 # gs://chromeos-image-archive/firmware-octopus-11297.B-
748 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
749
750 # Remove "gs://chromeos-image-archive".
751 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
752
753 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
754 return release.strip('/')
755 except error.CmdError:
756 # The directory might not exist. Let's try next candidate.
757 pass
758 else:
759 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800760
Brent Peterson1cb623a2020-01-09 13:14:28 -0800761 @staticmethod
762 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800763 """Get version string from binary image using regular expression.
764
765 @param image: Binary image to search
766 @param version_regex: Regular expression to search for
767
768 @return Version string
769
770 @raises TestFail if no version string is found in image
771 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800772 with open(image, 'rb') as f:
773 image_data = f.read()
774 match = re.findall(version_regex, image_data)
775 if match:
776 return match[0]
777 else:
778 raise error.TestFail('Failed to read version from %s.' % image)
779
780
Garry Wangad2a1712020-03-26 15:06:43 -0700781 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800782 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700783 try_scp=False, install_ec=True, install_bios=True,
784 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700785 """Install firmware to the DUT.
786
787 Use stateful update if the DUT is already running the same build.
788 Stateful update does not update kernel and tends to run much faster
789 than a full reimage. If the DUT is running a different build, or it
790 failed to do a stateful update, full update, including kernel update,
791 will be applied to the DUT.
792
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800793 Once a host enters firmware_install its fw[ro|rw]_version label will
794 be removed. After the firmware is updated successfully, a new
795 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700796
797 @param build: The build version to which we want to provision the
798 firmware of the machine,
799 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800800 @param rw_only: True to only install firmware to its RW portions. Keep
801 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700802 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800803 @param local_tarball: Path to local firmware image for installing
804 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800805 @param verify_version: True to verify EC and BIOS versions after
806 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800807 @param try_scp: False to always program using servo, true to try copying
808 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700809 @param install_ec: True to install EC FW, and False to skip it.
810 @param install_bios: True to install BIOS, and False to skip it.
811 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700812
813 TODO(dshi): After bug 381718 is fixed, update here with corresponding
814 exceptions that could be raised.
815
816 """
817 if not self.servo:
818 raise error.TestError('Host %s does not have servo.' %
819 self.hostname)
820
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700821 # Get the DUT board name from AFE.
822 info = self.host_info_store.get()
823 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700824 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800825
826 if board is None or board == '':
827 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700828
Namyoon Woo382e5892020-05-20 16:48:40 -0700829 # if board_as argument is passed, then use it instead of the original
830 # board name.
831 if board_as:
832 board = board_as
833
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700834 if model is None or model == '':
835 model = self.get_platform_from_fwid()
836
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800837 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700838 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800839 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700840 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800841
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700842 try:
843 ds = dev_server.ImageServer.resolve(build, self.hostname)
844 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800845
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700846 if not dest:
847 tmpd = autotemp.tempdir(unique_id='fwimage')
848 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800849
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700850 # Download firmware image
851 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
852 local_tarball = os.path.join(dest, os.path.basename(fwurl))
853 ds.download_file(fwurl, local_tarball)
854 except Exception as e:
855 raise error.TestError('Failed to download firmware package: %s'
856 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700857
Namyoon Woo382e5892020-05-20 16:48:40 -0700858 ec_image = None
859 if install_ec:
860 # Extract EC image from tarball
861 logging.info('Extracting EC image.')
862 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800863
Namyoon Woo382e5892020-05-20 16:48:40 -0700864 bios_image = None
865 if install_bios:
866 # Extract BIOS image from tarball
867 logging.info('Extracting BIOS image.')
868 bios_image = self.servo.extract_bios_image(board, model,
869 local_tarball)
870
871 if not bios_image and not ec_image:
872 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800873
Brent Petersonc70a1832020-01-24 15:54:35 -0800874 # Clear firmware version labels
875 self._clear_fw_version_labels(rw_only)
876
877 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800878 try:
Brent Petersonc70a1832020-01-24 15:54:35 -0800879 # Check if DUT is available and copying to DUT is enabled
880 if self.is_up() and try_scp:
881 # DUT is available, make temp firmware directory to store images
882 logging.info('Making temp folder.')
883 dest_folder = '/tmp/firmware'
884 self.run('mkdir -p ' + dest_folder)
885
Namyoon Woo68b68082020-06-02 13:13:14 -0700886 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800887
Namyoon Woo382e5892020-05-20 16:48:40 -0700888 if bios_image:
889 # Send BIOS firmware image to DUT
890 logging.info('Sending BIOS firmware.')
891 dest_bios_path = os.path.join(dest_folder,
892 os.path.basename(bios_image))
893 self.send_file(bios_image, dest_bios_path)
894
895 # Initialize firmware update command for BIOS image
896 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800897
898 # Send EC firmware image to DUT when EC image was found
899 if ec_image:
900 logging.info('Sending EC firmware.')
901 dest_ec_path = os.path.join(dest_folder,
902 os.path.basename(ec_image))
903 self.send_file(ec_image, dest_ec_path)
904
905 # Add EC image to firmware update command
906 fw_cmd += ' -e %s' % dest_ec_path
907
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700908 # Make sure command is allowed to finish even if ssh fails.
909 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
910
Brent Peterson669edf42020-02-07 15:07:54 -0800911 # Update firmware on DUT
912 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700913 try:
914 self.run(fw_cmd, options="LogLevel=verbose")
915 except error.AutoservRunError as e:
916 if e.result_obj.exit_status != 255:
917 raise
918 elif ec_image:
919 logging.warn("DUT network dropped during update"
920 " (often caused by EC resetting USB)")
921 else:
922 logging.error("DUT network dropped during update"
923 " (unexpected, since no EC image)")
924 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800925 else:
926 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800927 if ec_image:
928 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700929 if bios_image:
930 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800931 if utils.host_is_in_lab_zone(self.hostname):
932 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800933
934 # Reboot and wait for DUT after installing firmware
935 logging.info('Rebooting DUT.')
936 self.servo.get_power_state_controller().reset()
937 time.sleep(self.servo.BOOT_DELAY)
938 self.test_wait_for_boot()
939
940 # When enabled verify EC and BIOS firmware version after programming
941 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800942 # Check programmed EC firmware when EC image was found
943 if ec_image:
944 logging.info('Checking EC firmware version.')
945 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800946 ec_version_prefix = dest_ec_version.split('_', 1)[0]
947 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800948 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800949 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800950 if dest_ec_version != image_ec_version:
951 raise error.TestFail(
952 'Failed to update EC RO, version %s (expected %s)' %
953 (dest_ec_version, image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800954
Namyoon Woo382e5892020-05-20 16:48:40 -0700955 if bios_image:
956 # Check programmed BIOS firmware against expected version
957 logging.info('Checking BIOS firmware version.')
958 dest_bios_version = self.get_firmware_version()
959 bios_version_prefix = dest_bios_version.split('.', 1)[0]
960 bios_regex = self._BIOS_REGEX % bios_version_prefix
961 image_bios_version = self.get_version_from_image(bios_image,
962 bios_regex)
963 if dest_bios_version != image_bios_version:
964 raise error.TestFail(
965 'Failed to update BIOS RO, version %s '
966 '(expected %s)' % (dest_bios_version,
967 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700968 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700969 if tmpd:
970 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700971
972
beepsf079cfb2013-09-18 17:49:51 -0700973 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
974 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500975 """
976 Re-install the OS on the DUT by:
977 1) installing a test image on a USB storage device attached to the Servo
978 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800979 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700980 3) installing the image with chromeos-install.
981
Scott Zawalski62bacae2013-03-05 10:40:32 -0500982 @param image_url: If specified use as the url to install on the DUT.
983 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700984 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
985 Factory images need a longer usb_boot_timeout than regular
986 cros images.
987 @param install_timeout: The timeout to use when installing the chromeos
988 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800989
Scott Zawalski62bacae2013-03-05 10:40:32 -0500990 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700991
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800992 """
Garry Wang7b0e1b72020-03-25 19:08:59 -0700993 if image_url:
994 logging.info('Downloading image to USB, then booting from it.'
995 ' Usb boot timeout = %s', usb_boot_timeout)
996 else:
997 logging.info('Booting from USB directly. Usb boot timeout = %s',
998 usb_boot_timeout)
999
1000 metrics_field = {'download': bool(image_url)}
1001 metrics.Counter(
1002 'chromeos/autotest/provision/servo_install/download_image'
1003 ).increment(fields=metrics_field)
1004
Allen Li48a13fe2016-11-22 14:10:40 -08001005 with metrics.SecondsTimer(
1006 'chromeos/autotest/provision/servo_install/boot_duration'):
1007 self.servo.install_recovery_image(image_url)
1008 if not self.wait_up(timeout=usb_boot_timeout):
1009 raise hosts.AutoservRepairError(
1010 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001011 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001012
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001013 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1014 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001015 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001016 try:
1017 self.run('chromeos-tpm-recovery')
1018 except error.AutoservRunError:
1019 logging.warn('chromeos-tpm-recovery is too old.')
1020
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001021
Allen Li48a13fe2016-11-22 14:10:40 -08001022 with metrics.SecondsTimer(
1023 'chromeos/autotest/provision/servo_install/install_duration'):
1024 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001025 try:
1026 self.run('chromeos-install --yes',timeout=install_timeout)
1027 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001028 except Exception as e:
1029 storage_errors = [
1030 'No space left on device',
1031 'I/O error when trying to write primary GPT',
1032 'Input/output error while writing out',
1033 'cannot read GPT header',
1034 'can not determine destination device'
1035 ]
1036 has_error = [msg for msg in storage_errors if(msg in str(e))]
1037 if has_error:
1038 info = self.host_info_store.get()
1039 info.set_version_label(
1040 audit_const.DUT_STORAGE_STATE_PREFIX,
1041 audit_const.HW_STATE_NEED_REPLACEMENT)
1042 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001043 self.set_device_repair_state(
1044 DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001045 logging.debug(
1046 'Fail install image from USB; Storage error; %s', e)
1047 raise error.AutoservError(
1048 'Failed to install image from USB due to a suspect '
1049 'disk failure, DUT storage state changed to '
1050 'need_replacement, please check debug log '
1051 'for details.')
1052 else:
1053 logging.debug('Fail install image from USB; %s', e)
1054 raise error.AutoservError(
1055 'Failed to install image from USB due to unexpected '
1056 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001057 finally:
1058 # We need reset the DUT no matter re-install success or not,
1059 # as we don't want leave the DUT in boot from usb state.
1060 logging.info('Power cycling DUT through servo.')
1061 self.servo.get_power_state_controller().power_off()
1062 self.servo.switch_usbkey('off')
1063 # N.B. The Servo API requires that we use power_on() here
1064 # for two reasons:
1065 # 1) After turning on a DUT in recovery mode, you must turn
1066 # it off and then on with power_on() once more to
1067 # disable recovery mode (this is a Parrot specific
1068 # requirement).
1069 # 2) After power_off(), the only way to turn on is with
1070 # power_on() (this is a Storm specific requirement).
1071 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001072
1073 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001074 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001075 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1076 'test image after %d seconds' %
1077 self.BOOT_TIMEOUT,
1078 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001079
1080
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001081 def set_servo_host(self, host, servo_state = None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001082 """Set our servo host member, and associated servo.
1083
1084 @param host Our new `ServoHost`.
1085 """
1086 self._servo_host = host
1087 if self._servo_host is not None:
1088 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001089 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001090 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001091 else:
1092 self.servo = None
Otabek Kasimov41301a22020-05-10 15:28:21 -07001093 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001094 self.set_servo_state(servo_state)
1095
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001096
Richard Barnette9a26ad62016-06-10 12:03:08 -07001097 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001098 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001099 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001100
Richard Barnette9a26ad62016-06-10 12:03:08 -07001101 If the servo object is missing, attempt to repair the servo
1102 host. Repair failures are passed back to the caller.
1103
1104 @raise AutoservError: If there is no servo host for this CrOS
1105 host.
1106 """
1107 if self.servo:
1108 return
1109 if not self._servo_host:
1110 raise error.AutoservError('No servo host for %s.' %
1111 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001112 try:
1113 self._servo_host.repair()
1114 except:
1115 raise
1116 finally:
1117 self.set_servo_host(self._servo_host)
1118
1119
Otabek Kasimov41301a22020-05-10 15:28:21 -07001120 def set_servo_type(self):
1121 """Set servo info labels to dut host_info"""
1122 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001123 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001124 return
1125 servo_type = self.servo.get_servo_type()
1126 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001127 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001128 ' by `dut-control servo_type`! Please file a bug'
1129 ' and inform infra team as we are not expected '
1130 ' to reach this point.')
1131 return
1132 host_info = self.host_info_store.get()
1133 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1134 old_type = host_info.get_label_value(prefix)
1135 if old_type == servo_type:
1136 # do not need update
1137 return
1138 host_info.set_version_label(prefix, servo_type)
1139 self.host_info_store.commit(host_info)
1140 logging.info('ServoHost: servo_type updated to %s '
1141 '(previous: %s)', servo_type, old_type)
1142
1143
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001144 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001145 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001146 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001147 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001148 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001149 old_state = host_info.get_label_value(servo_state_prefix)
1150 if old_state == servo_state:
1151 # do not need update
1152 return
1153 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001154 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001155 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1156 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001157
1158
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001159 def get_servo_state(self):
1160 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001161 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001162 return host_info.get_label_value(servo_state_prefix)
1163
Otabek Kasimov41301a22020-05-10 15:28:21 -07001164
Garry Wang000c6c02020-05-11 21:27:23 -07001165 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1166 if smart_usbhub_detected is None:
1167 # skip the label update here as this indicate we wasn't able
1168 # to confirm usbhub type.
1169 return
1170 host_info = self.host_info_store.get()
1171 if (smart_usbhub_detected ==
1172 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1173 # skip label update if current label match the truth.
1174 return
1175 if smart_usbhub_detected:
1176 logging.info('Adding %s label to host %s',
1177 servo_constants.SMART_USBHUB_LABEL,
1178 self.hostname)
1179 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1180 else:
1181 logging.info('Removing %s label from host %s',
1182 servo_constants.SMART_USBHUB_LABEL,
1183 self.hostname)
1184 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1185 self.host_info_store.commit(host_info)
1186
1187
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001188 def repair(self):
1189 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001190
1191 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001192 not call back to the parent class, but instead relies on
1193 `self._repair_strategy` to coordinate the verification and
1194 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001195 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001196 message = 'Beginning repair for host %s board %s model %s'
1197 info = self.host_info_store.get()
1198 message %= (self.hostname, info.board, info.model)
1199 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001200 try:
1201 self._repair_strategy.repair(self)
1202 except hosts.AutoservVerifyDependencyError as e:
1203 # We don't want flag a DUT as failed if only non-critical
1204 # verifier(s) failed during the repair.
1205 if e.is_critical():
Otabek Kasimov6825b762020-06-23 23:42:44 -07001206 self.try_set_device_need_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001207 raise
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001208
Richard Barnette82c35912012-11-20 10:09:10 -08001209
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001210 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001211 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001212 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001213
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001214 if self._chameleon_host:
1215 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001216
1217 if self._servo_host:
1218 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001219
1220
Dan Shi49ca0932014-11-14 11:22:27 -08001221 def get_power_supply_info(self):
1222 """Get the output of power_supply_info.
1223
1224 power_supply_info outputs the info of each power supply, e.g.,
1225 Device: Line Power
1226 online: no
1227 type: Mains
1228 voltage (V): 0
1229 current (A): 0
1230 Device: Battery
1231 state: Discharging
1232 percentage: 95.9276
1233 technology: Li-ion
1234
1235 Above output shows two devices, Line Power and Battery, with details of
1236 each device listed. This function parses the output into a dictionary,
1237 with key being the device name, and value being a dictionary of details
1238 of the device info.
1239
1240 @return: The dictionary of power_supply_info, e.g.,
1241 {'Line Power': {'online': 'yes', 'type': 'main'},
1242 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001243 @raise error.AutoservRunError if power_supply_info tool is not found in
1244 the DUT. Caller should handle this error to avoid false failure
1245 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001246 """
1247 result = self.run('power_supply_info').stdout.strip()
1248 info = {}
1249 device_name = None
1250 device_info = {}
1251 for line in result.split('\n'):
1252 pair = [v.strip() for v in line.split(':')]
1253 if len(pair) != 2:
1254 continue
1255 if pair[0] == 'Device':
1256 if device_name:
1257 info[device_name] = device_info
1258 device_name = pair[1]
1259 device_info = {}
1260 else:
1261 device_info[pair[0]] = pair[1]
1262 if device_name and not device_name in info:
1263 info[device_name] = device_info
1264 return info
1265
1266
1267 def get_battery_percentage(self):
1268 """Get the battery percentage.
1269
1270 @return: The percentage of battery level, value range from 0-100. Return
1271 None if the battery info cannot be retrieved.
1272 """
1273 try:
1274 info = self.get_power_supply_info()
1275 logging.info(info)
1276 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001277 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001278 return None
1279
1280
Philip Chenaf69ead2020-03-27 13:06:42 -07001281 def get_battery_state(self):
1282 """Get the battery charging state.
1283
1284 @return: A string representing the battery charging state. It can be
1285 'Charging', 'Fully charged', or 'Discharging'.
1286 """
1287 try:
1288 info = self.get_power_supply_info()
1289 logging.info(info)
1290 return info['Battery']['state']
1291 except (KeyError, ValueError, error.AutoservRunError):
1292 return None
1293
1294
Daniel Campello8ca25c22019-12-13 16:48:26 -07001295 def get_battery_display_percentage(self):
1296 """Get the battery display percentage.
1297
1298 @return: The display percentage of battery level, value range from
1299 0-100. Return None if the battery info cannot be retrieved.
1300 """
1301 try:
1302 info = self.get_power_supply_info()
1303 logging.info(info)
1304 return float(info['Battery']['display percentage'])
1305 except (KeyError, ValueError, error.AutoservRunError):
1306 return None
1307
1308
Dan Shi49ca0932014-11-14 11:22:27 -08001309 def is_ac_connected(self):
1310 """Check if the dut has power adapter connected and charging.
1311
1312 @return: True if power adapter is connected and charging.
1313 """
1314 try:
1315 info = self.get_power_supply_info()
1316 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001317 except (KeyError, error.AutoservRunError):
1318 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001319
1320
Simran Basi5e6339a2013-03-21 11:34:32 -07001321 def _cleanup_poweron(self):
1322 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001323 info = self.host_info_store.get()
1324 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001325 return
1326 logging.debug('This host has recently interacted with the RPM'
1327 ' Infrastructure. Ensuring power is on.')
1328 try:
1329 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001330 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001331 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001332 logging.error('Failed to turn Power On for this host after '
1333 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001334
1335 battery_percentage = self.get_battery_percentage()
Gregory Nisbet02273172020-07-13 09:26:17 -07001336 if (battery_percentage and
1337 battery_percentage < cros_repair.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001338 raise
1339 elif self.is_ac_connected():
1340 logging.info('The device has power adapter connected and '
1341 'charging. No need to try to turn RPM on '
1342 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001343 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001344 logging.info('Battery level is now at %s%%. The device may '
1345 'still have enough power to run test, so no '
1346 'exception will be raised.', battery_percentage)
1347
Simran Basi5e6339a2013-03-21 11:34:32 -07001348
Garry Wangad4d4fd2019-01-30 17:00:38 -08001349 def _remove_rpm_changed_tag(self):
1350 info = self.host_info_store.get()
1351 del info.attributes[self._RPM_OUTLET_CHANGED]
1352 self.host_info_store.commit(info)
1353
1354
1355 def _add_rpm_changed_tag(self):
1356 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001357 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001358 self.host_info_store.commit(info)
1359
1360
1361
beepsc87ff602013-07-31 21:53:00 -07001362 def _is_factory_image(self):
1363 """Checks if the image on the DUT is a factory image.
1364
1365 @return: True if the image on the DUT is a factory image.
1366 False otherwise.
1367 """
1368 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1369 return result.exit_status == 0
1370
1371
1372 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001373 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001374
1375 @raises: FactoryImageCheckerException for factory images, since
1376 we cannot attempt to restart ui on them.
1377 error.AutoservRunError for any other type of error that
1378 occurs while restarting ui.
1379 """
1380 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001381 raise FactoryImageCheckerException('Cannot restart ui on factory '
1382 'images')
beepsc87ff602013-07-31 21:53:00 -07001383
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001384 # TODO(jrbarnette): The command to stop/start the ui job
1385 # should live inside cros_ui, too. However that would seem
1386 # to imply interface changes to the existing start()/restart()
1387 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001388 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001389 self.run('stop ui; start ui')
1390 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001391
1392
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001393 def _start_powerd_if_needed(self):
1394 """Start powerd if it isn't already running."""
1395 self.run('start powerd', ignore_status=True)
1396
1397
xixuana3bbc422017-05-04 15:57:21 -07001398 def _get_lsb_release_content(self):
1399 """Return the content of lsb-release file of host."""
1400 return self.run(
1401 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1402
1403
Dan Shi549fb822015-03-24 18:01:11 -07001404 def get_release_version(self):
1405 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1406
1407 @returns The version string in lsb-release, under attribute
1408 CHROMEOS_RELEASE_VERSION.
1409 """
Dan Shi549fb822015-03-24 18:01:11 -07001410 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001411 lsb_release_content=self._get_lsb_release_content())
1412
1413
Don Garrettb9f35802018-01-22 18:25:40 -08001414 def get_release_builder_path(self):
1415 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1416
1417 @returns The version string in lsb-release, under attribute
1418 CHROMEOS_RELEASE_BUILDER_PATH.
1419 """
1420 return lsbrelease_utils.get_chromeos_release_builder_path(
1421 lsb_release_content=self._get_lsb_release_content())
1422
1423
xixuana3bbc422017-05-04 15:57:21 -07001424 def get_chromeos_release_milestone(self):
1425 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1426 from lsb-release.
1427
1428 @returns The version string in lsb-release, under attribute
1429 CHROMEOS_RELEASE_BUILD_TYPE.
1430 """
1431 return lsbrelease_utils.get_chromeos_release_milestone(
1432 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001433
1434
1435 def verify_cros_version_label(self):
1436 """ Make sure host's cros-version label match the actual image in dut.
1437
1438 Remove any cros-version: label that doesn't match that installed in
1439 the dut.
1440
1441 @param raise_error: Set to True to raise exception if any mismatch found
1442
1443 @raise error.AutoservError: If any mismatch between cros-version label
1444 and the build installed in dut is found.
1445 """
Prathmesh Prabhuce2da3a2019-10-04 11:54:51 -07001446 # crbug.com/1007333: This check is being removed.
1447 return True
Dan Shi549fb822015-03-24 18:01:11 -07001448
1449
Laurence Goodby778c9a42017-05-24 19:24:07 -07001450 def cleanup_services(self):
1451 """Reinitializes the device for cleanup.
1452
1453 Subclasses may override this to customize the cleanup method.
1454
1455 To indicate failure of the reset, the implementation may raise
1456 any of:
1457 error.AutoservRunError
1458 error.AutotestRunError
1459 FactoryImageCheckerException
1460
1461 @raises error.AutoservRunError
1462 @raises error.AutotestRunError
1463 @raises error.FactoryImageCheckerException
1464 """
1465 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001466 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001467
1468
beepsc87ff602013-07-31 21:53:00 -07001469 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001470 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001471 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001472 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001473 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001474 except (error.AutotestRunError, error.AutoservRunError,
1475 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001476 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001477
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001478 # cleanup routines, i.e. reboot the machine.
1479 super(CrosHost, self).cleanup()
1480
Simran Basi5e6339a2013-03-21 11:34:32 -07001481 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001482 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001483 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001484 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001485
1486
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001487 def reboot(self, **dargs):
1488 """
1489 This function reboots the site host. The more generic
1490 RemoteHost.reboot() performs sync and sleeps for 5
1491 seconds. This is not necessary for Chrome OS devices as the
1492 sync should be finished in a short time during the reboot
1493 command.
1494 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001495 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001496 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001497 dargs['reboot_cmd'] = ('sleep 1; '
1498 'reboot & sleep %d; '
1499 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001500 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001501 if 'fastsync' not in dargs:
1502 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001503
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001504 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001505 # Record who called us
1506 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001507 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001508 'dut_host_name' : self.hostname,
1509 'success' : True}
1510 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001511 'caller' : "%s:%s" % (orig.co_filename,
1512 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001513 'success' : True,
1514 'error' : ''}
1515
Vincent Palatin80780b22016-07-27 16:02:37 +02001516 t0 = time.time()
1517 try:
1518 super(CrosHost, self).reboot(**dargs)
1519 except Exception as e:
1520 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001521 metric_debug_fields['success'] = False
1522 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001523 raise
1524 finally:
1525 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001526 metrics.Counter(
1527 'chromeos/autotest/autoserv/reboot_count').increment(
1528 fields=metric_fields)
1529 metrics.Counter(
1530 'chromeos/autotest/autoserv/reboot_debug').increment(
1531 fields=metric_debug_fields)
1532 metrics.SecondsDistribution(
1533 'chromeos/autotest/autoserv/reboot_duration').add(
1534 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001535
1536
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001537 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001538 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001539 """
1540 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001541
1542 @param suspend_time: How long to suspend as integer seconds.
1543 @param suspend_cmd: Suspend command to execute.
1544 @param allow_early_resume: If False and if device resumes before
1545 |suspend_time|, throw an error.
1546
1547 @exception AutoservSuspendError Host resumed earlier than
1548 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001549 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001550
1551 if suspend_cmd is None:
1552 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001553 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001554 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001555 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001556 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1557 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001558
1559
Simran Basiec564392014-08-25 16:48:09 -07001560 def upstart_status(self, service_name):
1561 """Check the status of an upstart init script.
1562
1563 @param service_name: Service to look up.
1564
1565 @returns True if the service is running, False otherwise.
1566 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001567 return 'start/running' in self.run('status %s' % service_name,
1568 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001569
Tom Hughese9552342018-12-18 14:29:25 -08001570 def upstart_stop(self, service_name):
1571 """Stops an upstart job if it's running.
1572
1573 @param service_name: Service to stop
1574
1575 @returns True if service has been stopped or was already stopped
1576 False otherwise.
1577 """
1578 if not self.upstart_status(service_name):
1579 return True
1580
1581 result = self.run('stop %s' % service_name, ignore_status=True)
1582 if result.exit_status != 0:
1583 return False
1584 return True
1585
1586 def upstart_restart(self, service_name):
1587 """Restarts (or starts) an upstart job.
1588
1589 @param service_name: Service to start/restart
1590
1591 @returns True if service has been started/restarted, False otherwise.
1592 """
1593 cmd = 'start'
1594 if self.upstart_status(service_name):
1595 cmd = 'restart'
1596 cmd = cmd + ' %s' % service_name
1597 result = self.run(cmd)
1598 if result.exit_status != 0:
1599 return False
1600 return True
Simran Basiec564392014-08-25 16:48:09 -07001601
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001602 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001603 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001604
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001605 Tests for the following conditions:
1606 1. All conditions tested by the parent version of this
1607 function.
1608 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001609 3. Sufficient space in /mnt/stateful_partition/encrypted.
1610 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001611
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001612 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001613 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001614 default_kilo_inodes_required = CONFIG.get_config_value(
1615 'SERVER', 'kilo_inodes_required', type=int, default=100)
1616 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1617 kilo_inodes_required = CONFIG.get_config_value(
1618 'SERVER', 'kilo_inodes_required_%s' % board,
1619 type=int, default=default_kilo_inodes_required)
1620 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001621 self.check_diskspace(
1622 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001623 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001624 'SERVER', 'gb_diskspace_required', type=float,
1625 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001626 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1627 # Not all targets build with encrypted stateful support.
1628 if self.path_exists(encrypted_stateful_path):
1629 self.check_diskspace(
1630 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001631 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001632 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1633 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001634
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001635 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001636
beepsc87ff602013-07-31 21:53:00 -07001637 # Factory images don't run update engine,
1638 # goofy controls dbus on these DUTs.
1639 if not self._is_factory_image():
1640 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001641
Dan Shi549fb822015-03-24 18:01:11 -07001642 self.verify_cros_version_label()
1643
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001644
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001645 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1646 def wait_for_system_services(self):
1647 """Waits for system-services to be running.
1648
1649 Sometimes, update_engine will take a while to update firmware, so we
1650 should give this some time to finish. See crbug.com/765686#c38 for
1651 details.
1652 """
1653 if not self.upstart_status('system-services'):
1654 raise error.AutoservError('Chrome failed to reach login. '
1655 'System services not running.')
1656
1657
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001658 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001659 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001660 message = 'Beginning verify for host %s board %s model %s'
1661 info = self.host_info_store.get()
1662 message %= (self.hostname, info.board, info.model)
1663 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001664 try:
1665 self._repair_strategy.verify(self)
1666 except hosts.AutoservVerifyDependencyError as e:
1667 # We don't want flag a DUT as failed if only non-critical
1668 # verifier(s) failed during the repair.
1669 if e.is_critical():
1670 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001671
1672
Fang Deng96667ca2013-08-01 17:46:18 -07001673 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001674 connect_timeout=None, alive_interval=None,
1675 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001676 """Override default make_ssh_command to use options tuned for Chrome OS.
1677
1678 Tuning changes:
1679 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1680 connection failure. Consistency with remote_access.sh.
1681
Samuel Tan2ce155b2015-06-23 18:24:38 -07001682 - ServerAliveInterval=900; which causes SSH to ping connection every
1683 900 seconds. In conjunction with ServerAliveCountMax ensures
1684 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001685 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001686 the test completed successfully. Later increased from 180 seconds to
1687 900 seconds to account for tests where the DUT is suspended for
1688 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001689
1690 - ServerAliveCountMax=3; consistency with remote_access.sh.
1691
1692 - ConnectAttempts=4; reduce flakiness in connection errors;
1693 consistency with remote_access.sh.
1694
1695 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1696 Host keys change with every new installation, don't waste
1697 memory/space saving them.
1698
1699 - SSH protocol forced to 2; needed for ServerAliveInterval.
1700
1701 @param user User name to use for the ssh connection.
1702 @param port Port on the target host to use for ssh connection.
1703 @param opts Additional options to the ssh command.
1704 @param hosts_file Ignored.
1705 @param connect_timeout Ignored.
1706 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001707 @param alive_count_max Ignored.
1708 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001709 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001710 options = ' '.join([opts, '-o Protocol=2'])
1711 return super(CrosHost, self).make_ssh_command(
1712 user=user, port=port, opts=options, hosts_file='/dev/null',
1713 connect_timeout=30, alive_interval=900, alive_count_max=3,
1714 connection_attempts=4)
1715
1716
Jason Abeleb6f924f2013-11-13 16:01:54 -08001717 def syslog(self, message, tag='autotest'):
1718 """Logs a message to syslog on host.
1719
1720 @param message String message to log into syslog
1721 @param tag String tag prefix for syslog
1722
1723 """
1724 self.run('logger -t "%s" "%s"' % (tag, message))
1725
1726
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001727 def _ping_check_status(self, status):
1728 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001729
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001730 @param status Check the ping status against this value.
1731 @return True iff `status` and the result of ping are the same
1732 (i.e. both True or both False).
1733
1734 """
1735 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1736 return not (status ^ (ping_val == 0))
1737
1738 def _ping_wait_for_status(self, status, timeout):
1739 """Wait for the host to have a given status (UP or DOWN).
1740
1741 Status is checked by polling. Polling will not last longer
1742 than the number of seconds in `timeout`. The polling
1743 interval will be long enough that only approximately
1744 _PING_WAIT_COUNT polling cycles will be executed, subject
1745 to a maximum interval of about one minute.
1746
1747 @param status Waiting will stop immediately if `ping` of the
1748 host returns this status.
1749 @param timeout Poll for at most this many seconds.
1750 @return True iff the host status from `ping` matched the
1751 requested status at the time of return.
1752
1753 """
1754 # _ping_check_status() takes about 1 second, hence the
1755 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001756 # FIXME: if the ping command errors then _ping_check_status()
1757 # returns instantly. If timeout is also smaller than twice
1758 # _PING_WAIT_COUNT then the while loop below forks many
1759 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1760 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1761 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001762 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1763 end_time = time.time() + timeout
1764 while time.time() <= end_time:
1765 if self._ping_check_status(status):
1766 return True
1767 if poll_interval > 0:
1768 time.sleep(poll_interval)
1769
1770 # The last thing we did was sleep(poll_interval), so it may
1771 # have been too long since the last `ping`. Check one more
1772 # time, just to be sure.
1773 return self._ping_check_status(status)
1774
1775 def ping_wait_up(self, timeout):
1776 """Wait for the host to respond to `ping`.
1777
1778 N.B. This method is not a reliable substitute for
1779 `wait_up()`, because a host that responds to ping will not
1780 necessarily respond to ssh. This method should only be used
1781 if the target DUT can be considered functional even if it
1782 can't be reached via ssh.
1783
1784 @param timeout Minimum time to allow before declaring the
1785 host to be non-responsive.
1786 @return True iff the host answered to ping before the timeout.
1787
1788 """
1789 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001790
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001791 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001792 """Wait until the host no longer responds to `ping`.
1793
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001794 This function can be used as a slightly faster version of
1795 `wait_down()`, by avoiding potentially long ssh timeouts.
1796
1797 @param timeout Minimum time to allow for the host to become
1798 non-responsive.
1799 @return True iff the host quit answering ping before the
1800 timeout.
1801
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001802 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001803 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001804
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001805 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001806 """Wait for the client to enter low-power sleep mode.
1807
1808 The test for "is asleep" can't distinguish a system that is
1809 powered off; to confirm that the unit was asleep, it is
1810 necessary to force resume, and then call
1811 `test_wait_for_resume()`.
1812
1813 This function is expected to be called from a test as part
1814 of a sequence like the following:
1815
1816 ~~~~~~~~
1817 boot_id = host.get_boot_id()
1818 # trigger sleep on the host
1819 host.test_wait_for_sleep()
1820 # trigger resume on the host
1821 host.test_wait_for_resume(boot_id)
1822 ~~~~~~~~
1823
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001824 @param sleep_timeout time limit in seconds to allow the host sleep.
1825
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001826 @exception TestFail The host did not go to sleep within
1827 the allowed time.
1828 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001829 if sleep_timeout is None:
1830 sleep_timeout = self.SLEEP_TIMEOUT
1831
1832 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001833 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001834 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001835
1836
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001837 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001838 """Wait for the client to resume from low-power sleep mode.
1839
1840 The `old_boot_id` parameter should be the value from
1841 `get_boot_id()` obtained prior to entering sleep mode. A
1842 `TestFail` exception is raised if the boot id changes.
1843
1844 See @ref test_wait_for_sleep for more on this function's
1845 usage.
1846
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001847 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001848 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001849 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001850
1851 @exception TestFail The host did not respond within the
1852 allowed time.
1853 @exception TestFail The host responded, but the boot id test
1854 indicated a reboot rather than a sleep
1855 cycle.
1856 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001857 if resume_timeout is None:
1858 resume_timeout = self.RESUME_TIMEOUT
1859
1860 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001861 raise error.TestFail(
1862 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001863 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001864 else:
1865 new_boot_id = self.get_boot_id()
1866 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001867 logging.error('client rebooted (old boot %s, new boot %s)',
1868 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001869 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001870 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001871
1872
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001873 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001874 """Wait for the client to shut down.
1875
1876 The test for "has shut down" can't distinguish a system that
1877 is merely asleep; to confirm that the unit was down, it is
1878 necessary to force boot, and then call test_wait_for_boot().
1879
1880 This function is expected to be called from a test as part
1881 of a sequence like the following:
1882
1883 ~~~~~~~~
1884 boot_id = host.get_boot_id()
1885 # trigger shutdown on the host
1886 host.test_wait_for_shutdown()
1887 # trigger boot on the host
1888 host.test_wait_for_boot(boot_id)
1889 ~~~~~~~~
1890
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001891 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001892 @exception TestFail The host did not shut down within the
1893 allowed time.
1894 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001895 if shutdown_timeout is None:
1896 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1897
1898 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001899 raise error.TestFail(
1900 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001901 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001902
1903
1904 def test_wait_for_boot(self, old_boot_id=None):
1905 """Wait for the client to boot from cold power.
1906
1907 The `old_boot_id` parameter should be the value from
1908 `get_boot_id()` obtained prior to shutting down. A
1909 `TestFail` exception is raised if the boot id does not
1910 change. The boot id test is omitted if `old_boot_id` is not
1911 specified.
1912
1913 See @ref test_wait_for_shutdown for more on this function's
1914 usage.
1915
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001916 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001917 shut down.
1918
1919 @exception TestFail The host did not respond within the
1920 allowed time.
1921 @exception TestFail The host responded, but the boot id test
1922 indicated that there was no reboot.
1923 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001924 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001925 raise error.TestFail(
1926 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001927 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001928 elif old_boot_id:
1929 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001930 logging.error('client not rebooted (boot %s)',
1931 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001932 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001933 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001934
1935
1936 @staticmethod
1937 def check_for_rpm_support(hostname):
1938 """For a given hostname, return whether or not it is powered by an RPM.
1939
Simran Basi1df55112013-09-06 11:25:09 -07001940 @param hostname: hostname to check for rpm support.
1941
Simran Basid5e5e272012-09-24 15:23:59 -07001942 @return None if this host does not follows the defined naming format
1943 for RPM powered DUT's in the lab. If it does follow the format,
1944 it returns a regular expression MatchObject instead.
1945 """
Fang Dengbaff9082015-01-06 13:46:15 -08001946 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001947
1948
1949 def has_power(self):
1950 """For this host, return whether or not it is powered by an RPM.
1951
1952 @return True if this host is in the CROS lab and follows the defined
1953 naming format.
1954 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001955 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001956
1957
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001958 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001959 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001960
1961 @param state Specifies which power state to set to DUT
1962 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001963 use. By default "RPM" or "CCD" will be used based
1964 on servo type. Valid values from
1965 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001966
1967 """
1968 ACCEPTABLE_STATES = ['ON', 'OFF']
1969
Garry Wang5e5538a2019-04-08 15:36:18 -07001970 if not power_method:
1971 power_method = self.get_default_power_method()
1972
1973 state = state.upper()
1974 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001975 raise error.TestError('State must be one of: %s.'
1976 % (ACCEPTABLE_STATES,))
1977
1978 if power_method == self.POWER_CONTROL_SERVO:
1979 logging.info('Setting servo port J10 to %s', state)
1980 self.servo.set('prtctl3_pwren', state.lower())
1981 time.sleep(self._USB_POWER_TIMEOUT)
1982 elif power_method == self.POWER_CONTROL_MANUAL:
1983 logging.info('You have %d seconds to set the AC power to %s.',
1984 self._POWER_CYCLE_TIMEOUT, state)
1985 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001986 elif power_method == self.POWER_CONTROL_CCD:
1987 servo_role = 'src' if state == 'ON' else 'snk'
1988 logging.info('servo ccd power pass through detected,'
1989 ' changing servo_role to %s.', servo_role)
1990 self.servo.set_servo_v4_role(servo_role)
1991 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001992 # Make sure we don't leave DUT with no power(servo_role=snk)
1993 # when DUT is not pingable, as we raise a exception here
1994 # that may break a power cycle in the middle.
1995 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001996 raise error.AutoservError(
1997 'DUT failed to regain network connection after %d seconds.'
1998 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001999 else:
2000 if not self.has_power():
2001 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002002 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07002003 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07002004
2005
Garry Wang5e5538a2019-04-08 15:36:18 -07002006 def power_off(self, power_method=None):
2007 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002008
2009 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002010 use. By default "RPM" or "CCD" will be used based
2011 on servo type. Valid values from
2012 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002013
2014 """
2015 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002016
2017
Garry Wang5e5538a2019-04-08 15:36:18 -07002018 def power_on(self, power_method=None):
2019 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002020
2021 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002022 use. By default "RPM" or "CCD" will be used based
2023 on servo type. Valid values from
2024 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002025
2026 """
2027 self._set_power('ON', power_method)
2028
2029
Garry Wang5e5538a2019-04-08 15:36:18 -07002030 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002031 """Cycle power to this host by turning it OFF, then ON.
2032
2033 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002034 use. By default "RPM" or "CCD" will be used based
2035 on servo type. Valid values from
2036 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002037
2038 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002039 if not power_method:
2040 power_method = self.get_default_power_method()
2041
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002042 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002043 self.POWER_CONTROL_MANUAL,
2044 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002045 self.power_off(power_method=power_method)
2046 time.sleep(self._POWER_CYCLE_TIMEOUT)
2047 self.power_on(power_method=power_method)
2048 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002049 self._add_rpm_changed_tag()
2050 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002051
2052
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002053 def get_platform_from_fwid(self):
2054 """Determine the platform from the crossystem fwid.
2055
2056 @returns a string representing this host's platform.
2057 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002058 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002059 crossystem = utils.Crossystem(self)
2060 crossystem.init()
2061 # Extract fwid value and use the leading part as the platform id.
2062 # fwid generally follow the format of {platform}.{firmware version}
2063 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2064 platform = crossystem.fwid().split('.')[0].lower()
2065 # Newer platforms start with 'Google_' while the older ones do not.
2066 return platform.replace('google_', '')
2067
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002068
Simran Basic6f1f7a2012-10-16 10:47:46 -07002069 def get_platform(self):
2070 """Determine the correct platform label for this host.
2071
2072 @returns a string representing this host's platform.
2073 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002074 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2075 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002076 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002077 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002078 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002079 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002080
2081
Greg Edelstona7b05d12020-04-01 16:00:51 -06002082 def get_model_from_cros_config(self):
2083 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002084
Greg Edelstona7b05d12020-04-01 16:00:51 -06002085 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002086 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002087 return cros_config.call_cros_config_get_output('/ name',
2088 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002089
2090
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002091 def get_architecture(self):
2092 """Determine the correct architecture label for this host.
2093
2094 @returns a string representing this host's architecture.
2095 """
2096 crossystem = utils.Crossystem(self)
2097 crossystem.init()
2098 return crossystem.arch()
2099
2100
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002101 def get_chrome_version(self):
2102 """Gets the Chrome version number and milestone as strings.
2103
2104 Invokes "chrome --version" to get the version number and milestone.
2105
2106 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2107 current Chrome version number as a string (in the form "W.X.Y.Z")
2108 and "milestone" is the first component of the version number
2109 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2110 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2111 of "chrome --version" and the milestone will be the empty string.
2112
2113 """
MK Ryu35d661e2014-09-25 17:44:10 -07002114 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002115 return utils.parse_chrome_version(version_string)
2116
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002117
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002118 def get_ec_version(self):
2119 """Get the ec version as strings.
2120
2121 @returns a string representing this host's ec version.
2122 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002123 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002124 result = self.run(command, ignore_status=True)
2125 if result.exit_status != 0:
2126 return ''
2127 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002128
2129
2130 def get_firmware_version(self):
2131 """Get the firmware version as strings.
2132
2133 @returns a string representing this host's firmware version.
2134 """
2135 crossystem = utils.Crossystem(self)
2136 crossystem.init()
2137 return crossystem.fwid()
2138
2139
2140 def get_hardware_revision(self):
2141 """Get the hardware revision as strings.
2142
2143 @returns a string representing this host's hardware revision.
2144 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002145 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002146 result = self.run(command, ignore_status=True)
2147 if result.exit_status != 0:
2148 return ''
2149 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002150
2151
2152 def get_kernel_version(self):
2153 """Get the kernel version as strings.
2154
2155 @returns a string representing this host's kernel version.
2156 """
2157 return self.run('uname -r').stdout.strip()
2158
2159
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002160 def get_cpu_name(self):
2161 """Get the cpu name as strings.
2162
2163 @returns a string representing this host's cpu name.
2164 """
2165
2166 # Try get cpu name from device tree first
2167 if self.path_exists('/proc/device-tree/compatible'):
2168 command = ' | '.join(
2169 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2170 'tail -1'])
2171 return self.run(command).stdout.strip().replace(',', ' ')
2172
2173 # Get cpu name from uname -p
2174 command = 'uname -p'
2175 ret = self.run(command).stdout.strip()
2176
2177 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2178 # Try get cpu name from /proc/cpuinfo instead
2179 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2180 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2181 self = self.run(command).stdout.strip()
2182
2183 # Remove bloat from CPU name, for example
2184 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2185 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2186 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2187 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2188 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2189 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2190
2191
2192 def get_screen_resolution(self):
2193 """Get the screen(s) resolution as strings.
2194 In case of more than 1 monitor, return resolution for each monitor
2195 separate with plus sign.
2196
2197 @returns a string representing this host's screen(s) resolution.
2198 """
2199 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2200 ret = self.run(command, ignore_status=True)
2201 # We might have Chromebox without a screen
2202 if ret.exit_status != 0:
2203 return ''
2204 return ret.stdout.strip().replace('\n', '+')
2205
2206
2207 def get_mem_total_gb(self):
2208 """Get total memory available in the system in GiB (2^20).
2209
2210 @returns an integer representing total memory
2211 """
2212 mem_total_kb = self.read_from_meminfo('MemTotal')
2213 kb_in_gb = float(2 ** 20)
2214 return int(round(mem_total_kb / kb_in_gb))
2215
2216
2217 def get_disk_size_gb(self):
2218 """Get size of disk in GB (10^9)
2219
2220 @returns an integer representing size of disk, 0 in Error Case
2221 """
2222 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2223 result = self.run(command, ignore_status=True)
2224 if result.exit_status != 0:
2225 return 0
2226 _, _, block, _ = re.split(r' +', result.stdout.strip())
2227 byte_per_block = 1024.0
2228 disk_kb_in_gb = 1e9
2229 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2230
2231
2232 def get_battery_size(self):
2233 """Get size of battery in Watt-hour via sysfs
2234
2235 This method assumes that battery support voltage_min_design and
2236 charge_full_design sysfs.
2237
2238 @returns a float representing Battery size, 0 if error.
2239 """
2240 # sysfs report data in micro scale
2241 battery_scale = 1e6
2242
2243 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2244 result = self.run(command, ignore_status=True)
2245 if result.exit_status != 0:
2246 return 0
2247 voltage = float(result.stdout.strip()) / battery_scale
2248
2249 command = 'cat /sys/class/power_supply/*/charge_full_design'
2250 result = self.run(command, ignore_status=True)
2251 if result.exit_status != 0:
2252 return 0
2253 amphereHour = float(result.stdout.strip()) / battery_scale
2254
2255 return voltage * amphereHour
2256
2257
2258 def get_low_battery_shutdown_percent(self):
2259 """Get the percent-based low-battery shutdown threshold.
2260
2261 @returns a float representing low-battery shutdown percent, 0 if error.
2262 """
2263 ret = 0.0
2264 try:
2265 command = 'check_powerd_config --low_battery_shutdown_percent'
2266 ret = float(self.run(command).stdout)
2267 except error.CmdError:
2268 logging.debug("Can't run %s", command)
2269 except ValueError:
2270 logging.debug("Didn't get number from %s", command)
2271
2272 return ret
2273
2274
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002275 def has_hammer(self):
2276 """Check whether DUT has hammer device or not.
2277
2278 @returns boolean whether device has hammer or not
2279 """
2280 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2281 return self.run(command, ignore_status=True).exit_status == 0
2282
2283
Niranjan Kumar34618872017-05-31 12:57:09 -07002284 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002285 """Returns True if the specified switch was provided to Chrome.
2286
2287 @param switch The chrome switch to search for.
2288 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002289
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002290 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2291 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002292
2293
2294 def oobe_triggers_update(self):
2295 """Returns True if this host has an OOBE flow during which
2296 it will perform an update check and perhaps an update.
2297 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2298 As more such flows are developed, code handling them needs
2299 to be added here.
2300
2301 @return Boolean indicating whether this host's OOBE triggers an update.
2302 """
2303 return self.is_chrome_switch_present(
2304 '--enterprise-enable-zero-touch-enrollment=hands-off')
2305
2306
Kevin Chenga2619dc2016-03-28 11:42:08 -07002307 # TODO(kevcheng): change this to just return the board without the
2308 # 'board:' prefix and fix up all the callers. Also look into removing the
2309 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002310 def get_board(self):
2311 """Determine the correct board label for this host.
2312
2313 @returns a string representing this host's board.
2314 """
2315 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2316 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002317 return (ds_constants.BOARD_PREFIX +
2318 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002319
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002320 def get_channel(self):
2321 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002322
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002323 @returns: a string represeting this host's build channel.
2324 (stable, dev, beta). None on fail.
2325 """
2326 return lsbrelease_utils.get_chromeos_channel(
2327 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002328
Kevin Chenga328da62016-03-31 10:49:04 -07002329 def get_power_supply(self):
2330 """
2331 Determine what type of power supply the host has
2332
2333 @returns a string representing this host's power supply.
2334 'power:battery' when the device has a battery intended for
2335 extended use
2336 'power:AC_primary' when the device has a battery not intended
2337 for extended use (for moving the machine, etc)
2338 'power:AC_only' when the device has no battery at all.
2339 """
2340 psu = self.run(command='mosys psu type', ignore_status=True)
2341 if psu.exit_status:
2342 # The psu command for mosys is not included for all platforms. The
2343 # assumption is that the device will have a battery if the command
2344 # is not found.
2345 return 'power:battery'
2346
2347 psu_str = psu.stdout.strip()
2348 if psu_str == 'unknown':
2349 return None
2350
2351 return 'power:%s' % psu_str
2352
2353
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002354 def has_battery(self):
2355 """Determine if DUT has a battery.
2356
2357 Returns:
2358 Boolean, False if known not to have battery, True otherwise.
2359 """
2360 rv = True
2361 power_supply = self.get_power_supply()
2362 if power_supply == 'power:battery':
2363 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2364 board_type = self.get_board_type()
2365 if board_type in _NO_BATTERY_BOARD_TYPE:
2366 logging.warn('Do NOT believe type %s has battery. '
2367 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002368 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002369 ignore_status=True)
2370 logging.debug(psu)
2371 rv = False
2372 elif power_supply == 'power:AC_only':
2373 rv = False
2374
2375 return rv
2376
2377
Kevin Chenga328da62016-03-31 10:49:04 -07002378 def get_servo(self):
2379 """Determine if the host has a servo attached.
2380
2381 If the host has a working servo attached, it should have a servo label.
2382
2383 @return: string 'servo' if the host has servo attached. Otherwise,
2384 returns None.
2385 """
2386 return 'servo' if self._servo_host else None
2387
2388
Kevin Chenga328da62016-03-31 10:49:04 -07002389 def has_internal_display(self):
2390 """Determine if the device under test is equipped with an internal
2391 display.
2392
2393 @return: 'internal_display' if one is present; None otherwise.
2394 """
2395 from autotest_lib.client.cros.graphics import graphics_utils
2396 from autotest_lib.client.common_lib import utils as common_utils
2397
2398 def __system_output(cmd):
2399 return self.run(cmd).stdout
2400
2401 def __read_file(remote_path):
2402 return self.run('cat %s' % remote_path).stdout
2403
2404 # Hijack the necessary client functions so that we can take advantage
2405 # of the client lib here.
2406 # FIXME: find a less hacky way than this
2407 original_system_output = utils.system_output
2408 original_read_file = common_utils.read_file
2409 utils.system_output = __system_output
2410 common_utils.read_file = __read_file
2411 try:
2412 return ('internal_display' if graphics_utils.has_internal_display()
2413 else None)
2414 finally:
2415 utils.system_output = original_system_output
2416 common_utils.read_file = original_read_file
2417
2418
Dan Shi85276d42014-04-08 22:11:45 -07002419 def is_boot_from_usb(self):
2420 """Check if DUT is boot from USB.
2421
2422 @return: True if DUT is boot from usb.
2423 """
2424 device = self.run('rootdev -s -d').stdout.strip()
2425 removable = int(self.run('cat /sys/block/%s/removable' %
2426 os.path.basename(device)).stdout.strip())
2427 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002428
2429
2430 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002431 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002432
2433 @param key: meminfo requested
2434
2435 @return the memory value as a string
2436
2437 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002438 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2439 logging.debug('%s', meminfo)
2440 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002441
2442
Rohit Makasana98e696f2016-06-03 18:48:10 -07002443 def get_cpu_arch(self):
2444 """Returns CPU arch of the device.
2445
2446 @return CPU architecture of the DUT.
2447 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002448 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002449 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2450 ignore_status=True).stdout:
2451 return 'x86_64'
2452 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2453 ignore_status=True).stdout:
2454 return 'arm'
2455 return 'i386'
2456
2457
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002458 def get_board_type(self):
2459 """
2460 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002461 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2462
2463 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002464 """
Danny Chan471a8d12015-08-18 14:57:41 -07002465 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2466 ignore_status=True).stdout
2467 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002468 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002469 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002470
2471
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002472 def get_arc_version(self):
2473 """Return ARC version installed on the DUT.
2474
2475 @returns ARC version as string if the CrOS build has ARC, else None.
2476 """
2477 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2478 ignore_status=True).stdout
2479 if arc_version:
2480 return arc_version.split('=')[-1].strip()
2481 return None
2482
2483
Gilad Arnolda76bef02015-09-29 13:55:15 -07002484 def get_os_type(self):
2485 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002486
2487
Kevin Chenga2619dc2016-03-28 11:42:08 -07002488 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002489 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002490 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002491
2492
2493 def get_default_power_method(self):
2494 """
2495 Get the default power method for power_on/off/cycle() methods.
2496 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2497 """
2498 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002499 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002500 if self.servo and self.servo.supports_built_in_pd_control():
2501 self._default_power_method = self.POWER_CONTROL_CCD
2502 else:
2503 logging.debug('Either servo is unitialized or the servo '
2504 'setup does not support pd controls. Falling '
2505 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002506 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002507
2508
2509 def find_usb_devices(self, idVendor, idProduct):
2510 """
2511 Get usb device sysfs name for specific device.
2512
2513 @param idVendor Vendor ID to search in sysfs directory.
2514 @param idProduct Product ID to search in sysfs directory.
2515
2516 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2517 """
2518 # Look for matching file and cut at position 7 to get dir name.
2519 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2520
2521 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2522 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2523
2524 # Use uniq -d to print duplicate line from both command
2525 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2526
2527 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2528
2529
2530 def bind_usb_device(self, usb_node):
2531 """
2532 Bind usb device
2533
2534 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2535 """
2536 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2537 self.run(cmd, ignore_status=True)
2538
2539
2540 def unbind_usb_device(self, usb_node):
2541 """
2542 Unbind usb device
2543
2544 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2545 """
2546 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2547 self.run(cmd, ignore_status=True)
2548
2549
2550 def get_wlan_ip(self):
2551 """
2552 Get ip address of wlan interface.
2553
2554 @return ip address of wlan or empty string if wlan is not connected.
2555 """
2556 cmds = [
2557 'iw dev', # List wlan physical device
2558 'grep Interface', # Grep only interface name
2559 'cut -f 2 -d" "', # Cut the name part
2560 'xargs ifconfig', # Feed it to ifconfig to get ip
2561 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2562 'cut -f 2 -d " "' # Cut the ip part
2563 ]
2564 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002565
2566 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2567 """
2568 Connect to wifi network
2569
2570 @param ssid SSID of the wifi network.
2571 @param passphrase Passphrase of the wifi network. None if not existed.
2572 @param security Security of the wifi network. Default to "psk" if
2573 passphase is given without security. Possible values
2574 are "none", "psk", "802_1x".
2575
2576 @return True if succeed, False if not.
2577 """
2578 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2579 if passphrase:
2580 cmd += ' ' + passphrase
2581 if security:
2582 cmd += ' ' + security
2583 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002584
2585 def get_device_repair_state(self):
2586 """Get device repair state"""
2587 return self._device_repair_state
2588
2589 def set_device_repair_state(self, state):
2590 """Set device repair state.
2591
2592 The special device state will be written to the 'dut_state.repair'
2593 file in result directory. The file will be read by Lucifer.
2594 """
2595 if self.job:
2596 target = os.path.join(self.job.resultdir, 'dut_state.repair')
2597 common_utils.open_write_close(target, state)
2598 else:
2599 logging.debug('Cannot write the device state due missing info '
2600 'about result dir.')
2601 self._device_repair_state = state
2602
2603 def try_set_device_need_manual_repair(self):
2604 """Check if device require manual attention to be fixed.
2605
2606 The state 'needs_manual_repair' can be set when auto repair cannot
2607 fix the device due hardware or cable issues.
2608 """
2609 # ignore the logic if state present
2610 # state can be set by any cros repair actions
2611 if self.get_device_repair_state():
2612 return
2613
2614 # set need manual attention if servo has hardware issue
2615 servo_state_required_manual_fix = [
2616 servo_constants.SERVO_STATE_NOT_CONNECTED,
2617 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
2618 servo_constants.SERVO_STATE_LID_OPEN_FAILED,
2619 servo_constants.SERVO_STATE_BAD_RIBBON_CABLE,
2620 servo_constants.SERVO_STATE_EC_BROKEN,
2621 ]
2622 if self.get_servo_state() in servo_state_required_manual_fix:
Otabek Kasimovde8eea32020-07-01 12:12:22 -07002623 data = {'host': self.hostname,
2624 'state': DEVICE_STATE_NEEDS_MANUAL_REPAIR}
Otabek Kasimov6825b762020-06-23 23:42:44 -07002625 metrics.Counter(
2626 'chromeos/autotest/repair/special_dut_state'
2627 ).increment(fields=data)
2628 # TODO (otabek) unblock when be sure that we do not have flakiness
2629 # self.set_device_repair_state(DEVICE_STATE_NEEDS_MANUAL_REPAIR)