blob: 3c096f94a6cf4326110d92d296f6bbfe62835f46 [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
Otabek Kasimov27bb2862020-08-10 14:40:45 -070041from autotest_lib.site_utils.admin_audit import verifiers as audit_verify
Simran Basid5e5e272012-09-24 15:23:59 -070042
Simran Basi382506b2016-09-13 14:58:15 -070043# In case cros_host is being ran via SSP on an older Moblab version with an
44# older chromite version.
45try:
46 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080047except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080048 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080049
Simran Basid5e5e272012-09-24 15:23:59 -070050
Dan Shib8540a52015-07-16 14:18:23 -070051CONFIG = global_config.global_config
52
beepsc87ff602013-07-31 21:53:00 -070053class FactoryImageCheckerException(error.AutoservError):
54 """Exception raised when an image is a factory image."""
55 pass
56
57
Fang Deng0ca40e22013-08-27 17:47:44 -070058class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070059 """Chromium OS specific subclass of Host."""
60
Simran Basi5ace6f22016-01-06 17:30:44 -080061 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
62
Scott Zawalski62bacae2013-03-05 10:40:32 -050063 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070064
Richard Barnette03a0c132012-11-05 12:40:35 -080065 # Timeout values (in seconds) associated with various Chrome OS
66 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070067 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080068 # In general, a good rule of thumb is that the timeout can be up
69 # to twice the typical measured value on the slowest platform.
70 # The times here have not necessarily been empirically tested to
71 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070072 #
73 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080074 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
75 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080076 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070077 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080078 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070079 # screen delay, time to start the network on the DUT, and the
80 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070081 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080082 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080083 # network.
beepsf079cfb2013-09-18 17:49:51 -070084 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080085 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
86 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070087
88 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080089 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080090 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070091 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070092 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070093 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080094 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070095
Dan Shica503482015-03-30 17:23:25 -070096 # Minimum OS version that supports server side packaging. Older builds may
97 # not have server side package built or with Autotest code change to support
98 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070099 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -0700100 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700101
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800102 # REBOOT_TIMEOUT: How long to wait for a reboot.
103 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700104 # We have a long timeout to ensure we don't flakily fail due to other
105 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700106 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
107 # return from reboot' bug is solved.
108 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700109
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800110 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
111 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700112 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
113 # since changing servo role will reset USB state
114 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800115 _USB_POWER_TIMEOUT = 5
116 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700117 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800118
Fang Dengdeba14f2014-11-14 11:54:09 -0800119 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
120 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700121
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800122 # Constants used in ping_wait_up() and ping_wait_down().
123 #
124 # _PING_WAIT_COUNT is the approximate number of polling
125 # cycles to use when waiting for a host state change.
126 #
127 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
128 # for arguments to the internal _ping_wait_for_status()
129 # method.
130 _PING_WAIT_COUNT = 40
131 _PING_STATUS_DOWN = False
132 _PING_STATUS_UP = True
133
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800134 # Allowed values for the power_method argument.
135
Garry Wang5e5538a2019-04-08 15:36:18 -0700136 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
137 # DUTs except those with servo_v4 CCD.
138 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
139 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800140 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
141 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
142 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700143 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800144 POWER_CONTROL_SERVO = 'servoj10'
145 POWER_CONTROL_MANUAL = 'manual'
146
147 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700148 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800149 POWER_CONTROL_SERVO,
150 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800151
Simran Basi5e6339a2013-03-21 11:34:32 -0700152 _RPM_OUTLET_CHANGED = 'outlet_changed'
153
Dan Shi9cb0eec2014-06-03 09:04:50 -0700154 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700155 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700156 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700157
Brent Peterson1cb623a2020-01-09 13:14:28 -0800158 # Regular expression for extracting EC version string
159 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
160
161 # Regular expression for extracting BIOS version string
162 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
163
Brent Petersonc70a1832020-01-24 15:54:35 -0800164 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700165 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800166
J. Richard Barnette964fba02012-10-24 17:34:29 -0700167 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800168 def check_host(host, timeout=10):
169 """
170 Check if the given host is a chrome-os host.
171
172 @param host: An ssh host representing a device.
173 @param timeout: The timeout for the run command.
174
175 @return: True if the host device is chromeos.
176
beeps46dadc92013-11-07 14:07:10 -0800177 """
178 try:
Allen Liad719c12017-06-27 23:48:04 +0000179 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700180 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700181 '! grep -q moblab /etc/lsb-release && '
182 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700183 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700184 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000185 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700186 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
187 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800188 return not (
189 lsbrelease_utils.is_jetstream(
190 lsb_release_content=lsb_release_content) or
191 lsbrelease_utils.is_gce_board(
192 lsb_release_content=lsb_release_content))
193
beeps46dadc92013-11-07 14:07:10 -0800194 except (error.AutoservRunError, error.AutoservSSHTimeout):
195 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700196
197 return False
beeps46dadc92013-11-07 14:07:10 -0800198
199
200 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800201 def get_chameleon_arguments(args_dict):
202 """Extract chameleon options from `args_dict` and return the result.
203
204 Recommended usage:
205 ~~~~~~~~
206 args_dict = utils.args_to_dict(args)
207 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
208 host = hosts.create_host(machine, chameleon_args=chameleon_args)
209 ~~~~~~~~
210
211 @param args_dict Dictionary from which to extract the chameleon
212 arguments.
213 """
Sam McNally66594ca2019-12-09 12:45:44 +1100214 chameleon_args = {key: args_dict[key]
215 for key in ('chameleon_host', 'chameleon_port')
216 if key in args_dict}
217 if 'chameleon_ssh_port' in args_dict:
218 chameleon_args['port'] = int(args_dict['chameleon_ssh_port'])
219 return chameleon_args
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800220
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800221 @staticmethod
222 def get_btpeer_arguments(args_dict):
223 """Extract btpeer options from `args_dict` and return the result.
224
225 This is used to parse details of Bluetooth peer.
226 Recommended usage:
227 ~~~~~~~~
228 args_dict = utils.args_to_dict(args)
229 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700230 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800231 ~~~~~~~~
232
233 @param args_dict: Dictionary from which to extract the btpeer
234 arguments.
235 """
236 if 'btpeer_host_list' in args_dict:
237 result = []
238 for btpeer in args_dict['btpeer_host_list'].split(','):
239 result.append({key: value for key,value in
240 zip(('btpeer_host','btpeer_port'),
241 btpeer.split(':'))})
242 return result
243 else:
Anand K Mistrye8933092020-08-05 14:49:41 +1000244 return {key: args_dict[key]
245 for key in ('btpeer_host', 'btpeer_port', 'btpeer_ssh_port')
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800246 if key in args_dict}
247
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800248
249 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700250 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800251 """Extract chameleon options from `args_dict` and return the result.
252
253 Recommended usage:
254 ~~~~~~~~
255 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700256 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
257 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800258 ~~~~~~~~
259
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700260 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800261 arguments.
262 """
Allen Li083866b2016-08-18 10:07:10 -0700263 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700264 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700265 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800266
267
268 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800269 def get_servo_arguments(args_dict):
270 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800271
272 Recommended usage:
273 ~~~~~~~~
274 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700275 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800276 host = hosts.create_host(machine, servo_args=servo_args)
277 ~~~~~~~~
278
279 @param args_dict Dictionary from which to extract the servo
280 arguments.
281 """
Garry Wang11b5e872020-03-11 15:14:08 -0700282 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
283 servo_constants.SERVO_PORT_ATTR,
284 servo_constants.SERVO_BOARD_ATTR,
285 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200286 servo_args = {key: args_dict[key]
287 for key in servo_attrs
288 if key in args_dict}
289 return (
290 None
Garry Wang11b5e872020-03-11 15:14:08 -0700291 if servo_constants.SERVO_HOST_ATTR in servo_args
292 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200293 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700294
J. Richard Barnette964fba02012-10-24 17:34:29 -0700295
J. Richard Barnette91137f02016-03-10 16:52:26 -0800296 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700297 pdtester_args=None, try_lab_servo=False,
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700298 try_servo_repair=False, btpeer_args=[],
J. Richard Barnette91137f02016-03-10 16:52:26 -0800299 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700300 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800301 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700302
Fang Denge545abb2014-12-30 18:43:47 -0800303 This method will attempt to create the test-assistant object
304 (chameleon/servo) when it is needed by the test. Check
305 the docstring of chameleon_host.create_chameleon_host and
306 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700307
Fang Denge545abb2014-12-30 18:43:47 -0800308 @param hostname: Hostname of the dut.
309 @param chameleon_args: A dictionary that contains args for creating
310 a ChameleonHost. See chameleon_host for details.
311 @param servo_args: A dictionary that contains args for creating
312 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700313 @param try_lab_servo: When true, indicates that an attempt should
314 be made to create a ServoHost for a DUT in
315 the test lab, even if not required by
316 `servo_args`. See servo_host for details.
317 @param try_servo_repair: If a servo host is created, check it
318 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800319 See servo_host for details.
320 @param ssh_verbosity_flag: String, to pass to the ssh command to control
321 verbosity.
322 @param ssh_options: String, other ssh options to pass to the ssh
323 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700324 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700325 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700326 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800327 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700328 # hold special dut_state for repair process
329 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700330 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700331 # self.env is a dictionary of environment variable settings
332 # to be exported for commands run on the host.
333 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
334 # errors that might happen.
335 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700336 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700337 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700338 _servo_host, servo_state = servo_host.create_servo_host(
339 dut=self,
340 servo_args=servo_args,
341 try_lab_servo=try_lab_servo,
342 try_servo_repair=try_servo_repair,
343 dut_host_info=self.host_info_store.get())
344 self.set_servo_host(_servo_host, servo_state)
Garry Wang5e5538a2019-04-08 15:36:18 -0700345 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700346
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800347 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800348 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700349 dut=self.hostname,
350 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800351 if self._chameleon_host:
352 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800353 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800354 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700355
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700356 # Initialize Bluetooth peers.
357 try:
358 self.initialize_btpeer(btpeer_args)
359 except Exception as e:
360 logging.error('Exception %s in initialize_btpeer', str(e))
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800361
howardchung83e55272019-08-08 14:08:05 +0800362 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700363 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700364 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800365
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700366 if self._pdtester_host:
367 self.pdtester_servo = self._pdtester_host.get_servo()
368 logging.info('pdtester_servo: %r', self.pdtester_servo)
369 # Create the pdtester object used to access the ec uart
370 self.pdtester = pdtester.PDTester(self.pdtester_servo,
371 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800372 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700373 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800374
Fang Deng5d518f42013-08-02 14:04:32 -0700375
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800376 def initialize_btpeer(self, btpeer_args):
377 """ Initialize the Bluetooth peers
378
379 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
380 is chameleon host on Raspberry Pi.
381 @param btpeer_args: A dictionary that contains args for creating
382 a ChameleonHost. See chameleon_host for details.
383
384 """
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700385 #TODO (b:142486063) Remove the try..except
386 try:
387 self._btpeer_host_list = []
388 self.btpeer_list = []
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800389 self.btpeer = None
390
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700391 if type(btpeer_args) is list:
392 btpeer_args_list = btpeer_args
393 else:
394 btpeer_args_list = [btpeer_args]
395
396 self._btpeer_host_list = chameleon_host.create_btpeer_host(
397 dut=self.hostname, btpeer_args_list=btpeer_args_list)
398 logging.debug('Bluetooth peer hosts are %s',
399 self._btpeer_host_list)
400 self.btpeer_list = [_host.create_chameleon_board() for _host in
401 self._btpeer_host_list if _host is not None]
402
403 if len(self.btpeer_list) > 0:
404 self.btpeer = self.btpeer_list[0]
405
406 logging.debug('After initialize_btpeer btpeer_list %s '
407 'btpeer_host_list is %s and btpeer is %s',
408 self.btpeer_list, self._btpeer_host_list,
409 self.btpeer)
410 except Exception as e:
411 logging.error('Exception %s in initialize_btpeer', str(e))
412
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800413
414
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000415 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700416 """Get latest stable cros image name from AFE.
417
418 Use the board name from the info store. Should that fail, try to
419 retrieve the board name from the host's installed image itself.
420
421 @returns: current stable cros image name for this host.
422 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700423 info = self.host_info_store.get()
424 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700425 logging.warn('No board label value found. Trying to infer '
426 'from the host itself.')
427 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700428 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700429 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
430 logging.error('Also failed to get the board name from the DUT '
431 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700432 raise error.AutoservError('Cannot determine board of the DUT'
433 ' while getting repair image name.')
434 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500435
436
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700437 def host_version_prefix(self, image):
438 """Return version label prefix.
439
440 In case the CrOS provisioning version is something other than the
441 standard CrOS version e.g. CrOS TH version, this function will
442 find the prefix from provision.py.
443
444 @param image: The image name to find its version prefix.
445 @returns: A prefix string for the image type.
446 """
447 return provision.get_version_label_prefix(image)
448
Andrew Luo3332ab22020-04-28 16:42:03 -0700449 def stage_build_to_usb(self, build):
450 """Stage the current ChromeOS image on the USB stick connected to the
451 servo.
452
453 @param build: The build to download and send to USB.
454 """
455 if not self.servo:
456 raise error.TestError('Host %s does not have servo.' %
457 self.hostname)
458
459 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700460
461 try:
462 self.servo.image_to_servo_usb(update_url)
463 finally:
464 # servo.image_to_servo_usb turned the DUT off, so turn it back on
465 logging.debug('Turn DUT power back on.')
466 self.servo.get_power_state_controller().power_on()
467
Andrew Luo3332ab22020-04-28 16:42:03 -0700468 logging.debug('ChromeOS image %s is staged on the USB stick.',
469 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700470
beepsdae65fd2013-07-26 16:24:41 -0700471 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700472 """
473 Make sure job_repo_url of this host is valid.
474
joychen03eaad92013-06-26 09:55:21 -0700475 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700476 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
477 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
478 download and extract it. If the devserver embedded in the url is
479 unresponsive, update the job_repo_url of the host after staging it on
480 another devserver.
481
482 @param job_repo_url: A url pointing to the devserver where the autotest
483 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700484 @param tag: The tag from the server job, in the format
485 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700486
487 @raises DevServerException: If we could not resolve a devserver.
488 @raises AutoservError: If we're unable to save the new job_repo_url as
489 a result of choosing a new devserver because the old one failed to
490 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700491 @raises urllib2.URLError: If the devserver embedded in job_repo_url
492 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700493 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800494 info = self.host_info_store.get()
495 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700496 if not job_repo_url:
497 logging.warning('No job repo url set on host %s', self.hostname)
498 return
499
500 logging.info('Verifying job repo url %s', job_repo_url)
501 devserver_url, image_name = tools.get_devserver_build_from_package_url(
502 job_repo_url)
503
beeps0c865032013-07-30 11:37:06 -0700504 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700505
506 logging.info('Staging autotest artifacts for %s on devserver %s',
507 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700508
509 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700510 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700511 stage_time = time.time() - start_time
512
513 # Record how much of the verification time comes from a devserver
514 # restage. If we're doing things right we should not see multiple
515 # devservers for a given board/build/branch path.
516 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800517 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700518 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800519 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700520 pass
521 else:
beeps0c865032013-07-30 11:37:06 -0700522 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700523 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700524 stats_key = {
525 'board': board,
526 'build_type': build_type,
527 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700528 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700529 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800530
531 monarch_fields = {
532 'board': board,
533 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800534 'branch': branch,
535 'dev_server': devserver,
536 }
537 metrics.Counter(
538 'chromeos/autotest/provision/verify_url'
539 ).increment(fields=monarch_fields)
540 metrics.SecondsDistribution(
541 'chromeos/autotest/provision/verify_url_duration'
542 ).add(stage_time, fields=monarch_fields)
543
544
Dan Shicf4d2032015-03-12 15:04:21 -0700545 def stage_server_side_package(self, image=None):
546 """Stage autotest server-side package on devserver.
547
548 @param image: Full path of an OS image to install or a build name.
549
550 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700551
552 @raise: error.AutoservError if fail to locate the build to test with, or
553 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700554 """
Dan Shid37736b2016-07-06 15:10:29 -0700555 # If enable_drone_in_restricted_subnet is False, do not set hostname
556 # in devserver.resolve call, so a devserver in non-restricted subnet
557 # is picked to stage autotest server package for drone to download.
558 hostname = self.hostname
559 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
560 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700561 if image:
562 image_name = tools.get_build_from_image(image)
563 if not image_name:
564 raise error.AutoservError(
565 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700566 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700567 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700568 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800569 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700570 if job_repo_url:
571 devserver_url, image_name = (
572 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700573 # If enable_drone_in_restricted_subnet is True, use the
574 # existing devserver. Otherwise, resolve a new one in
575 # non-restricted subnet.
576 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
577 ds = dev_server.ImageServer(devserver_url)
578 else:
579 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800580 elif info.build is not None:
581 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700582 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700583 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800584 raise error.AutoservError(
585 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700586 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700587
588 # Get the OS version of the build, for any build older than
589 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
590 match = re.match('.*/R\d+-(\d+)\.', image_name)
591 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700592 raise error.AutoservError(
593 'Build %s is older than %s. Server side packaging is '
594 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700595
Dan Shicf4d2032015-03-12 15:04:21 -0700596 ds.stage_artifacts(image_name, ['autotest_server_package'])
597 return '%s/static/%s/%s' % (ds.url(), image_name,
598 'autotest_server_package.tar.bz2')
599
600
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700601 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700602 """Stage a build on a devserver and return the update_url.
603
604 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700605 @param artifact: a string like 'test_image'. Requests
606 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700607 @returns a tuple of (image_name, URL) like
608 (lumpy-release/R27-3837.0.0,
609 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700610 """
611 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000612 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700613 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800614 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700615 devserver.stage_artifacts(image_name, [artifact])
616 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700617 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700618 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700619 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700620 else:
621 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700622
623
beepse539be02013-07-31 21:57:39 -0700624 def stage_factory_image_for_servo(self, image_name):
625 """Stage a build on a devserver and return the update_url.
626
627 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700628
beepse539be02013-07-31 21:57:39 -0700629 @return: An update URL, eg:
630 http://<devserver>/static/canary-channel/\
631 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700632
633 @raises: ValueError if the factory artifact name is missing from
634 the config.
635
beepse539be02013-07-31 21:57:39 -0700636 """
637 if not image_name:
638 logging.error('Need an image_name to stage a factory image.')
639 return
640
Dan Shib8540a52015-07-16 14:18:23 -0700641 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700642 'CROS', 'factory_artifact', type=str, default='')
643 if not factory_artifact:
644 raise ValueError('Cannot retrieve the factory artifact name from '
645 'autotest config, and hence cannot stage factory '
646 'artifacts.')
647
beepse539be02013-07-31 21:57:39 -0700648 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800649 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700650 devserver.stage_artifacts(
651 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700652 [factory_artifact],
653 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700654
655 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
656
657
Laurence Goodby778c9a42017-05-24 19:24:07 -0700658 def prepare_for_update(self):
659 """Prepares the DUT for an update.
660
661 Subclasses may override this to perform any special actions
662 required before updating.
663 """
Laurence Goodby468de252017-06-08 17:22:53 -0700664 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700665
666
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800667 def _clear_fw_version_labels(self, rw_only):
668 """Clear firmware version labels from the machine.
669
670 @param rw_only: True to only clear fwrw_version; otherewise, clear
671 both fwro_version and fwrw_version.
672 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700673 info = self.host_info_store.get()
674 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800675 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700676 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
677 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700678
679
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800680 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700681 """Add firmware version label to the machine.
682
683 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800684 @param rw_only: True to only add fwrw_version; otherwise, add both
685 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700686
687 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700688 info = self.host_info_store.get()
689 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800690 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700691 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
692 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700693
694
Namyoon Woo33f38852020-04-13 17:26:58 -0700695 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800696 """Search for the latest package release version from the image archive,
697 and return it.
698
Namyoon Woo33f38852020-04-13 17:26:58 -0700699 @param platform: platform name, a.k.a. board or model
700 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800701
Namyoon Woo33f38852020-04-13 17:26:58 -0700702 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
703 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800704 or None if LATEST release file does not exist.
705 """
706
Namyoon Woo33f38852020-04-13 17:26:58 -0700707 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800708
Namyoon Woo33f38852020-04-13 17:26:58 -0700709 # Search the image path in reference board archive as well.
710 # For example, bob has its binary image under its reference board (gru)
711 # image archive.
712 if ref_board:
713 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800714
Namyoon Woo33f38852020-04-13 17:26:58 -0700715 for board in platforms:
716 # Read 'LATEST-1.0.0' file
717 branch_dir = provision.FW_BRANCH_GLOB % board
718 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
719 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800720
Namyoon Woo33f38852020-04-13 17:26:58 -0700721 try:
722 # The result could be one or more.
723 result = utils.system_output('gsutil ls -d ' + latest_file)
724
725 candidates = re.findall('gs://.*', result)
726
727 # Found the directory candidates. No need to check the other
728 # board name cadidates. Let's break the loop.
729 break
730 except error.CmdError:
731 # It doesn't exist. Let's move on to the next item.
732 pass
733 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800734 logging.error('No LATEST release info is available.')
735 return None
736
Namyoon Woo406c7d42020-01-24 15:57:11 -0800737 for cand_dir in candidates:
738 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800739
Namyoon Woo406c7d42020-01-24 15:57:11 -0800740 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700741 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800742 try:
743 # Check if release_path does exist.
744 release = utils.system_output('gsutil ls -d ' + release_path)
745 # Now 'release' has a full directory path: e.g.
746 # gs://chromeos-image-archive/firmware-octopus-11297.B-
747 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
748
749 # Remove "gs://chromeos-image-archive".
750 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
751
752 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
753 return release.strip('/')
754 except error.CmdError:
755 # The directory might not exist. Let's try next candidate.
756 pass
757 else:
758 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800759
Brent Peterson1cb623a2020-01-09 13:14:28 -0800760 @staticmethod
761 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800762 """Get version string from binary image using regular expression.
763
764 @param image: Binary image to search
765 @param version_regex: Regular expression to search for
766
767 @return Version string
768
769 @raises TestFail if no version string is found in image
770 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800771 with open(image, 'rb') as f:
772 image_data = f.read()
773 match = re.findall(version_regex, image_data)
774 if match:
775 return match[0]
776 else:
777 raise error.TestFail('Failed to read version from %s.' % image)
778
779
Garry Wangad2a1712020-03-26 15:06:43 -0700780 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800781 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700782 try_scp=False, install_ec=True, install_bios=True,
783 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700784 """Install firmware to the DUT.
785
786 Use stateful update if the DUT is already running the same build.
787 Stateful update does not update kernel and tends to run much faster
788 than a full reimage. If the DUT is running a different build, or it
789 failed to do a stateful update, full update, including kernel update,
790 will be applied to the DUT.
791
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800792 Once a host enters firmware_install its fw[ro|rw]_version label will
793 be removed. After the firmware is updated successfully, a new
794 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700795
796 @param build: The build version to which we want to provision the
797 firmware of the machine,
798 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800799 @param rw_only: True to only install firmware to its RW portions. Keep
800 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700801 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800802 @param local_tarball: Path to local firmware image for installing
803 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800804 @param verify_version: True to verify EC and BIOS versions after
805 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800806 @param try_scp: False to always program using servo, true to try copying
807 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700808 @param install_ec: True to install EC FW, and False to skip it.
809 @param install_bios: True to install BIOS, and False to skip it.
810 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700811
812 TODO(dshi): After bug 381718 is fixed, update here with corresponding
813 exceptions that could be raised.
814
815 """
816 if not self.servo:
817 raise error.TestError('Host %s does not have servo.' %
818 self.hostname)
819
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700820 # Get the DUT board name from AFE.
821 info = self.host_info_store.get()
822 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700823 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800824
825 if board is None or board == '':
826 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700827
Namyoon Woo382e5892020-05-20 16:48:40 -0700828 # if board_as argument is passed, then use it instead of the original
829 # board name.
830 if board_as:
831 board = board_as
832
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700833 if model is None or model == '':
Namyoon Woofb16eae2020-08-14 10:02:39 -0700834 try:
835 model = self.get_platform_from_fwid()
836 except Exception as e:
837 logging.warn('Dut is unresponsive: %s', str(e))
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700838
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800839 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700840 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800841 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700842 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800843
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700844 try:
845 ds = dev_server.ImageServer.resolve(build, self.hostname)
846 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800847
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700848 if not dest:
849 tmpd = autotemp.tempdir(unique_id='fwimage')
850 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800851
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700852 # Download firmware image
853 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
854 local_tarball = os.path.join(dest, os.path.basename(fwurl))
855 ds.download_file(fwurl, local_tarball)
856 except Exception as e:
857 raise error.TestError('Failed to download firmware package: %s'
858 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700859
Namyoon Woo382e5892020-05-20 16:48:40 -0700860 ec_image = None
861 if install_ec:
862 # Extract EC image from tarball
863 logging.info('Extracting EC image.')
864 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800865
Namyoon Woo382e5892020-05-20 16:48:40 -0700866 bios_image = None
867 if install_bios:
868 # Extract BIOS image from tarball
869 logging.info('Extracting BIOS image.')
870 bios_image = self.servo.extract_bios_image(board, model,
871 local_tarball)
872
873 if not bios_image and not ec_image:
874 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800875
Brent Petersonc70a1832020-01-24 15:54:35 -0800876 # Clear firmware version labels
877 self._clear_fw_version_labels(rw_only)
878
879 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800880 try:
Garry Wang50e4a492020-08-05 12:29:57 -0700881 # Check if copying to DUT is enabled and DUT is available
882 if try_scp and self.is_up():
Brent Petersonc70a1832020-01-24 15:54:35 -0800883 # DUT is available, make temp firmware directory to store images
884 logging.info('Making temp folder.')
885 dest_folder = '/tmp/firmware'
886 self.run('mkdir -p ' + dest_folder)
887
Namyoon Woo68b68082020-06-02 13:13:14 -0700888 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800889
Namyoon Woo382e5892020-05-20 16:48:40 -0700890 if bios_image:
891 # Send BIOS firmware image to DUT
892 logging.info('Sending BIOS firmware.')
893 dest_bios_path = os.path.join(dest_folder,
894 os.path.basename(bios_image))
895 self.send_file(bios_image, dest_bios_path)
896
897 # Initialize firmware update command for BIOS image
898 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800899
900 # Send EC firmware image to DUT when EC image was found
901 if ec_image:
902 logging.info('Sending EC firmware.')
903 dest_ec_path = os.path.join(dest_folder,
904 os.path.basename(ec_image))
905 self.send_file(ec_image, dest_ec_path)
906
907 # Add EC image to firmware update command
908 fw_cmd += ' -e %s' % dest_ec_path
909
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700910 # Make sure command is allowed to finish even if ssh fails.
911 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
912
Brent Peterson669edf42020-02-07 15:07:54 -0800913 # Update firmware on DUT
914 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700915 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700916 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700917 except error.AutoservRunError as e:
918 if e.result_obj.exit_status != 255:
919 raise
920 elif ec_image:
921 logging.warn("DUT network dropped during update"
922 " (often caused by EC resetting USB)")
923 else:
924 logging.error("DUT network dropped during update"
925 " (unexpected, since no EC image)")
926 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800927 else:
928 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800929 if ec_image:
930 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700931 if bios_image:
932 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800933 if utils.host_is_in_lab_zone(self.hostname):
934 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800935
936 # Reboot and wait for DUT after installing firmware
937 logging.info('Rebooting DUT.')
938 self.servo.get_power_state_controller().reset()
939 time.sleep(self.servo.BOOT_DELAY)
940 self.test_wait_for_boot()
941
942 # When enabled verify EC and BIOS firmware version after programming
943 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800944 # Check programmed EC firmware when EC image was found
945 if ec_image:
946 logging.info('Checking EC firmware version.')
947 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800948 ec_version_prefix = dest_ec_version.split('_', 1)[0]
949 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800950 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800951 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800952 if dest_ec_version != image_ec_version:
953 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700954 'Failed to update EC firmware, version %s '
955 '(expected %s)' % (dest_ec_version,
956 image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800957
Namyoon Woo382e5892020-05-20 16:48:40 -0700958 if bios_image:
959 # Check programmed BIOS firmware against expected version
960 logging.info('Checking BIOS firmware version.')
961 dest_bios_version = self.get_firmware_version()
962 bios_version_prefix = dest_bios_version.split('.', 1)[0]
963 bios_regex = self._BIOS_REGEX % bios_version_prefix
964 image_bios_version = self.get_version_from_image(bios_image,
965 bios_regex)
966 if dest_bios_version != image_bios_version:
967 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700968 'Failed to update BIOS, version %s '
Namyoon Woo382e5892020-05-20 16:48:40 -0700969 '(expected %s)' % (dest_bios_version,
970 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700971 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700972 if tmpd:
973 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700974
975
beepsf079cfb2013-09-18 17:49:51 -0700976 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
977 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500978 """
979 Re-install the OS on the DUT by:
980 1) installing a test image on a USB storage device attached to the Servo
981 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800982 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700983 3) installing the image with chromeos-install.
984
Scott Zawalski62bacae2013-03-05 10:40:32 -0500985 @param image_url: If specified use as the url to install on the DUT.
986 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700987 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
988 Factory images need a longer usb_boot_timeout than regular
989 cros images.
990 @param install_timeout: The timeout to use when installing the chromeos
991 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800992
Scott Zawalski62bacae2013-03-05 10:40:32 -0500993 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700994
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800995 """
Garry Wang7b0e1b72020-03-25 19:08:59 -0700996 if image_url:
997 logging.info('Downloading image to USB, then booting from it.'
998 ' Usb boot timeout = %s', usb_boot_timeout)
999 else:
1000 logging.info('Booting from USB directly. Usb boot timeout = %s',
1001 usb_boot_timeout)
1002
1003 metrics_field = {'download': bool(image_url)}
1004 metrics.Counter(
1005 'chromeos/autotest/provision/servo_install/download_image'
1006 ).increment(fields=metrics_field)
1007
Allen Li48a13fe2016-11-22 14:10:40 -08001008 with metrics.SecondsTimer(
1009 'chromeos/autotest/provision/servo_install/boot_duration'):
1010 self.servo.install_recovery_image(image_url)
1011 if not self.wait_up(timeout=usb_boot_timeout):
1012 raise hosts.AutoservRepairError(
1013 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001014 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001015
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001016 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1017 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001018 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001019 try:
1020 self.run('chromeos-tpm-recovery')
1021 except error.AutoservRunError:
1022 logging.warn('chromeos-tpm-recovery is too old.')
1023
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001024
Allen Li48a13fe2016-11-22 14:10:40 -08001025 with metrics.SecondsTimer(
1026 'chromeos/autotest/provision/servo_install/install_duration'):
1027 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001028 try:
1029 self.run('chromeos-install --yes',timeout=install_timeout)
1030 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001031 except Exception as e:
1032 storage_errors = [
1033 'No space left on device',
1034 'I/O error when trying to write primary GPT',
1035 'Input/output error while writing out',
1036 'cannot read GPT header',
Otabek Kasimov2b7e8302020-08-21 09:23:31 -07001037 'can not determine destination device',
1038 'wrong fs type',
1039 'bad superblock on',
Otabek Kasimov808cd832020-05-28 18:27:46 -07001040 ]
1041 has_error = [msg for msg in storage_errors if(msg in str(e))]
1042 if has_error:
1043 info = self.host_info_store.get()
1044 info.set_version_label(
1045 audit_const.DUT_STORAGE_STATE_PREFIX,
1046 audit_const.HW_STATE_NEED_REPLACEMENT)
1047 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001048 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001049 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001050 logging.debug(
1051 'Fail install image from USB; Storage error; %s', e)
1052 raise error.AutoservError(
1053 'Failed to install image from USB due to a suspect '
1054 'disk failure, DUT storage state changed to '
1055 'need_replacement, please check debug log '
1056 'for details.')
1057 else:
Otabek Kasimov27bb2862020-08-10 14:40:45 -07001058 # DUT will be marked for replacement if storage is bad.
1059 audit_verify.VerifyDutStorage(self).verify()
1060
Otabek Kasimov808cd832020-05-28 18:27:46 -07001061 logging.debug('Fail install image from USB; %s', e)
1062 raise error.AutoservError(
1063 'Failed to install image from USB due to unexpected '
1064 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001065 finally:
1066 # We need reset the DUT no matter re-install success or not,
1067 # as we don't want leave the DUT in boot from usb state.
1068 logging.info('Power cycling DUT through servo.')
1069 self.servo.get_power_state_controller().power_off()
1070 self.servo.switch_usbkey('off')
1071 # N.B. The Servo API requires that we use power_on() here
1072 # for two reasons:
1073 # 1) After turning on a DUT in recovery mode, you must turn
1074 # it off and then on with power_on() once more to
1075 # disable recovery mode (this is a Parrot specific
1076 # requirement).
1077 # 2) After power_off(), the only way to turn on is with
1078 # power_on() (this is a Storm specific requirement).
1079 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001080
1081 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001082 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001083 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1084 'test image after %d seconds' %
1085 self.BOOT_TIMEOUT,
1086 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001087
1088
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001089 def set_servo_host(self, host, servo_state = None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001090 """Set our servo host member, and associated servo.
1091
1092 @param host Our new `ServoHost`.
1093 """
1094 self._servo_host = host
1095 if self._servo_host is not None:
1096 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001097 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001098 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001099 else:
1100 self.servo = None
Otabek Kasimov41301a22020-05-10 15:28:21 -07001101 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001102 self.set_servo_state(servo_state)
1103
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001104
Richard Barnette9a26ad62016-06-10 12:03:08 -07001105 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001106 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001107 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001108
Richard Barnette9a26ad62016-06-10 12:03:08 -07001109 If the servo object is missing, attempt to repair the servo
1110 host. Repair failures are passed back to the caller.
1111
1112 @raise AutoservError: If there is no servo host for this CrOS
1113 host.
1114 """
1115 if self.servo:
1116 return
1117 if not self._servo_host:
1118 raise error.AutoservError('No servo host for %s.' %
1119 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001120 try:
1121 self._servo_host.repair()
1122 except:
1123 raise
1124 finally:
1125 self.set_servo_host(self._servo_host)
1126
1127
Otabek Kasimov41301a22020-05-10 15:28:21 -07001128 def set_servo_type(self):
1129 """Set servo info labels to dut host_info"""
1130 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001131 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001132 return
1133 servo_type = self.servo.get_servo_type()
1134 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001135 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001136 ' by `dut-control servo_type`! Please file a bug'
1137 ' and inform infra team as we are not expected '
1138 ' to reach this point.')
1139 return
1140 host_info = self.host_info_store.get()
1141 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1142 old_type = host_info.get_label_value(prefix)
1143 if old_type == servo_type:
1144 # do not need update
1145 return
1146 host_info.set_version_label(prefix, servo_type)
1147 self.host_info_store.commit(host_info)
1148 logging.info('ServoHost: servo_type updated to %s '
1149 '(previous: %s)', servo_type, old_type)
1150
1151
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001152 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001153 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001154 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001155 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001156 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001157 old_state = host_info.get_label_value(servo_state_prefix)
1158 if old_state == servo_state:
1159 # do not need update
1160 return
1161 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001162 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001163 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1164 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001165
1166
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001167 def get_servo_state(self):
1168 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001169 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001170 return host_info.get_label_value(servo_state_prefix)
1171
Otabek Kasimov41301a22020-05-10 15:28:21 -07001172
Garry Wang000c6c02020-05-11 21:27:23 -07001173 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1174 if smart_usbhub_detected is None:
1175 # skip the label update here as this indicate we wasn't able
1176 # to confirm usbhub type.
1177 return
1178 host_info = self.host_info_store.get()
1179 if (smart_usbhub_detected ==
1180 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1181 # skip label update if current label match the truth.
1182 return
1183 if smart_usbhub_detected:
1184 logging.info('Adding %s label to host %s',
1185 servo_constants.SMART_USBHUB_LABEL,
1186 self.hostname)
1187 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1188 else:
1189 logging.info('Removing %s label from host %s',
1190 servo_constants.SMART_USBHUB_LABEL,
1191 self.hostname)
1192 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1193 self.host_info_store.commit(host_info)
1194
1195
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001196 def repair(self):
1197 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001198
1199 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001200 not call back to the parent class, but instead relies on
1201 `self._repair_strategy` to coordinate the verification and
1202 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001203 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001204 message = 'Beginning repair for host %s board %s model %s'
1205 info = self.host_info_store.get()
1206 message %= (self.hostname, info.board, info.model)
1207 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001208 try:
1209 self._repair_strategy.repair(self)
1210 except hosts.AutoservVerifyDependencyError as e:
1211 # We don't want flag a DUT as failed if only non-critical
1212 # verifier(s) failed during the repair.
1213 if e.is_critical():
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001214 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001215 raise
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001216
Richard Barnette82c35912012-11-20 10:09:10 -08001217
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001218 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001219 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001220 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001221
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001222 if self._chameleon_host:
1223 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001224
1225 if self._servo_host:
1226 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001227
1228
Dan Shi49ca0932014-11-14 11:22:27 -08001229 def get_power_supply_info(self):
1230 """Get the output of power_supply_info.
1231
1232 power_supply_info outputs the info of each power supply, e.g.,
1233 Device: Line Power
1234 online: no
1235 type: Mains
1236 voltage (V): 0
1237 current (A): 0
1238 Device: Battery
1239 state: Discharging
1240 percentage: 95.9276
1241 technology: Li-ion
1242
1243 Above output shows two devices, Line Power and Battery, with details of
1244 each device listed. This function parses the output into a dictionary,
1245 with key being the device name, and value being a dictionary of details
1246 of the device info.
1247
1248 @return: The dictionary of power_supply_info, e.g.,
1249 {'Line Power': {'online': 'yes', 'type': 'main'},
1250 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001251 @raise error.AutoservRunError if power_supply_info tool is not found in
1252 the DUT. Caller should handle this error to avoid false failure
1253 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001254 """
1255 result = self.run('power_supply_info').stdout.strip()
1256 info = {}
1257 device_name = None
1258 device_info = {}
1259 for line in result.split('\n'):
1260 pair = [v.strip() for v in line.split(':')]
1261 if len(pair) != 2:
1262 continue
1263 if pair[0] == 'Device':
1264 if device_name:
1265 info[device_name] = device_info
1266 device_name = pair[1]
1267 device_info = {}
1268 else:
1269 device_info[pair[0]] = pair[1]
1270 if device_name and not device_name in info:
1271 info[device_name] = device_info
1272 return info
1273
1274
1275 def get_battery_percentage(self):
1276 """Get the battery percentage.
1277
1278 @return: The percentage of battery level, value range from 0-100. Return
1279 None if the battery info cannot be retrieved.
1280 """
1281 try:
1282 info = self.get_power_supply_info()
1283 logging.info(info)
1284 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001285 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001286 return None
1287
1288
Philip Chenaf69ead2020-03-27 13:06:42 -07001289 def get_battery_state(self):
1290 """Get the battery charging state.
1291
1292 @return: A string representing the battery charging state. It can be
1293 'Charging', 'Fully charged', or 'Discharging'.
1294 """
1295 try:
1296 info = self.get_power_supply_info()
1297 logging.info(info)
1298 return info['Battery']['state']
1299 except (KeyError, ValueError, error.AutoservRunError):
1300 return None
1301
1302
Daniel Campello8ca25c22019-12-13 16:48:26 -07001303 def get_battery_display_percentage(self):
1304 """Get the battery display percentage.
1305
1306 @return: The display percentage of battery level, value range from
1307 0-100. Return None if the battery info cannot be retrieved.
1308 """
1309 try:
1310 info = self.get_power_supply_info()
1311 logging.info(info)
1312 return float(info['Battery']['display percentage'])
1313 except (KeyError, ValueError, error.AutoservRunError):
1314 return None
1315
1316
Dan Shi49ca0932014-11-14 11:22:27 -08001317 def is_ac_connected(self):
1318 """Check if the dut has power adapter connected and charging.
1319
1320 @return: True if power adapter is connected and charging.
1321 """
1322 try:
1323 info = self.get_power_supply_info()
1324 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001325 except (KeyError, error.AutoservRunError):
1326 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001327
1328
Simran Basi5e6339a2013-03-21 11:34:32 -07001329 def _cleanup_poweron(self):
1330 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001331 info = self.host_info_store.get()
1332 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001333 return
1334 logging.debug('This host has recently interacted with the RPM'
1335 ' Infrastructure. Ensuring power is on.')
1336 try:
1337 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001338 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001339 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001340 logging.error('Failed to turn Power On for this host after '
1341 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001342
1343 battery_percentage = self.get_battery_percentage()
Gregory Nisbet02273172020-07-13 09:26:17 -07001344 if (battery_percentage and
1345 battery_percentage < cros_repair.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001346 raise
1347 elif self.is_ac_connected():
1348 logging.info('The device has power adapter connected and '
1349 'charging. No need to try to turn RPM on '
1350 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001351 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001352 logging.info('Battery level is now at %s%%. The device may '
1353 'still have enough power to run test, so no '
1354 'exception will be raised.', battery_percentage)
1355
Simran Basi5e6339a2013-03-21 11:34:32 -07001356
Garry Wangad4d4fd2019-01-30 17:00:38 -08001357 def _remove_rpm_changed_tag(self):
1358 info = self.host_info_store.get()
1359 del info.attributes[self._RPM_OUTLET_CHANGED]
1360 self.host_info_store.commit(info)
1361
1362
1363 def _add_rpm_changed_tag(self):
1364 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001365 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001366 self.host_info_store.commit(info)
1367
1368
1369
beepsc87ff602013-07-31 21:53:00 -07001370 def _is_factory_image(self):
1371 """Checks if the image on the DUT is a factory image.
1372
1373 @return: True if the image on the DUT is a factory image.
1374 False otherwise.
1375 """
1376 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1377 return result.exit_status == 0
1378
1379
1380 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001381 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001382
1383 @raises: FactoryImageCheckerException for factory images, since
1384 we cannot attempt to restart ui on them.
1385 error.AutoservRunError for any other type of error that
1386 occurs while restarting ui.
1387 """
1388 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001389 raise FactoryImageCheckerException('Cannot restart ui on factory '
1390 'images')
beepsc87ff602013-07-31 21:53:00 -07001391
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001392 # TODO(jrbarnette): The command to stop/start the ui job
1393 # should live inside cros_ui, too. However that would seem
1394 # to imply interface changes to the existing start()/restart()
1395 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001396 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001397 self.run('stop ui; start ui')
1398 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001399
1400
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001401 def _start_powerd_if_needed(self):
1402 """Start powerd if it isn't already running."""
1403 self.run('start powerd', ignore_status=True)
1404
1405
xixuana3bbc422017-05-04 15:57:21 -07001406 def _get_lsb_release_content(self):
1407 """Return the content of lsb-release file of host."""
1408 return self.run(
1409 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1410
1411
Dan Shi549fb822015-03-24 18:01:11 -07001412 def get_release_version(self):
1413 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1414
1415 @returns The version string in lsb-release, under attribute
1416 CHROMEOS_RELEASE_VERSION.
1417 """
Dan Shi549fb822015-03-24 18:01:11 -07001418 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001419 lsb_release_content=self._get_lsb_release_content())
1420
1421
Don Garrettb9f35802018-01-22 18:25:40 -08001422 def get_release_builder_path(self):
1423 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1424
1425 @returns The version string in lsb-release, under attribute
1426 CHROMEOS_RELEASE_BUILDER_PATH.
1427 """
1428 return lsbrelease_utils.get_chromeos_release_builder_path(
1429 lsb_release_content=self._get_lsb_release_content())
1430
1431
xixuana3bbc422017-05-04 15:57:21 -07001432 def get_chromeos_release_milestone(self):
1433 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1434 from lsb-release.
1435
1436 @returns The version string in lsb-release, under attribute
1437 CHROMEOS_RELEASE_BUILD_TYPE.
1438 """
1439 return lsbrelease_utils.get_chromeos_release_milestone(
1440 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001441
1442
1443 def verify_cros_version_label(self):
Garry Wangd18e7b32020-08-07 18:31:44 -07001444 """Verify if host's cros-version label match the actual image in dut.
Dan Shi549fb822015-03-24 18:01:11 -07001445
Garry Wangd18e7b32020-08-07 18:31:44 -07001446 @returns True if the label match with image in dut, otherwise False
Dan Shi549fb822015-03-24 18:01:11 -07001447 """
Garry Wangd18e7b32020-08-07 18:31:44 -07001448 os_from_host = self.get_release_builder_path()
1449 info = self.host_info_store.get()
1450 os_from_label = info.get_label_value(self.VERSION_PREFIX)
1451 if not os_from_label:
1452 logging.debug('No existing %s label detected', self.VERSION_PREFIX)
1453 return True
1454
1455 # known cases where the version label will not match the
1456 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1457 # * Tests for the `arc-presubmit` append "-cheetsth" to the label.
1458 if os_from_label.endswith(provision.CHEETS_SUFFIX):
1459 logging.debug('%s label with %s suffix detected, this suffix will'
1460 ' be ignored when comparing label.',
1461 self.VERSION_PREFIX, provision.CHEETS_SUFFIX)
1462 os_from_label = os_from_label[:-len(provision.CHEETS_SUFFIX)]
1463 logging.debug('OS version from host: %s; OS verision cached in '
1464 'label: %s', os_from_host, os_from_label)
1465 return os_from_label == os_from_host
Dan Shi549fb822015-03-24 18:01:11 -07001466
1467
Laurence Goodby778c9a42017-05-24 19:24:07 -07001468 def cleanup_services(self):
1469 """Reinitializes the device for cleanup.
1470
1471 Subclasses may override this to customize the cleanup method.
1472
1473 To indicate failure of the reset, the implementation may raise
1474 any of:
1475 error.AutoservRunError
1476 error.AutotestRunError
1477 FactoryImageCheckerException
1478
1479 @raises error.AutoservRunError
1480 @raises error.AutotestRunError
1481 @raises error.FactoryImageCheckerException
1482 """
1483 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001484 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001485
1486
beepsc87ff602013-07-31 21:53:00 -07001487 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001488 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001489 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001490 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001491 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001492 except (error.AutotestRunError, error.AutoservRunError,
1493 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001494 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001495
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001496 # cleanup routines, i.e. reboot the machine.
1497 super(CrosHost, self).cleanup()
1498
Simran Basi5e6339a2013-03-21 11:34:32 -07001499 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001500 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001501 self._cleanup_poweron()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001502
1503
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001504 def reboot(self, **dargs):
1505 """
1506 This function reboots the site host. The more generic
1507 RemoteHost.reboot() performs sync and sleeps for 5
1508 seconds. This is not necessary for Chrome OS devices as the
1509 sync should be finished in a short time during the reboot
1510 command.
1511 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001512 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001513 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001514 dargs['reboot_cmd'] = ('sleep 1; '
1515 'reboot & sleep %d; '
1516 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001517 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001518 if 'fastsync' not in dargs:
1519 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001520
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001521 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001522 # Record who called us
1523 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001524 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001525 'dut_host_name' : self.hostname,
1526 'success' : True}
1527 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001528 'caller' : "%s:%s" % (orig.co_filename,
1529 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001530 'success' : True,
1531 'error' : ''}
1532
Vincent Palatin80780b22016-07-27 16:02:37 +02001533 t0 = time.time()
1534 try:
1535 super(CrosHost, self).reboot(**dargs)
1536 except Exception as e:
1537 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001538 metric_debug_fields['success'] = False
1539 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001540 raise
1541 finally:
1542 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001543 metrics.Counter(
1544 'chromeos/autotest/autoserv/reboot_count').increment(
1545 fields=metric_fields)
1546 metrics.Counter(
1547 'chromeos/autotest/autoserv/reboot_debug').increment(
1548 fields=metric_debug_fields)
1549 metrics.SecondsDistribution(
1550 'chromeos/autotest/autoserv/reboot_duration').add(
1551 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001552
1553
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001554 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001555 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001556 """
1557 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001558
1559 @param suspend_time: How long to suspend as integer seconds.
1560 @param suspend_cmd: Suspend command to execute.
1561 @param allow_early_resume: If False and if device resumes before
1562 |suspend_time|, throw an error.
1563
1564 @exception AutoservSuspendError Host resumed earlier than
1565 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001566 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001567
1568 if suspend_cmd is None:
1569 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001570 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001571 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001572 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001573 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1574 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001575
1576
Simran Basiec564392014-08-25 16:48:09 -07001577 def upstart_status(self, service_name):
1578 """Check the status of an upstart init script.
1579
1580 @param service_name: Service to look up.
1581
1582 @returns True if the service is running, False otherwise.
1583 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001584 return 'start/running' in self.run('status %s' % service_name,
1585 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001586
Tom Hughese9552342018-12-18 14:29:25 -08001587 def upstart_stop(self, service_name):
1588 """Stops an upstart job if it's running.
1589
1590 @param service_name: Service to stop
1591
1592 @returns True if service has been stopped or was already stopped
1593 False otherwise.
1594 """
1595 if not self.upstart_status(service_name):
1596 return True
1597
1598 result = self.run('stop %s' % service_name, ignore_status=True)
1599 if result.exit_status != 0:
1600 return False
1601 return True
1602
1603 def upstart_restart(self, service_name):
1604 """Restarts (or starts) an upstart job.
1605
1606 @param service_name: Service to start/restart
1607
1608 @returns True if service has been started/restarted, False otherwise.
1609 """
1610 cmd = 'start'
1611 if self.upstart_status(service_name):
1612 cmd = 'restart'
1613 cmd = cmd + ' %s' % service_name
1614 result = self.run(cmd)
1615 if result.exit_status != 0:
1616 return False
1617 return True
Simran Basiec564392014-08-25 16:48:09 -07001618
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001619 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001620 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001621
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001622 Tests for the following conditions:
1623 1. All conditions tested by the parent version of this
1624 function.
1625 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001626 3. Sufficient space in /mnt/stateful_partition/encrypted.
1627 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001628
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001629 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001630 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001631 default_kilo_inodes_required = CONFIG.get_config_value(
1632 'SERVER', 'kilo_inodes_required', type=int, default=100)
1633 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1634 kilo_inodes_required = CONFIG.get_config_value(
1635 'SERVER', 'kilo_inodes_required_%s' % board,
1636 type=int, default=default_kilo_inodes_required)
1637 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001638 self.check_diskspace(
1639 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001640 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001641 'SERVER', 'gb_diskspace_required', type=float,
1642 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001643 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1644 # Not all targets build with encrypted stateful support.
1645 if self.path_exists(encrypted_stateful_path):
1646 self.check_diskspace(
1647 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001648 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001649 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1650 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001651
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001652 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001653
beepsc87ff602013-07-31 21:53:00 -07001654 # Factory images don't run update engine,
1655 # goofy controls dbus on these DUTs.
1656 if not self._is_factory_image():
1657 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001658
1659
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001660 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1661 def wait_for_system_services(self):
1662 """Waits for system-services to be running.
1663
1664 Sometimes, update_engine will take a while to update firmware, so we
1665 should give this some time to finish. See crbug.com/765686#c38 for
1666 details.
1667 """
1668 if not self.upstart_status('system-services'):
1669 raise error.AutoservError('Chrome failed to reach login. '
1670 'System services not running.')
1671
1672
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001673 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001674 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001675 message = 'Beginning verify for host %s board %s model %s'
1676 info = self.host_info_store.get()
1677 message %= (self.hostname, info.board, info.model)
1678 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001679 try:
1680 self._repair_strategy.verify(self)
1681 except hosts.AutoservVerifyDependencyError as e:
1682 # We don't want flag a DUT as failed if only non-critical
1683 # verifier(s) failed during the repair.
1684 if e.is_critical():
1685 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001686
1687
Fang Deng96667ca2013-08-01 17:46:18 -07001688 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001689 connect_timeout=None, alive_interval=None,
1690 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001691 """Override default make_ssh_command to use options tuned for Chrome OS.
1692
1693 Tuning changes:
1694 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1695 connection failure. Consistency with remote_access.sh.
1696
Samuel Tan2ce155b2015-06-23 18:24:38 -07001697 - ServerAliveInterval=900; which causes SSH to ping connection every
1698 900 seconds. In conjunction with ServerAliveCountMax ensures
1699 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001700 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001701 the test completed successfully. Later increased from 180 seconds to
1702 900 seconds to account for tests where the DUT is suspended for
1703 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001704
1705 - ServerAliveCountMax=3; consistency with remote_access.sh.
1706
1707 - ConnectAttempts=4; reduce flakiness in connection errors;
1708 consistency with remote_access.sh.
1709
1710 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1711 Host keys change with every new installation, don't waste
1712 memory/space saving them.
1713
1714 - SSH protocol forced to 2; needed for ServerAliveInterval.
1715
1716 @param user User name to use for the ssh connection.
1717 @param port Port on the target host to use for ssh connection.
1718 @param opts Additional options to the ssh command.
1719 @param hosts_file Ignored.
1720 @param connect_timeout Ignored.
1721 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001722 @param alive_count_max Ignored.
1723 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001724 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001725 options = ' '.join([opts, '-o Protocol=2'])
1726 return super(CrosHost, self).make_ssh_command(
1727 user=user, port=port, opts=options, hosts_file='/dev/null',
1728 connect_timeout=30, alive_interval=900, alive_count_max=3,
1729 connection_attempts=4)
1730
1731
Jason Abeleb6f924f2013-11-13 16:01:54 -08001732 def syslog(self, message, tag='autotest'):
1733 """Logs a message to syslog on host.
1734
1735 @param message String message to log into syslog
1736 @param tag String tag prefix for syslog
1737
1738 """
1739 self.run('logger -t "%s" "%s"' % (tag, message))
1740
1741
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001742 def _ping_check_status(self, status):
1743 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001744
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001745 @param status Check the ping status against this value.
1746 @return True iff `status` and the result of ping are the same
1747 (i.e. both True or both False).
1748
1749 """
1750 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1751 return not (status ^ (ping_val == 0))
1752
1753 def _ping_wait_for_status(self, status, timeout):
1754 """Wait for the host to have a given status (UP or DOWN).
1755
1756 Status is checked by polling. Polling will not last longer
1757 than the number of seconds in `timeout`. The polling
1758 interval will be long enough that only approximately
1759 _PING_WAIT_COUNT polling cycles will be executed, subject
1760 to a maximum interval of about one minute.
1761
1762 @param status Waiting will stop immediately if `ping` of the
1763 host returns this status.
1764 @param timeout Poll for at most this many seconds.
1765 @return True iff the host status from `ping` matched the
1766 requested status at the time of return.
1767
1768 """
1769 # _ping_check_status() takes about 1 second, hence the
1770 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001771 # FIXME: if the ping command errors then _ping_check_status()
1772 # returns instantly. If timeout is also smaller than twice
1773 # _PING_WAIT_COUNT then the while loop below forks many
1774 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1775 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1776 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001777 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1778 end_time = time.time() + timeout
1779 while time.time() <= end_time:
1780 if self._ping_check_status(status):
1781 return True
1782 if poll_interval > 0:
1783 time.sleep(poll_interval)
1784
1785 # The last thing we did was sleep(poll_interval), so it may
1786 # have been too long since the last `ping`. Check one more
1787 # time, just to be sure.
1788 return self._ping_check_status(status)
1789
1790 def ping_wait_up(self, timeout):
1791 """Wait for the host to respond to `ping`.
1792
1793 N.B. This method is not a reliable substitute for
1794 `wait_up()`, because a host that responds to ping will not
1795 necessarily respond to ssh. This method should only be used
1796 if the target DUT can be considered functional even if it
1797 can't be reached via ssh.
1798
1799 @param timeout Minimum time to allow before declaring the
1800 host to be non-responsive.
1801 @return True iff the host answered to ping before the timeout.
1802
1803 """
1804 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001805
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001806 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001807 """Wait until the host no longer responds to `ping`.
1808
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001809 This function can be used as a slightly faster version of
1810 `wait_down()`, by avoiding potentially long ssh timeouts.
1811
1812 @param timeout Minimum time to allow for the host to become
1813 non-responsive.
1814 @return True iff the host quit answering ping before the
1815 timeout.
1816
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001817 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001818 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001819
Anand K Mistry50f218e2020-07-31 14:50:15 +10001820 def _is_host_port_forwarded(self):
1821 """Checks if the dut is connected over port forwarding.
1822
1823 N.B. This method does not detect all situations where port forwarding is
1824 occurring. Namely, running autotest on the dut may result in a
1825 false-positive, and port forwarding using a different machine on the
1826 same network will be a false-negative.
1827
1828 @return True if the dut is connected over port forwarding
1829 False otherwise
1830 """
1831 is_localhost = self.hostname in ['localhost', '127.0.0.1']
1832 is_forwarded = is_localhost and not self.is_default_port
1833 if is_forwarded:
1834 logging.info('Detected DUT connected by port forwarding')
1835 return is_forwarded
1836
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001837 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001838 """Wait for the client to enter low-power sleep mode.
1839
1840 The test for "is asleep" can't distinguish a system that is
1841 powered off; to confirm that the unit was asleep, it is
1842 necessary to force resume, and then call
1843 `test_wait_for_resume()`.
1844
1845 This function is expected to be called from a test as part
1846 of a sequence like the following:
1847
1848 ~~~~~~~~
1849 boot_id = host.get_boot_id()
1850 # trigger sleep on the host
1851 host.test_wait_for_sleep()
1852 # trigger resume on the host
1853 host.test_wait_for_resume(boot_id)
1854 ~~~~~~~~
1855
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001856 @param sleep_timeout time limit in seconds to allow the host sleep.
1857
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001858 @exception TestFail The host did not go to sleep within
1859 the allowed time.
1860 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001861 if sleep_timeout is None:
1862 sleep_timeout = self.SLEEP_TIMEOUT
1863
Anand K Mistry50f218e2020-07-31 14:50:15 +10001864 # If the dut is accessed over SSH port-forwarding, `ping` is not useful
1865 # for detecting the dut is down since a ping to localhost will always
1866 # succeed. In this case, fall back to wait_down() which uses SSH.
1867 if self._is_host_port_forwarded():
1868 success = self.wait_down(timeout=sleep_timeout)
1869 else:
1870 success = self.ping_wait_down(timeout=sleep_timeout)
1871
1872 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001873 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001874 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001875
1876
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001877 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001878 """Wait for the client to resume from low-power sleep mode.
1879
1880 The `old_boot_id` parameter should be the value from
1881 `get_boot_id()` obtained prior to entering sleep mode. A
1882 `TestFail` exception is raised if the boot id changes.
1883
1884 See @ref test_wait_for_sleep for more on this function's
1885 usage.
1886
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001887 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001888 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001889 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001890
1891 @exception TestFail The host did not respond within the
1892 allowed time.
1893 @exception TestFail The host responded, but the boot id test
1894 indicated a reboot rather than a sleep
1895 cycle.
1896 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001897 if resume_timeout is None:
1898 resume_timeout = self.RESUME_TIMEOUT
1899
1900 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001901 raise error.TestFail(
1902 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001903 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001904 else:
1905 new_boot_id = self.get_boot_id()
1906 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001907 logging.error('client rebooted (old boot %s, new boot %s)',
1908 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001909 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001910 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001911
1912
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001913 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001914 """Wait for the client to shut down.
1915
1916 The test for "has shut down" can't distinguish a system that
1917 is merely asleep; to confirm that the unit was down, it is
1918 necessary to force boot, and then call test_wait_for_boot().
1919
1920 This function is expected to be called from a test as part
1921 of a sequence like the following:
1922
1923 ~~~~~~~~
1924 boot_id = host.get_boot_id()
1925 # trigger shutdown on the host
1926 host.test_wait_for_shutdown()
1927 # trigger boot on the host
1928 host.test_wait_for_boot(boot_id)
1929 ~~~~~~~~
1930
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001931 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001932 @exception TestFail The host did not shut down within the
1933 allowed time.
1934 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001935 if shutdown_timeout is None:
1936 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1937
Anand K Mistry50f218e2020-07-31 14:50:15 +10001938 if self._is_host_port_forwarded():
1939 success = self.wait_down(timeout=shutdown_timeout)
1940 else:
1941 success = self.ping_wait_down(timeout=shutdown_timeout)
1942
1943 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001944 raise error.TestFail(
1945 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001946 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001947
1948
1949 def test_wait_for_boot(self, old_boot_id=None):
1950 """Wait for the client to boot from cold power.
1951
1952 The `old_boot_id` parameter should be the value from
1953 `get_boot_id()` obtained prior to shutting down. A
1954 `TestFail` exception is raised if the boot id does not
1955 change. The boot id test is omitted if `old_boot_id` is not
1956 specified.
1957
1958 See @ref test_wait_for_shutdown for more on this function's
1959 usage.
1960
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001961 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001962 shut down.
1963
1964 @exception TestFail The host did not respond within the
1965 allowed time.
1966 @exception TestFail The host responded, but the boot id test
1967 indicated that there was no reboot.
1968 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001969 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001970 raise error.TestFail(
1971 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001972 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001973 elif old_boot_id:
1974 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001975 logging.error('client not rebooted (boot %s)',
1976 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001977 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001978 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001979
1980
1981 @staticmethod
1982 def check_for_rpm_support(hostname):
1983 """For a given hostname, return whether or not it is powered by an RPM.
1984
Simran Basi1df55112013-09-06 11:25:09 -07001985 @param hostname: hostname to check for rpm support.
1986
Simran Basid5e5e272012-09-24 15:23:59 -07001987 @return None if this host does not follows the defined naming format
1988 for RPM powered DUT's in the lab. If it does follow the format,
1989 it returns a regular expression MatchObject instead.
1990 """
Fang Dengbaff9082015-01-06 13:46:15 -08001991 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001992
1993
1994 def has_power(self):
1995 """For this host, return whether or not it is powered by an RPM.
1996
1997 @return True if this host is in the CROS lab and follows the defined
1998 naming format.
1999 """
Fang Deng0ca40e22013-08-27 17:47:44 -07002000 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002001
2002
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002003 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07002004 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002005
2006 @param state Specifies which power state to set to DUT
2007 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002008 use. By default "RPM" or "CCD" will be used based
2009 on servo type. Valid values from
2010 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002011
2012 """
2013 ACCEPTABLE_STATES = ['ON', 'OFF']
2014
Garry Wang5e5538a2019-04-08 15:36:18 -07002015 if not power_method:
2016 power_method = self.get_default_power_method()
2017
2018 state = state.upper()
2019 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002020 raise error.TestError('State must be one of: %s.'
2021 % (ACCEPTABLE_STATES,))
2022
2023 if power_method == self.POWER_CONTROL_SERVO:
2024 logging.info('Setting servo port J10 to %s', state)
2025 self.servo.set('prtctl3_pwren', state.lower())
2026 time.sleep(self._USB_POWER_TIMEOUT)
2027 elif power_method == self.POWER_CONTROL_MANUAL:
2028 logging.info('You have %d seconds to set the AC power to %s.',
2029 self._POWER_CYCLE_TIMEOUT, state)
2030 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07002031 elif power_method == self.POWER_CONTROL_CCD:
2032 servo_role = 'src' if state == 'ON' else 'snk'
2033 logging.info('servo ccd power pass through detected,'
2034 ' changing servo_role to %s.', servo_role)
2035 self.servo.set_servo_v4_role(servo_role)
2036 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07002037 # Make sure we don't leave DUT with no power(servo_role=snk)
2038 # when DUT is not pingable, as we raise a exception here
2039 # that may break a power cycle in the middle.
2040 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07002041 raise error.AutoservError(
2042 'DUT failed to regain network connection after %d seconds.'
2043 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002044 else:
2045 if not self.has_power():
2046 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002047 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07002048 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07002049
2050
Garry Wang5e5538a2019-04-08 15:36:18 -07002051 def power_off(self, power_method=None):
2052 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002053
2054 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002055 use. By default "RPM" or "CCD" will be used based
2056 on servo type. Valid values from
2057 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002058
2059 """
2060 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002061
2062
Garry Wang5e5538a2019-04-08 15:36:18 -07002063 def power_on(self, power_method=None):
2064 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002065
2066 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002067 use. By default "RPM" or "CCD" will be used based
2068 on servo type. Valid values from
2069 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002070
2071 """
2072 self._set_power('ON', power_method)
2073
2074
Garry Wang5e5538a2019-04-08 15:36:18 -07002075 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002076 """Cycle power to this host by turning it OFF, then ON.
2077
2078 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002079 use. By default "RPM" or "CCD" will be used based
2080 on servo type. Valid values from
2081 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002082
2083 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002084 if not power_method:
2085 power_method = self.get_default_power_method()
2086
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002087 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002088 self.POWER_CONTROL_MANUAL,
2089 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002090 self.power_off(power_method=power_method)
2091 time.sleep(self._POWER_CYCLE_TIMEOUT)
2092 self.power_on(power_method=power_method)
2093 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002094 self._add_rpm_changed_tag()
2095 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002096
2097
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002098 def get_platform_from_fwid(self):
2099 """Determine the platform from the crossystem fwid.
2100
2101 @returns a string representing this host's platform.
2102 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002103 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002104 crossystem = utils.Crossystem(self)
2105 crossystem.init()
2106 # Extract fwid value and use the leading part as the platform id.
2107 # fwid generally follow the format of {platform}.{firmware version}
2108 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2109 platform = crossystem.fwid().split('.')[0].lower()
2110 # Newer platforms start with 'Google_' while the older ones do not.
2111 return platform.replace('google_', '')
2112
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002113
Simran Basic6f1f7a2012-10-16 10:47:46 -07002114 def get_platform(self):
2115 """Determine the correct platform label for this host.
2116
2117 @returns a string representing this host's platform.
2118 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002119 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2120 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002121 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002122 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002123 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002124 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002125
2126
Greg Edelstona7b05d12020-04-01 16:00:51 -06002127 def get_model_from_cros_config(self):
2128 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002129
Greg Edelstona7b05d12020-04-01 16:00:51 -06002130 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002131 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002132 return cros_config.call_cros_config_get_output('/ name',
2133 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002134
2135
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002136 def get_architecture(self):
2137 """Determine the correct architecture label for this host.
2138
2139 @returns a string representing this host's architecture.
2140 """
2141 crossystem = utils.Crossystem(self)
2142 crossystem.init()
2143 return crossystem.arch()
2144
2145
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002146 def get_chrome_version(self):
2147 """Gets the Chrome version number and milestone as strings.
2148
2149 Invokes "chrome --version" to get the version number and milestone.
2150
2151 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2152 current Chrome version number as a string (in the form "W.X.Y.Z")
2153 and "milestone" is the first component of the version number
2154 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2155 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2156 of "chrome --version" and the milestone will be the empty string.
2157
2158 """
MK Ryu35d661e2014-09-25 17:44:10 -07002159 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002160 return utils.parse_chrome_version(version_string)
2161
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002162
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002163 def get_ec_version(self):
2164 """Get the ec version as strings.
2165
2166 @returns a string representing this host's ec version.
2167 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002168 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002169 result = self.run(command, ignore_status=True)
2170 if result.exit_status != 0:
2171 return ''
2172 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002173
2174
2175 def get_firmware_version(self):
2176 """Get the firmware version as strings.
2177
2178 @returns a string representing this host's firmware version.
2179 """
2180 crossystem = utils.Crossystem(self)
2181 crossystem.init()
2182 return crossystem.fwid()
2183
2184
2185 def get_hardware_revision(self):
2186 """Get the hardware revision as strings.
2187
2188 @returns a string representing this host's hardware revision.
2189 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002190 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002191 result = self.run(command, ignore_status=True)
2192 if result.exit_status != 0:
2193 return ''
2194 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002195
2196
2197 def get_kernel_version(self):
2198 """Get the kernel version as strings.
2199
2200 @returns a string representing this host's kernel version.
2201 """
2202 return self.run('uname -r').stdout.strip()
2203
2204
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002205 def get_cpu_name(self):
2206 """Get the cpu name as strings.
2207
2208 @returns a string representing this host's cpu name.
2209 """
2210
2211 # Try get cpu name from device tree first
2212 if self.path_exists('/proc/device-tree/compatible'):
2213 command = ' | '.join(
2214 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2215 'tail -1'])
2216 return self.run(command).stdout.strip().replace(',', ' ')
2217
2218 # Get cpu name from uname -p
2219 command = 'uname -p'
2220 ret = self.run(command).stdout.strip()
2221
2222 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2223 # Try get cpu name from /proc/cpuinfo instead
2224 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2225 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2226 self = self.run(command).stdout.strip()
2227
2228 # Remove bloat from CPU name, for example
2229 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2230 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2231 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2232 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2233 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2234 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2235
2236
2237 def get_screen_resolution(self):
2238 """Get the screen(s) resolution as strings.
2239 In case of more than 1 monitor, return resolution for each monitor
2240 separate with plus sign.
2241
2242 @returns a string representing this host's screen(s) resolution.
2243 """
2244 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2245 ret = self.run(command, ignore_status=True)
2246 # We might have Chromebox without a screen
2247 if ret.exit_status != 0:
2248 return ''
2249 return ret.stdout.strip().replace('\n', '+')
2250
2251
2252 def get_mem_total_gb(self):
2253 """Get total memory available in the system in GiB (2^20).
2254
2255 @returns an integer representing total memory
2256 """
2257 mem_total_kb = self.read_from_meminfo('MemTotal')
2258 kb_in_gb = float(2 ** 20)
2259 return int(round(mem_total_kb / kb_in_gb))
2260
2261
2262 def get_disk_size_gb(self):
2263 """Get size of disk in GB (10^9)
2264
2265 @returns an integer representing size of disk, 0 in Error Case
2266 """
2267 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2268 result = self.run(command, ignore_status=True)
2269 if result.exit_status != 0:
2270 return 0
2271 _, _, block, _ = re.split(r' +', result.stdout.strip())
2272 byte_per_block = 1024.0
2273 disk_kb_in_gb = 1e9
2274 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2275
2276
2277 def get_battery_size(self):
2278 """Get size of battery in Watt-hour via sysfs
2279
2280 This method assumes that battery support voltage_min_design and
2281 charge_full_design sysfs.
2282
2283 @returns a float representing Battery size, 0 if error.
2284 """
2285 # sysfs report data in micro scale
2286 battery_scale = 1e6
2287
2288 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2289 result = self.run(command, ignore_status=True)
2290 if result.exit_status != 0:
2291 return 0
2292 voltage = float(result.stdout.strip()) / battery_scale
2293
2294 command = 'cat /sys/class/power_supply/*/charge_full_design'
2295 result = self.run(command, ignore_status=True)
2296 if result.exit_status != 0:
2297 return 0
2298 amphereHour = float(result.stdout.strip()) / battery_scale
2299
2300 return voltage * amphereHour
2301
2302
2303 def get_low_battery_shutdown_percent(self):
2304 """Get the percent-based low-battery shutdown threshold.
2305
2306 @returns a float representing low-battery shutdown percent, 0 if error.
2307 """
2308 ret = 0.0
2309 try:
2310 command = 'check_powerd_config --low_battery_shutdown_percent'
2311 ret = float(self.run(command).stdout)
2312 except error.CmdError:
2313 logging.debug("Can't run %s", command)
2314 except ValueError:
2315 logging.debug("Didn't get number from %s", command)
2316
2317 return ret
2318
2319
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002320 def has_hammer(self):
2321 """Check whether DUT has hammer device or not.
2322
2323 @returns boolean whether device has hammer or not
2324 """
2325 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2326 return self.run(command, ignore_status=True).exit_status == 0
2327
2328
Niranjan Kumar34618872017-05-31 12:57:09 -07002329 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002330 """Returns True if the specified switch was provided to Chrome.
2331
2332 @param switch The chrome switch to search for.
2333 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002334
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002335 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2336 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002337
2338
2339 def oobe_triggers_update(self):
2340 """Returns True if this host has an OOBE flow during which
2341 it will perform an update check and perhaps an update.
2342 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2343 As more such flows are developed, code handling them needs
2344 to be added here.
2345
2346 @return Boolean indicating whether this host's OOBE triggers an update.
2347 """
2348 return self.is_chrome_switch_present(
2349 '--enterprise-enable-zero-touch-enrollment=hands-off')
2350
2351
Kevin Chenga2619dc2016-03-28 11:42:08 -07002352 # TODO(kevcheng): change this to just return the board without the
2353 # 'board:' prefix and fix up all the callers. Also look into removing the
2354 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002355 def get_board(self):
2356 """Determine the correct board label for this host.
2357
2358 @returns a string representing this host's board.
2359 """
2360 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2361 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002362 return (ds_constants.BOARD_PREFIX +
2363 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002364
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002365 def get_channel(self):
2366 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002367
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002368 @returns: a string represeting this host's build channel.
2369 (stable, dev, beta). None on fail.
2370 """
2371 return lsbrelease_utils.get_chromeos_channel(
2372 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002373
Kevin Chenga328da62016-03-31 10:49:04 -07002374 def get_power_supply(self):
2375 """
2376 Determine what type of power supply the host has
2377
2378 @returns a string representing this host's power supply.
2379 'power:battery' when the device has a battery intended for
2380 extended use
2381 'power:AC_primary' when the device has a battery not intended
2382 for extended use (for moving the machine, etc)
2383 'power:AC_only' when the device has no battery at all.
2384 """
2385 psu = self.run(command='mosys psu type', ignore_status=True)
2386 if psu.exit_status:
2387 # The psu command for mosys is not included for all platforms. The
2388 # assumption is that the device will have a battery if the command
2389 # is not found.
2390 return 'power:battery'
2391
2392 psu_str = psu.stdout.strip()
2393 if psu_str == 'unknown':
2394 return None
2395
2396 return 'power:%s' % psu_str
2397
2398
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002399 def has_battery(self):
2400 """Determine if DUT has a battery.
2401
2402 Returns:
2403 Boolean, False if known not to have battery, True otherwise.
2404 """
2405 rv = True
2406 power_supply = self.get_power_supply()
2407 if power_supply == 'power:battery':
2408 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2409 board_type = self.get_board_type()
2410 if board_type in _NO_BATTERY_BOARD_TYPE:
2411 logging.warn('Do NOT believe type %s has battery. '
2412 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002413 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002414 ignore_status=True)
2415 logging.debug(psu)
2416 rv = False
2417 elif power_supply == 'power:AC_only':
2418 rv = False
2419
2420 return rv
2421
2422
Kevin Chenga328da62016-03-31 10:49:04 -07002423 def get_servo(self):
2424 """Determine if the host has a servo attached.
2425
2426 If the host has a working servo attached, it should have a servo label.
2427
2428 @return: string 'servo' if the host has servo attached. Otherwise,
2429 returns None.
2430 """
2431 return 'servo' if self._servo_host else None
2432
2433
Kevin Chenga328da62016-03-31 10:49:04 -07002434 def has_internal_display(self):
2435 """Determine if the device under test is equipped with an internal
2436 display.
2437
2438 @return: 'internal_display' if one is present; None otherwise.
2439 """
2440 from autotest_lib.client.cros.graphics import graphics_utils
2441 from autotest_lib.client.common_lib import utils as common_utils
2442
2443 def __system_output(cmd):
2444 return self.run(cmd).stdout
2445
2446 def __read_file(remote_path):
2447 return self.run('cat %s' % remote_path).stdout
2448
2449 # Hijack the necessary client functions so that we can take advantage
2450 # of the client lib here.
2451 # FIXME: find a less hacky way than this
2452 original_system_output = utils.system_output
2453 original_read_file = common_utils.read_file
2454 utils.system_output = __system_output
2455 common_utils.read_file = __read_file
2456 try:
2457 return ('internal_display' if graphics_utils.has_internal_display()
2458 else None)
2459 finally:
2460 utils.system_output = original_system_output
2461 common_utils.read_file = original_read_file
2462
2463
Dan Shi85276d42014-04-08 22:11:45 -07002464 def is_boot_from_usb(self):
2465 """Check if DUT is boot from USB.
2466
2467 @return: True if DUT is boot from usb.
2468 """
2469 device = self.run('rootdev -s -d').stdout.strip()
2470 removable = int(self.run('cat /sys/block/%s/removable' %
2471 os.path.basename(device)).stdout.strip())
2472 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002473
2474
2475 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002476 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002477
2478 @param key: meminfo requested
2479
2480 @return the memory value as a string
2481
2482 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002483 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2484 logging.debug('%s', meminfo)
2485 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002486
2487
Rohit Makasana98e696f2016-06-03 18:48:10 -07002488 def get_cpu_arch(self):
2489 """Returns CPU arch of the device.
2490
2491 @return CPU architecture of the DUT.
2492 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002493 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002494 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2495 ignore_status=True).stdout:
2496 return 'x86_64'
2497 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2498 ignore_status=True).stdout:
2499 return 'arm'
2500 return 'i386'
2501
2502
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002503 def get_board_type(self):
2504 """
2505 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002506 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2507
2508 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002509 """
Danny Chan471a8d12015-08-18 14:57:41 -07002510 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2511 ignore_status=True).stdout
2512 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002513 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002514 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002515
2516
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002517 def get_arc_version(self):
2518 """Return ARC version installed on the DUT.
2519
2520 @returns ARC version as string if the CrOS build has ARC, else None.
2521 """
2522 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2523 ignore_status=True).stdout
2524 if arc_version:
2525 return arc_version.split('=')[-1].strip()
2526 return None
2527
2528
Gilad Arnolda76bef02015-09-29 13:55:15 -07002529 def get_os_type(self):
2530 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002531
2532
Kevin Chenga2619dc2016-03-28 11:42:08 -07002533 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002534 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002535 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002536
2537
2538 def get_default_power_method(self):
2539 """
2540 Get the default power method for power_on/off/cycle() methods.
2541 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2542 """
2543 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002544 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002545 if self.servo and self.servo.supports_built_in_pd_control():
2546 self._default_power_method = self.POWER_CONTROL_CCD
2547 else:
2548 logging.debug('Either servo is unitialized or the servo '
2549 'setup does not support pd controls. Falling '
2550 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002551 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002552
2553
2554 def find_usb_devices(self, idVendor, idProduct):
2555 """
2556 Get usb device sysfs name for specific device.
2557
2558 @param idVendor Vendor ID to search in sysfs directory.
2559 @param idProduct Product ID to search in sysfs directory.
2560
2561 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2562 """
2563 # Look for matching file and cut at position 7 to get dir name.
2564 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2565
2566 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2567 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2568
2569 # Use uniq -d to print duplicate line from both command
2570 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2571
2572 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2573
2574
2575 def bind_usb_device(self, usb_node):
2576 """
2577 Bind usb device
2578
2579 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2580 """
2581 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2582 self.run(cmd, ignore_status=True)
2583
2584
2585 def unbind_usb_device(self, usb_node):
2586 """
2587 Unbind usb device
2588
2589 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2590 """
2591 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2592 self.run(cmd, ignore_status=True)
2593
2594
2595 def get_wlan_ip(self):
2596 """
2597 Get ip address of wlan interface.
2598
2599 @return ip address of wlan or empty string if wlan is not connected.
2600 """
2601 cmds = [
2602 'iw dev', # List wlan physical device
2603 'grep Interface', # Grep only interface name
2604 'cut -f 2 -d" "', # Cut the name part
2605 'xargs ifconfig', # Feed it to ifconfig to get ip
2606 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2607 'cut -f 2 -d " "' # Cut the ip part
2608 ]
2609 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002610
2611 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2612 """
2613 Connect to wifi network
2614
2615 @param ssid SSID of the wifi network.
2616 @param passphrase Passphrase of the wifi network. None if not existed.
2617 @param security Security of the wifi network. Default to "psk" if
2618 passphase is given without security. Possible values
2619 are "none", "psk", "802_1x".
2620
2621 @return True if succeed, False if not.
2622 """
2623 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2624 if passphrase:
2625 cmd += ' ' + passphrase
2626 if security:
2627 cmd += ' ' + security
2628 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002629
2630 def get_device_repair_state(self):
2631 """Get device repair state"""
2632 return self._device_repair_state
2633
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002634 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002635 """Set device repair state.
2636
2637 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002638 file in result directory. The file will be read by Lucifer. The
2639 file will not be created if result directory not specified.
2640
2641 @params state: The new state for the device.
2642 @params resultdir: The path to result directory. If path not provided
2643 will be attempt to get retrieve it from job
2644 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002645 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002646 resultdir = resultdir or getattr(self.job, 'resultdir', '')
2647 if resultdir:
2648 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07002649 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002650 logging.info('Set device state as %s. '
2651 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07002652 else:
2653 logging.debug('Cannot write the device state due missing info '
2654 'about result dir.')
2655 self._device_repair_state = state
2656
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002657 def set_device_needs_replacement(self, resultdir=None):
2658 """Set device as required replacement.
2659
2660 @params resultdir: The path to result directory. If path not provided
2661 will be attempt to get retrieve it from job
2662 if present.
2663 """
2664 self.set_device_repair_state(
2665 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
2666 resultdir=resultdir)
2667
2668 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002669 """Check if device require manual attention to be fixed.
2670
2671 The state 'needs_manual_repair' can be set when auto repair cannot
2672 fix the device due hardware or cable issues.
2673 """
2674 # ignore the logic if state present
2675 # state can be set by any cros repair actions
2676 if self.get_device_repair_state():
2677 return
2678
2679 # set need manual attention if servo has hardware issue
2680 servo_state_required_manual_fix = [
2681 servo_constants.SERVO_STATE_NOT_CONNECTED,
2682 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
2683 servo_constants.SERVO_STATE_LID_OPEN_FAILED,
2684 servo_constants.SERVO_STATE_BAD_RIBBON_CABLE,
2685 servo_constants.SERVO_STATE_EC_BROKEN,
2686 ]
2687 if self.get_servo_state() in servo_state_required_manual_fix:
Otabek Kasimovde8eea32020-07-01 12:12:22 -07002688 data = {'host': self.hostname,
Otabek Kasimov832d9162020-07-27 19:24:57 -07002689 'state': cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR}
Otabek Kasimov6825b762020-06-23 23:42:44 -07002690 metrics.Counter(
2691 'chromeos/autotest/repair/special_dut_state'
2692 ).increment(fields=data)
2693 # TODO (otabek) unblock when be sure that we do not have flakiness
Otabek Kasimov832d9162020-07-27 19:24:57 -07002694 # self.set_device_repair_state(
2695 # cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)
Otabek Kasimov42506d02020-07-29 14:44:57 -07002696
2697 def is_file_system_writable(self, testdirs=None):
2698 """Check is the file systems are writable.
2699
2700 The standard linux response to certain unexpected file system errors
2701 (including hardware errors in block devices) is to change the file
2702 system status to read-only. This checks that that hasn't happened.
2703
2704 @param testdirs: List of directories to check. If no data provided
2705 then '/mnt/stateful_partition' and '/var/tmp'
2706 directories will be checked.
2707
2708 @returns boolean whether file-system writable.
2709 """
2710 def _check_dir(testdir):
2711 # check if we can create a file
2712 filename = os.path.join(testdir, 'writable_my_test_file')
2713 command = 'touch %s && rm %s' % (filename, filename)
2714 rv = self.run(command=command,
2715 timeout=30,
2716 ignore_status=True)
2717 is_writable = rv.exit_status == 0
2718 if not is_writable:
2719 logging.info('Cannot create a file in "%s"!'
2720 ' Probably the FS is read-only', testdir)
2721 logging.info("FileSystem is not writable!")
2722 return False
2723 return True
2724
2725 if not testdirs or len(testdirs) == 0:
2726 # N.B. Order matters here: Encrypted stateful is loop-mounted
2727 # from a file in unencrypted stateful, so we don't test for
2728 # errors in encrypted stateful if unencrypted fails.
2729 testdirs = ['/mnt/stateful_partition', '/var/tmp']
2730
2731 for dir in testdirs:
2732 # loop will be stopped if any directory fill fail the check
2733 try:
2734 if not _check_dir(dir):
2735 return False
2736 except Exception as e:
2737 # here expected only timeout error, all other will
2738 # be catch by 'ignore_status=True'
2739 logging.debug('Fail to check %s to write in it', dir)
2740 return False
2741 return True