blob: b182ad17c4c1aeb93767410effd1a45b79e16c1f [file] [log] [blame]
J. Richard Barnette24adbf42012-04-11 15:04:53 -07001# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Dale Curtisaa5eedb2011-08-23 16:18:52 -07002# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
J. Richard Barnette1d78b012012-05-15 13:56:30 -07005import logging
Dan Shi0f466e82013-02-22 15:44:58 -08006import os
Simran Basid5e5e272012-09-24 15:23:59 -07007import re
Vincent Palatindf2372c2016-10-07 17:03:00 +02008import sys
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07009import time
10
mussa584b4462014-06-20 15:13:28 -070011import common
J. Richard Barnette45e93de2012-04-11 17:24:15 -070012from autotest_lib.client.bin import utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070013from autotest_lib.client.common_lib import autotemp
Richard Barnette0c73ffc2012-11-19 15:21:18 -080014from autotest_lib.client.common_lib import error
15from autotest_lib.client.common_lib import global_config
J. Richard Barnette91137f02016-03-10 16:52:26 -080016from autotest_lib.client.common_lib import hosts
Dan Shi549fb822015-03-24 18:01:11 -070017from autotest_lib.client.common_lib import lsbrelease_utils
Otabek Kasimov6825b762020-06-23 23:42:44 -070018from autotest_lib.client.common_lib import utils as common_utils
Greg Edelstona7b05d12020-04-01 16:00:51 -060019from autotest_lib.client.common_lib.cros import cros_config
Richard Barnette03a0c132012-11-05 12:40:35 -080020from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070021from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000022from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080023from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080024from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070025from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070026from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050027from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070028from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070029from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070030from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070031from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080032from autotest_lib.server.hosts import chameleon_host
Otabek Kasimov832d9162020-07-27 19:24:57 -070033from autotest_lib.server.hosts import cros_constants
Richard Barnetted31580e2018-05-14 19:58:00 +000034from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080035from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070036from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070037from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070038from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080039from autotest_lib.site_utils.rpm_control_system import rpm_client
Otabek Kasimov808cd832020-05-28 18:27:46 -070040from autotest_lib.site_utils.admin_audit import constants as audit_const
Simran Basid5e5e272012-09-24 15:23:59 -070041
Simran Basi382506b2016-09-13 14:58:15 -070042# In case cros_host is being ran via SSP on an older Moblab version with an
43# older chromite version.
44try:
45 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080046except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080047 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080048
Simran Basid5e5e272012-09-24 15:23:59 -070049
Dan Shib8540a52015-07-16 14:18:23 -070050CONFIG = global_config.global_config
51
beepsc87ff602013-07-31 21:53:00 -070052class FactoryImageCheckerException(error.AutoservError):
53 """Exception raised when an image is a factory image."""
54 pass
55
56
Fang Deng0ca40e22013-08-27 17:47:44 -070057class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070058 """Chromium OS specific subclass of Host."""
59
Simran Basi5ace6f22016-01-06 17:30:44 -080060 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
61
Scott Zawalski62bacae2013-03-05 10:40:32 -050062 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070063
Richard Barnette03a0c132012-11-05 12:40:35 -080064 # Timeout values (in seconds) associated with various Chrome OS
65 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070066 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080067 # In general, a good rule of thumb is that the timeout can be up
68 # to twice the typical measured value on the slowest platform.
69 # The times here have not necessarily been empirically tested to
70 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070071 #
72 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080073 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
74 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080075 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070076 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080077 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070078 # screen delay, time to start the network on the DUT, and the
79 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070080 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080081 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080082 # network.
beepsf079cfb2013-09-18 17:49:51 -070083 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080084 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
85 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070086
87 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080088 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080089 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070090 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070091 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070092 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080093 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070094
Dan Shica503482015-03-30 17:23:25 -070095 # Minimum OS version that supports server side packaging. Older builds may
96 # not have server side package built or with Autotest code change to support
97 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070098 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -070099 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700100
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800101 # REBOOT_TIMEOUT: How long to wait for a reboot.
102 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700103 # We have a long timeout to ensure we don't flakily fail due to other
104 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700105 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
106 # return from reboot' bug is solved.
107 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700108
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800109 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
110 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700111 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
112 # since changing servo role will reset USB state
113 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800114 _USB_POWER_TIMEOUT = 5
115 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700116 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800117
Fang Dengdeba14f2014-11-14 11:54:09 -0800118 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
119 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700120
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800121 # Constants used in ping_wait_up() and ping_wait_down().
122 #
123 # _PING_WAIT_COUNT is the approximate number of polling
124 # cycles to use when waiting for a host state change.
125 #
126 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
127 # for arguments to the internal _ping_wait_for_status()
128 # method.
129 _PING_WAIT_COUNT = 40
130 _PING_STATUS_DOWN = False
131 _PING_STATUS_UP = True
132
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800133 # Allowed values for the power_method argument.
134
Garry Wang5e5538a2019-04-08 15:36:18 -0700135 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
136 # DUTs except those with servo_v4 CCD.
137 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
138 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800139 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
140 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
141 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700142 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800143 POWER_CONTROL_SERVO = 'servoj10'
144 POWER_CONTROL_MANUAL = 'manual'
145
146 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700147 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800148 POWER_CONTROL_SERVO,
149 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800150
Simran Basi5e6339a2013-03-21 11:34:32 -0700151 _RPM_OUTLET_CHANGED = 'outlet_changed'
152
Dan Shi9cb0eec2014-06-03 09:04:50 -0700153 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700154 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700155 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700156
Brent Peterson1cb623a2020-01-09 13:14:28 -0800157 # Regular expression for extracting EC version string
158 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
159
160 # Regular expression for extracting BIOS version string
161 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
162
Brent Petersonc70a1832020-01-24 15:54:35 -0800163 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700164 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800165
J. Richard Barnette964fba02012-10-24 17:34:29 -0700166 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800167 def check_host(host, timeout=10):
168 """
169 Check if the given host is a chrome-os host.
170
171 @param host: An ssh host representing a device.
172 @param timeout: The timeout for the run command.
173
174 @return: True if the host device is chromeos.
175
beeps46dadc92013-11-07 14:07:10 -0800176 """
177 try:
Allen Liad719c12017-06-27 23:48:04 +0000178 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700179 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700180 '! grep -q moblab /etc/lsb-release && '
181 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700182 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700183 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000184 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700185 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
186 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800187 return not (
188 lsbrelease_utils.is_jetstream(
189 lsb_release_content=lsb_release_content) or
190 lsbrelease_utils.is_gce_board(
191 lsb_release_content=lsb_release_content))
192
beeps46dadc92013-11-07 14:07:10 -0800193 except (error.AutoservRunError, error.AutoservSSHTimeout):
194 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700195
196 return False
beeps46dadc92013-11-07 14:07:10 -0800197
198
199 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800200 def get_chameleon_arguments(args_dict):
201 """Extract chameleon options from `args_dict` and return the result.
202
203 Recommended usage:
204 ~~~~~~~~
205 args_dict = utils.args_to_dict(args)
206 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
207 host = hosts.create_host(machine, chameleon_args=chameleon_args)
208 ~~~~~~~~
209
210 @param args_dict Dictionary from which to extract the chameleon
211 arguments.
212 """
Sam McNally66594ca2019-12-09 12:45:44 +1100213 chameleon_args = {key: args_dict[key]
214 for key in ('chameleon_host', 'chameleon_port')
215 if key in args_dict}
216 if 'chameleon_ssh_port' in args_dict:
217 chameleon_args['port'] = int(args_dict['chameleon_ssh_port'])
218 return chameleon_args
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800219
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800220 @staticmethod
221 def get_btpeer_arguments(args_dict):
222 """Extract btpeer options from `args_dict` and return the result.
223
224 This is used to parse details of Bluetooth peer.
225 Recommended usage:
226 ~~~~~~~~
227 args_dict = utils.args_to_dict(args)
228 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700229 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800230 ~~~~~~~~
231
232 @param args_dict: Dictionary from which to extract the btpeer
233 arguments.
234 """
235 if 'btpeer_host_list' in args_dict:
236 result = []
237 for btpeer in args_dict['btpeer_host_list'].split(','):
238 result.append({key: value for key,value in
239 zip(('btpeer_host','btpeer_port'),
240 btpeer.split(':'))})
241 return result
242 else:
243 return {key: args_dict[key]
244 for key in ('btpeer_host', 'btpeer_port')
245 if key in args_dict}
246
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800247
248 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700249 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800250 """Extract chameleon options from `args_dict` and return the result.
251
252 Recommended usage:
253 ~~~~~~~~
254 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700255 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
256 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800257 ~~~~~~~~
258
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700259 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800260 arguments.
261 """
Allen Li083866b2016-08-18 10:07:10 -0700262 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700263 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700264 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800265
266
267 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800268 def get_servo_arguments(args_dict):
269 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800270
271 Recommended usage:
272 ~~~~~~~~
273 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700274 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800275 host = hosts.create_host(machine, servo_args=servo_args)
276 ~~~~~~~~
277
278 @param args_dict Dictionary from which to extract the servo
279 arguments.
280 """
Garry Wang11b5e872020-03-11 15:14:08 -0700281 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
282 servo_constants.SERVO_PORT_ATTR,
283 servo_constants.SERVO_BOARD_ATTR,
284 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200285 servo_args = {key: args_dict[key]
286 for key in servo_attrs
287 if key in args_dict}
288 return (
289 None
Garry Wang11b5e872020-03-11 15:14:08 -0700290 if servo_constants.SERVO_HOST_ATTR in servo_args
291 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200292 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700293
J. Richard Barnette964fba02012-10-24 17:34:29 -0700294
J. Richard Barnette91137f02016-03-10 16:52:26 -0800295 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700296 pdtester_args=None, try_lab_servo=False,
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700297 try_servo_repair=False, btpeer_args=[],
J. Richard Barnette91137f02016-03-10 16:52:26 -0800298 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700299 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800300 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700301
Fang Denge545abb2014-12-30 18:43:47 -0800302 This method will attempt to create the test-assistant object
303 (chameleon/servo) when it is needed by the test. Check
304 the docstring of chameleon_host.create_chameleon_host and
305 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700306
Fang Denge545abb2014-12-30 18:43:47 -0800307 @param hostname: Hostname of the dut.
308 @param chameleon_args: A dictionary that contains args for creating
309 a ChameleonHost. See chameleon_host for details.
310 @param servo_args: A dictionary that contains args for creating
311 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700312 @param try_lab_servo: When true, indicates that an attempt should
313 be made to create a ServoHost for a DUT in
314 the test lab, even if not required by
315 `servo_args`. See servo_host for details.
316 @param try_servo_repair: If a servo host is created, check it
317 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800318 See servo_host for details.
319 @param ssh_verbosity_flag: String, to pass to the ssh command to control
320 verbosity.
321 @param ssh_options: String, other ssh options to pass to the ssh
322 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700323 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700324 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700325 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800326 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700327 # hold special dut_state for repair process
328 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700329 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700330 # self.env is a dictionary of environment variable settings
331 # to be exported for commands run on the host.
332 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
333 # errors that might happen.
334 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700335 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700336 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700337 _servo_host, servo_state = servo_host.create_servo_host(
338 dut=self,
339 servo_args=servo_args,
340 try_lab_servo=try_lab_servo,
341 try_servo_repair=try_servo_repair,
342 dut_host_info=self.host_info_store.get())
343 self.set_servo_host(_servo_host, servo_state)
Garry Wang5e5538a2019-04-08 15:36:18 -0700344 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700345
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800346 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800347 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700348 dut=self.hostname,
349 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800350 if self._chameleon_host:
351 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800352 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800353 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700354
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700355 # Initialize Bluetooth peers.
356 try:
357 self.initialize_btpeer(btpeer_args)
358 except Exception as e:
359 logging.error('Exception %s in initialize_btpeer', str(e))
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800360
howardchung83e55272019-08-08 14:08:05 +0800361 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700362 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700363 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800364
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700365 if self._pdtester_host:
366 self.pdtester_servo = self._pdtester_host.get_servo()
367 logging.info('pdtester_servo: %r', self.pdtester_servo)
368 # Create the pdtester object used to access the ec uart
369 self.pdtester = pdtester.PDTester(self.pdtester_servo,
370 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800371 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700372 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800373
Fang Deng5d518f42013-08-02 14:04:32 -0700374
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800375 def initialize_btpeer(self, btpeer_args):
376 """ Initialize the Bluetooth peers
377
378 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
379 is chameleon host on Raspberry Pi.
380 @param btpeer_args: A dictionary that contains args for creating
381 a ChameleonHost. See chameleon_host for details.
382
383 """
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700384 #TODO (b:142486063) Remove the try..except
385 try:
386 self._btpeer_host_list = []
387 self.btpeer_list = []
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800388 self.btpeer = None
389
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700390 if type(btpeer_args) is list:
391 btpeer_args_list = btpeer_args
392 else:
393 btpeer_args_list = [btpeer_args]
394
395 self._btpeer_host_list = chameleon_host.create_btpeer_host(
396 dut=self.hostname, btpeer_args_list=btpeer_args_list)
397 logging.debug('Bluetooth peer hosts are %s',
398 self._btpeer_host_list)
399 self.btpeer_list = [_host.create_chameleon_board() for _host in
400 self._btpeer_host_list if _host is not None]
401
402 if len(self.btpeer_list) > 0:
403 self.btpeer = self.btpeer_list[0]
404
405 logging.debug('After initialize_btpeer btpeer_list %s '
406 'btpeer_host_list is %s and btpeer is %s',
407 self.btpeer_list, self._btpeer_host_list,
408 self.btpeer)
409 except Exception as e:
410 logging.error('Exception %s in initialize_btpeer', str(e))
411
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800412
413
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000414 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700415 """Get latest stable cros image name from AFE.
416
417 Use the board name from the info store. Should that fail, try to
418 retrieve the board name from the host's installed image itself.
419
420 @returns: current stable cros image name for this host.
421 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700422 info = self.host_info_store.get()
423 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700424 logging.warn('No board label value found. Trying to infer '
425 'from the host itself.')
426 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700427 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700428 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
429 logging.error('Also failed to get the board name from the DUT '
430 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700431 raise error.AutoservError('Cannot determine board of the DUT'
432 ' while getting repair image name.')
433 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500434
435
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700436 def host_version_prefix(self, image):
437 """Return version label prefix.
438
439 In case the CrOS provisioning version is something other than the
440 standard CrOS version e.g. CrOS TH version, this function will
441 find the prefix from provision.py.
442
443 @param image: The image name to find its version prefix.
444 @returns: A prefix string for the image type.
445 """
446 return provision.get_version_label_prefix(image)
447
Andrew Luo3332ab22020-04-28 16:42:03 -0700448 def stage_build_to_usb(self, build):
449 """Stage the current ChromeOS image on the USB stick connected to the
450 servo.
451
452 @param build: The build to download and send to USB.
453 """
454 if not self.servo:
455 raise error.TestError('Host %s does not have servo.' %
456 self.hostname)
457
458 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700459
460 try:
461 self.servo.image_to_servo_usb(update_url)
462 finally:
463 # servo.image_to_servo_usb turned the DUT off, so turn it back on
464 logging.debug('Turn DUT power back on.')
465 self.servo.get_power_state_controller().power_on()
466
Andrew Luo3332ab22020-04-28 16:42:03 -0700467 logging.debug('ChromeOS image %s is staged on the USB stick.',
468 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700469
beepsdae65fd2013-07-26 16:24:41 -0700470 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700471 """
472 Make sure job_repo_url of this host is valid.
473
joychen03eaad92013-06-26 09:55:21 -0700474 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700475 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
476 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
477 download and extract it. If the devserver embedded in the url is
478 unresponsive, update the job_repo_url of the host after staging it on
479 another devserver.
480
481 @param job_repo_url: A url pointing to the devserver where the autotest
482 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700483 @param tag: The tag from the server job, in the format
484 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700485
486 @raises DevServerException: If we could not resolve a devserver.
487 @raises AutoservError: If we're unable to save the new job_repo_url as
488 a result of choosing a new devserver because the old one failed to
489 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700490 @raises urllib2.URLError: If the devserver embedded in job_repo_url
491 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700492 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800493 info = self.host_info_store.get()
494 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700495 if not job_repo_url:
496 logging.warning('No job repo url set on host %s', self.hostname)
497 return
498
499 logging.info('Verifying job repo url %s', job_repo_url)
500 devserver_url, image_name = tools.get_devserver_build_from_package_url(
501 job_repo_url)
502
beeps0c865032013-07-30 11:37:06 -0700503 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700504
505 logging.info('Staging autotest artifacts for %s on devserver %s',
506 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700507
508 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700509 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700510 stage_time = time.time() - start_time
511
512 # Record how much of the verification time comes from a devserver
513 # restage. If we're doing things right we should not see multiple
514 # devservers for a given board/build/branch path.
515 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800516 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700517 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800518 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700519 pass
520 else:
beeps0c865032013-07-30 11:37:06 -0700521 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700522 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700523 stats_key = {
524 'board': board,
525 'build_type': build_type,
526 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700527 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700528 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800529
530 monarch_fields = {
531 'board': board,
532 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800533 'branch': branch,
534 'dev_server': devserver,
535 }
536 metrics.Counter(
537 'chromeos/autotest/provision/verify_url'
538 ).increment(fields=monarch_fields)
539 metrics.SecondsDistribution(
540 'chromeos/autotest/provision/verify_url_duration'
541 ).add(stage_time, fields=monarch_fields)
542
543
Dan Shicf4d2032015-03-12 15:04:21 -0700544 def stage_server_side_package(self, image=None):
545 """Stage autotest server-side package on devserver.
546
547 @param image: Full path of an OS image to install or a build name.
548
549 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700550
551 @raise: error.AutoservError if fail to locate the build to test with, or
552 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700553 """
Dan Shid37736b2016-07-06 15:10:29 -0700554 # If enable_drone_in_restricted_subnet is False, do not set hostname
555 # in devserver.resolve call, so a devserver in non-restricted subnet
556 # is picked to stage autotest server package for drone to download.
557 hostname = self.hostname
558 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
559 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700560 if image:
561 image_name = tools.get_build_from_image(image)
562 if not image_name:
563 raise error.AutoservError(
564 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700565 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700566 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700567 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800568 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700569 if job_repo_url:
570 devserver_url, image_name = (
571 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700572 # If enable_drone_in_restricted_subnet is True, use the
573 # existing devserver. Otherwise, resolve a new one in
574 # non-restricted subnet.
575 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
576 ds = dev_server.ImageServer(devserver_url)
577 else:
578 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800579 elif info.build is not None:
580 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700581 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700582 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800583 raise error.AutoservError(
584 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700585 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700586
587 # Get the OS version of the build, for any build older than
588 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
589 match = re.match('.*/R\d+-(\d+)\.', image_name)
590 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700591 raise error.AutoservError(
592 'Build %s is older than %s. Server side packaging is '
593 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700594
Dan Shicf4d2032015-03-12 15:04:21 -0700595 ds.stage_artifacts(image_name, ['autotest_server_package'])
596 return '%s/static/%s/%s' % (ds.url(), image_name,
597 'autotest_server_package.tar.bz2')
598
599
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700600 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700601 """Stage a build on a devserver and return the update_url.
602
603 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700604 @param artifact: a string like 'test_image'. Requests
605 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700606 @returns a tuple of (image_name, URL) like
607 (lumpy-release/R27-3837.0.0,
608 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700609 """
610 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000611 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700612 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800613 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700614 devserver.stage_artifacts(image_name, [artifact])
615 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700616 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700617 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700618 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700619 else:
620 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700621
622
beepse539be02013-07-31 21:57:39 -0700623 def stage_factory_image_for_servo(self, image_name):
624 """Stage a build on a devserver and return the update_url.
625
626 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700627
beepse539be02013-07-31 21:57:39 -0700628 @return: An update URL, eg:
629 http://<devserver>/static/canary-channel/\
630 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700631
632 @raises: ValueError if the factory artifact name is missing from
633 the config.
634
beepse539be02013-07-31 21:57:39 -0700635 """
636 if not image_name:
637 logging.error('Need an image_name to stage a factory image.')
638 return
639
Dan Shib8540a52015-07-16 14:18:23 -0700640 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700641 'CROS', 'factory_artifact', type=str, default='')
642 if not factory_artifact:
643 raise ValueError('Cannot retrieve the factory artifact name from '
644 'autotest config, and hence cannot stage factory '
645 'artifacts.')
646
beepse539be02013-07-31 21:57:39 -0700647 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800648 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700649 devserver.stage_artifacts(
650 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700651 [factory_artifact],
652 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700653
654 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
655
656
Laurence Goodby778c9a42017-05-24 19:24:07 -0700657 def prepare_for_update(self):
658 """Prepares the DUT for an update.
659
660 Subclasses may override this to perform any special actions
661 required before updating.
662 """
Laurence Goodby468de252017-06-08 17:22:53 -0700663 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700664
665
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800666 def _clear_fw_version_labels(self, rw_only):
667 """Clear firmware version labels from the machine.
668
669 @param rw_only: True to only clear fwrw_version; otherewise, clear
670 both fwro_version and fwrw_version.
671 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700672 info = self.host_info_store.get()
673 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800674 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700675 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
676 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700677
678
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800679 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700680 """Add firmware version label to the machine.
681
682 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800683 @param rw_only: True to only add fwrw_version; otherwise, add both
684 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700685
686 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700687 info = self.host_info_store.get()
688 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800689 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700690 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
691 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700692
693
Namyoon Woo33f38852020-04-13 17:26:58 -0700694 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800695 """Search for the latest package release version from the image archive,
696 and return it.
697
Namyoon Woo33f38852020-04-13 17:26:58 -0700698 @param platform: platform name, a.k.a. board or model
699 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800700
Namyoon Woo33f38852020-04-13 17:26:58 -0700701 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
702 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800703 or None if LATEST release file does not exist.
704 """
705
Namyoon Woo33f38852020-04-13 17:26:58 -0700706 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800707
Namyoon Woo33f38852020-04-13 17:26:58 -0700708 # Search the image path in reference board archive as well.
709 # For example, bob has its binary image under its reference board (gru)
710 # image archive.
711 if ref_board:
712 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800713
Namyoon Woo33f38852020-04-13 17:26:58 -0700714 for board in platforms:
715 # Read 'LATEST-1.0.0' file
716 branch_dir = provision.FW_BRANCH_GLOB % board
717 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
718 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800719
Namyoon Woo33f38852020-04-13 17:26:58 -0700720 try:
721 # The result could be one or more.
722 result = utils.system_output('gsutil ls -d ' + latest_file)
723
724 candidates = re.findall('gs://.*', result)
725
726 # Found the directory candidates. No need to check the other
727 # board name cadidates. Let's break the loop.
728 break
729 except error.CmdError:
730 # It doesn't exist. Let's move on to the next item.
731 pass
732 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800733 logging.error('No LATEST release info is available.')
734 return None
735
Namyoon Woo406c7d42020-01-24 15:57:11 -0800736 for cand_dir in candidates:
737 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800738
Namyoon Woo406c7d42020-01-24 15:57:11 -0800739 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700740 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800741 try:
742 # Check if release_path does exist.
743 release = utils.system_output('gsutil ls -d ' + release_path)
744 # Now 'release' has a full directory path: e.g.
745 # gs://chromeos-image-archive/firmware-octopus-11297.B-
746 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
747
748 # Remove "gs://chromeos-image-archive".
749 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
750
751 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
752 return release.strip('/')
753 except error.CmdError:
754 # The directory might not exist. Let's try next candidate.
755 pass
756 else:
757 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800758
Brent Peterson1cb623a2020-01-09 13:14:28 -0800759 @staticmethod
760 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800761 """Get version string from binary image using regular expression.
762
763 @param image: Binary image to search
764 @param version_regex: Regular expression to search for
765
766 @return Version string
767
768 @raises TestFail if no version string is found in image
769 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800770 with open(image, 'rb') as f:
771 image_data = f.read()
772 match = re.findall(version_regex, image_data)
773 if match:
774 return match[0]
775 else:
776 raise error.TestFail('Failed to read version from %s.' % image)
777
778
Garry Wangad2a1712020-03-26 15:06:43 -0700779 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800780 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700781 try_scp=False, install_ec=True, install_bios=True,
782 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700783 """Install firmware to the DUT.
784
785 Use stateful update if the DUT is already running the same build.
786 Stateful update does not update kernel and tends to run much faster
787 than a full reimage. If the DUT is running a different build, or it
788 failed to do a stateful update, full update, including kernel update,
789 will be applied to the DUT.
790
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800791 Once a host enters firmware_install its fw[ro|rw]_version label will
792 be removed. After the firmware is updated successfully, a new
793 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700794
795 @param build: The build version to which we want to provision the
796 firmware of the machine,
797 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800798 @param rw_only: True to only install firmware to its RW portions. Keep
799 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700800 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800801 @param local_tarball: Path to local firmware image for installing
802 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800803 @param verify_version: True to verify EC and BIOS versions after
804 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800805 @param try_scp: False to always program using servo, true to try copying
806 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700807 @param install_ec: True to install EC FW, and False to skip it.
808 @param install_bios: True to install BIOS, and False to skip it.
809 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700810
811 TODO(dshi): After bug 381718 is fixed, update here with corresponding
812 exceptions that could be raised.
813
814 """
815 if not self.servo:
816 raise error.TestError('Host %s does not have servo.' %
817 self.hostname)
818
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700819 # Get the DUT board name from AFE.
820 info = self.host_info_store.get()
821 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700822 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800823
824 if board is None or board == '':
825 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700826
Namyoon Woo382e5892020-05-20 16:48:40 -0700827 # if board_as argument is passed, then use it instead of the original
828 # board name.
829 if board_as:
830 board = board_as
831
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700832 if model is None or model == '':
833 model = self.get_platform_from_fwid()
834
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800835 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700836 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800837 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700838 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800839
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700840 try:
841 ds = dev_server.ImageServer.resolve(build, self.hostname)
842 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800843
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700844 if not dest:
845 tmpd = autotemp.tempdir(unique_id='fwimage')
846 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800847
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700848 # Download firmware image
849 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
850 local_tarball = os.path.join(dest, os.path.basename(fwurl))
851 ds.download_file(fwurl, local_tarball)
852 except Exception as e:
853 raise error.TestError('Failed to download firmware package: %s'
854 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700855
Namyoon Woo382e5892020-05-20 16:48:40 -0700856 ec_image = None
857 if install_ec:
858 # Extract EC image from tarball
859 logging.info('Extracting EC image.')
860 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800861
Namyoon Woo382e5892020-05-20 16:48:40 -0700862 bios_image = None
863 if install_bios:
864 # Extract BIOS image from tarball
865 logging.info('Extracting BIOS image.')
866 bios_image = self.servo.extract_bios_image(board, model,
867 local_tarball)
868
869 if not bios_image and not ec_image:
870 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800871
Brent Petersonc70a1832020-01-24 15:54:35 -0800872 # Clear firmware version labels
873 self._clear_fw_version_labels(rw_only)
874
875 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800876 try:
Garry Wang50e4a492020-08-05 12:29:57 -0700877 # Check if copying to DUT is enabled and DUT is available
878 if try_scp and self.is_up():
Brent Petersonc70a1832020-01-24 15:54:35 -0800879 # DUT is available, make temp firmware directory to store images
880 logging.info('Making temp folder.')
881 dest_folder = '/tmp/firmware'
882 self.run('mkdir -p ' + dest_folder)
883
Namyoon Woo68b68082020-06-02 13:13:14 -0700884 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800885
Namyoon Woo382e5892020-05-20 16:48:40 -0700886 if bios_image:
887 # Send BIOS firmware image to DUT
888 logging.info('Sending BIOS firmware.')
889 dest_bios_path = os.path.join(dest_folder,
890 os.path.basename(bios_image))
891 self.send_file(bios_image, dest_bios_path)
892
893 # Initialize firmware update command for BIOS image
894 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800895
896 # Send EC firmware image to DUT when EC image was found
897 if ec_image:
898 logging.info('Sending EC firmware.')
899 dest_ec_path = os.path.join(dest_folder,
900 os.path.basename(ec_image))
901 self.send_file(ec_image, dest_ec_path)
902
903 # Add EC image to firmware update command
904 fw_cmd += ' -e %s' % dest_ec_path
905
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700906 # Make sure command is allowed to finish even if ssh fails.
907 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
908
Brent Peterson669edf42020-02-07 15:07:54 -0800909 # Update firmware on DUT
910 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700911 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700912 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700913 except error.AutoservRunError as e:
914 if e.result_obj.exit_status != 255:
915 raise
916 elif ec_image:
917 logging.warn("DUT network dropped during update"
918 " (often caused by EC resetting USB)")
919 else:
920 logging.error("DUT network dropped during update"
921 " (unexpected, since no EC image)")
922 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800923 else:
924 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800925 if ec_image:
926 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700927 if bios_image:
928 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800929 if utils.host_is_in_lab_zone(self.hostname):
930 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800931
932 # Reboot and wait for DUT after installing firmware
933 logging.info('Rebooting DUT.')
934 self.servo.get_power_state_controller().reset()
935 time.sleep(self.servo.BOOT_DELAY)
936 self.test_wait_for_boot()
937
938 # When enabled verify EC and BIOS firmware version after programming
939 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800940 # Check programmed EC firmware when EC image was found
941 if ec_image:
942 logging.info('Checking EC firmware version.')
943 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800944 ec_version_prefix = dest_ec_version.split('_', 1)[0]
945 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800946 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800947 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800948 if dest_ec_version != image_ec_version:
949 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700950 'Failed to update EC firmware, version %s '
951 '(expected %s)' % (dest_ec_version,
952 image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800953
Namyoon Woo382e5892020-05-20 16:48:40 -0700954 if bios_image:
955 # Check programmed BIOS firmware against expected version
956 logging.info('Checking BIOS firmware version.')
957 dest_bios_version = self.get_firmware_version()
958 bios_version_prefix = dest_bios_version.split('.', 1)[0]
959 bios_regex = self._BIOS_REGEX % bios_version_prefix
960 image_bios_version = self.get_version_from_image(bios_image,
961 bios_regex)
962 if dest_bios_version != image_bios_version:
963 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700964 'Failed to update BIOS, version %s '
Namyoon Woo382e5892020-05-20 16:48:40 -0700965 '(expected %s)' % (dest_bios_version,
966 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700967 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700968 if tmpd:
969 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700970
971
beepsf079cfb2013-09-18 17:49:51 -0700972 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
973 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500974 """
975 Re-install the OS on the DUT by:
976 1) installing a test image on a USB storage device attached to the Servo
977 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800978 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700979 3) installing the image with chromeos-install.
980
Scott Zawalski62bacae2013-03-05 10:40:32 -0500981 @param image_url: If specified use as the url to install on the DUT.
982 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700983 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
984 Factory images need a longer usb_boot_timeout than regular
985 cros images.
986 @param install_timeout: The timeout to use when installing the chromeos
987 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800988
Scott Zawalski62bacae2013-03-05 10:40:32 -0500989 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700990
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800991 """
Garry Wang7b0e1b72020-03-25 19:08:59 -0700992 if image_url:
993 logging.info('Downloading image to USB, then booting from it.'
994 ' Usb boot timeout = %s', usb_boot_timeout)
995 else:
996 logging.info('Booting from USB directly. Usb boot timeout = %s',
997 usb_boot_timeout)
998
999 metrics_field = {'download': bool(image_url)}
1000 metrics.Counter(
1001 'chromeos/autotest/provision/servo_install/download_image'
1002 ).increment(fields=metrics_field)
1003
Allen Li48a13fe2016-11-22 14:10:40 -08001004 with metrics.SecondsTimer(
1005 'chromeos/autotest/provision/servo_install/boot_duration'):
1006 self.servo.install_recovery_image(image_url)
1007 if not self.wait_up(timeout=usb_boot_timeout):
1008 raise hosts.AutoservRepairError(
1009 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001010 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001011
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001012 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1013 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001014 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001015 try:
1016 self.run('chromeos-tpm-recovery')
1017 except error.AutoservRunError:
1018 logging.warn('chromeos-tpm-recovery is too old.')
1019
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001020
Allen Li48a13fe2016-11-22 14:10:40 -08001021 with metrics.SecondsTimer(
1022 'chromeos/autotest/provision/servo_install/install_duration'):
1023 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001024 try:
1025 self.run('chromeos-install --yes',timeout=install_timeout)
1026 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001027 except Exception as e:
1028 storage_errors = [
1029 'No space left on device',
1030 'I/O error when trying to write primary GPT',
1031 'Input/output error while writing out',
1032 'cannot read GPT header',
1033 'can not determine destination device'
1034 ]
1035 has_error = [msg for msg in storage_errors if(msg in str(e))]
1036 if has_error:
1037 info = self.host_info_store.get()
1038 info.set_version_label(
1039 audit_const.DUT_STORAGE_STATE_PREFIX,
1040 audit_const.HW_STATE_NEED_REPLACEMENT)
1041 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001042 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001043 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001044 logging.debug(
1045 'Fail install image from USB; Storage error; %s', e)
1046 raise error.AutoservError(
1047 'Failed to install image from USB due to a suspect '
1048 'disk failure, DUT storage state changed to '
1049 'need_replacement, please check debug log '
1050 'for details.')
1051 else:
1052 logging.debug('Fail install image from USB; %s', e)
1053 raise error.AutoservError(
1054 'Failed to install image from USB due to unexpected '
1055 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001056 finally:
1057 # We need reset the DUT no matter re-install success or not,
1058 # as we don't want leave the DUT in boot from usb state.
1059 logging.info('Power cycling DUT through servo.')
1060 self.servo.get_power_state_controller().power_off()
1061 self.servo.switch_usbkey('off')
1062 # N.B. The Servo API requires that we use power_on() here
1063 # for two reasons:
1064 # 1) After turning on a DUT in recovery mode, you must turn
1065 # it off and then on with power_on() once more to
1066 # disable recovery mode (this is a Parrot specific
1067 # requirement).
1068 # 2) After power_off(), the only way to turn on is with
1069 # power_on() (this is a Storm specific requirement).
1070 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001071
1072 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001073 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001074 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1075 'test image after %d seconds' %
1076 self.BOOT_TIMEOUT,
1077 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001078
1079
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001080 def set_servo_host(self, host, servo_state = None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001081 """Set our servo host member, and associated servo.
1082
1083 @param host Our new `ServoHost`.
1084 """
1085 self._servo_host = host
1086 if self._servo_host is not None:
1087 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001088 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001089 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001090 else:
1091 self.servo = None
Otabek Kasimov41301a22020-05-10 15:28:21 -07001092 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001093 self.set_servo_state(servo_state)
1094
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001095
Richard Barnette9a26ad62016-06-10 12:03:08 -07001096 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001097 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001098 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001099
Richard Barnette9a26ad62016-06-10 12:03:08 -07001100 If the servo object is missing, attempt to repair the servo
1101 host. Repair failures are passed back to the caller.
1102
1103 @raise AutoservError: If there is no servo host for this CrOS
1104 host.
1105 """
1106 if self.servo:
1107 return
1108 if not self._servo_host:
1109 raise error.AutoservError('No servo host for %s.' %
1110 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001111 try:
1112 self._servo_host.repair()
1113 except:
1114 raise
1115 finally:
1116 self.set_servo_host(self._servo_host)
1117
1118
Otabek Kasimov41301a22020-05-10 15:28:21 -07001119 def set_servo_type(self):
1120 """Set servo info labels to dut host_info"""
1121 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001122 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001123 return
1124 servo_type = self.servo.get_servo_type()
1125 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001126 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001127 ' by `dut-control servo_type`! Please file a bug'
1128 ' and inform infra team as we are not expected '
1129 ' to reach this point.')
1130 return
1131 host_info = self.host_info_store.get()
1132 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1133 old_type = host_info.get_label_value(prefix)
1134 if old_type == servo_type:
1135 # do not need update
1136 return
1137 host_info.set_version_label(prefix, servo_type)
1138 self.host_info_store.commit(host_info)
1139 logging.info('ServoHost: servo_type updated to %s '
1140 '(previous: %s)', servo_type, old_type)
1141
1142
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001143 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001144 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001145 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001146 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001147 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001148 old_state = host_info.get_label_value(servo_state_prefix)
1149 if old_state == servo_state:
1150 # do not need update
1151 return
1152 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001153 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001154 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1155 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001156
1157
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001158 def get_servo_state(self):
1159 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001160 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001161 return host_info.get_label_value(servo_state_prefix)
1162
Otabek Kasimov41301a22020-05-10 15:28:21 -07001163
Garry Wang000c6c02020-05-11 21:27:23 -07001164 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1165 if smart_usbhub_detected is None:
1166 # skip the label update here as this indicate we wasn't able
1167 # to confirm usbhub type.
1168 return
1169 host_info = self.host_info_store.get()
1170 if (smart_usbhub_detected ==
1171 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1172 # skip label update if current label match the truth.
1173 return
1174 if smart_usbhub_detected:
1175 logging.info('Adding %s label to host %s',
1176 servo_constants.SMART_USBHUB_LABEL,
1177 self.hostname)
1178 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1179 else:
1180 logging.info('Removing %s label from host %s',
1181 servo_constants.SMART_USBHUB_LABEL,
1182 self.hostname)
1183 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1184 self.host_info_store.commit(host_info)
1185
1186
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001187 def repair(self):
1188 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001189
1190 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001191 not call back to the parent class, but instead relies on
1192 `self._repair_strategy` to coordinate the verification and
1193 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001194 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001195 message = 'Beginning repair for host %s board %s model %s'
1196 info = self.host_info_store.get()
1197 message %= (self.hostname, info.board, info.model)
1198 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001199 try:
1200 self._repair_strategy.repair(self)
1201 except hosts.AutoservVerifyDependencyError as e:
1202 # We don't want flag a DUT as failed if only non-critical
1203 # verifier(s) failed during the repair.
1204 if e.is_critical():
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001205 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001206 raise
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001207
Richard Barnette82c35912012-11-20 10:09:10 -08001208
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001209 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001210 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001211 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001212
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001213 if self._chameleon_host:
1214 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001215
1216 if self._servo_host:
1217 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001218
1219
Dan Shi49ca0932014-11-14 11:22:27 -08001220 def get_power_supply_info(self):
1221 """Get the output of power_supply_info.
1222
1223 power_supply_info outputs the info of each power supply, e.g.,
1224 Device: Line Power
1225 online: no
1226 type: Mains
1227 voltage (V): 0
1228 current (A): 0
1229 Device: Battery
1230 state: Discharging
1231 percentage: 95.9276
1232 technology: Li-ion
1233
1234 Above output shows two devices, Line Power and Battery, with details of
1235 each device listed. This function parses the output into a dictionary,
1236 with key being the device name, and value being a dictionary of details
1237 of the device info.
1238
1239 @return: The dictionary of power_supply_info, e.g.,
1240 {'Line Power': {'online': 'yes', 'type': 'main'},
1241 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001242 @raise error.AutoservRunError if power_supply_info tool is not found in
1243 the DUT. Caller should handle this error to avoid false failure
1244 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001245 """
1246 result = self.run('power_supply_info').stdout.strip()
1247 info = {}
1248 device_name = None
1249 device_info = {}
1250 for line in result.split('\n'):
1251 pair = [v.strip() for v in line.split(':')]
1252 if len(pair) != 2:
1253 continue
1254 if pair[0] == 'Device':
1255 if device_name:
1256 info[device_name] = device_info
1257 device_name = pair[1]
1258 device_info = {}
1259 else:
1260 device_info[pair[0]] = pair[1]
1261 if device_name and not device_name in info:
1262 info[device_name] = device_info
1263 return info
1264
1265
1266 def get_battery_percentage(self):
1267 """Get the battery percentage.
1268
1269 @return: The percentage of battery level, value range from 0-100. Return
1270 None if the battery info cannot be retrieved.
1271 """
1272 try:
1273 info = self.get_power_supply_info()
1274 logging.info(info)
1275 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001276 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001277 return None
1278
1279
Philip Chenaf69ead2020-03-27 13:06:42 -07001280 def get_battery_state(self):
1281 """Get the battery charging state.
1282
1283 @return: A string representing the battery charging state. It can be
1284 'Charging', 'Fully charged', or 'Discharging'.
1285 """
1286 try:
1287 info = self.get_power_supply_info()
1288 logging.info(info)
1289 return info['Battery']['state']
1290 except (KeyError, ValueError, error.AutoservRunError):
1291 return None
1292
1293
Daniel Campello8ca25c22019-12-13 16:48:26 -07001294 def get_battery_display_percentage(self):
1295 """Get the battery display percentage.
1296
1297 @return: The display percentage of battery level, value range from
1298 0-100. Return None if the battery info cannot be retrieved.
1299 """
1300 try:
1301 info = self.get_power_supply_info()
1302 logging.info(info)
1303 return float(info['Battery']['display percentage'])
1304 except (KeyError, ValueError, error.AutoservRunError):
1305 return None
1306
1307
Dan Shi49ca0932014-11-14 11:22:27 -08001308 def is_ac_connected(self):
1309 """Check if the dut has power adapter connected and charging.
1310
1311 @return: True if power adapter is connected and charging.
1312 """
1313 try:
1314 info = self.get_power_supply_info()
1315 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001316 except (KeyError, error.AutoservRunError):
1317 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001318
1319
Simran Basi5e6339a2013-03-21 11:34:32 -07001320 def _cleanup_poweron(self):
1321 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001322 info = self.host_info_store.get()
1323 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001324 return
1325 logging.debug('This host has recently interacted with the RPM'
1326 ' Infrastructure. Ensuring power is on.')
1327 try:
1328 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001329 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001330 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001331 logging.error('Failed to turn Power On for this host after '
1332 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001333
1334 battery_percentage = self.get_battery_percentage()
Gregory Nisbet02273172020-07-13 09:26:17 -07001335 if (battery_percentage and
1336 battery_percentage < cros_repair.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001337 raise
1338 elif self.is_ac_connected():
1339 logging.info('The device has power adapter connected and '
1340 'charging. No need to try to turn RPM on '
1341 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001342 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001343 logging.info('Battery level is now at %s%%. The device may '
1344 'still have enough power to run test, so no '
1345 'exception will be raised.', battery_percentage)
1346
Simran Basi5e6339a2013-03-21 11:34:32 -07001347
Garry Wangad4d4fd2019-01-30 17:00:38 -08001348 def _remove_rpm_changed_tag(self):
1349 info = self.host_info_store.get()
1350 del info.attributes[self._RPM_OUTLET_CHANGED]
1351 self.host_info_store.commit(info)
1352
1353
1354 def _add_rpm_changed_tag(self):
1355 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001356 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001357 self.host_info_store.commit(info)
1358
1359
1360
beepsc87ff602013-07-31 21:53:00 -07001361 def _is_factory_image(self):
1362 """Checks if the image on the DUT is a factory image.
1363
1364 @return: True if the image on the DUT is a factory image.
1365 False otherwise.
1366 """
1367 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1368 return result.exit_status == 0
1369
1370
1371 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001372 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001373
1374 @raises: FactoryImageCheckerException for factory images, since
1375 we cannot attempt to restart ui on them.
1376 error.AutoservRunError for any other type of error that
1377 occurs while restarting ui.
1378 """
1379 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001380 raise FactoryImageCheckerException('Cannot restart ui on factory '
1381 'images')
beepsc87ff602013-07-31 21:53:00 -07001382
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001383 # TODO(jrbarnette): The command to stop/start the ui job
1384 # should live inside cros_ui, too. However that would seem
1385 # to imply interface changes to the existing start()/restart()
1386 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001387 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001388 self.run('stop ui; start ui')
1389 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001390
1391
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001392 def _start_powerd_if_needed(self):
1393 """Start powerd if it isn't already running."""
1394 self.run('start powerd', ignore_status=True)
1395
1396
xixuana3bbc422017-05-04 15:57:21 -07001397 def _get_lsb_release_content(self):
1398 """Return the content of lsb-release file of host."""
1399 return self.run(
1400 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1401
1402
Dan Shi549fb822015-03-24 18:01:11 -07001403 def get_release_version(self):
1404 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1405
1406 @returns The version string in lsb-release, under attribute
1407 CHROMEOS_RELEASE_VERSION.
1408 """
Dan Shi549fb822015-03-24 18:01:11 -07001409 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001410 lsb_release_content=self._get_lsb_release_content())
1411
1412
Don Garrettb9f35802018-01-22 18:25:40 -08001413 def get_release_builder_path(self):
1414 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1415
1416 @returns The version string in lsb-release, under attribute
1417 CHROMEOS_RELEASE_BUILDER_PATH.
1418 """
1419 return lsbrelease_utils.get_chromeos_release_builder_path(
1420 lsb_release_content=self._get_lsb_release_content())
1421
1422
xixuana3bbc422017-05-04 15:57:21 -07001423 def get_chromeos_release_milestone(self):
1424 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1425 from lsb-release.
1426
1427 @returns The version string in lsb-release, under attribute
1428 CHROMEOS_RELEASE_BUILD_TYPE.
1429 """
1430 return lsbrelease_utils.get_chromeos_release_milestone(
1431 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001432
1433
1434 def verify_cros_version_label(self):
Garry Wangd18e7b32020-08-07 18:31:44 -07001435 """Verify if host's cros-version label match the actual image in dut.
Dan Shi549fb822015-03-24 18:01:11 -07001436
Garry Wangd18e7b32020-08-07 18:31:44 -07001437 @returns True if the label match with image in dut, otherwise False
Dan Shi549fb822015-03-24 18:01:11 -07001438 """
Garry Wangd18e7b32020-08-07 18:31:44 -07001439 os_from_host = self.get_release_builder_path()
1440 info = self.host_info_store.get()
1441 os_from_label = info.get_label_value(self.VERSION_PREFIX)
1442 if not os_from_label:
1443 logging.debug('No existing %s label detected', self.VERSION_PREFIX)
1444 return True
1445
1446 # known cases where the version label will not match the
1447 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1448 # * Tests for the `arc-presubmit` append "-cheetsth" to the label.
1449 if os_from_label.endswith(provision.CHEETS_SUFFIX):
1450 logging.debug('%s label with %s suffix detected, this suffix will'
1451 ' be ignored when comparing label.',
1452 self.VERSION_PREFIX, provision.CHEETS_SUFFIX)
1453 os_from_label = os_from_label[:-len(provision.CHEETS_SUFFIX)]
1454 logging.debug('OS version from host: %s; OS verision cached in '
1455 'label: %s', os_from_host, os_from_label)
1456 return os_from_label == os_from_host
Dan Shi549fb822015-03-24 18:01:11 -07001457
1458
Laurence Goodby778c9a42017-05-24 19:24:07 -07001459 def cleanup_services(self):
1460 """Reinitializes the device for cleanup.
1461
1462 Subclasses may override this to customize the cleanup method.
1463
1464 To indicate failure of the reset, the implementation may raise
1465 any of:
1466 error.AutoservRunError
1467 error.AutotestRunError
1468 FactoryImageCheckerException
1469
1470 @raises error.AutoservRunError
1471 @raises error.AutotestRunError
1472 @raises error.FactoryImageCheckerException
1473 """
1474 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001475 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001476
1477
beepsc87ff602013-07-31 21:53:00 -07001478 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001479 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001480 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001481 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001482 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001483 except (error.AutotestRunError, error.AutoservRunError,
1484 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001485 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001486
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001487 # cleanup routines, i.e. reboot the machine.
1488 super(CrosHost, self).cleanup()
1489
Simran Basi5e6339a2013-03-21 11:34:32 -07001490 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001491 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001492 self._cleanup_poweron()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001493
1494
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001495 def reboot(self, **dargs):
1496 """
1497 This function reboots the site host. The more generic
1498 RemoteHost.reboot() performs sync and sleeps for 5
1499 seconds. This is not necessary for Chrome OS devices as the
1500 sync should be finished in a short time during the reboot
1501 command.
1502 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001503 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001504 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001505 dargs['reboot_cmd'] = ('sleep 1; '
1506 'reboot & sleep %d; '
1507 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001508 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001509 if 'fastsync' not in dargs:
1510 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001511
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001512 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001513 # Record who called us
1514 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001515 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001516 'dut_host_name' : self.hostname,
1517 'success' : True}
1518 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001519 'caller' : "%s:%s" % (orig.co_filename,
1520 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001521 'success' : True,
1522 'error' : ''}
1523
Vincent Palatin80780b22016-07-27 16:02:37 +02001524 t0 = time.time()
1525 try:
1526 super(CrosHost, self).reboot(**dargs)
1527 except Exception as e:
1528 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001529 metric_debug_fields['success'] = False
1530 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001531 raise
1532 finally:
1533 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001534 metrics.Counter(
1535 'chromeos/autotest/autoserv/reboot_count').increment(
1536 fields=metric_fields)
1537 metrics.Counter(
1538 'chromeos/autotest/autoserv/reboot_debug').increment(
1539 fields=metric_debug_fields)
1540 metrics.SecondsDistribution(
1541 'chromeos/autotest/autoserv/reboot_duration').add(
1542 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001543
1544
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001545 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001546 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001547 """
1548 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001549
1550 @param suspend_time: How long to suspend as integer seconds.
1551 @param suspend_cmd: Suspend command to execute.
1552 @param allow_early_resume: If False and if device resumes before
1553 |suspend_time|, throw an error.
1554
1555 @exception AutoservSuspendError Host resumed earlier than
1556 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001557 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001558
1559 if suspend_cmd is None:
1560 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001561 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001562 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001563 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001564 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1565 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001566
1567
Simran Basiec564392014-08-25 16:48:09 -07001568 def upstart_status(self, service_name):
1569 """Check the status of an upstart init script.
1570
1571 @param service_name: Service to look up.
1572
1573 @returns True if the service is running, False otherwise.
1574 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001575 return 'start/running' in self.run('status %s' % service_name,
1576 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001577
Tom Hughese9552342018-12-18 14:29:25 -08001578 def upstart_stop(self, service_name):
1579 """Stops an upstart job if it's running.
1580
1581 @param service_name: Service to stop
1582
1583 @returns True if service has been stopped or was already stopped
1584 False otherwise.
1585 """
1586 if not self.upstart_status(service_name):
1587 return True
1588
1589 result = self.run('stop %s' % service_name, ignore_status=True)
1590 if result.exit_status != 0:
1591 return False
1592 return True
1593
1594 def upstart_restart(self, service_name):
1595 """Restarts (or starts) an upstart job.
1596
1597 @param service_name: Service to start/restart
1598
1599 @returns True if service has been started/restarted, False otherwise.
1600 """
1601 cmd = 'start'
1602 if self.upstart_status(service_name):
1603 cmd = 'restart'
1604 cmd = cmd + ' %s' % service_name
1605 result = self.run(cmd)
1606 if result.exit_status != 0:
1607 return False
1608 return True
Simran Basiec564392014-08-25 16:48:09 -07001609
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001610 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001611 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001612
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001613 Tests for the following conditions:
1614 1. All conditions tested by the parent version of this
1615 function.
1616 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001617 3. Sufficient space in /mnt/stateful_partition/encrypted.
1618 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001619
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001620 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001621 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001622 default_kilo_inodes_required = CONFIG.get_config_value(
1623 'SERVER', 'kilo_inodes_required', type=int, default=100)
1624 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1625 kilo_inodes_required = CONFIG.get_config_value(
1626 'SERVER', 'kilo_inodes_required_%s' % board,
1627 type=int, default=default_kilo_inodes_required)
1628 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001629 self.check_diskspace(
1630 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001631 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001632 'SERVER', 'gb_diskspace_required', type=float,
1633 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001634 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1635 # Not all targets build with encrypted stateful support.
1636 if self.path_exists(encrypted_stateful_path):
1637 self.check_diskspace(
1638 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001639 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001640 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1641 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001642
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001643 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001644
beepsc87ff602013-07-31 21:53:00 -07001645 # Factory images don't run update engine,
1646 # goofy controls dbus on these DUTs.
1647 if not self._is_factory_image():
1648 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001649
1650
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001651 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1652 def wait_for_system_services(self):
1653 """Waits for system-services to be running.
1654
1655 Sometimes, update_engine will take a while to update firmware, so we
1656 should give this some time to finish. See crbug.com/765686#c38 for
1657 details.
1658 """
1659 if not self.upstart_status('system-services'):
1660 raise error.AutoservError('Chrome failed to reach login. '
1661 'System services not running.')
1662
1663
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001664 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001665 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001666 message = 'Beginning verify for host %s board %s model %s'
1667 info = self.host_info_store.get()
1668 message %= (self.hostname, info.board, info.model)
1669 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001670 try:
1671 self._repair_strategy.verify(self)
1672 except hosts.AutoservVerifyDependencyError as e:
1673 # We don't want flag a DUT as failed if only non-critical
1674 # verifier(s) failed during the repair.
1675 if e.is_critical():
1676 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001677
1678
Fang Deng96667ca2013-08-01 17:46:18 -07001679 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001680 connect_timeout=None, alive_interval=None,
1681 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001682 """Override default make_ssh_command to use options tuned for Chrome OS.
1683
1684 Tuning changes:
1685 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1686 connection failure. Consistency with remote_access.sh.
1687
Samuel Tan2ce155b2015-06-23 18:24:38 -07001688 - ServerAliveInterval=900; which causes SSH to ping connection every
1689 900 seconds. In conjunction with ServerAliveCountMax ensures
1690 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001691 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001692 the test completed successfully. Later increased from 180 seconds to
1693 900 seconds to account for tests where the DUT is suspended for
1694 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001695
1696 - ServerAliveCountMax=3; consistency with remote_access.sh.
1697
1698 - ConnectAttempts=4; reduce flakiness in connection errors;
1699 consistency with remote_access.sh.
1700
1701 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1702 Host keys change with every new installation, don't waste
1703 memory/space saving them.
1704
1705 - SSH protocol forced to 2; needed for ServerAliveInterval.
1706
1707 @param user User name to use for the ssh connection.
1708 @param port Port on the target host to use for ssh connection.
1709 @param opts Additional options to the ssh command.
1710 @param hosts_file Ignored.
1711 @param connect_timeout Ignored.
1712 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001713 @param alive_count_max Ignored.
1714 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001715 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001716 options = ' '.join([opts, '-o Protocol=2'])
1717 return super(CrosHost, self).make_ssh_command(
1718 user=user, port=port, opts=options, hosts_file='/dev/null',
1719 connect_timeout=30, alive_interval=900, alive_count_max=3,
1720 connection_attempts=4)
1721
1722
Jason Abeleb6f924f2013-11-13 16:01:54 -08001723 def syslog(self, message, tag='autotest'):
1724 """Logs a message to syslog on host.
1725
1726 @param message String message to log into syslog
1727 @param tag String tag prefix for syslog
1728
1729 """
1730 self.run('logger -t "%s" "%s"' % (tag, message))
1731
1732
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001733 def _ping_check_status(self, status):
1734 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001735
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001736 @param status Check the ping status against this value.
1737 @return True iff `status` and the result of ping are the same
1738 (i.e. both True or both False).
1739
1740 """
1741 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1742 return not (status ^ (ping_val == 0))
1743
1744 def _ping_wait_for_status(self, status, timeout):
1745 """Wait for the host to have a given status (UP or DOWN).
1746
1747 Status is checked by polling. Polling will not last longer
1748 than the number of seconds in `timeout`. The polling
1749 interval will be long enough that only approximately
1750 _PING_WAIT_COUNT polling cycles will be executed, subject
1751 to a maximum interval of about one minute.
1752
1753 @param status Waiting will stop immediately if `ping` of the
1754 host returns this status.
1755 @param timeout Poll for at most this many seconds.
1756 @return True iff the host status from `ping` matched the
1757 requested status at the time of return.
1758
1759 """
1760 # _ping_check_status() takes about 1 second, hence the
1761 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001762 # FIXME: if the ping command errors then _ping_check_status()
1763 # returns instantly. If timeout is also smaller than twice
1764 # _PING_WAIT_COUNT then the while loop below forks many
1765 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1766 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1767 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001768 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1769 end_time = time.time() + timeout
1770 while time.time() <= end_time:
1771 if self._ping_check_status(status):
1772 return True
1773 if poll_interval > 0:
1774 time.sleep(poll_interval)
1775
1776 # The last thing we did was sleep(poll_interval), so it may
1777 # have been too long since the last `ping`. Check one more
1778 # time, just to be sure.
1779 return self._ping_check_status(status)
1780
1781 def ping_wait_up(self, timeout):
1782 """Wait for the host to respond to `ping`.
1783
1784 N.B. This method is not a reliable substitute for
1785 `wait_up()`, because a host that responds to ping will not
1786 necessarily respond to ssh. This method should only be used
1787 if the target DUT can be considered functional even if it
1788 can't be reached via ssh.
1789
1790 @param timeout Minimum time to allow before declaring the
1791 host to be non-responsive.
1792 @return True iff the host answered to ping before the timeout.
1793
1794 """
1795 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001796
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001797 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001798 """Wait until the host no longer responds to `ping`.
1799
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001800 This function can be used as a slightly faster version of
1801 `wait_down()`, by avoiding potentially long ssh timeouts.
1802
1803 @param timeout Minimum time to allow for the host to become
1804 non-responsive.
1805 @return True iff the host quit answering ping before the
1806 timeout.
1807
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001808 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001809 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001810
Anand K Mistry50f218e2020-07-31 14:50:15 +10001811 def _is_host_port_forwarded(self):
1812 """Checks if the dut is connected over port forwarding.
1813
1814 N.B. This method does not detect all situations where port forwarding is
1815 occurring. Namely, running autotest on the dut may result in a
1816 false-positive, and port forwarding using a different machine on the
1817 same network will be a false-negative.
1818
1819 @return True if the dut is connected over port forwarding
1820 False otherwise
1821 """
1822 is_localhost = self.hostname in ['localhost', '127.0.0.1']
1823 is_forwarded = is_localhost and not self.is_default_port
1824 if is_forwarded:
1825 logging.info('Detected DUT connected by port forwarding')
1826 return is_forwarded
1827
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001828 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001829 """Wait for the client to enter low-power sleep mode.
1830
1831 The test for "is asleep" can't distinguish a system that is
1832 powered off; to confirm that the unit was asleep, it is
1833 necessary to force resume, and then call
1834 `test_wait_for_resume()`.
1835
1836 This function is expected to be called from a test as part
1837 of a sequence like the following:
1838
1839 ~~~~~~~~
1840 boot_id = host.get_boot_id()
1841 # trigger sleep on the host
1842 host.test_wait_for_sleep()
1843 # trigger resume on the host
1844 host.test_wait_for_resume(boot_id)
1845 ~~~~~~~~
1846
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001847 @param sleep_timeout time limit in seconds to allow the host sleep.
1848
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001849 @exception TestFail The host did not go to sleep within
1850 the allowed time.
1851 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001852 if sleep_timeout is None:
1853 sleep_timeout = self.SLEEP_TIMEOUT
1854
Anand K Mistry50f218e2020-07-31 14:50:15 +10001855 # If the dut is accessed over SSH port-forwarding, `ping` is not useful
1856 # for detecting the dut is down since a ping to localhost will always
1857 # succeed. In this case, fall back to wait_down() which uses SSH.
1858 if self._is_host_port_forwarded():
1859 success = self.wait_down(timeout=sleep_timeout)
1860 else:
1861 success = self.ping_wait_down(timeout=sleep_timeout)
1862
1863 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001864 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001865 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001866
1867
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001868 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001869 """Wait for the client to resume from low-power sleep mode.
1870
1871 The `old_boot_id` parameter should be the value from
1872 `get_boot_id()` obtained prior to entering sleep mode. A
1873 `TestFail` exception is raised if the boot id changes.
1874
1875 See @ref test_wait_for_sleep for more on this function's
1876 usage.
1877
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001878 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001879 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001880 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001881
1882 @exception TestFail The host did not respond within the
1883 allowed time.
1884 @exception TestFail The host responded, but the boot id test
1885 indicated a reboot rather than a sleep
1886 cycle.
1887 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001888 if resume_timeout is None:
1889 resume_timeout = self.RESUME_TIMEOUT
1890
1891 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001892 raise error.TestFail(
1893 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001894 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001895 else:
1896 new_boot_id = self.get_boot_id()
1897 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001898 logging.error('client rebooted (old boot %s, new boot %s)',
1899 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001900 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001901 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001902
1903
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001904 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001905 """Wait for the client to shut down.
1906
1907 The test for "has shut down" can't distinguish a system that
1908 is merely asleep; to confirm that the unit was down, it is
1909 necessary to force boot, and then call test_wait_for_boot().
1910
1911 This function is expected to be called from a test as part
1912 of a sequence like the following:
1913
1914 ~~~~~~~~
1915 boot_id = host.get_boot_id()
1916 # trigger shutdown on the host
1917 host.test_wait_for_shutdown()
1918 # trigger boot on the host
1919 host.test_wait_for_boot(boot_id)
1920 ~~~~~~~~
1921
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001922 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001923 @exception TestFail The host did not shut down within the
1924 allowed time.
1925 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001926 if shutdown_timeout is None:
1927 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1928
Anand K Mistry50f218e2020-07-31 14:50:15 +10001929 if self._is_host_port_forwarded():
1930 success = self.wait_down(timeout=shutdown_timeout)
1931 else:
1932 success = self.ping_wait_down(timeout=shutdown_timeout)
1933
1934 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001935 raise error.TestFail(
1936 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001937 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001938
1939
1940 def test_wait_for_boot(self, old_boot_id=None):
1941 """Wait for the client to boot from cold power.
1942
1943 The `old_boot_id` parameter should be the value from
1944 `get_boot_id()` obtained prior to shutting down. A
1945 `TestFail` exception is raised if the boot id does not
1946 change. The boot id test is omitted if `old_boot_id` is not
1947 specified.
1948
1949 See @ref test_wait_for_shutdown for more on this function's
1950 usage.
1951
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001952 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001953 shut down.
1954
1955 @exception TestFail The host did not respond within the
1956 allowed time.
1957 @exception TestFail The host responded, but the boot id test
1958 indicated that there was no reboot.
1959 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001960 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001961 raise error.TestFail(
1962 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001963 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001964 elif old_boot_id:
1965 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001966 logging.error('client not rebooted (boot %s)',
1967 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001968 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001969 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001970
1971
1972 @staticmethod
1973 def check_for_rpm_support(hostname):
1974 """For a given hostname, return whether or not it is powered by an RPM.
1975
Simran Basi1df55112013-09-06 11:25:09 -07001976 @param hostname: hostname to check for rpm support.
1977
Simran Basid5e5e272012-09-24 15:23:59 -07001978 @return None if this host does not follows the defined naming format
1979 for RPM powered DUT's in the lab. If it does follow the format,
1980 it returns a regular expression MatchObject instead.
1981 """
Fang Dengbaff9082015-01-06 13:46:15 -08001982 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001983
1984
1985 def has_power(self):
1986 """For this host, return whether or not it is powered by an RPM.
1987
1988 @return True if this host is in the CROS lab and follows the defined
1989 naming format.
1990 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001991 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001992
1993
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001994 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001995 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001996
1997 @param state Specifies which power state to set to DUT
1998 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001999 use. By default "RPM" or "CCD" will be used based
2000 on servo type. Valid values from
2001 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002002
2003 """
2004 ACCEPTABLE_STATES = ['ON', 'OFF']
2005
Garry Wang5e5538a2019-04-08 15:36:18 -07002006 if not power_method:
2007 power_method = self.get_default_power_method()
2008
2009 state = state.upper()
2010 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002011 raise error.TestError('State must be one of: %s.'
2012 % (ACCEPTABLE_STATES,))
2013
2014 if power_method == self.POWER_CONTROL_SERVO:
2015 logging.info('Setting servo port J10 to %s', state)
2016 self.servo.set('prtctl3_pwren', state.lower())
2017 time.sleep(self._USB_POWER_TIMEOUT)
2018 elif power_method == self.POWER_CONTROL_MANUAL:
2019 logging.info('You have %d seconds to set the AC power to %s.',
2020 self._POWER_CYCLE_TIMEOUT, state)
2021 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07002022 elif power_method == self.POWER_CONTROL_CCD:
2023 servo_role = 'src' if state == 'ON' else 'snk'
2024 logging.info('servo ccd power pass through detected,'
2025 ' changing servo_role to %s.', servo_role)
2026 self.servo.set_servo_v4_role(servo_role)
2027 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07002028 # Make sure we don't leave DUT with no power(servo_role=snk)
2029 # when DUT is not pingable, as we raise a exception here
2030 # that may break a power cycle in the middle.
2031 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07002032 raise error.AutoservError(
2033 'DUT failed to regain network connection after %d seconds.'
2034 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002035 else:
2036 if not self.has_power():
2037 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002038 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07002039 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07002040
2041
Garry Wang5e5538a2019-04-08 15:36:18 -07002042 def power_off(self, power_method=None):
2043 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002044
2045 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002046 use. By default "RPM" or "CCD" will be used based
2047 on servo type. Valid values from
2048 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002049
2050 """
2051 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002052
2053
Garry Wang5e5538a2019-04-08 15:36:18 -07002054 def power_on(self, power_method=None):
2055 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002056
2057 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002058 use. By default "RPM" or "CCD" will be used based
2059 on servo type. Valid values from
2060 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002061
2062 """
2063 self._set_power('ON', power_method)
2064
2065
Garry Wang5e5538a2019-04-08 15:36:18 -07002066 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002067 """Cycle power to this host by turning it OFF, then ON.
2068
2069 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002070 use. By default "RPM" or "CCD" will be used based
2071 on servo type. Valid values from
2072 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002073
2074 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002075 if not power_method:
2076 power_method = self.get_default_power_method()
2077
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002078 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002079 self.POWER_CONTROL_MANUAL,
2080 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002081 self.power_off(power_method=power_method)
2082 time.sleep(self._POWER_CYCLE_TIMEOUT)
2083 self.power_on(power_method=power_method)
2084 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002085 self._add_rpm_changed_tag()
2086 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002087
2088
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002089 def get_platform_from_fwid(self):
2090 """Determine the platform from the crossystem fwid.
2091
2092 @returns a string representing this host's platform.
2093 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002094 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002095 crossystem = utils.Crossystem(self)
2096 crossystem.init()
2097 # Extract fwid value and use the leading part as the platform id.
2098 # fwid generally follow the format of {platform}.{firmware version}
2099 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2100 platform = crossystem.fwid().split('.')[0].lower()
2101 # Newer platforms start with 'Google_' while the older ones do not.
2102 return platform.replace('google_', '')
2103
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002104
Simran Basic6f1f7a2012-10-16 10:47:46 -07002105 def get_platform(self):
2106 """Determine the correct platform label for this host.
2107
2108 @returns a string representing this host's platform.
2109 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002110 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2111 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002112 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002113 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002114 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002115 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002116
2117
Greg Edelstona7b05d12020-04-01 16:00:51 -06002118 def get_model_from_cros_config(self):
2119 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002120
Greg Edelstona7b05d12020-04-01 16:00:51 -06002121 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002122 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002123 return cros_config.call_cros_config_get_output('/ name',
2124 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002125
2126
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002127 def get_architecture(self):
2128 """Determine the correct architecture label for this host.
2129
2130 @returns a string representing this host's architecture.
2131 """
2132 crossystem = utils.Crossystem(self)
2133 crossystem.init()
2134 return crossystem.arch()
2135
2136
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002137 def get_chrome_version(self):
2138 """Gets the Chrome version number and milestone as strings.
2139
2140 Invokes "chrome --version" to get the version number and milestone.
2141
2142 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2143 current Chrome version number as a string (in the form "W.X.Y.Z")
2144 and "milestone" is the first component of the version number
2145 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2146 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2147 of "chrome --version" and the milestone will be the empty string.
2148
2149 """
MK Ryu35d661e2014-09-25 17:44:10 -07002150 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002151 return utils.parse_chrome_version(version_string)
2152
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002153
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002154 def get_ec_version(self):
2155 """Get the ec version as strings.
2156
2157 @returns a string representing this host's ec version.
2158 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002159 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002160 result = self.run(command, ignore_status=True)
2161 if result.exit_status != 0:
2162 return ''
2163 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002164
2165
2166 def get_firmware_version(self):
2167 """Get the firmware version as strings.
2168
2169 @returns a string representing this host's firmware version.
2170 """
2171 crossystem = utils.Crossystem(self)
2172 crossystem.init()
2173 return crossystem.fwid()
2174
2175
2176 def get_hardware_revision(self):
2177 """Get the hardware revision as strings.
2178
2179 @returns a string representing this host's hardware revision.
2180 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002181 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002182 result = self.run(command, ignore_status=True)
2183 if result.exit_status != 0:
2184 return ''
2185 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002186
2187
2188 def get_kernel_version(self):
2189 """Get the kernel version as strings.
2190
2191 @returns a string representing this host's kernel version.
2192 """
2193 return self.run('uname -r').stdout.strip()
2194
2195
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002196 def get_cpu_name(self):
2197 """Get the cpu name as strings.
2198
2199 @returns a string representing this host's cpu name.
2200 """
2201
2202 # Try get cpu name from device tree first
2203 if self.path_exists('/proc/device-tree/compatible'):
2204 command = ' | '.join(
2205 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2206 'tail -1'])
2207 return self.run(command).stdout.strip().replace(',', ' ')
2208
2209 # Get cpu name from uname -p
2210 command = 'uname -p'
2211 ret = self.run(command).stdout.strip()
2212
2213 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2214 # Try get cpu name from /proc/cpuinfo instead
2215 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2216 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2217 self = self.run(command).stdout.strip()
2218
2219 # Remove bloat from CPU name, for example
2220 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2221 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2222 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2223 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2224 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2225 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2226
2227
2228 def get_screen_resolution(self):
2229 """Get the screen(s) resolution as strings.
2230 In case of more than 1 monitor, return resolution for each monitor
2231 separate with plus sign.
2232
2233 @returns a string representing this host's screen(s) resolution.
2234 """
2235 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2236 ret = self.run(command, ignore_status=True)
2237 # We might have Chromebox without a screen
2238 if ret.exit_status != 0:
2239 return ''
2240 return ret.stdout.strip().replace('\n', '+')
2241
2242
2243 def get_mem_total_gb(self):
2244 """Get total memory available in the system in GiB (2^20).
2245
2246 @returns an integer representing total memory
2247 """
2248 mem_total_kb = self.read_from_meminfo('MemTotal')
2249 kb_in_gb = float(2 ** 20)
2250 return int(round(mem_total_kb / kb_in_gb))
2251
2252
2253 def get_disk_size_gb(self):
2254 """Get size of disk in GB (10^9)
2255
2256 @returns an integer representing size of disk, 0 in Error Case
2257 """
2258 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2259 result = self.run(command, ignore_status=True)
2260 if result.exit_status != 0:
2261 return 0
2262 _, _, block, _ = re.split(r' +', result.stdout.strip())
2263 byte_per_block = 1024.0
2264 disk_kb_in_gb = 1e9
2265 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2266
2267
2268 def get_battery_size(self):
2269 """Get size of battery in Watt-hour via sysfs
2270
2271 This method assumes that battery support voltage_min_design and
2272 charge_full_design sysfs.
2273
2274 @returns a float representing Battery size, 0 if error.
2275 """
2276 # sysfs report data in micro scale
2277 battery_scale = 1e6
2278
2279 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2280 result = self.run(command, ignore_status=True)
2281 if result.exit_status != 0:
2282 return 0
2283 voltage = float(result.stdout.strip()) / battery_scale
2284
2285 command = 'cat /sys/class/power_supply/*/charge_full_design'
2286 result = self.run(command, ignore_status=True)
2287 if result.exit_status != 0:
2288 return 0
2289 amphereHour = float(result.stdout.strip()) / battery_scale
2290
2291 return voltage * amphereHour
2292
2293
2294 def get_low_battery_shutdown_percent(self):
2295 """Get the percent-based low-battery shutdown threshold.
2296
2297 @returns a float representing low-battery shutdown percent, 0 if error.
2298 """
2299 ret = 0.0
2300 try:
2301 command = 'check_powerd_config --low_battery_shutdown_percent'
2302 ret = float(self.run(command).stdout)
2303 except error.CmdError:
2304 logging.debug("Can't run %s", command)
2305 except ValueError:
2306 logging.debug("Didn't get number from %s", command)
2307
2308 return ret
2309
2310
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002311 def has_hammer(self):
2312 """Check whether DUT has hammer device or not.
2313
2314 @returns boolean whether device has hammer or not
2315 """
2316 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2317 return self.run(command, ignore_status=True).exit_status == 0
2318
2319
Niranjan Kumar34618872017-05-31 12:57:09 -07002320 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002321 """Returns True if the specified switch was provided to Chrome.
2322
2323 @param switch The chrome switch to search for.
2324 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002325
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002326 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2327 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002328
2329
2330 def oobe_triggers_update(self):
2331 """Returns True if this host has an OOBE flow during which
2332 it will perform an update check and perhaps an update.
2333 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2334 As more such flows are developed, code handling them needs
2335 to be added here.
2336
2337 @return Boolean indicating whether this host's OOBE triggers an update.
2338 """
2339 return self.is_chrome_switch_present(
2340 '--enterprise-enable-zero-touch-enrollment=hands-off')
2341
2342
Kevin Chenga2619dc2016-03-28 11:42:08 -07002343 # TODO(kevcheng): change this to just return the board without the
2344 # 'board:' prefix and fix up all the callers. Also look into removing the
2345 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002346 def get_board(self):
2347 """Determine the correct board label for this host.
2348
2349 @returns a string representing this host's board.
2350 """
2351 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2352 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002353 return (ds_constants.BOARD_PREFIX +
2354 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002355
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002356 def get_channel(self):
2357 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002358
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002359 @returns: a string represeting this host's build channel.
2360 (stable, dev, beta). None on fail.
2361 """
2362 return lsbrelease_utils.get_chromeos_channel(
2363 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002364
Kevin Chenga328da62016-03-31 10:49:04 -07002365 def get_power_supply(self):
2366 """
2367 Determine what type of power supply the host has
2368
2369 @returns a string representing this host's power supply.
2370 'power:battery' when the device has a battery intended for
2371 extended use
2372 'power:AC_primary' when the device has a battery not intended
2373 for extended use (for moving the machine, etc)
2374 'power:AC_only' when the device has no battery at all.
2375 """
2376 psu = self.run(command='mosys psu type', ignore_status=True)
2377 if psu.exit_status:
2378 # The psu command for mosys is not included for all platforms. The
2379 # assumption is that the device will have a battery if the command
2380 # is not found.
2381 return 'power:battery'
2382
2383 psu_str = psu.stdout.strip()
2384 if psu_str == 'unknown':
2385 return None
2386
2387 return 'power:%s' % psu_str
2388
2389
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002390 def has_battery(self):
2391 """Determine if DUT has a battery.
2392
2393 Returns:
2394 Boolean, False if known not to have battery, True otherwise.
2395 """
2396 rv = True
2397 power_supply = self.get_power_supply()
2398 if power_supply == 'power:battery':
2399 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2400 board_type = self.get_board_type()
2401 if board_type in _NO_BATTERY_BOARD_TYPE:
2402 logging.warn('Do NOT believe type %s has battery. '
2403 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002404 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002405 ignore_status=True)
2406 logging.debug(psu)
2407 rv = False
2408 elif power_supply == 'power:AC_only':
2409 rv = False
2410
2411 return rv
2412
2413
Kevin Chenga328da62016-03-31 10:49:04 -07002414 def get_servo(self):
2415 """Determine if the host has a servo attached.
2416
2417 If the host has a working servo attached, it should have a servo label.
2418
2419 @return: string 'servo' if the host has servo attached. Otherwise,
2420 returns None.
2421 """
2422 return 'servo' if self._servo_host else None
2423
2424
Kevin Chenga328da62016-03-31 10:49:04 -07002425 def has_internal_display(self):
2426 """Determine if the device under test is equipped with an internal
2427 display.
2428
2429 @return: 'internal_display' if one is present; None otherwise.
2430 """
2431 from autotest_lib.client.cros.graphics import graphics_utils
2432 from autotest_lib.client.common_lib import utils as common_utils
2433
2434 def __system_output(cmd):
2435 return self.run(cmd).stdout
2436
2437 def __read_file(remote_path):
2438 return self.run('cat %s' % remote_path).stdout
2439
2440 # Hijack the necessary client functions so that we can take advantage
2441 # of the client lib here.
2442 # FIXME: find a less hacky way than this
2443 original_system_output = utils.system_output
2444 original_read_file = common_utils.read_file
2445 utils.system_output = __system_output
2446 common_utils.read_file = __read_file
2447 try:
2448 return ('internal_display' if graphics_utils.has_internal_display()
2449 else None)
2450 finally:
2451 utils.system_output = original_system_output
2452 common_utils.read_file = original_read_file
2453
2454
Dan Shi85276d42014-04-08 22:11:45 -07002455 def is_boot_from_usb(self):
2456 """Check if DUT is boot from USB.
2457
2458 @return: True if DUT is boot from usb.
2459 """
2460 device = self.run('rootdev -s -d').stdout.strip()
2461 removable = int(self.run('cat /sys/block/%s/removable' %
2462 os.path.basename(device)).stdout.strip())
2463 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002464
2465
2466 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002467 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002468
2469 @param key: meminfo requested
2470
2471 @return the memory value as a string
2472
2473 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002474 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2475 logging.debug('%s', meminfo)
2476 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002477
2478
Rohit Makasana98e696f2016-06-03 18:48:10 -07002479 def get_cpu_arch(self):
2480 """Returns CPU arch of the device.
2481
2482 @return CPU architecture of the DUT.
2483 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002484 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002485 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2486 ignore_status=True).stdout:
2487 return 'x86_64'
2488 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2489 ignore_status=True).stdout:
2490 return 'arm'
2491 return 'i386'
2492
2493
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002494 def get_board_type(self):
2495 """
2496 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002497 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2498
2499 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002500 """
Danny Chan471a8d12015-08-18 14:57:41 -07002501 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2502 ignore_status=True).stdout
2503 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002504 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002505 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002506
2507
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002508 def get_arc_version(self):
2509 """Return ARC version installed on the DUT.
2510
2511 @returns ARC version as string if the CrOS build has ARC, else None.
2512 """
2513 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2514 ignore_status=True).stdout
2515 if arc_version:
2516 return arc_version.split('=')[-1].strip()
2517 return None
2518
2519
Gilad Arnolda76bef02015-09-29 13:55:15 -07002520 def get_os_type(self):
2521 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002522
2523
Kevin Chenga2619dc2016-03-28 11:42:08 -07002524 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002525 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002526 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002527
2528
2529 def get_default_power_method(self):
2530 """
2531 Get the default power method for power_on/off/cycle() methods.
2532 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2533 """
2534 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002535 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002536 if self.servo and self.servo.supports_built_in_pd_control():
2537 self._default_power_method = self.POWER_CONTROL_CCD
2538 else:
2539 logging.debug('Either servo is unitialized or the servo '
2540 'setup does not support pd controls. Falling '
2541 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002542 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002543
2544
2545 def find_usb_devices(self, idVendor, idProduct):
2546 """
2547 Get usb device sysfs name for specific device.
2548
2549 @param idVendor Vendor ID to search in sysfs directory.
2550 @param idProduct Product ID to search in sysfs directory.
2551
2552 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2553 """
2554 # Look for matching file and cut at position 7 to get dir name.
2555 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2556
2557 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2558 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2559
2560 # Use uniq -d to print duplicate line from both command
2561 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2562
2563 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2564
2565
2566 def bind_usb_device(self, usb_node):
2567 """
2568 Bind usb device
2569
2570 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2571 """
2572 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2573 self.run(cmd, ignore_status=True)
2574
2575
2576 def unbind_usb_device(self, usb_node):
2577 """
2578 Unbind usb device
2579
2580 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2581 """
2582 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2583 self.run(cmd, ignore_status=True)
2584
2585
2586 def get_wlan_ip(self):
2587 """
2588 Get ip address of wlan interface.
2589
2590 @return ip address of wlan or empty string if wlan is not connected.
2591 """
2592 cmds = [
2593 'iw dev', # List wlan physical device
2594 'grep Interface', # Grep only interface name
2595 'cut -f 2 -d" "', # Cut the name part
2596 'xargs ifconfig', # Feed it to ifconfig to get ip
2597 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2598 'cut -f 2 -d " "' # Cut the ip part
2599 ]
2600 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002601
2602 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2603 """
2604 Connect to wifi network
2605
2606 @param ssid SSID of the wifi network.
2607 @param passphrase Passphrase of the wifi network. None if not existed.
2608 @param security Security of the wifi network. Default to "psk" if
2609 passphase is given without security. Possible values
2610 are "none", "psk", "802_1x".
2611
2612 @return True if succeed, False if not.
2613 """
2614 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2615 if passphrase:
2616 cmd += ' ' + passphrase
2617 if security:
2618 cmd += ' ' + security
2619 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002620
2621 def get_device_repair_state(self):
2622 """Get device repair state"""
2623 return self._device_repair_state
2624
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002625 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002626 """Set device repair state.
2627
2628 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002629 file in result directory. The file will be read by Lucifer. The
2630 file will not be created if result directory not specified.
2631
2632 @params state: The new state for the device.
2633 @params resultdir: The path to result directory. If path not provided
2634 will be attempt to get retrieve it from job
2635 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002636 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002637 resultdir = resultdir or getattr(self.job, 'resultdir', '')
2638 if resultdir:
2639 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07002640 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002641 logging.info('Set device state as %s. '
2642 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07002643 else:
2644 logging.debug('Cannot write the device state due missing info '
2645 'about result dir.')
2646 self._device_repair_state = state
2647
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002648 def set_device_needs_replacement(self, resultdir=None):
2649 """Set device as required replacement.
2650
2651 @params resultdir: The path to result directory. If path not provided
2652 will be attempt to get retrieve it from job
2653 if present.
2654 """
2655 self.set_device_repair_state(
2656 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
2657 resultdir=resultdir)
2658
2659 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002660 """Check if device require manual attention to be fixed.
2661
2662 The state 'needs_manual_repair' can be set when auto repair cannot
2663 fix the device due hardware or cable issues.
2664 """
2665 # ignore the logic if state present
2666 # state can be set by any cros repair actions
2667 if self.get_device_repair_state():
2668 return
2669
2670 # set need manual attention if servo has hardware issue
2671 servo_state_required_manual_fix = [
2672 servo_constants.SERVO_STATE_NOT_CONNECTED,
2673 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
2674 servo_constants.SERVO_STATE_LID_OPEN_FAILED,
2675 servo_constants.SERVO_STATE_BAD_RIBBON_CABLE,
2676 servo_constants.SERVO_STATE_EC_BROKEN,
2677 ]
2678 if self.get_servo_state() in servo_state_required_manual_fix:
Otabek Kasimovde8eea32020-07-01 12:12:22 -07002679 data = {'host': self.hostname,
Otabek Kasimov832d9162020-07-27 19:24:57 -07002680 'state': cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR}
Otabek Kasimov6825b762020-06-23 23:42:44 -07002681 metrics.Counter(
2682 'chromeos/autotest/repair/special_dut_state'
2683 ).increment(fields=data)
2684 # TODO (otabek) unblock when be sure that we do not have flakiness
Otabek Kasimov832d9162020-07-27 19:24:57 -07002685 # self.set_device_repair_state(
2686 # cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)
Otabek Kasimov42506d02020-07-29 14:44:57 -07002687
2688 def is_file_system_writable(self, testdirs=None):
2689 """Check is the file systems are writable.
2690
2691 The standard linux response to certain unexpected file system errors
2692 (including hardware errors in block devices) is to change the file
2693 system status to read-only. This checks that that hasn't happened.
2694
2695 @param testdirs: List of directories to check. If no data provided
2696 then '/mnt/stateful_partition' and '/var/tmp'
2697 directories will be checked.
2698
2699 @returns boolean whether file-system writable.
2700 """
2701 def _check_dir(testdir):
2702 # check if we can create a file
2703 filename = os.path.join(testdir, 'writable_my_test_file')
2704 command = 'touch %s && rm %s' % (filename, filename)
2705 rv = self.run(command=command,
2706 timeout=30,
2707 ignore_status=True)
2708 is_writable = rv.exit_status == 0
2709 if not is_writable:
2710 logging.info('Cannot create a file in "%s"!'
2711 ' Probably the FS is read-only', testdir)
2712 logging.info("FileSystem is not writable!")
2713 return False
2714 return True
2715
2716 if not testdirs or len(testdirs) == 0:
2717 # N.B. Order matters here: Encrypted stateful is loop-mounted
2718 # from a file in unencrypted stateful, so we don't test for
2719 # errors in encrypted stateful if unencrypted fails.
2720 testdirs = ['/mnt/stateful_partition', '/var/tmp']
2721
2722 for dir in testdirs:
2723 # loop will be stopped if any directory fill fail the check
2724 try:
2725 if not _check_dir(dir):
2726 return False
2727 except Exception as e:
2728 # here expected only timeout error, all other will
2729 # be catch by 'ignore_status=True'
2730 logging.debug('Fail to check %s to write in it', dir)
2731 return False
2732 return True