blob: 7ffe5cd6f27bbc4346915e6df07cddcdc4138969 [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
Greg Edelstona7b05d12020-04-01 16:00:51 -060018from autotest_lib.client.common_lib.cros import cros_config
Richard Barnette03a0c132012-11-05 12:40:35 -080019from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070020from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000021from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080022from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080023from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070024from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070025from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050026from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070027from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070028from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070029from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070030from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080031from autotest_lib.server.hosts import chameleon_host
Richard Barnetted31580e2018-05-14 19:58:00 +000032from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080033from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070034from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070035from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070036from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080037from autotest_lib.site_utils.rpm_control_system import rpm_client
Otabek Kasimov808cd832020-05-28 18:27:46 -070038from autotest_lib.site_utils.admin_audit import constants as audit_const
Simran Basid5e5e272012-09-24 15:23:59 -070039
Simran Basi382506b2016-09-13 14:58:15 -070040# In case cros_host is being ran via SSP on an older Moblab version with an
41# older chromite version.
42try:
43 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080044except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080045 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080046
Simran Basid5e5e272012-09-24 15:23:59 -070047
Dan Shib8540a52015-07-16 14:18:23 -070048CONFIG = global_config.global_config
49
Dan Shid07ee2e2015-09-24 14:49:25 -070050
beepsc87ff602013-07-31 21:53:00 -070051class FactoryImageCheckerException(error.AutoservError):
52 """Exception raised when an image is a factory image."""
53 pass
54
55
Fang Deng0ca40e22013-08-27 17:47:44 -070056class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070057 """Chromium OS specific subclass of Host."""
58
Simran Basi5ace6f22016-01-06 17:30:44 -080059 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
60
Scott Zawalski62bacae2013-03-05 10:40:32 -050061 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070062
Richard Barnette03a0c132012-11-05 12:40:35 -080063 # Timeout values (in seconds) associated with various Chrome OS
64 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070065 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080066 # In general, a good rule of thumb is that the timeout can be up
67 # to twice the typical measured value on the slowest platform.
68 # The times here have not necessarily been empirically tested to
69 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070070 #
71 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080072 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
73 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080074 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070075 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080076 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070077 # screen delay, time to start the network on the DUT, and the
78 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070079 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080080 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080081 # network.
beepsf079cfb2013-09-18 17:49:51 -070082 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080083 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
84 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070085
86 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080087 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080088 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070089 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070090 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070091 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080092 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070093
Dan Shica503482015-03-30 17:23:25 -070094 # Minimum OS version that supports server side packaging. Older builds may
95 # not have server side package built or with Autotest code change to support
96 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070097 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -070098 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -070099
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800100 # REBOOT_TIMEOUT: How long to wait for a reboot.
101 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700102 # We have a long timeout to ensure we don't flakily fail due to other
103 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700104 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
105 # return from reboot' bug is solved.
106 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700107
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800108 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
109 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700110 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
111 # since changing servo role will reset USB state
112 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800113 _USB_POWER_TIMEOUT = 5
114 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700115 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800116
Fang Dengdeba14f2014-11-14 11:54:09 -0800117 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
118 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700119
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800120 # Constants used in ping_wait_up() and ping_wait_down().
121 #
122 # _PING_WAIT_COUNT is the approximate number of polling
123 # cycles to use when waiting for a host state change.
124 #
125 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
126 # for arguments to the internal _ping_wait_for_status()
127 # method.
128 _PING_WAIT_COUNT = 40
129 _PING_STATUS_DOWN = False
130 _PING_STATUS_UP = True
131
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800132 # Allowed values for the power_method argument.
133
Garry Wang5e5538a2019-04-08 15:36:18 -0700134 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
135 # DUTs except those with servo_v4 CCD.
136 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
137 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800138 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
139 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
140 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700141 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800142 POWER_CONTROL_SERVO = 'servoj10'
143 POWER_CONTROL_MANUAL = 'manual'
144
145 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700146 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800147 POWER_CONTROL_SERVO,
148 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800149
Simran Basi5e6339a2013-03-21 11:34:32 -0700150 _RPM_OUTLET_CHANGED = 'outlet_changed'
151
Dan Shi9cb0eec2014-06-03 09:04:50 -0700152 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700153 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700154 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700155
Brent Peterson1cb623a2020-01-09 13:14:28 -0800156 # Regular expression for extracting EC version string
157 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
158
159 # Regular expression for extracting BIOS version string
160 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
161
Brent Petersonc70a1832020-01-24 15:54:35 -0800162 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700163 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800164
J. Richard Barnette964fba02012-10-24 17:34:29 -0700165 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800166 def check_host(host, timeout=10):
167 """
168 Check if the given host is a chrome-os host.
169
170 @param host: An ssh host representing a device.
171 @param timeout: The timeout for the run command.
172
173 @return: True if the host device is chromeos.
174
beeps46dadc92013-11-07 14:07:10 -0800175 """
176 try:
Allen Liad719c12017-06-27 23:48:04 +0000177 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700178 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700179 '! grep -q moblab /etc/lsb-release && '
180 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700181 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700182 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000183 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700184 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
185 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800186 return not (
187 lsbrelease_utils.is_jetstream(
188 lsb_release_content=lsb_release_content) or
189 lsbrelease_utils.is_gce_board(
190 lsb_release_content=lsb_release_content))
191
beeps46dadc92013-11-07 14:07:10 -0800192 except (error.AutoservRunError, error.AutoservSSHTimeout):
193 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700194
195 return False
beeps46dadc92013-11-07 14:07:10 -0800196
197
198 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800199 def get_chameleon_arguments(args_dict):
200 """Extract chameleon options from `args_dict` and return the result.
201
202 Recommended usage:
203 ~~~~~~~~
204 args_dict = utils.args_to_dict(args)
205 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
206 host = hosts.create_host(machine, chameleon_args=chameleon_args)
207 ~~~~~~~~
208
209 @param args_dict Dictionary from which to extract the chameleon
210 arguments.
211 """
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800212 return {key: args_dict[key]
Allen Li083866b2016-08-18 10:07:10 -0700213 for key in ('chameleon_host', 'chameleon_port')
214 if key in args_dict}
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800215
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800216 @staticmethod
217 def get_btpeer_arguments(args_dict):
218 """Extract btpeer options from `args_dict` and return the result.
219
220 This is used to parse details of Bluetooth peer.
221 Recommended usage:
222 ~~~~~~~~
223 args_dict = utils.args_to_dict(args)
224 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700225 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800226 ~~~~~~~~
227
228 @param args_dict: Dictionary from which to extract the btpeer
229 arguments.
230 """
231 if 'btpeer_host_list' in args_dict:
232 result = []
233 for btpeer in args_dict['btpeer_host_list'].split(','):
234 result.append({key: value for key,value in
235 zip(('btpeer_host','btpeer_port'),
236 btpeer.split(':'))})
237 return result
238 else:
239 return {key: args_dict[key]
240 for key in ('btpeer_host', 'btpeer_port')
241 if key in args_dict}
242
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800243
244 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700245 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800246 """Extract chameleon options from `args_dict` and return the result.
247
248 Recommended usage:
249 ~~~~~~~~
250 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700251 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
252 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800253 ~~~~~~~~
254
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700255 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800256 arguments.
257 """
Allen Li083866b2016-08-18 10:07:10 -0700258 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700259 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700260 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800261
262
263 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800264 def get_servo_arguments(args_dict):
265 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800266
267 Recommended usage:
268 ~~~~~~~~
269 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700270 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800271 host = hosts.create_host(machine, servo_args=servo_args)
272 ~~~~~~~~
273
274 @param args_dict Dictionary from which to extract the servo
275 arguments.
276 """
Garry Wang11b5e872020-03-11 15:14:08 -0700277 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
278 servo_constants.SERVO_PORT_ATTR,
279 servo_constants.SERVO_BOARD_ATTR,
280 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200281 servo_args = {key: args_dict[key]
282 for key in servo_attrs
283 if key in args_dict}
284 return (
285 None
Garry Wang11b5e872020-03-11 15:14:08 -0700286 if servo_constants.SERVO_HOST_ATTR in servo_args
287 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200288 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700289
J. Richard Barnette964fba02012-10-24 17:34:29 -0700290
J. Richard Barnette91137f02016-03-10 16:52:26 -0800291 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700292 pdtester_args=None, try_lab_servo=False,
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700293 try_servo_repair=False, btpeer_args=[],
J. Richard Barnette91137f02016-03-10 16:52:26 -0800294 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700295 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800296 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700297
Fang Denge545abb2014-12-30 18:43:47 -0800298 This method will attempt to create the test-assistant object
299 (chameleon/servo) when it is needed by the test. Check
300 the docstring of chameleon_host.create_chameleon_host and
301 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700302
Fang Denge545abb2014-12-30 18:43:47 -0800303 @param hostname: Hostname of the dut.
304 @param chameleon_args: A dictionary that contains args for creating
305 a ChameleonHost. See chameleon_host for details.
306 @param servo_args: A dictionary that contains args for creating
307 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700308 @param try_lab_servo: When true, indicates that an attempt should
309 be made to create a ServoHost for a DUT in
310 the test lab, even if not required by
311 `servo_args`. See servo_host for details.
312 @param try_servo_repair: If a servo host is created, check it
313 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800314 See servo_host for details.
315 @param ssh_verbosity_flag: String, to pass to the ssh command to control
316 verbosity.
317 @param ssh_options: String, other ssh options to pass to the ssh
318 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700319 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700320 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700321 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800322 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Kevin Chenga2619dc2016-03-28 11:42:08 -0700323 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700324 # self.env is a dictionary of environment variable settings
325 # to be exported for commands run on the host.
326 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
327 # errors that might happen.
328 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700329 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700330 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700331 _servo_host, servo_state = servo_host.create_servo_host(
332 dut=self,
333 servo_args=servo_args,
334 try_lab_servo=try_lab_servo,
335 try_servo_repair=try_servo_repair,
336 dut_host_info=self.host_info_store.get())
337 self.set_servo_host(_servo_host, servo_state)
Garry Wang5e5538a2019-04-08 15:36:18 -0700338 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700339
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800340 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800341 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700342 dut=self.hostname,
343 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800344 if self._chameleon_host:
345 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800346 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800347 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700348
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700349 # Initialize Bluetooth peers.
350 try:
351 self.initialize_btpeer(btpeer_args)
352 except Exception as e:
353 logging.error('Exception %s in initialize_btpeer', str(e))
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800354
howardchung83e55272019-08-08 14:08:05 +0800355 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700356 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700357 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800358
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700359 if self._pdtester_host:
360 self.pdtester_servo = self._pdtester_host.get_servo()
361 logging.info('pdtester_servo: %r', self.pdtester_servo)
362 # Create the pdtester object used to access the ec uart
363 self.pdtester = pdtester.PDTester(self.pdtester_servo,
364 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800365 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700366 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800367
Fang Deng5d518f42013-08-02 14:04:32 -0700368
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800369 def initialize_btpeer(self, btpeer_args):
370 """ Initialize the Bluetooth peers
371
372 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
373 is chameleon host on Raspberry Pi.
374 @param btpeer_args: A dictionary that contains args for creating
375 a ChameleonHost. See chameleon_host for details.
376
377 """
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700378 #TODO (b:142486063) Remove the try..except
379 try:
380 self._btpeer_host_list = []
381 self.btpeer_list = []
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800382 self.btpeer = None
383
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700384 if type(btpeer_args) is list:
385 btpeer_args_list = btpeer_args
386 else:
387 btpeer_args_list = [btpeer_args]
388
389 self._btpeer_host_list = chameleon_host.create_btpeer_host(
390 dut=self.hostname, btpeer_args_list=btpeer_args_list)
391 logging.debug('Bluetooth peer hosts are %s',
392 self._btpeer_host_list)
393 self.btpeer_list = [_host.create_chameleon_board() for _host in
394 self._btpeer_host_list if _host is not None]
395
396 if len(self.btpeer_list) > 0:
397 self.btpeer = self.btpeer_list[0]
398
399 logging.debug('After initialize_btpeer btpeer_list %s '
400 'btpeer_host_list is %s and btpeer is %s',
401 self.btpeer_list, self._btpeer_host_list,
402 self.btpeer)
403 except Exception as e:
404 logging.error('Exception %s in initialize_btpeer', str(e))
405
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800406
407
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000408 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700409 """Get latest stable cros image name from AFE.
410
411 Use the board name from the info store. Should that fail, try to
412 retrieve the board name from the host's installed image itself.
413
414 @returns: current stable cros image name for this host.
415 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700416 info = self.host_info_store.get()
417 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700418 logging.warn('No board label value found. Trying to infer '
419 'from the host itself.')
420 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700421 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700422 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
423 logging.error('Also failed to get the board name from the DUT '
424 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700425 raise error.AutoservError('Cannot determine board of the DUT'
426 ' while getting repair image name.')
427 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500428
429
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700430 def host_version_prefix(self, image):
431 """Return version label prefix.
432
433 In case the CrOS provisioning version is something other than the
434 standard CrOS version e.g. CrOS TH version, this function will
435 find the prefix from provision.py.
436
437 @param image: The image name to find its version prefix.
438 @returns: A prefix string for the image type.
439 """
440 return provision.get_version_label_prefix(image)
441
Andrew Luo3332ab22020-04-28 16:42:03 -0700442 def stage_build_to_usb(self, build):
443 """Stage the current ChromeOS image on the USB stick connected to the
444 servo.
445
446 @param build: The build to download and send to USB.
447 """
448 if not self.servo:
449 raise error.TestError('Host %s does not have servo.' %
450 self.hostname)
451
452 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700453
454 try:
455 self.servo.image_to_servo_usb(update_url)
456 finally:
457 # servo.image_to_servo_usb turned the DUT off, so turn it back on
458 logging.debug('Turn DUT power back on.')
459 self.servo.get_power_state_controller().power_on()
460
Andrew Luo3332ab22020-04-28 16:42:03 -0700461 logging.debug('ChromeOS image %s is staged on the USB stick.',
462 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700463
beepsdae65fd2013-07-26 16:24:41 -0700464 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700465 """
466 Make sure job_repo_url of this host is valid.
467
joychen03eaad92013-06-26 09:55:21 -0700468 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700469 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
470 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
471 download and extract it. If the devserver embedded in the url is
472 unresponsive, update the job_repo_url of the host after staging it on
473 another devserver.
474
475 @param job_repo_url: A url pointing to the devserver where the autotest
476 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700477 @param tag: The tag from the server job, in the format
478 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700479
480 @raises DevServerException: If we could not resolve a devserver.
481 @raises AutoservError: If we're unable to save the new job_repo_url as
482 a result of choosing a new devserver because the old one failed to
483 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700484 @raises urllib2.URLError: If the devserver embedded in job_repo_url
485 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700486 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800487 info = self.host_info_store.get()
488 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700489 if not job_repo_url:
490 logging.warning('No job repo url set on host %s', self.hostname)
491 return
492
493 logging.info('Verifying job repo url %s', job_repo_url)
494 devserver_url, image_name = tools.get_devserver_build_from_package_url(
495 job_repo_url)
496
beeps0c865032013-07-30 11:37:06 -0700497 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700498
499 logging.info('Staging autotest artifacts for %s on devserver %s',
500 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700501
502 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700503 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700504 stage_time = time.time() - start_time
505
506 # Record how much of the verification time comes from a devserver
507 # restage. If we're doing things right we should not see multiple
508 # devservers for a given board/build/branch path.
509 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800510 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700511 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800512 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700513 pass
514 else:
beeps0c865032013-07-30 11:37:06 -0700515 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700516 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700517 stats_key = {
518 'board': board,
519 'build_type': build_type,
520 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700521 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700522 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800523
524 monarch_fields = {
525 'board': board,
526 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800527 'branch': branch,
528 'dev_server': devserver,
529 }
530 metrics.Counter(
531 'chromeos/autotest/provision/verify_url'
532 ).increment(fields=monarch_fields)
533 metrics.SecondsDistribution(
534 'chromeos/autotest/provision/verify_url_duration'
535 ).add(stage_time, fields=monarch_fields)
536
537
Dan Shicf4d2032015-03-12 15:04:21 -0700538 def stage_server_side_package(self, image=None):
539 """Stage autotest server-side package on devserver.
540
541 @param image: Full path of an OS image to install or a build name.
542
543 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700544
545 @raise: error.AutoservError if fail to locate the build to test with, or
546 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700547 """
Dan Shid37736b2016-07-06 15:10:29 -0700548 # If enable_drone_in_restricted_subnet is False, do not set hostname
549 # in devserver.resolve call, so a devserver in non-restricted subnet
550 # is picked to stage autotest server package for drone to download.
551 hostname = self.hostname
552 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
553 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700554 if image:
555 image_name = tools.get_build_from_image(image)
556 if not image_name:
557 raise error.AutoservError(
558 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700559 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700560 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700561 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800562 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700563 if job_repo_url:
564 devserver_url, image_name = (
565 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700566 # If enable_drone_in_restricted_subnet is True, use the
567 # existing devserver. Otherwise, resolve a new one in
568 # non-restricted subnet.
569 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
570 ds = dev_server.ImageServer(devserver_url)
571 else:
572 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800573 elif info.build is not None:
574 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700575 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700576 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800577 raise error.AutoservError(
578 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700579 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700580
581 # Get the OS version of the build, for any build older than
582 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
583 match = re.match('.*/R\d+-(\d+)\.', image_name)
584 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700585 raise error.AutoservError(
586 'Build %s is older than %s. Server side packaging is '
587 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700588
Dan Shicf4d2032015-03-12 15:04:21 -0700589 ds.stage_artifacts(image_name, ['autotest_server_package'])
590 return '%s/static/%s/%s' % (ds.url(), image_name,
591 'autotest_server_package.tar.bz2')
592
593
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700594 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700595 """Stage a build on a devserver and return the update_url.
596
597 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700598 @param artifact: a string like 'test_image'. Requests
599 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700600 @returns a tuple of (image_name, URL) like
601 (lumpy-release/R27-3837.0.0,
602 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700603 """
604 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000605 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700606 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800607 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700608 devserver.stage_artifacts(image_name, [artifact])
609 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700610 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700611 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700612 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700613 else:
614 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700615
616
beepse539be02013-07-31 21:57:39 -0700617 def stage_factory_image_for_servo(self, image_name):
618 """Stage a build on a devserver and return the update_url.
619
620 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700621
beepse539be02013-07-31 21:57:39 -0700622 @return: An update URL, eg:
623 http://<devserver>/static/canary-channel/\
624 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700625
626 @raises: ValueError if the factory artifact name is missing from
627 the config.
628
beepse539be02013-07-31 21:57:39 -0700629 """
630 if not image_name:
631 logging.error('Need an image_name to stage a factory image.')
632 return
633
Dan Shib8540a52015-07-16 14:18:23 -0700634 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700635 'CROS', 'factory_artifact', type=str, default='')
636 if not factory_artifact:
637 raise ValueError('Cannot retrieve the factory artifact name from '
638 'autotest config, and hence cannot stage factory '
639 'artifacts.')
640
beepse539be02013-07-31 21:57:39 -0700641 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800642 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700643 devserver.stage_artifacts(
644 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700645 [factory_artifact],
646 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700647
648 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
649
650
Laurence Goodby778c9a42017-05-24 19:24:07 -0700651 def prepare_for_update(self):
652 """Prepares the DUT for an update.
653
654 Subclasses may override this to perform any special actions
655 required before updating.
656 """
Laurence Goodby468de252017-06-08 17:22:53 -0700657 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700658
659
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800660 def _clear_fw_version_labels(self, rw_only):
661 """Clear firmware version labels from the machine.
662
663 @param rw_only: True to only clear fwrw_version; otherewise, clear
664 both fwro_version and fwrw_version.
665 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700666 info = self.host_info_store.get()
667 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800668 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700669 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
670 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700671
672
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800673 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700674 """Add firmware version label to the machine.
675
676 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800677 @param rw_only: True to only add fwrw_version; otherwise, add both
678 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700679
680 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700681 info = self.host_info_store.get()
682 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800683 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700684 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
685 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700686
687
Namyoon Woo33f38852020-04-13 17:26:58 -0700688 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800689 """Search for the latest package release version from the image archive,
690 and return it.
691
Namyoon Woo33f38852020-04-13 17:26:58 -0700692 @param platform: platform name, a.k.a. board or model
693 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800694
Namyoon Woo33f38852020-04-13 17:26:58 -0700695 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
696 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800697 or None if LATEST release file does not exist.
698 """
699
Namyoon Woo33f38852020-04-13 17:26:58 -0700700 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800701
Namyoon Woo33f38852020-04-13 17:26:58 -0700702 # Search the image path in reference board archive as well.
703 # For example, bob has its binary image under its reference board (gru)
704 # image archive.
705 if ref_board:
706 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800707
Namyoon Woo33f38852020-04-13 17:26:58 -0700708 for board in platforms:
709 # Read 'LATEST-1.0.0' file
710 branch_dir = provision.FW_BRANCH_GLOB % board
711 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
712 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800713
Namyoon Woo33f38852020-04-13 17:26:58 -0700714 try:
715 # The result could be one or more.
716 result = utils.system_output('gsutil ls -d ' + latest_file)
717
718 candidates = re.findall('gs://.*', result)
719
720 # Found the directory candidates. No need to check the other
721 # board name cadidates. Let's break the loop.
722 break
723 except error.CmdError:
724 # It doesn't exist. Let's move on to the next item.
725 pass
726 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800727 logging.error('No LATEST release info is available.')
728 return None
729
Namyoon Woo406c7d42020-01-24 15:57:11 -0800730 for cand_dir in candidates:
731 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800732
Namyoon Woo406c7d42020-01-24 15:57:11 -0800733 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700734 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800735 try:
736 # Check if release_path does exist.
737 release = utils.system_output('gsutil ls -d ' + release_path)
738 # Now 'release' has a full directory path: e.g.
739 # gs://chromeos-image-archive/firmware-octopus-11297.B-
740 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
741
742 # Remove "gs://chromeos-image-archive".
743 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
744
745 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
746 return release.strip('/')
747 except error.CmdError:
748 # The directory might not exist. Let's try next candidate.
749 pass
750 else:
751 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800752
Brent Peterson1cb623a2020-01-09 13:14:28 -0800753 @staticmethod
754 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800755 """Get version string from binary image using regular expression.
756
757 @param image: Binary image to search
758 @param version_regex: Regular expression to search for
759
760 @return Version string
761
762 @raises TestFail if no version string is found in image
763 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800764 with open(image, 'rb') as f:
765 image_data = f.read()
766 match = re.findall(version_regex, image_data)
767 if match:
768 return match[0]
769 else:
770 raise error.TestFail('Failed to read version from %s.' % image)
771
772
Garry Wangad2a1712020-03-26 15:06:43 -0700773 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800774 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700775 try_scp=False, install_ec=True, install_bios=True,
776 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700777 """Install firmware to the DUT.
778
779 Use stateful update if the DUT is already running the same build.
780 Stateful update does not update kernel and tends to run much faster
781 than a full reimage. If the DUT is running a different build, or it
782 failed to do a stateful update, full update, including kernel update,
783 will be applied to the DUT.
784
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800785 Once a host enters firmware_install its fw[ro|rw]_version label will
786 be removed. After the firmware is updated successfully, a new
787 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700788
789 @param build: The build version to which we want to provision the
790 firmware of the machine,
791 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800792 @param rw_only: True to only install firmware to its RW portions. Keep
793 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700794 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800795 @param local_tarball: Path to local firmware image for installing
796 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800797 @param verify_version: True to verify EC and BIOS versions after
798 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800799 @param try_scp: False to always program using servo, true to try copying
800 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700801 @param install_ec: True to install EC FW, and False to skip it.
802 @param install_bios: True to install BIOS, and False to skip it.
803 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700804
805 TODO(dshi): After bug 381718 is fixed, update here with corresponding
806 exceptions that could be raised.
807
808 """
809 if not self.servo:
810 raise error.TestError('Host %s does not have servo.' %
811 self.hostname)
812
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700813 # Get the DUT board name from AFE.
814 info = self.host_info_store.get()
815 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700816 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800817
818 if board is None or board == '':
819 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700820
Namyoon Woo382e5892020-05-20 16:48:40 -0700821 # if board_as argument is passed, then use it instead of the original
822 # board name.
823 if board_as:
824 board = board_as
825
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700826 if model is None or model == '':
827 model = self.get_platform_from_fwid()
828
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800829 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700830 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800831 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700832 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800833
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700834 try:
835 ds = dev_server.ImageServer.resolve(build, self.hostname)
836 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800837
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700838 if not dest:
839 tmpd = autotemp.tempdir(unique_id='fwimage')
840 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800841
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700842 # Download firmware image
843 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
844 local_tarball = os.path.join(dest, os.path.basename(fwurl))
845 ds.download_file(fwurl, local_tarball)
846 except Exception as e:
847 raise error.TestError('Failed to download firmware package: %s'
848 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700849
Namyoon Woo382e5892020-05-20 16:48:40 -0700850 ec_image = None
851 if install_ec:
852 # Extract EC image from tarball
853 logging.info('Extracting EC image.')
854 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800855
Namyoon Woo382e5892020-05-20 16:48:40 -0700856 bios_image = None
857 if install_bios:
858 # Extract BIOS image from tarball
859 logging.info('Extracting BIOS image.')
860 bios_image = self.servo.extract_bios_image(board, model,
861 local_tarball)
862
863 if not bios_image and not ec_image:
864 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800865
Brent Petersonc70a1832020-01-24 15:54:35 -0800866 # Clear firmware version labels
867 self._clear_fw_version_labels(rw_only)
868
869 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800870 try:
Brent Petersonc70a1832020-01-24 15:54:35 -0800871 # Check if DUT is available and copying to DUT is enabled
872 if self.is_up() and try_scp:
873 # DUT is available, make temp firmware directory to store images
874 logging.info('Making temp folder.')
875 dest_folder = '/tmp/firmware'
876 self.run('mkdir -p ' + dest_folder)
877
Namyoon Woo68b68082020-06-02 13:13:14 -0700878 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800879
Namyoon Woo382e5892020-05-20 16:48:40 -0700880 if bios_image:
881 # Send BIOS firmware image to DUT
882 logging.info('Sending BIOS firmware.')
883 dest_bios_path = os.path.join(dest_folder,
884 os.path.basename(bios_image))
885 self.send_file(bios_image, dest_bios_path)
886
887 # Initialize firmware update command for BIOS image
888 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800889
890 # Send EC firmware image to DUT when EC image was found
891 if ec_image:
892 logging.info('Sending EC firmware.')
893 dest_ec_path = os.path.join(dest_folder,
894 os.path.basename(ec_image))
895 self.send_file(ec_image, dest_ec_path)
896
897 # Add EC image to firmware update command
898 fw_cmd += ' -e %s' % dest_ec_path
899
900 # Update firmware on DUT
901 logging.info('Updating firmware.')
Brent Petersonc70a1832020-01-24 15:54:35 -0800902 self.run(fw_cmd)
903 else:
904 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800905 if ec_image:
906 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700907 if bios_image:
908 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800909 if utils.host_is_in_lab_zone(self.hostname):
910 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800911
912 # Reboot and wait for DUT after installing firmware
913 logging.info('Rebooting DUT.')
914 self.servo.get_power_state_controller().reset()
915 time.sleep(self.servo.BOOT_DELAY)
916 self.test_wait_for_boot()
917
918 # When enabled verify EC and BIOS firmware version after programming
919 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800920 # Check programmed EC firmware when EC image was found
921 if ec_image:
922 logging.info('Checking EC firmware version.')
923 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800924 ec_version_prefix = dest_ec_version.split('_', 1)[0]
925 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800926 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800927 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800928 if dest_ec_version != image_ec_version:
929 raise error.TestFail(
930 'Failed to update EC RO, version %s (expected %s)' %
931 (dest_ec_version, image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800932
Namyoon Woo382e5892020-05-20 16:48:40 -0700933 if bios_image:
934 # Check programmed BIOS firmware against expected version
935 logging.info('Checking BIOS firmware version.')
936 dest_bios_version = self.get_firmware_version()
937 bios_version_prefix = dest_bios_version.split('.', 1)[0]
938 bios_regex = self._BIOS_REGEX % bios_version_prefix
939 image_bios_version = self.get_version_from_image(bios_image,
940 bios_regex)
941 if dest_bios_version != image_bios_version:
942 raise error.TestFail(
943 'Failed to update BIOS RO, version %s '
944 '(expected %s)' % (dest_bios_version,
945 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700946 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700947 if tmpd:
948 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700949
950
beepsf079cfb2013-09-18 17:49:51 -0700951 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
952 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500953 """
954 Re-install the OS on the DUT by:
955 1) installing a test image on a USB storage device attached to the Servo
956 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800957 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700958 3) installing the image with chromeos-install.
959
Scott Zawalski62bacae2013-03-05 10:40:32 -0500960 @param image_url: If specified use as the url to install on the DUT.
961 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700962 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
963 Factory images need a longer usb_boot_timeout than regular
964 cros images.
965 @param install_timeout: The timeout to use when installing the chromeos
966 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800967
Scott Zawalski62bacae2013-03-05 10:40:32 -0500968 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700969
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800970 """
Garry Wang7b0e1b72020-03-25 19:08:59 -0700971 if image_url:
972 logging.info('Downloading image to USB, then booting from it.'
973 ' Usb boot timeout = %s', usb_boot_timeout)
974 else:
975 logging.info('Booting from USB directly. Usb boot timeout = %s',
976 usb_boot_timeout)
977
978 metrics_field = {'download': bool(image_url)}
979 metrics.Counter(
980 'chromeos/autotest/provision/servo_install/download_image'
981 ).increment(fields=metrics_field)
982
Allen Li48a13fe2016-11-22 14:10:40 -0800983 with metrics.SecondsTimer(
984 'chromeos/autotest/provision/servo_install/boot_duration'):
985 self.servo.install_recovery_image(image_url)
986 if not self.wait_up(timeout=usb_boot_timeout):
987 raise hosts.AutoservRepairError(
988 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -0700989 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500990
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800991 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
992 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800993 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800994 try:
995 self.run('chromeos-tpm-recovery')
996 except error.AutoservRunError:
997 logging.warn('chromeos-tpm-recovery is too old.')
998
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800999
Allen Li48a13fe2016-11-22 14:10:40 -08001000 with metrics.SecondsTimer(
1001 'chromeos/autotest/provision/servo_install/install_duration'):
1002 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001003 try:
1004 self.run('chromeos-install --yes',timeout=install_timeout)
1005 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001006 except Exception as e:
1007 storage_errors = [
1008 'No space left on device',
1009 'I/O error when trying to write primary GPT',
1010 'Input/output error while writing out',
1011 'cannot read GPT header',
1012 'can not determine destination device'
1013 ]
1014 has_error = [msg for msg in storage_errors if(msg in str(e))]
1015 if has_error:
1016 info = self.host_info_store.get()
1017 info.set_version_label(
1018 audit_const.DUT_STORAGE_STATE_PREFIX,
1019 audit_const.HW_STATE_NEED_REPLACEMENT)
1020 self.host_info_store.commit(info)
1021 logging.debug(
1022 'Fail install image from USB; Storage error; %s', e)
1023 raise error.AutoservError(
1024 'Failed to install image from USB due to a suspect '
1025 'disk failure, DUT storage state changed to '
1026 'need_replacement, please check debug log '
1027 'for details.')
1028 else:
1029 logging.debug('Fail install image from USB; %s', e)
1030 raise error.AutoservError(
1031 'Failed to install image from USB due to unexpected '
1032 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001033 finally:
1034 # We need reset the DUT no matter re-install success or not,
1035 # as we don't want leave the DUT in boot from usb state.
1036 logging.info('Power cycling DUT through servo.')
1037 self.servo.get_power_state_controller().power_off()
1038 self.servo.switch_usbkey('off')
1039 # N.B. The Servo API requires that we use power_on() here
1040 # for two reasons:
1041 # 1) After turning on a DUT in recovery mode, you must turn
1042 # it off and then on with power_on() once more to
1043 # disable recovery mode (this is a Parrot specific
1044 # requirement).
1045 # 2) After power_off(), the only way to turn on is with
1046 # power_on() (this is a Storm specific requirement).
1047 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001048
1049 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001050 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001051 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1052 'test image after %d seconds' %
1053 self.BOOT_TIMEOUT,
1054 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001055
1056
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001057 def set_servo_host(self, host, servo_state = None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001058 """Set our servo host member, and associated servo.
1059
1060 @param host Our new `ServoHost`.
1061 """
1062 self._servo_host = host
1063 if self._servo_host is not None:
1064 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001065 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001066 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001067 else:
1068 self.servo = None
Otabek Kasimov41301a22020-05-10 15:28:21 -07001069 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001070 self.set_servo_state(servo_state)
1071
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001072
Richard Barnette9a26ad62016-06-10 12:03:08 -07001073 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001074 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001075 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001076
Richard Barnette9a26ad62016-06-10 12:03:08 -07001077 If the servo object is missing, attempt to repair the servo
1078 host. Repair failures are passed back to the caller.
1079
1080 @raise AutoservError: If there is no servo host for this CrOS
1081 host.
1082 """
1083 if self.servo:
1084 return
1085 if not self._servo_host:
1086 raise error.AutoservError('No servo host for %s.' %
1087 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001088 try:
1089 self._servo_host.repair()
1090 except:
1091 raise
1092 finally:
1093 self.set_servo_host(self._servo_host)
1094
1095
Otabek Kasimov41301a22020-05-10 15:28:21 -07001096 def set_servo_type(self):
1097 """Set servo info labels to dut host_info"""
1098 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001099 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001100 return
1101 servo_type = self.servo.get_servo_type()
1102 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001103 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001104 ' by `dut-control servo_type`! Please file a bug'
1105 ' and inform infra team as we are not expected '
1106 ' to reach this point.')
1107 return
1108 host_info = self.host_info_store.get()
1109 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1110 old_type = host_info.get_label_value(prefix)
1111 if old_type == servo_type:
1112 # do not need update
1113 return
1114 host_info.set_version_label(prefix, servo_type)
1115 self.host_info_store.commit(host_info)
1116 logging.info('ServoHost: servo_type updated to %s '
1117 '(previous: %s)', servo_type, old_type)
1118
1119
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001120 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001121 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001122 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001123 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001124 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001125 old_state = host_info.get_label_value(servo_state_prefix)
1126 if old_state == servo_state:
1127 # do not need update
1128 return
1129 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001130 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001131 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1132 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001133
1134
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001135 def get_servo_state(self):
1136 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001137 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001138 return host_info.get_label_value(servo_state_prefix)
1139
Otabek Kasimov41301a22020-05-10 15:28:21 -07001140
Garry Wang000c6c02020-05-11 21:27:23 -07001141 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1142 if smart_usbhub_detected is None:
1143 # skip the label update here as this indicate we wasn't able
1144 # to confirm usbhub type.
1145 return
1146 host_info = self.host_info_store.get()
1147 if (smart_usbhub_detected ==
1148 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1149 # skip label update if current label match the truth.
1150 return
1151 if smart_usbhub_detected:
1152 logging.info('Adding %s label to host %s',
1153 servo_constants.SMART_USBHUB_LABEL,
1154 self.hostname)
1155 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1156 else:
1157 logging.info('Removing %s label from host %s',
1158 servo_constants.SMART_USBHUB_LABEL,
1159 self.hostname)
1160 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1161 self.host_info_store.commit(host_info)
1162
1163
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001164 def repair(self):
1165 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001166
1167 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001168 not call back to the parent class, but instead relies on
1169 `self._repair_strategy` to coordinate the verification and
1170 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001171 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001172 message = 'Beginning repair for host %s board %s model %s'
1173 info = self.host_info_store.get()
1174 message %= (self.hostname, info.board, info.model)
1175 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001176 try:
1177 self._repair_strategy.repair(self)
1178 except hosts.AutoservVerifyDependencyError as e:
1179 # We don't want flag a DUT as failed if only non-critical
1180 # verifier(s) failed during the repair.
1181 if e.is_critical():
1182 raise
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001183
Richard Barnette82c35912012-11-20 10:09:10 -08001184
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001185 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001186 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001187 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001188
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001189 if self._chameleon_host:
1190 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001191
1192 if self._servo_host:
1193 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001194
1195
Dan Shi49ca0932014-11-14 11:22:27 -08001196 def get_power_supply_info(self):
1197 """Get the output of power_supply_info.
1198
1199 power_supply_info outputs the info of each power supply, e.g.,
1200 Device: Line Power
1201 online: no
1202 type: Mains
1203 voltage (V): 0
1204 current (A): 0
1205 Device: Battery
1206 state: Discharging
1207 percentage: 95.9276
1208 technology: Li-ion
1209
1210 Above output shows two devices, Line Power and Battery, with details of
1211 each device listed. This function parses the output into a dictionary,
1212 with key being the device name, and value being a dictionary of details
1213 of the device info.
1214
1215 @return: The dictionary of power_supply_info, e.g.,
1216 {'Line Power': {'online': 'yes', 'type': 'main'},
1217 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001218 @raise error.AutoservRunError if power_supply_info tool is not found in
1219 the DUT. Caller should handle this error to avoid false failure
1220 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001221 """
1222 result = self.run('power_supply_info').stdout.strip()
1223 info = {}
1224 device_name = None
1225 device_info = {}
1226 for line in result.split('\n'):
1227 pair = [v.strip() for v in line.split(':')]
1228 if len(pair) != 2:
1229 continue
1230 if pair[0] == 'Device':
1231 if device_name:
1232 info[device_name] = device_info
1233 device_name = pair[1]
1234 device_info = {}
1235 else:
1236 device_info[pair[0]] = pair[1]
1237 if device_name and not device_name in info:
1238 info[device_name] = device_info
1239 return info
1240
1241
1242 def get_battery_percentage(self):
1243 """Get the battery percentage.
1244
1245 @return: The percentage of battery level, value range from 0-100. Return
1246 None if the battery info cannot be retrieved.
1247 """
1248 try:
1249 info = self.get_power_supply_info()
1250 logging.info(info)
1251 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001252 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001253 return None
1254
1255
Philip Chenaf69ead2020-03-27 13:06:42 -07001256 def get_battery_state(self):
1257 """Get the battery charging state.
1258
1259 @return: A string representing the battery charging state. It can be
1260 'Charging', 'Fully charged', or 'Discharging'.
1261 """
1262 try:
1263 info = self.get_power_supply_info()
1264 logging.info(info)
1265 return info['Battery']['state']
1266 except (KeyError, ValueError, error.AutoservRunError):
1267 return None
1268
1269
Daniel Campello8ca25c22019-12-13 16:48:26 -07001270 def get_battery_display_percentage(self):
1271 """Get the battery display percentage.
1272
1273 @return: The display percentage of battery level, value range from
1274 0-100. Return None if the battery info cannot be retrieved.
1275 """
1276 try:
1277 info = self.get_power_supply_info()
1278 logging.info(info)
1279 return float(info['Battery']['display percentage'])
1280 except (KeyError, ValueError, error.AutoservRunError):
1281 return None
1282
1283
Dan Shi49ca0932014-11-14 11:22:27 -08001284 def is_ac_connected(self):
1285 """Check if the dut has power adapter connected and charging.
1286
1287 @return: True if power adapter is connected and charging.
1288 """
1289 try:
1290 info = self.get_power_supply_info()
1291 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001292 except (KeyError, error.AutoservRunError):
1293 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001294
1295
Simran Basi5e6339a2013-03-21 11:34:32 -07001296 def _cleanup_poweron(self):
1297 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001298 info = self.host_info_store.get()
1299 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001300 return
1301 logging.debug('This host has recently interacted with the RPM'
1302 ' Infrastructure. Ensuring power is on.')
1303 try:
1304 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001305 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001306 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001307 logging.error('Failed to turn Power On for this host after '
1308 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001309
1310 battery_percentage = self.get_battery_percentage()
Dan Shif01ebe22014-12-05 13:10:57 -08001311 if battery_percentage and battery_percentage < 50:
Dan Shi49ca0932014-11-14 11:22:27 -08001312 raise
1313 elif self.is_ac_connected():
1314 logging.info('The device has power adapter connected and '
1315 'charging. No need to try to turn RPM on '
1316 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001317 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001318 logging.info('Battery level is now at %s%%. The device may '
1319 'still have enough power to run test, so no '
1320 'exception will be raised.', battery_percentage)
1321
Simran Basi5e6339a2013-03-21 11:34:32 -07001322
Garry Wangad4d4fd2019-01-30 17:00:38 -08001323 def _remove_rpm_changed_tag(self):
1324 info = self.host_info_store.get()
1325 del info.attributes[self._RPM_OUTLET_CHANGED]
1326 self.host_info_store.commit(info)
1327
1328
1329 def _add_rpm_changed_tag(self):
1330 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001331 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001332 self.host_info_store.commit(info)
1333
1334
1335
beepsc87ff602013-07-31 21:53:00 -07001336 def _is_factory_image(self):
1337 """Checks if the image on the DUT is a factory image.
1338
1339 @return: True if the image on the DUT is a factory image.
1340 False otherwise.
1341 """
1342 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1343 return result.exit_status == 0
1344
1345
1346 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001347 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001348
1349 @raises: FactoryImageCheckerException for factory images, since
1350 we cannot attempt to restart ui on them.
1351 error.AutoservRunError for any other type of error that
1352 occurs while restarting ui.
1353 """
1354 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001355 raise FactoryImageCheckerException('Cannot restart ui on factory '
1356 'images')
beepsc87ff602013-07-31 21:53:00 -07001357
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001358 # TODO(jrbarnette): The command to stop/start the ui job
1359 # should live inside cros_ui, too. However that would seem
1360 # to imply interface changes to the existing start()/restart()
1361 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001362 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001363 self.run('stop ui; start ui')
1364 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001365
1366
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001367 def _start_powerd_if_needed(self):
1368 """Start powerd if it isn't already running."""
1369 self.run('start powerd', ignore_status=True)
1370
1371
xixuana3bbc422017-05-04 15:57:21 -07001372 def _get_lsb_release_content(self):
1373 """Return the content of lsb-release file of host."""
1374 return self.run(
1375 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1376
1377
Dan Shi549fb822015-03-24 18:01:11 -07001378 def get_release_version(self):
1379 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1380
1381 @returns The version string in lsb-release, under attribute
1382 CHROMEOS_RELEASE_VERSION.
1383 """
Dan Shi549fb822015-03-24 18:01:11 -07001384 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001385 lsb_release_content=self._get_lsb_release_content())
1386
1387
Don Garrettb9f35802018-01-22 18:25:40 -08001388 def get_release_builder_path(self):
1389 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1390
1391 @returns The version string in lsb-release, under attribute
1392 CHROMEOS_RELEASE_BUILDER_PATH.
1393 """
1394 return lsbrelease_utils.get_chromeos_release_builder_path(
1395 lsb_release_content=self._get_lsb_release_content())
1396
1397
xixuana3bbc422017-05-04 15:57:21 -07001398 def get_chromeos_release_milestone(self):
1399 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1400 from lsb-release.
1401
1402 @returns The version string in lsb-release, under attribute
1403 CHROMEOS_RELEASE_BUILD_TYPE.
1404 """
1405 return lsbrelease_utils.get_chromeos_release_milestone(
1406 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001407
1408
1409 def verify_cros_version_label(self):
1410 """ Make sure host's cros-version label match the actual image in dut.
1411
1412 Remove any cros-version: label that doesn't match that installed in
1413 the dut.
1414
1415 @param raise_error: Set to True to raise exception if any mismatch found
1416
1417 @raise error.AutoservError: If any mismatch between cros-version label
1418 and the build installed in dut is found.
1419 """
Prathmesh Prabhuce2da3a2019-10-04 11:54:51 -07001420 # crbug.com/1007333: This check is being removed.
1421 return True
Dan Shi549fb822015-03-24 18:01:11 -07001422
1423
Laurence Goodby778c9a42017-05-24 19:24:07 -07001424 def cleanup_services(self):
1425 """Reinitializes the device for cleanup.
1426
1427 Subclasses may override this to customize the cleanup method.
1428
1429 To indicate failure of the reset, the implementation may raise
1430 any of:
1431 error.AutoservRunError
1432 error.AutotestRunError
1433 FactoryImageCheckerException
1434
1435 @raises error.AutoservRunError
1436 @raises error.AutotestRunError
1437 @raises error.FactoryImageCheckerException
1438 """
1439 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001440 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001441
1442
beepsc87ff602013-07-31 21:53:00 -07001443 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001444 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001445 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001446 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001447 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001448 except (error.AutotestRunError, error.AutoservRunError,
1449 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001450 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001451
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001452 # cleanup routines, i.e. reboot the machine.
1453 super(CrosHost, self).cleanup()
1454
Simran Basi5e6339a2013-03-21 11:34:32 -07001455 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001456 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001457 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001458 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001459
1460
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001461 def reboot(self, **dargs):
1462 """
1463 This function reboots the site host. The more generic
1464 RemoteHost.reboot() performs sync and sleeps for 5
1465 seconds. This is not necessary for Chrome OS devices as the
1466 sync should be finished in a short time during the reboot
1467 command.
1468 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001469 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001470 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001471 dargs['reboot_cmd'] = ('sleep 1; '
1472 'reboot & sleep %d; '
1473 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001474 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001475 if 'fastsync' not in dargs:
1476 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001477
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001478 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001479 # Record who called us
1480 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001481 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001482 'dut_host_name' : self.hostname,
1483 'success' : True}
1484 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001485 'caller' : "%s:%s" % (orig.co_filename,
1486 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001487 'success' : True,
1488 'error' : ''}
1489
Vincent Palatin80780b22016-07-27 16:02:37 +02001490 t0 = time.time()
1491 try:
1492 super(CrosHost, self).reboot(**dargs)
1493 except Exception as e:
1494 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001495 metric_debug_fields['success'] = False
1496 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001497 raise
1498 finally:
1499 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001500 metrics.Counter(
1501 'chromeos/autotest/autoserv/reboot_count').increment(
1502 fields=metric_fields)
1503 metrics.Counter(
1504 'chromeos/autotest/autoserv/reboot_debug').increment(
1505 fields=metric_debug_fields)
1506 metrics.SecondsDistribution(
1507 'chromeos/autotest/autoserv/reboot_duration').add(
1508 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001509
1510
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001511 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001512 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001513 """
1514 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001515
1516 @param suspend_time: How long to suspend as integer seconds.
1517 @param suspend_cmd: Suspend command to execute.
1518 @param allow_early_resume: If False and if device resumes before
1519 |suspend_time|, throw an error.
1520
1521 @exception AutoservSuspendError Host resumed earlier than
1522 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001523 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001524
1525 if suspend_cmd is None:
1526 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001527 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001528 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001529 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001530 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1531 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001532
1533
Simran Basiec564392014-08-25 16:48:09 -07001534 def upstart_status(self, service_name):
1535 """Check the status of an upstart init script.
1536
1537 @param service_name: Service to look up.
1538
1539 @returns True if the service is running, False otherwise.
1540 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001541 return 'start/running' in self.run('status %s' % service_name,
1542 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001543
Tom Hughese9552342018-12-18 14:29:25 -08001544 def upstart_stop(self, service_name):
1545 """Stops an upstart job if it's running.
1546
1547 @param service_name: Service to stop
1548
1549 @returns True if service has been stopped or was already stopped
1550 False otherwise.
1551 """
1552 if not self.upstart_status(service_name):
1553 return True
1554
1555 result = self.run('stop %s' % service_name, ignore_status=True)
1556 if result.exit_status != 0:
1557 return False
1558 return True
1559
1560 def upstart_restart(self, service_name):
1561 """Restarts (or starts) an upstart job.
1562
1563 @param service_name: Service to start/restart
1564
1565 @returns True if service has been started/restarted, False otherwise.
1566 """
1567 cmd = 'start'
1568 if self.upstart_status(service_name):
1569 cmd = 'restart'
1570 cmd = cmd + ' %s' % service_name
1571 result = self.run(cmd)
1572 if result.exit_status != 0:
1573 return False
1574 return True
Simran Basiec564392014-08-25 16:48:09 -07001575
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001576 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001577 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001578
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001579 Tests for the following conditions:
1580 1. All conditions tested by the parent version of this
1581 function.
1582 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001583 3. Sufficient space in /mnt/stateful_partition/encrypted.
1584 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001585
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001586 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001587 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001588 default_kilo_inodes_required = CONFIG.get_config_value(
1589 'SERVER', 'kilo_inodes_required', type=int, default=100)
1590 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1591 kilo_inodes_required = CONFIG.get_config_value(
1592 'SERVER', 'kilo_inodes_required_%s' % board,
1593 type=int, default=default_kilo_inodes_required)
1594 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001595 self.check_diskspace(
1596 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001597 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001598 'SERVER', 'gb_diskspace_required', type=float,
1599 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001600 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1601 # Not all targets build with encrypted stateful support.
1602 if self.path_exists(encrypted_stateful_path):
1603 self.check_diskspace(
1604 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001605 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001606 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1607 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001608
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001609 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001610
beepsc87ff602013-07-31 21:53:00 -07001611 # Factory images don't run update engine,
1612 # goofy controls dbus on these DUTs.
1613 if not self._is_factory_image():
1614 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001615
Dan Shi549fb822015-03-24 18:01:11 -07001616 self.verify_cros_version_label()
1617
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001618
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001619 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1620 def wait_for_system_services(self):
1621 """Waits for system-services to be running.
1622
1623 Sometimes, update_engine will take a while to update firmware, so we
1624 should give this some time to finish. See crbug.com/765686#c38 for
1625 details.
1626 """
1627 if not self.upstart_status('system-services'):
1628 raise error.AutoservError('Chrome failed to reach login. '
1629 'System services not running.')
1630
1631
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001632 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001633 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001634 message = 'Beginning verify for host %s board %s model %s'
1635 info = self.host_info_store.get()
1636 message %= (self.hostname, info.board, info.model)
1637 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001638 try:
1639 self._repair_strategy.verify(self)
1640 except hosts.AutoservVerifyDependencyError as e:
1641 # We don't want flag a DUT as failed if only non-critical
1642 # verifier(s) failed during the repair.
1643 if e.is_critical():
1644 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001645
1646
Fang Deng96667ca2013-08-01 17:46:18 -07001647 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001648 connect_timeout=None, alive_interval=None,
1649 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001650 """Override default make_ssh_command to use options tuned for Chrome OS.
1651
1652 Tuning changes:
1653 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1654 connection failure. Consistency with remote_access.sh.
1655
Samuel Tan2ce155b2015-06-23 18:24:38 -07001656 - ServerAliveInterval=900; which causes SSH to ping connection every
1657 900 seconds. In conjunction with ServerAliveCountMax ensures
1658 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001659 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001660 the test completed successfully. Later increased from 180 seconds to
1661 900 seconds to account for tests where the DUT is suspended for
1662 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001663
1664 - ServerAliveCountMax=3; consistency with remote_access.sh.
1665
1666 - ConnectAttempts=4; reduce flakiness in connection errors;
1667 consistency with remote_access.sh.
1668
1669 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1670 Host keys change with every new installation, don't waste
1671 memory/space saving them.
1672
1673 - SSH protocol forced to 2; needed for ServerAliveInterval.
1674
1675 @param user User name to use for the ssh connection.
1676 @param port Port on the target host to use for ssh connection.
1677 @param opts Additional options to the ssh command.
1678 @param hosts_file Ignored.
1679 @param connect_timeout Ignored.
1680 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001681 @param alive_count_max Ignored.
1682 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001683 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001684 options = ' '.join([opts, '-o Protocol=2'])
1685 return super(CrosHost, self).make_ssh_command(
1686 user=user, port=port, opts=options, hosts_file='/dev/null',
1687 connect_timeout=30, alive_interval=900, alive_count_max=3,
1688 connection_attempts=4)
1689
1690
Jason Abeleb6f924f2013-11-13 16:01:54 -08001691 def syslog(self, message, tag='autotest'):
1692 """Logs a message to syslog on host.
1693
1694 @param message String message to log into syslog
1695 @param tag String tag prefix for syslog
1696
1697 """
1698 self.run('logger -t "%s" "%s"' % (tag, message))
1699
1700
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001701 def _ping_check_status(self, status):
1702 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001703
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001704 @param status Check the ping status against this value.
1705 @return True iff `status` and the result of ping are the same
1706 (i.e. both True or both False).
1707
1708 """
1709 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1710 return not (status ^ (ping_val == 0))
1711
1712 def _ping_wait_for_status(self, status, timeout):
1713 """Wait for the host to have a given status (UP or DOWN).
1714
1715 Status is checked by polling. Polling will not last longer
1716 than the number of seconds in `timeout`. The polling
1717 interval will be long enough that only approximately
1718 _PING_WAIT_COUNT polling cycles will be executed, subject
1719 to a maximum interval of about one minute.
1720
1721 @param status Waiting will stop immediately if `ping` of the
1722 host returns this status.
1723 @param timeout Poll for at most this many seconds.
1724 @return True iff the host status from `ping` matched the
1725 requested status at the time of return.
1726
1727 """
1728 # _ping_check_status() takes about 1 second, hence the
1729 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001730 # FIXME: if the ping command errors then _ping_check_status()
1731 # returns instantly. If timeout is also smaller than twice
1732 # _PING_WAIT_COUNT then the while loop below forks many
1733 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1734 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1735 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001736 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1737 end_time = time.time() + timeout
1738 while time.time() <= end_time:
1739 if self._ping_check_status(status):
1740 return True
1741 if poll_interval > 0:
1742 time.sleep(poll_interval)
1743
1744 # The last thing we did was sleep(poll_interval), so it may
1745 # have been too long since the last `ping`. Check one more
1746 # time, just to be sure.
1747 return self._ping_check_status(status)
1748
1749 def ping_wait_up(self, timeout):
1750 """Wait for the host to respond to `ping`.
1751
1752 N.B. This method is not a reliable substitute for
1753 `wait_up()`, because a host that responds to ping will not
1754 necessarily respond to ssh. This method should only be used
1755 if the target DUT can be considered functional even if it
1756 can't be reached via ssh.
1757
1758 @param timeout Minimum time to allow before declaring the
1759 host to be non-responsive.
1760 @return True iff the host answered to ping before the timeout.
1761
1762 """
1763 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001764
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001765 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001766 """Wait until the host no longer responds to `ping`.
1767
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001768 This function can be used as a slightly faster version of
1769 `wait_down()`, by avoiding potentially long ssh timeouts.
1770
1771 @param timeout Minimum time to allow for the host to become
1772 non-responsive.
1773 @return True iff the host quit answering ping before the
1774 timeout.
1775
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001776 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001777 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001778
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001779 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001780 """Wait for the client to enter low-power sleep mode.
1781
1782 The test for "is asleep" can't distinguish a system that is
1783 powered off; to confirm that the unit was asleep, it is
1784 necessary to force resume, and then call
1785 `test_wait_for_resume()`.
1786
1787 This function is expected to be called from a test as part
1788 of a sequence like the following:
1789
1790 ~~~~~~~~
1791 boot_id = host.get_boot_id()
1792 # trigger sleep on the host
1793 host.test_wait_for_sleep()
1794 # trigger resume on the host
1795 host.test_wait_for_resume(boot_id)
1796 ~~~~~~~~
1797
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001798 @param sleep_timeout time limit in seconds to allow the host sleep.
1799
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001800 @exception TestFail The host did not go to sleep within
1801 the allowed time.
1802 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001803 if sleep_timeout is None:
1804 sleep_timeout = self.SLEEP_TIMEOUT
1805
1806 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001807 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001808 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001809
1810
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001811 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001812 """Wait for the client to resume from low-power sleep mode.
1813
1814 The `old_boot_id` parameter should be the value from
1815 `get_boot_id()` obtained prior to entering sleep mode. A
1816 `TestFail` exception is raised if the boot id changes.
1817
1818 See @ref test_wait_for_sleep for more on this function's
1819 usage.
1820
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001821 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001822 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001823 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001824
1825 @exception TestFail The host did not respond within the
1826 allowed time.
1827 @exception TestFail The host responded, but the boot id test
1828 indicated a reboot rather than a sleep
1829 cycle.
1830 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001831 if resume_timeout is None:
1832 resume_timeout = self.RESUME_TIMEOUT
1833
1834 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001835 raise error.TestFail(
1836 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001837 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001838 else:
1839 new_boot_id = self.get_boot_id()
1840 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001841 logging.error('client rebooted (old boot %s, new boot %s)',
1842 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001843 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001844 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001845
1846
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001847 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001848 """Wait for the client to shut down.
1849
1850 The test for "has shut down" can't distinguish a system that
1851 is merely asleep; to confirm that the unit was down, it is
1852 necessary to force boot, and then call test_wait_for_boot().
1853
1854 This function is expected to be called from a test as part
1855 of a sequence like the following:
1856
1857 ~~~~~~~~
1858 boot_id = host.get_boot_id()
1859 # trigger shutdown on the host
1860 host.test_wait_for_shutdown()
1861 # trigger boot on the host
1862 host.test_wait_for_boot(boot_id)
1863 ~~~~~~~~
1864
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001865 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001866 @exception TestFail The host did not shut down within the
1867 allowed time.
1868 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001869 if shutdown_timeout is None:
1870 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1871
1872 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001873 raise error.TestFail(
1874 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001875 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001876
1877
1878 def test_wait_for_boot(self, old_boot_id=None):
1879 """Wait for the client to boot from cold power.
1880
1881 The `old_boot_id` parameter should be the value from
1882 `get_boot_id()` obtained prior to shutting down. A
1883 `TestFail` exception is raised if the boot id does not
1884 change. The boot id test is omitted if `old_boot_id` is not
1885 specified.
1886
1887 See @ref test_wait_for_shutdown for more on this function's
1888 usage.
1889
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001890 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001891 shut down.
1892
1893 @exception TestFail The host did not respond within the
1894 allowed time.
1895 @exception TestFail The host responded, but the boot id test
1896 indicated that there was no reboot.
1897 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001898 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001899 raise error.TestFail(
1900 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001901 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001902 elif old_boot_id:
1903 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001904 logging.error('client not rebooted (boot %s)',
1905 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001906 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001907 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001908
1909
1910 @staticmethod
1911 def check_for_rpm_support(hostname):
1912 """For a given hostname, return whether or not it is powered by an RPM.
1913
Simran Basi1df55112013-09-06 11:25:09 -07001914 @param hostname: hostname to check for rpm support.
1915
Simran Basid5e5e272012-09-24 15:23:59 -07001916 @return None if this host does not follows the defined naming format
1917 for RPM powered DUT's in the lab. If it does follow the format,
1918 it returns a regular expression MatchObject instead.
1919 """
Fang Dengbaff9082015-01-06 13:46:15 -08001920 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001921
1922
1923 def has_power(self):
1924 """For this host, return whether or not it is powered by an RPM.
1925
1926 @return True if this host is in the CROS lab and follows the defined
1927 naming format.
1928 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001929 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001930
1931
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001932 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001933 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001934
1935 @param state Specifies which power state to set to DUT
1936 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001937 use. By default "RPM" or "CCD" will be used based
1938 on servo type. Valid values from
1939 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001940
1941 """
1942 ACCEPTABLE_STATES = ['ON', 'OFF']
1943
Garry Wang5e5538a2019-04-08 15:36:18 -07001944 if not power_method:
1945 power_method = self.get_default_power_method()
1946
1947 state = state.upper()
1948 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001949 raise error.TestError('State must be one of: %s.'
1950 % (ACCEPTABLE_STATES,))
1951
1952 if power_method == self.POWER_CONTROL_SERVO:
1953 logging.info('Setting servo port J10 to %s', state)
1954 self.servo.set('prtctl3_pwren', state.lower())
1955 time.sleep(self._USB_POWER_TIMEOUT)
1956 elif power_method == self.POWER_CONTROL_MANUAL:
1957 logging.info('You have %d seconds to set the AC power to %s.',
1958 self._POWER_CYCLE_TIMEOUT, state)
1959 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001960 elif power_method == self.POWER_CONTROL_CCD:
1961 servo_role = 'src' if state == 'ON' else 'snk'
1962 logging.info('servo ccd power pass through detected,'
1963 ' changing servo_role to %s.', servo_role)
1964 self.servo.set_servo_v4_role(servo_role)
1965 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001966 # Make sure we don't leave DUT with no power(servo_role=snk)
1967 # when DUT is not pingable, as we raise a exception here
1968 # that may break a power cycle in the middle.
1969 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001970 raise error.AutoservError(
1971 'DUT failed to regain network connection after %d seconds.'
1972 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001973 else:
1974 if not self.has_power():
1975 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001976 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07001977 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07001978
1979
Garry Wang5e5538a2019-04-08 15:36:18 -07001980 def power_off(self, power_method=None):
1981 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001982
1983 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001984 use. By default "RPM" or "CCD" will be used based
1985 on servo type. Valid values from
1986 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001987
1988 """
1989 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07001990
1991
Garry Wang5e5538a2019-04-08 15:36:18 -07001992 def power_on(self, power_method=None):
1993 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001994
1995 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001996 use. By default "RPM" or "CCD" will be used based
1997 on servo type. Valid values from
1998 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001999
2000 """
2001 self._set_power('ON', power_method)
2002
2003
Garry Wang5e5538a2019-04-08 15:36:18 -07002004 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002005 """Cycle power to this host by turning it OFF, then ON.
2006
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 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002013 if not power_method:
2014 power_method = self.get_default_power_method()
2015
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002016 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002017 self.POWER_CONTROL_MANUAL,
2018 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002019 self.power_off(power_method=power_method)
2020 time.sleep(self._POWER_CYCLE_TIMEOUT)
2021 self.power_on(power_method=power_method)
2022 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002023 self._add_rpm_changed_tag()
2024 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002025
2026
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002027 def get_platform_from_fwid(self):
2028 """Determine the platform from the crossystem fwid.
2029
2030 @returns a string representing this host's platform.
2031 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002032 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002033 crossystem = utils.Crossystem(self)
2034 crossystem.init()
2035 # Extract fwid value and use the leading part as the platform id.
2036 # fwid generally follow the format of {platform}.{firmware version}
2037 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2038 platform = crossystem.fwid().split('.')[0].lower()
2039 # Newer platforms start with 'Google_' while the older ones do not.
2040 return platform.replace('google_', '')
2041
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002042
Simran Basic6f1f7a2012-10-16 10:47:46 -07002043 def get_platform(self):
2044 """Determine the correct platform label for this host.
2045
2046 @returns a string representing this host's platform.
2047 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002048 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2049 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002050 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002051 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002052 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002053 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002054
2055
Greg Edelstona7b05d12020-04-01 16:00:51 -06002056 def get_model_from_cros_config(self):
2057 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002058
Greg Edelstona7b05d12020-04-01 16:00:51 -06002059 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002060 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002061 return cros_config.call_cros_config_get_output('/ name',
2062 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002063
2064
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002065 def get_architecture(self):
2066 """Determine the correct architecture label for this host.
2067
2068 @returns a string representing this host's architecture.
2069 """
2070 crossystem = utils.Crossystem(self)
2071 crossystem.init()
2072 return crossystem.arch()
2073
2074
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002075 def get_chrome_version(self):
2076 """Gets the Chrome version number and milestone as strings.
2077
2078 Invokes "chrome --version" to get the version number and milestone.
2079
2080 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2081 current Chrome version number as a string (in the form "W.X.Y.Z")
2082 and "milestone" is the first component of the version number
2083 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2084 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2085 of "chrome --version" and the milestone will be the empty string.
2086
2087 """
MK Ryu35d661e2014-09-25 17:44:10 -07002088 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002089 return utils.parse_chrome_version(version_string)
2090
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002091
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002092 def get_ec_version(self):
2093 """Get the ec version as strings.
2094
2095 @returns a string representing this host's ec version.
2096 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002097 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002098 result = self.run(command, ignore_status=True)
2099 if result.exit_status != 0:
2100 return ''
2101 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002102
2103
2104 def get_firmware_version(self):
2105 """Get the firmware version as strings.
2106
2107 @returns a string representing this host's firmware version.
2108 """
2109 crossystem = utils.Crossystem(self)
2110 crossystem.init()
2111 return crossystem.fwid()
2112
2113
2114 def get_hardware_revision(self):
2115 """Get the hardware revision as strings.
2116
2117 @returns a string representing this host's hardware revision.
2118 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002119 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002120 result = self.run(command, ignore_status=True)
2121 if result.exit_status != 0:
2122 return ''
2123 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002124
2125
2126 def get_kernel_version(self):
2127 """Get the kernel version as strings.
2128
2129 @returns a string representing this host's kernel version.
2130 """
2131 return self.run('uname -r').stdout.strip()
2132
2133
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002134 def get_cpu_name(self):
2135 """Get the cpu name as strings.
2136
2137 @returns a string representing this host's cpu name.
2138 """
2139
2140 # Try get cpu name from device tree first
2141 if self.path_exists('/proc/device-tree/compatible'):
2142 command = ' | '.join(
2143 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2144 'tail -1'])
2145 return self.run(command).stdout.strip().replace(',', ' ')
2146
2147 # Get cpu name from uname -p
2148 command = 'uname -p'
2149 ret = self.run(command).stdout.strip()
2150
2151 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2152 # Try get cpu name from /proc/cpuinfo instead
2153 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2154 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2155 self = self.run(command).stdout.strip()
2156
2157 # Remove bloat from CPU name, for example
2158 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2159 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2160 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2161 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2162 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2163 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2164
2165
2166 def get_screen_resolution(self):
2167 """Get the screen(s) resolution as strings.
2168 In case of more than 1 monitor, return resolution for each monitor
2169 separate with plus sign.
2170
2171 @returns a string representing this host's screen(s) resolution.
2172 """
2173 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2174 ret = self.run(command, ignore_status=True)
2175 # We might have Chromebox without a screen
2176 if ret.exit_status != 0:
2177 return ''
2178 return ret.stdout.strip().replace('\n', '+')
2179
2180
2181 def get_mem_total_gb(self):
2182 """Get total memory available in the system in GiB (2^20).
2183
2184 @returns an integer representing total memory
2185 """
2186 mem_total_kb = self.read_from_meminfo('MemTotal')
2187 kb_in_gb = float(2 ** 20)
2188 return int(round(mem_total_kb / kb_in_gb))
2189
2190
2191 def get_disk_size_gb(self):
2192 """Get size of disk in GB (10^9)
2193
2194 @returns an integer representing size of disk, 0 in Error Case
2195 """
2196 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2197 result = self.run(command, ignore_status=True)
2198 if result.exit_status != 0:
2199 return 0
2200 _, _, block, _ = re.split(r' +', result.stdout.strip())
2201 byte_per_block = 1024.0
2202 disk_kb_in_gb = 1e9
2203 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2204
2205
2206 def get_battery_size(self):
2207 """Get size of battery in Watt-hour via sysfs
2208
2209 This method assumes that battery support voltage_min_design and
2210 charge_full_design sysfs.
2211
2212 @returns a float representing Battery size, 0 if error.
2213 """
2214 # sysfs report data in micro scale
2215 battery_scale = 1e6
2216
2217 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2218 result = self.run(command, ignore_status=True)
2219 if result.exit_status != 0:
2220 return 0
2221 voltage = float(result.stdout.strip()) / battery_scale
2222
2223 command = 'cat /sys/class/power_supply/*/charge_full_design'
2224 result = self.run(command, ignore_status=True)
2225 if result.exit_status != 0:
2226 return 0
2227 amphereHour = float(result.stdout.strip()) / battery_scale
2228
2229 return voltage * amphereHour
2230
2231
2232 def get_low_battery_shutdown_percent(self):
2233 """Get the percent-based low-battery shutdown threshold.
2234
2235 @returns a float representing low-battery shutdown percent, 0 if error.
2236 """
2237 ret = 0.0
2238 try:
2239 command = 'check_powerd_config --low_battery_shutdown_percent'
2240 ret = float(self.run(command).stdout)
2241 except error.CmdError:
2242 logging.debug("Can't run %s", command)
2243 except ValueError:
2244 logging.debug("Didn't get number from %s", command)
2245
2246 return ret
2247
2248
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002249 def has_hammer(self):
2250 """Check whether DUT has hammer device or not.
2251
2252 @returns boolean whether device has hammer or not
2253 """
2254 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2255 return self.run(command, ignore_status=True).exit_status == 0
2256
2257
Niranjan Kumar34618872017-05-31 12:57:09 -07002258 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002259 """Returns True if the specified switch was provided to Chrome.
2260
2261 @param switch The chrome switch to search for.
2262 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002263
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002264 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2265 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002266
2267
2268 def oobe_triggers_update(self):
2269 """Returns True if this host has an OOBE flow during which
2270 it will perform an update check and perhaps an update.
2271 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2272 As more such flows are developed, code handling them needs
2273 to be added here.
2274
2275 @return Boolean indicating whether this host's OOBE triggers an update.
2276 """
2277 return self.is_chrome_switch_present(
2278 '--enterprise-enable-zero-touch-enrollment=hands-off')
2279
2280
Kevin Chenga2619dc2016-03-28 11:42:08 -07002281 # TODO(kevcheng): change this to just return the board without the
2282 # 'board:' prefix and fix up all the callers. Also look into removing the
2283 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002284 def get_board(self):
2285 """Determine the correct board label for this host.
2286
2287 @returns a string representing this host's board.
2288 """
2289 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2290 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002291 return (ds_constants.BOARD_PREFIX +
2292 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002293
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002294 def get_channel(self):
2295 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002296
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002297 @returns: a string represeting this host's build channel.
2298 (stable, dev, beta). None on fail.
2299 """
2300 return lsbrelease_utils.get_chromeos_channel(
2301 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002302
Kevin Chenga328da62016-03-31 10:49:04 -07002303 def get_power_supply(self):
2304 """
2305 Determine what type of power supply the host has
2306
2307 @returns a string representing this host's power supply.
2308 'power:battery' when the device has a battery intended for
2309 extended use
2310 'power:AC_primary' when the device has a battery not intended
2311 for extended use (for moving the machine, etc)
2312 'power:AC_only' when the device has no battery at all.
2313 """
2314 psu = self.run(command='mosys psu type', ignore_status=True)
2315 if psu.exit_status:
2316 # The psu command for mosys is not included for all platforms. The
2317 # assumption is that the device will have a battery if the command
2318 # is not found.
2319 return 'power:battery'
2320
2321 psu_str = psu.stdout.strip()
2322 if psu_str == 'unknown':
2323 return None
2324
2325 return 'power:%s' % psu_str
2326
2327
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002328 def has_battery(self):
2329 """Determine if DUT has a battery.
2330
2331 Returns:
2332 Boolean, False if known not to have battery, True otherwise.
2333 """
2334 rv = True
2335 power_supply = self.get_power_supply()
2336 if power_supply == 'power:battery':
2337 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2338 board_type = self.get_board_type()
2339 if board_type in _NO_BATTERY_BOARD_TYPE:
2340 logging.warn('Do NOT believe type %s has battery. '
2341 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002342 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002343 ignore_status=True)
2344 logging.debug(psu)
2345 rv = False
2346 elif power_supply == 'power:AC_only':
2347 rv = False
2348
2349 return rv
2350
2351
Kevin Chenga328da62016-03-31 10:49:04 -07002352 def get_servo(self):
2353 """Determine if the host has a servo attached.
2354
2355 If the host has a working servo attached, it should have a servo label.
2356
2357 @return: string 'servo' if the host has servo attached. Otherwise,
2358 returns None.
2359 """
2360 return 'servo' if self._servo_host else None
2361
2362
Kevin Chenga328da62016-03-31 10:49:04 -07002363 def has_internal_display(self):
2364 """Determine if the device under test is equipped with an internal
2365 display.
2366
2367 @return: 'internal_display' if one is present; None otherwise.
2368 """
2369 from autotest_lib.client.cros.graphics import graphics_utils
2370 from autotest_lib.client.common_lib import utils as common_utils
2371
2372 def __system_output(cmd):
2373 return self.run(cmd).stdout
2374
2375 def __read_file(remote_path):
2376 return self.run('cat %s' % remote_path).stdout
2377
2378 # Hijack the necessary client functions so that we can take advantage
2379 # of the client lib here.
2380 # FIXME: find a less hacky way than this
2381 original_system_output = utils.system_output
2382 original_read_file = common_utils.read_file
2383 utils.system_output = __system_output
2384 common_utils.read_file = __read_file
2385 try:
2386 return ('internal_display' if graphics_utils.has_internal_display()
2387 else None)
2388 finally:
2389 utils.system_output = original_system_output
2390 common_utils.read_file = original_read_file
2391
2392
Dan Shi85276d42014-04-08 22:11:45 -07002393 def is_boot_from_usb(self):
2394 """Check if DUT is boot from USB.
2395
2396 @return: True if DUT is boot from usb.
2397 """
2398 device = self.run('rootdev -s -d').stdout.strip()
2399 removable = int(self.run('cat /sys/block/%s/removable' %
2400 os.path.basename(device)).stdout.strip())
2401 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002402
2403
2404 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002405 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002406
2407 @param key: meminfo requested
2408
2409 @return the memory value as a string
2410
2411 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002412 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2413 logging.debug('%s', meminfo)
2414 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002415
2416
Rohit Makasana98e696f2016-06-03 18:48:10 -07002417 def get_cpu_arch(self):
2418 """Returns CPU arch of the device.
2419
2420 @return CPU architecture of the DUT.
2421 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002422 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002423 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2424 ignore_status=True).stdout:
2425 return 'x86_64'
2426 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2427 ignore_status=True).stdout:
2428 return 'arm'
2429 return 'i386'
2430
2431
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002432 def get_board_type(self):
2433 """
2434 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002435 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2436
2437 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002438 """
Danny Chan471a8d12015-08-18 14:57:41 -07002439 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2440 ignore_status=True).stdout
2441 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002442 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002443 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002444
2445
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002446 def get_arc_version(self):
2447 """Return ARC version installed on the DUT.
2448
2449 @returns ARC version as string if the CrOS build has ARC, else None.
2450 """
2451 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2452 ignore_status=True).stdout
2453 if arc_version:
2454 return arc_version.split('=')[-1].strip()
2455 return None
2456
2457
Gilad Arnolda76bef02015-09-29 13:55:15 -07002458 def get_os_type(self):
2459 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002460
2461
Kevin Chenga2619dc2016-03-28 11:42:08 -07002462 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002463 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002464 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002465
2466
2467 def get_default_power_method(self):
2468 """
2469 Get the default power method for power_on/off/cycle() methods.
2470 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2471 """
2472 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002473 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002474 if self.servo and self.servo.supports_built_in_pd_control():
2475 self._default_power_method = self.POWER_CONTROL_CCD
2476 else:
2477 logging.debug('Either servo is unitialized or the servo '
2478 'setup does not support pd controls. Falling '
2479 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002480 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002481
2482
2483 def find_usb_devices(self, idVendor, idProduct):
2484 """
2485 Get usb device sysfs name for specific device.
2486
2487 @param idVendor Vendor ID to search in sysfs directory.
2488 @param idProduct Product ID to search in sysfs directory.
2489
2490 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2491 """
2492 # Look for matching file and cut at position 7 to get dir name.
2493 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2494
2495 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2496 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2497
2498 # Use uniq -d to print duplicate line from both command
2499 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2500
2501 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2502
2503
2504 def bind_usb_device(self, usb_node):
2505 """
2506 Bind usb device
2507
2508 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2509 """
2510 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2511 self.run(cmd, ignore_status=True)
2512
2513
2514 def unbind_usb_device(self, usb_node):
2515 """
2516 Unbind usb device
2517
2518 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2519 """
2520 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2521 self.run(cmd, ignore_status=True)
2522
2523
2524 def get_wlan_ip(self):
2525 """
2526 Get ip address of wlan interface.
2527
2528 @return ip address of wlan or empty string if wlan is not connected.
2529 """
2530 cmds = [
2531 'iw dev', # List wlan physical device
2532 'grep Interface', # Grep only interface name
2533 'cut -f 2 -d" "', # Cut the name part
2534 'xargs ifconfig', # Feed it to ifconfig to get ip
2535 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2536 'cut -f 2 -d " "' # Cut the ip part
2537 ]
2538 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002539
2540 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2541 """
2542 Connect to wifi network
2543
2544 @param ssid SSID of the wifi network.
2545 @param passphrase Passphrase of the wifi network. None if not existed.
2546 @param security Security of the wifi network. Default to "psk" if
2547 passphase is given without security. Possible values
2548 are "none", "psk", "802_1x".
2549
2550 @return True if succeed, False if not.
2551 """
2552 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2553 if passphrase:
2554 cmd += ' ' + passphrase
2555 if security:
2556 cmd += ' ' + security
2557 return self.run(cmd, ignore_status=True).exit_status == 0