blob: cabca07cb081d0b6b169a8b10afb8c62dbf4c68b [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
Richard Barnette03a0c132012-11-05 12:40:35 -080018from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070019from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000020from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080021from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080022from autotest_lib.server import afe_utils
Anh Le3ad780d2019-05-28 10:54:51 -070023from autotest_lib.server import crashcollect
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
Simran Basidcff4252012-11-20 16:13:20 -080036from autotest_lib.site_utils.rpm_control_system import rpm_client
Simran Basid5e5e272012-09-24 15:23:59 -070037
Simran Basi382506b2016-09-13 14:58:15 -070038# In case cros_host is being ran via SSP on an older Moblab version with an
39# older chromite version.
40try:
41 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080042except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080043 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080044
Simran Basid5e5e272012-09-24 15:23:59 -070045
Dan Shib8540a52015-07-16 14:18:23 -070046CONFIG = global_config.global_config
47
Dan Shid07ee2e2015-09-24 14:49:25 -070048
beepsc87ff602013-07-31 21:53:00 -070049class FactoryImageCheckerException(error.AutoservError):
50 """Exception raised when an image is a factory image."""
51 pass
52
53
Fang Deng0ca40e22013-08-27 17:47:44 -070054class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070055 """Chromium OS specific subclass of Host."""
56
Simran Basi5ace6f22016-01-06 17:30:44 -080057 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
58
Scott Zawalski62bacae2013-03-05 10:40:32 -050059 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070060
Richard Barnette03a0c132012-11-05 12:40:35 -080061 # Timeout values (in seconds) associated with various Chrome OS
62 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070063 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080064 # In general, a good rule of thumb is that the timeout can be up
65 # to twice the typical measured value on the slowest platform.
66 # The times here have not necessarily been empirically tested to
67 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070068 #
69 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080070 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
71 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080072 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070073 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080074 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070075 # screen delay, time to start the network on the DUT, and the
76 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070077 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080078 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080079 # network.
beepsf079cfb2013-09-18 17:49:51 -070080 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080081 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
82 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070083
84 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080085 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080086 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070087 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070088 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070089 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080090 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070091
Dan Shica503482015-03-30 17:23:25 -070092 # Minimum OS version that supports server side packaging. Older builds may
93 # not have server side package built or with Autotest code change to support
94 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070095 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -070096 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -070097
J. Richard Barnette84890bd2014-02-21 11:05:47 -080098 # REBOOT_TIMEOUT: How long to wait for a reboot.
99 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700100 # We have a long timeout to ensure we don't flakily fail due to other
101 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700102 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
103 # return from reboot' bug is solved.
104 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700105
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800106 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
107 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700108 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
109 # since changing servo role will reset USB state
110 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800111 _USB_POWER_TIMEOUT = 5
112 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700113 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800114
Fang Dengdeba14f2014-11-14 11:54:09 -0800115 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
116 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700117
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800118 # Constants used in ping_wait_up() and ping_wait_down().
119 #
120 # _PING_WAIT_COUNT is the approximate number of polling
121 # cycles to use when waiting for a host state change.
122 #
123 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
124 # for arguments to the internal _ping_wait_for_status()
125 # method.
126 _PING_WAIT_COUNT = 40
127 _PING_STATUS_DOWN = False
128 _PING_STATUS_UP = True
129
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800130 # Allowed values for the power_method argument.
131
Garry Wang5e5538a2019-04-08 15:36:18 -0700132 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
133 # DUTs except those with servo_v4 CCD.
134 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
135 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800136 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
137 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
138 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700139 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800140 POWER_CONTROL_SERVO = 'servoj10'
141 POWER_CONTROL_MANUAL = 'manual'
142
143 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700144 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800145 POWER_CONTROL_SERVO,
146 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800147
Garry Wang5e5538a2019-04-08 15:36:18 -0700148 # CCD_SERVO: The string of servo type to compare with the return value of
149 # self.servo.get_servo_version() to decide default power method.
150 CCD_SERVO = 'servo_v4_with_ccd_cr50'
151
Simran Basi5e6339a2013-03-21 11:34:32 -0700152 _RPM_OUTLET_CHANGED = 'outlet_changed'
153
Dan Shi9cb0eec2014-06-03 09:04:50 -0700154 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700155 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700156 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700157
Anh Le3ad780d2019-05-28 10:54:51 -0700158 # DUT_LOG_LOCATION: the directory in the DUT that the log is saved
159 # after re-imaging using chromeos-install.
160 # The location is specified in chromeos-install script.
161 DUT_LOG_LOCATION = '/mnt/stateful_partition/unencrypted/prior_logs'
J. Richard Barnette91137f02016-03-10 16:52:26 -0800162
J. Richard Barnette964fba02012-10-24 17:34:29 -0700163 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800164 def check_host(host, timeout=10):
165 """
166 Check if the given host is a chrome-os host.
167
168 @param host: An ssh host representing a device.
169 @param timeout: The timeout for the run command.
170
171 @return: True if the host device is chromeos.
172
beeps46dadc92013-11-07 14:07:10 -0800173 """
174 try:
Allen Liad719c12017-06-27 23:48:04 +0000175 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700176 'grep -q CHROMEOS /etc/lsb-release && '
177 '! test -f /mnt/stateful_partition/.android_tester && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700178 '! grep -q moblab /etc/lsb-release && '
179 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700180 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700181 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000182 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700183 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
184 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800185 return not (
186 lsbrelease_utils.is_jetstream(
187 lsb_release_content=lsb_release_content) or
188 lsbrelease_utils.is_gce_board(
189 lsb_release_content=lsb_release_content))
190
beeps46dadc92013-11-07 14:07:10 -0800191 except (error.AutoservRunError, error.AutoservSSHTimeout):
192 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700193
194 return False
beeps46dadc92013-11-07 14:07:10 -0800195
196
197 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800198 def get_chameleon_arguments(args_dict):
199 """Extract chameleon options from `args_dict` and return the result.
200
201 Recommended usage:
202 ~~~~~~~~
203 args_dict = utils.args_to_dict(args)
204 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
205 host = hosts.create_host(machine, chameleon_args=chameleon_args)
206 ~~~~~~~~
207
208 @param args_dict Dictionary from which to extract the chameleon
209 arguments.
210 """
Allen Li083866b2016-08-18 10:07:10 -0700211 return {key: args_dict[key]
212 for key in ('chameleon_host', 'chameleon_port')
213 if key in args_dict}
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800214
215
216 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700217 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800218 """Extract chameleon options from `args_dict` and return the result.
219
220 Recommended usage:
221 ~~~~~~~~
222 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700223 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
224 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800225 ~~~~~~~~
226
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700227 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800228 arguments.
229 """
Allen Li083866b2016-08-18 10:07:10 -0700230 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700231 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700232 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800233
234
235 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800236 def get_servo_arguments(args_dict):
237 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800238
239 Recommended usage:
240 ~~~~~~~~
241 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700242 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800243 host = hosts.create_host(machine, servo_args=servo_args)
244 ~~~~~~~~
245
246 @param args_dict Dictionary from which to extract the servo
247 arguments.
248 """
Richard Barnettee519dcd2016-08-15 17:37:17 -0700249 servo_attrs = (servo_host.SERVO_HOST_ATTR,
250 servo_host.SERVO_PORT_ATTR,
Nick Sanders2f3c9852018-10-24 12:10:24 -0700251 servo_host.SERVO_BOARD_ATTR,
252 servo_host.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200253 servo_args = {key: args_dict[key]
254 for key in servo_attrs
255 if key in args_dict}
256 return (
257 None
258 if servo_host.SERVO_HOST_ATTR in servo_args
259 and not servo_args[servo_host.SERVO_HOST_ATTR]
260 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700261
J. Richard Barnette964fba02012-10-24 17:34:29 -0700262
J. Richard Barnette91137f02016-03-10 16:52:26 -0800263 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700264 pdtester_args=None, try_lab_servo=False,
Richard Barnette9a26ad62016-06-10 12:03:08 -0700265 try_servo_repair=False,
J. Richard Barnette91137f02016-03-10 16:52:26 -0800266 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700267 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800268 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700269
Fang Denge545abb2014-12-30 18:43:47 -0800270 This method will attempt to create the test-assistant object
271 (chameleon/servo) when it is needed by the test. Check
272 the docstring of chameleon_host.create_chameleon_host and
273 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700274
Fang Denge545abb2014-12-30 18:43:47 -0800275 @param hostname: Hostname of the dut.
276 @param chameleon_args: A dictionary that contains args for creating
277 a ChameleonHost. See chameleon_host for details.
278 @param servo_args: A dictionary that contains args for creating
279 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700280 @param try_lab_servo: When true, indicates that an attempt should
281 be made to create a ServoHost for a DUT in
282 the test lab, even if not required by
283 `servo_args`. See servo_host for details.
284 @param try_servo_repair: If a servo host is created, check it
285 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800286 See servo_host for details.
287 @param ssh_verbosity_flag: String, to pass to the ssh command to control
288 verbosity.
289 @param ssh_options: String, other ssh options to pass to the ssh
290 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700291 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700292 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700293 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800294 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Kevin Chenga2619dc2016-03-28 11:42:08 -0700295 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700296 # self.env is a dictionary of environment variable settings
297 # to be exported for commands run on the host.
298 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
299 # errors that might happen.
300 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700301 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700302 self._ssh_options = ssh_options
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700303 self.set_servo_host(
304 servo_host.create_servo_host(
Richard Barnetteea3e4602016-06-10 12:36:41 -0700305 dut=self, servo_args=servo_args,
Richard Barnette9a26ad62016-06-10 12:03:08 -0700306 try_lab_servo=try_lab_servo,
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700307 try_servo_repair=try_servo_repair))
Garry Wang5e5538a2019-04-08 15:36:18 -0700308 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700309
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800310 # TODO(waihong): Do the simplication on Chameleon too.
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800311 self._chameleon_host = chameleon_host.create_chameleon_host(
312 dut=self.hostname, chameleon_args=chameleon_args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700313 # Add pdtester host if pdtester args were added on command line
314 self._pdtester_host = pdtester_host.create_pdtester_host(pdtester_args)
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800315
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800316 if self._chameleon_host:
Tom Wai-Hong Tameaee3402014-01-22 08:52:10 +0800317 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800318 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800319 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700320
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700321 if self._pdtester_host:
322 self.pdtester_servo = self._pdtester_host.get_servo()
323 logging.info('pdtester_servo: %r', self.pdtester_servo)
324 # Create the pdtester object used to access the ec uart
325 self.pdtester = pdtester.PDTester(self.pdtester_servo,
326 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800327 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700328 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800329
Fang Deng5d518f42013-08-02 14:04:32 -0700330
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000331 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700332 """Get latest stable cros image name from AFE.
333
334 Use the board name from the info store. Should that fail, try to
335 retrieve the board name from the host's installed image itself.
336
337 @returns: current stable cros image name for this host.
338 """
339 board = self.host_info_store.get().board
340 if not board:
341 logging.warn('No board label value found. Trying to infer '
342 'from the host itself.')
343 try:
344 board = self.get_board().split(':')[1]
345 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
346 logging.error('Also failed to get the board name from the DUT '
347 'itself. %s.', str(e))
348 raise error.AutoservError('Cannot obtain repair image name.')
349 return afe_utils.get_stable_cros_image_name(board)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500350
351
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700352 def host_version_prefix(self, image):
353 """Return version label prefix.
354
355 In case the CrOS provisioning version is something other than the
356 standard CrOS version e.g. CrOS TH version, this function will
357 find the prefix from provision.py.
358
359 @param image: The image name to find its version prefix.
360 @returns: A prefix string for the image type.
361 """
362 return provision.get_version_label_prefix(image)
363
364
beepsdae65fd2013-07-26 16:24:41 -0700365 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700366 """
367 Make sure job_repo_url of this host is valid.
368
joychen03eaad92013-06-26 09:55:21 -0700369 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700370 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
371 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
372 download and extract it. If the devserver embedded in the url is
373 unresponsive, update the job_repo_url of the host after staging it on
374 another devserver.
375
376 @param job_repo_url: A url pointing to the devserver where the autotest
377 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700378 @param tag: The tag from the server job, in the format
379 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700380
381 @raises DevServerException: If we could not resolve a devserver.
382 @raises AutoservError: If we're unable to save the new job_repo_url as
383 a result of choosing a new devserver because the old one failed to
384 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700385 @raises urllib2.URLError: If the devserver embedded in job_repo_url
386 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700387 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800388 info = self.host_info_store.get()
389 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700390 if not job_repo_url:
391 logging.warning('No job repo url set on host %s', self.hostname)
392 return
393
394 logging.info('Verifying job repo url %s', job_repo_url)
395 devserver_url, image_name = tools.get_devserver_build_from_package_url(
396 job_repo_url)
397
beeps0c865032013-07-30 11:37:06 -0700398 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700399
400 logging.info('Staging autotest artifacts for %s on devserver %s',
401 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700402
403 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700404 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700405 stage_time = time.time() - start_time
406
407 # Record how much of the verification time comes from a devserver
408 # restage. If we're doing things right we should not see multiple
409 # devservers for a given board/build/branch path.
410 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800411 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700412 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800413 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700414 pass
415 else:
beeps0c865032013-07-30 11:37:06 -0700416 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700417 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700418 stats_key = {
419 'board': board,
420 'build_type': build_type,
421 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700422 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700423 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800424
425 monarch_fields = {
426 'board': board,
427 'build_type': build_type,
428 # TODO(akeshet): To be consistent with most other metrics,
429 # consider changing the following field to be named
430 # 'milestone'.
431 'branch': branch,
432 'dev_server': devserver,
433 }
434 metrics.Counter(
435 'chromeos/autotest/provision/verify_url'
436 ).increment(fields=monarch_fields)
437 metrics.SecondsDistribution(
438 'chromeos/autotest/provision/verify_url_duration'
439 ).add(stage_time, fields=monarch_fields)
440
441
Dan Shicf4d2032015-03-12 15:04:21 -0700442 def stage_server_side_package(self, image=None):
443 """Stage autotest server-side package on devserver.
444
445 @param image: Full path of an OS image to install or a build name.
446
447 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700448
449 @raise: error.AutoservError if fail to locate the build to test with, or
450 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700451 """
Dan Shid37736b2016-07-06 15:10:29 -0700452 # If enable_drone_in_restricted_subnet is False, do not set hostname
453 # in devserver.resolve call, so a devserver in non-restricted subnet
454 # is picked to stage autotest server package for drone to download.
455 hostname = self.hostname
456 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
457 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700458 if image:
459 image_name = tools.get_build_from_image(image)
460 if not image_name:
461 raise error.AutoservError(
462 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700463 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700464 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700465 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800466 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700467 if job_repo_url:
468 devserver_url, image_name = (
469 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700470 # If enable_drone_in_restricted_subnet is True, use the
471 # existing devserver. Otherwise, resolve a new one in
472 # non-restricted subnet.
473 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
474 ds = dev_server.ImageServer(devserver_url)
475 else:
476 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800477 elif info.build is not None:
478 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700479 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700480 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800481 raise error.AutoservError(
482 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700483 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700484
485 # Get the OS version of the build, for any build older than
486 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
487 match = re.match('.*/R\d+-(\d+)\.', image_name)
488 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700489 raise error.AutoservError(
490 'Build %s is older than %s. Server side packaging is '
491 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700492
Dan Shicf4d2032015-03-12 15:04:21 -0700493 ds.stage_artifacts(image_name, ['autotest_server_package'])
494 return '%s/static/%s/%s' % (ds.url(), image_name,
495 'autotest_server_package.tar.bz2')
496
497
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700498 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700499 """Stage a build on a devserver and return the update_url.
500
501 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700502 @param artifact: a string like 'test_image'. Requests
503 appropriate image to be staged.
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700504 @returns an update URL like:
505 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0
506 """
507 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000508 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700509 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800510 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700511 devserver.stage_artifacts(image_name, [artifact])
512 if artifact == 'test_image':
513 return devserver.get_test_image_url(image_name)
514 elif artifact == 'recovery_image':
515 return devserver.get_recovery_image_url(image_name)
516 else:
517 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700518
519
beepse539be02013-07-31 21:57:39 -0700520 def stage_factory_image_for_servo(self, image_name):
521 """Stage a build on a devserver and return the update_url.
522
523 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700524
beepse539be02013-07-31 21:57:39 -0700525 @return: An update URL, eg:
526 http://<devserver>/static/canary-channel/\
527 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700528
529 @raises: ValueError if the factory artifact name is missing from
530 the config.
531
beepse539be02013-07-31 21:57:39 -0700532 """
533 if not image_name:
534 logging.error('Need an image_name to stage a factory image.')
535 return
536
Dan Shib8540a52015-07-16 14:18:23 -0700537 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700538 'CROS', 'factory_artifact', type=str, default='')
539 if not factory_artifact:
540 raise ValueError('Cannot retrieve the factory artifact name from '
541 'autotest config, and hence cannot stage factory '
542 'artifacts.')
543
beepse539be02013-07-31 21:57:39 -0700544 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800545 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700546 devserver.stage_artifacts(
547 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700548 [factory_artifact],
549 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700550
551 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
552
553
Laurence Goodby778c9a42017-05-24 19:24:07 -0700554 def prepare_for_update(self):
555 """Prepares the DUT for an update.
556
557 Subclasses may override this to perform any special actions
558 required before updating.
559 """
Laurence Goodby468de252017-06-08 17:22:53 -0700560 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700561
562
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800563 def _clear_fw_version_labels(self, rw_only):
564 """Clear firmware version labels from the machine.
565
566 @param rw_only: True to only clear fwrw_version; otherewise, clear
567 both fwro_version and fwrw_version.
568 """
Dan Shi9cb0eec2014-06-03 09:04:50 -0700569 labels = self._AFE.get_labels(
Dan Shi0723bf52015-06-24 10:52:38 -0700570 name__startswith=provision.FW_RW_VERSION_PREFIX,
Dan Shi9cb0eec2014-06-03 09:04:50 -0700571 host__hostname=self.hostname)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800572 if not rw_only:
573 labels = labels + self._AFE.get_labels(
574 name__startswith=provision.FW_RO_VERSION_PREFIX,
575 host__hostname=self.hostname)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700576 for label in labels:
577 label.remove_hosts(hosts=[self.hostname])
578
579
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800580 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700581 """Add firmware version label to the machine.
582
583 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800584 @param rw_only: True to only add fwrw_version; otherwise, add both
585 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700586
587 """
Prathmesh Prabhu2c7471d2016-11-15 20:19:57 +0000588 fw_label = provision.fwrw_version_to_label(build)
MK Ryu73be9862015-07-06 12:25:00 -0700589 self._AFE.run('label_add_hosts', id=fw_label, hosts=[self.hostname])
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800590 if not rw_only:
Prathmesh Prabhu2c7471d2016-11-15 20:19:57 +0000591 fw_label = provision.fwro_version_to_label(build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800592 self._AFE.run('label_add_hosts', id=fw_label, hosts=[self.hostname])
Dan Shi9cb0eec2014-06-03 09:04:50 -0700593
594
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800595 def firmware_install(self, build=None, rw_only=False):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700596 """Install firmware to the DUT.
597
598 Use stateful update if the DUT is already running the same build.
599 Stateful update does not update kernel and tends to run much faster
600 than a full reimage. If the DUT is running a different build, or it
601 failed to do a stateful update, full update, including kernel update,
602 will be applied to the DUT.
603
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800604 Once a host enters firmware_install its fw[ro|rw]_version label will
605 be removed. After the firmware is updated successfully, a new
606 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700607
608 @param build: The build version to which we want to provision the
609 firmware of the machine,
610 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800611 @param rw_only: True to only install firmware to its RW portions. Keep
612 the RO portions unchanged.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700613
614 TODO(dshi): After bug 381718 is fixed, update here with corresponding
615 exceptions that could be raised.
616
617 """
618 if not self.servo:
619 raise error.TestError('Host %s does not have servo.' %
620 self.hostname)
621
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700622 # Get the DUT board name from AFE.
623 info = self.host_info_store.get()
624 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700625 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800626
627 if board is None or board == '':
628 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700629
Chris Sosae92399e2015-04-24 11:32:59 -0700630 # If build is not set, try to install firmware from stable CrOS.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700631 if not build:
Richard Barnette260cbd02016-10-06 12:23:28 -0700632 build = afe_utils.get_stable_faft_version(board)
Dan Shi3d7a0e12015-10-12 11:55:45 -0700633 if not build:
634 raise error.TestError(
635 'Failed to find stable firmware build for %s.',
636 self.hostname)
637 logging.info('Will install firmware from build %s.', build)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700638
Dan Shi216389c2015-12-22 11:03:06 -0800639 ds = dev_server.ImageServer.resolve(build, self.hostname)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700640 ds.stage_artifacts(build, ['firmware'])
641
642 tmpd = autotemp.tempdir(unique_id='fwimage')
643 try:
644 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
645 local_tarball = os.path.join(tmpd.name, os.path.basename(fwurl))
xixuan4e116822016-11-17 15:32:10 -0800646 ds.download_file(fwurl, local_tarball)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700647
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800648 self._clear_fw_version_labels(rw_only)
Shelley Chenac61d5a2019-06-24 15:35:46 -0700649 self.servo.program_firmware(board, model, local_tarball, rw_only)
Wai-Hong Tamb5f66ce2016-11-10 15:45:30 -0800650 if utils.host_is_in_lab_zone(self.hostname):
651 self._add_fw_version_label(build, rw_only)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700652 finally:
653 tmpd.clean()
654
655
beepsf079cfb2013-09-18 17:49:51 -0700656 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
657 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500658 """
659 Re-install the OS on the DUT by:
660 1) installing a test image on a USB storage device attached to the Servo
661 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800662 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700663 3) installing the image with chromeos-install.
664
Scott Zawalski62bacae2013-03-05 10:40:32 -0500665 @param image_url: If specified use as the url to install on the DUT.
666 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700667 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
668 Factory images need a longer usb_boot_timeout than regular
669 cros images.
670 @param install_timeout: The timeout to use when installing the chromeos
671 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800672
Scott Zawalski62bacae2013-03-05 10:40:32 -0500673 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700674
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800675 """
beepsf079cfb2013-09-18 17:49:51 -0700676 logging.info('Downloading image to USB, then booting from it. Usb boot '
677 'timeout = %s', usb_boot_timeout)
Allen Li48a13fe2016-11-22 14:10:40 -0800678 with metrics.SecondsTimer(
679 'chromeos/autotest/provision/servo_install/boot_duration'):
680 self.servo.install_recovery_image(image_url)
681 if not self.wait_up(timeout=usb_boot_timeout):
682 raise hosts.AutoservRepairError(
683 'DUT failed to boot from USB after %d seconds' %
Garry Wang954f8382019-01-23 13:49:29 -0800684 usb_boot_timeout, 'failed_to_reboot')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500685
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800686 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
687 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800688 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800689 try:
690 self.run('chromeos-tpm-recovery')
691 except error.AutoservRunError:
692 logging.warn('chromeos-tpm-recovery is too old.')
693
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800694
Allen Li48a13fe2016-11-22 14:10:40 -0800695 with metrics.SecondsTimer(
696 'chromeos/autotest/provision/servo_install/install_duration'):
697 logging.info('Installing image through chromeos-install.')
Anh Lee21e4032019-07-11 15:01:06 -0700698 try:
699 # Re-imaging the DUT with log collecting.
700 self.run(
701 'chromeos-install --yes '
702 '--lab_preserve_logs='
703 '"/usr/local/autotest/common_lib/logs_to_collect"',
704 timeout=install_timeout)
705 except Exception as e:
706 logging.exception(
707 'Fail to collect log from DUT.'
708 'Retry to fix DUT without collecting log.')
709 self.run(
710 'chromeos-install --yes',
711 timeout=install_timeout)
712
Allen Li48a13fe2016-11-22 14:10:40 -0800713 self.halt()
beepsf079cfb2013-09-18 17:49:51 -0700714
715 logging.info('Power cycling DUT through servo.')
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800716 self.servo.get_power_state_controller().power_off()
Fang Dengafb88142013-05-30 17:44:31 -0700717 self.servo.switch_usbkey('off')
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800718 # N.B. The Servo API requires that we use power_on() here
719 # for two reasons:
720 # 1) After turning on a DUT in recovery mode, you must turn
721 # it off and then on with power_on() once more to
722 # disable recovery mode (this is a Parrot specific
723 # requirement).
724 # 2) After power_off(), the only way to turn on is with
725 # power_on() (this is a Storm specific requirement).
J. Richard Barnettefbcc7122013-07-24 18:24:59 -0700726 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -0700727
728 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -0800729 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
730 raise error.AutoservError('DUT failed to reboot installed '
731 'test image after %d seconds' %
Scott Zawalski62bacae2013-03-05 10:40:32 -0500732 self.BOOT_TIMEOUT)
733
Anh Le3ad780d2019-05-28 10:54:51 -0700734 # The log saved after re-imaging process is transferred to shard
735 # result directory when the job instance exists.
736 # When we run repair manually, the result directory is created
737 # within local host.
738 try:
739 local_dir = crashcollect.get_crashinfo_dir(
740 self,
Anh Lee21e4032019-07-11 15:01:06 -0700741 'prior_log'
Anh Le3ad780d2019-05-28 10:54:51 -0700742 )
743
744 self.collect_logs(self.DUT_LOG_LOCATION, local_dir)
745 except OSError:
746 logging.exception('Fail to collect log. '
747 'The destination log directory does not exist')
748
Scott Zawalski62bacae2013-03-05 10:40:32 -0500749
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700750 def set_servo_host(self, host):
751 """Set our servo host member, and associated servo.
752
753 @param host Our new `ServoHost`.
754 """
755 self._servo_host = host
756 if self._servo_host is not None:
757 self.servo = self._servo_host.get_servo()
758 else:
759 self.servo = None
760
761
Richard Barnette9a26ad62016-06-10 12:03:08 -0700762 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -0700763 """
Richard Barnette9a26ad62016-06-10 12:03:08 -0700764 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -0700765
Richard Barnette9a26ad62016-06-10 12:03:08 -0700766 If the servo object is missing, attempt to repair the servo
767 host. Repair failures are passed back to the caller.
768
769 @raise AutoservError: If there is no servo host for this CrOS
770 host.
771 """
772 if self.servo:
773 return
774 if not self._servo_host:
775 raise error.AutoservError('No servo host for %s.' %
776 self.hostname)
777 self._servo_host.repair()
778 self.servo = self._servo_host.get_servo()
Dan Shi90466352015-09-22 15:01:05 -0700779
780
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -0800781 def repair(self):
782 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -0800783
784 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -0800785 not call back to the parent class, but instead relies on
786 `self._repair_strategy` to coordinate the verification and
787 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -0800788 """
Richard Barnetteabbdc252018-07-26 16:57:42 -0700789 message = 'Beginning repair for host %s board %s model %s'
790 info = self.host_info_store.get()
791 message %= (self.hostname, info.board, info.model)
792 self.record('INFO', None, None, message)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800793 self._repair_strategy.repair(self)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500794
Richard Barnette82c35912012-11-20 10:09:10 -0800795
J. Richard Barnette1d78b012012-05-15 13:56:30 -0700796 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -0800797 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -0700798 super(CrosHost, self).close()
xixuand6011f12016-12-08 15:01:58 -0800799 if self._chameleon_host:
800 self._chameleon_host.close()
801
802 if self._servo_host:
803 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -0700804
805
Dan Shi49ca0932014-11-14 11:22:27 -0800806 def get_power_supply_info(self):
807 """Get the output of power_supply_info.
808
809 power_supply_info outputs the info of each power supply, e.g.,
810 Device: Line Power
811 online: no
812 type: Mains
813 voltage (V): 0
814 current (A): 0
815 Device: Battery
816 state: Discharging
817 percentage: 95.9276
818 technology: Li-ion
819
820 Above output shows two devices, Line Power and Battery, with details of
821 each device listed. This function parses the output into a dictionary,
822 with key being the device name, and value being a dictionary of details
823 of the device info.
824
825 @return: The dictionary of power_supply_info, e.g.,
826 {'Line Power': {'online': 'yes', 'type': 'main'},
827 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -0800828 @raise error.AutoservRunError if power_supply_info tool is not found in
829 the DUT. Caller should handle this error to avoid false failure
830 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -0800831 """
832 result = self.run('power_supply_info').stdout.strip()
833 info = {}
834 device_name = None
835 device_info = {}
836 for line in result.split('\n'):
837 pair = [v.strip() for v in line.split(':')]
838 if len(pair) != 2:
839 continue
840 if pair[0] == 'Device':
841 if device_name:
842 info[device_name] = device_info
843 device_name = pair[1]
844 device_info = {}
845 else:
846 device_info[pair[0]] = pair[1]
847 if device_name and not device_name in info:
848 info[device_name] = device_info
849 return info
850
851
852 def get_battery_percentage(self):
853 """Get the battery percentage.
854
855 @return: The percentage of battery level, value range from 0-100. Return
856 None if the battery info cannot be retrieved.
857 """
858 try:
859 info = self.get_power_supply_info()
860 logging.info(info)
861 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -0800862 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -0800863 return None
864
865
866 def is_ac_connected(self):
867 """Check if the dut has power adapter connected and charging.
868
869 @return: True if power adapter is connected and charging.
870 """
871 try:
872 info = self.get_power_supply_info()
873 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -0800874 except (KeyError, error.AutoservRunError):
875 return None
Dan Shi49ca0932014-11-14 11:22:27 -0800876
877
Simran Basi5e6339a2013-03-21 11:34:32 -0700878 def _cleanup_poweron(self):
879 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -0800880 info = self.host_info_store.get()
881 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -0700882 return
883 logging.debug('This host has recently interacted with the RPM'
884 ' Infrastructure. Ensuring power is on.')
885 try:
886 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -0800887 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -0700888 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -0700889 logging.error('Failed to turn Power On for this host after '
890 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -0800891
892 battery_percentage = self.get_battery_percentage()
Dan Shif01ebe22014-12-05 13:10:57 -0800893 if battery_percentage and battery_percentage < 50:
Dan Shi49ca0932014-11-14 11:22:27 -0800894 raise
895 elif self.is_ac_connected():
896 logging.info('The device has power adapter connected and '
897 'charging. No need to try to turn RPM on '
898 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -0800899 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -0800900 logging.info('Battery level is now at %s%%. The device may '
901 'still have enough power to run test, so no '
902 'exception will be raised.', battery_percentage)
903
Simran Basi5e6339a2013-03-21 11:34:32 -0700904
Garry Wangad4d4fd2019-01-30 17:00:38 -0800905 def _remove_rpm_changed_tag(self):
906 info = self.host_info_store.get()
907 del info.attributes[self._RPM_OUTLET_CHANGED]
908 self.host_info_store.commit(info)
909
910
911 def _add_rpm_changed_tag(self):
912 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -0800913 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -0800914 self.host_info_store.commit(info)
915
916
917
beepsc87ff602013-07-31 21:53:00 -0700918 def _is_factory_image(self):
919 """Checks if the image on the DUT is a factory image.
920
921 @return: True if the image on the DUT is a factory image.
922 False otherwise.
923 """
924 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
925 return result.exit_status == 0
926
927
928 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800929 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -0700930
931 @raises: FactoryImageCheckerException for factory images, since
932 we cannot attempt to restart ui on them.
933 error.AutoservRunError for any other type of error that
934 occurs while restarting ui.
935 """
936 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -0700937 raise FactoryImageCheckerException('Cannot restart ui on factory '
938 'images')
beepsc87ff602013-07-31 21:53:00 -0700939
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800940 # TODO(jrbarnette): The command to stop/start the ui job
941 # should live inside cros_ui, too. However that would seem
942 # to imply interface changes to the existing start()/restart()
943 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -0700944 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800945 self.run('stop ui; start ui')
946 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -0700947
948
Daniel Erat4b5f7e02018-07-04 22:05:30 -0700949 def _start_powerd_if_needed(self):
950 """Start powerd if it isn't already running."""
951 self.run('start powerd', ignore_status=True)
952
953
xixuana3bbc422017-05-04 15:57:21 -0700954 def _get_lsb_release_content(self):
955 """Return the content of lsb-release file of host."""
956 return self.run(
957 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
958
959
Dan Shi549fb822015-03-24 18:01:11 -0700960 def get_release_version(self):
961 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
962
963 @returns The version string in lsb-release, under attribute
964 CHROMEOS_RELEASE_VERSION.
965 """
Dan Shi549fb822015-03-24 18:01:11 -0700966 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -0700967 lsb_release_content=self._get_lsb_release_content())
968
969
Don Garrettb9f35802018-01-22 18:25:40 -0800970 def get_release_builder_path(self):
971 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
972
973 @returns The version string in lsb-release, under attribute
974 CHROMEOS_RELEASE_BUILDER_PATH.
975 """
976 return lsbrelease_utils.get_chromeos_release_builder_path(
977 lsb_release_content=self._get_lsb_release_content())
978
979
xixuana3bbc422017-05-04 15:57:21 -0700980 def get_chromeos_release_milestone(self):
981 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
982 from lsb-release.
983
984 @returns The version string in lsb-release, under attribute
985 CHROMEOS_RELEASE_BUILD_TYPE.
986 """
987 return lsbrelease_utils.get_chromeos_release_milestone(
988 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -0700989
990
991 def verify_cros_version_label(self):
992 """ Make sure host's cros-version label match the actual image in dut.
993
994 Remove any cros-version: label that doesn't match that installed in
995 the dut.
996
997 @param raise_error: Set to True to raise exception if any mismatch found
998
999 @raise error.AutoservError: If any mismatch between cros-version label
1000 and the build installed in dut is found.
1001 """
1002 labels = self._AFE.get_labels(
1003 name__startswith=ds_constants.VERSION_PREFIX,
1004 host__hostname=self.hostname)
1005 mismatch_found = False
1006 if labels:
Richard Barnette147dda42018-02-15 10:54:02 -08001007 # Ask the DUT for its canonical image name. This will be in
1008 # a form like this: kevin-release/R66-10405.0.0
Don Garrettb9f35802018-01-22 18:25:40 -08001009 release_builder_path = self.get_release_builder_path()
Dan Shi549fb822015-03-24 18:01:11 -07001010 host_list = [self.hostname]
1011 for label in labels:
1012 # Remove any cros-version label that does not match
Richard Barnette147dda42018-02-15 10:54:02 -08001013 # the DUT's installed image.
1014 #
Richard Barnettec92805b2018-06-26 14:07:07 -07001015 # TODO(jrbarnette): We make exceptions for certain
1016 # known cases where the version label will not match the
1017 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1018 # * Tests for the `arc-presubmit` pool append
1019 # "-cheetsth" to the label.
1020 # * Moblab use cases based on `cros stage` store images
1021 # under a name with the string "-custom" embedded.
1022 # It's not reliable to match such an image name to the
1023 # label.
1024 label_version = label.name[len(ds_constants.VERSION_PREFIX):]
1025 if '-custom' in label_version:
1026 continue
1027 if label_version.endswith('-cheetsth'):
1028 label_version = label_version[:-len('-cheetsth')]
1029 if label_version != release_builder_path:
Don Garrettb9f35802018-01-22 18:25:40 -08001030 logging.warn(
Aviv Keshet3ccfc912019-04-10 16:50:49 -07001031 'version according to cros-version label "%s" does not '
1032 'match DUT-determined version %s. Removing the label.',
1033 label_version, release_builder_path)
Dan Shi549fb822015-03-24 18:01:11 -07001034 label.remove_hosts(hosts=host_list)
1035 mismatch_found = True
1036 if mismatch_found:
1037 raise error.AutoservError('The host has wrong cros-version label.')
1038
1039
Laurence Goodby778c9a42017-05-24 19:24:07 -07001040 def cleanup_services(self):
1041 """Reinitializes the device for cleanup.
1042
1043 Subclasses may override this to customize the cleanup method.
1044
1045 To indicate failure of the reset, the implementation may raise
1046 any of:
1047 error.AutoservRunError
1048 error.AutotestRunError
1049 FactoryImageCheckerException
1050
1051 @raises error.AutoservRunError
1052 @raises error.AutotestRunError
1053 @raises error.FactoryImageCheckerException
1054 """
1055 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001056 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001057
1058
beepsc87ff602013-07-31 21:53:00 -07001059 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001060 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001061 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001062 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001063 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001064 except (error.AutotestRunError, error.AutoservRunError,
1065 FactoryImageCheckerException):
Ilja H. Friedel04be2bd2014-05-07 21:29:59 -07001066 logging.warning('Unable to restart ui, rebooting device.')
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001067 # Since restarting the UI fails fall back to normal Autotest
1068 # cleanup routines, i.e. reboot the machine.
Fang Deng0ca40e22013-08-27 17:47:44 -07001069 super(CrosHost, self).cleanup()
Simran Basi5e6339a2013-03-21 11:34:32 -07001070 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001071 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001072 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001073 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001074
1075
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001076 def reboot(self, **dargs):
1077 """
1078 This function reboots the site host. The more generic
1079 RemoteHost.reboot() performs sync and sleeps for 5
1080 seconds. This is not necessary for Chrome OS devices as the
1081 sync should be finished in a short time during the reboot
1082 command.
1083 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001084 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001085 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001086 dargs['reboot_cmd'] = ('sleep 1; '
1087 'reboot & sleep %d; '
1088 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001089 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001090 if 'fastsync' not in dargs:
1091 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001092
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001093 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001094 # Record who called us
1095 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001096 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001097 'dut_host_name' : self.hostname,
1098 'success' : True}
1099 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001100 'caller' : "%s:%s" % (orig.co_filename,
1101 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001102 'success' : True,
1103 'error' : ''}
1104
Vincent Palatin80780b22016-07-27 16:02:37 +02001105 t0 = time.time()
1106 try:
1107 super(CrosHost, self).reboot(**dargs)
1108 except Exception as e:
1109 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001110 metric_debug_fields['success'] = False
1111 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001112 raise
1113 finally:
1114 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001115 metrics.Counter(
1116 'chromeos/autotest/autoserv/reboot_count').increment(
1117 fields=metric_fields)
1118 metrics.Counter(
1119 'chromeos/autotest/autoserv/reboot_debug').increment(
1120 fields=metric_debug_fields)
1121 metrics.SecondsDistribution(
1122 'chromeos/autotest/autoserv/reboot_duration').add(
1123 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001124
1125
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001126 def suspend(self, suspend_time=60,
1127 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001128 """
1129 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001130
1131 @param suspend_time: How long to suspend as integer seconds.
1132 @param suspend_cmd: Suspend command to execute.
1133 @param allow_early_resume: If False and if device resumes before
1134 |suspend_time|, throw an error.
1135
1136 @exception AutoservSuspendError Host resumed earlier than
1137 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001138 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001139
1140 if suspend_cmd is None:
1141 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001142 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001143 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
J. Richard Barnette9af19632015-09-25 12:18:03 -07001144 'powerd_dbus_suspend --delay=0'])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001145 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1146 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001147
1148
Simran Basiec564392014-08-25 16:48:09 -07001149 def upstart_status(self, service_name):
1150 """Check the status of an upstart init script.
1151
1152 @param service_name: Service to look up.
1153
1154 @returns True if the service is running, False otherwise.
1155 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001156 return 'start/running' in self.run('status %s' % service_name,
1157 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001158
Tom Hughese9552342018-12-18 14:29:25 -08001159 def upstart_stop(self, service_name):
1160 """Stops an upstart job if it's running.
1161
1162 @param service_name: Service to stop
1163
1164 @returns True if service has been stopped or was already stopped
1165 False otherwise.
1166 """
1167 if not self.upstart_status(service_name):
1168 return True
1169
1170 result = self.run('stop %s' % service_name, ignore_status=True)
1171 if result.exit_status != 0:
1172 return False
1173 return True
1174
1175 def upstart_restart(self, service_name):
1176 """Restarts (or starts) an upstart job.
1177
1178 @param service_name: Service to start/restart
1179
1180 @returns True if service has been started/restarted, False otherwise.
1181 """
1182 cmd = 'start'
1183 if self.upstart_status(service_name):
1184 cmd = 'restart'
1185 cmd = cmd + ' %s' % service_name
1186 result = self.run(cmd)
1187 if result.exit_status != 0:
1188 return False
1189 return True
Simran Basiec564392014-08-25 16:48:09 -07001190
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001191 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001192 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001193
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001194 Tests for the following conditions:
1195 1. All conditions tested by the parent version of this
1196 function.
1197 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001198 3. Sufficient space in /mnt/stateful_partition/encrypted.
1199 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001200
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001201 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001202 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001203 default_kilo_inodes_required = CONFIG.get_config_value(
1204 'SERVER', 'kilo_inodes_required', type=int, default=100)
1205 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1206 kilo_inodes_required = CONFIG.get_config_value(
1207 'SERVER', 'kilo_inodes_required_%s' % board,
1208 type=int, default=default_kilo_inodes_required)
1209 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001210 self.check_diskspace(
1211 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001212 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001213 'SERVER', 'gb_diskspace_required', type=float,
1214 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001215 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1216 # Not all targets build with encrypted stateful support.
1217 if self.path_exists(encrypted_stateful_path):
1218 self.check_diskspace(
1219 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001220 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001221 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1222 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001223
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001224 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001225
beepsc87ff602013-07-31 21:53:00 -07001226 # Factory images don't run update engine,
1227 # goofy controls dbus on these DUTs.
1228 if not self._is_factory_image():
1229 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001230
Dan Shi549fb822015-03-24 18:01:11 -07001231 self.verify_cros_version_label()
1232
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001233
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001234 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1235 def wait_for_system_services(self):
1236 """Waits for system-services to be running.
1237
1238 Sometimes, update_engine will take a while to update firmware, so we
1239 should give this some time to finish. See crbug.com/765686#c38 for
1240 details.
1241 """
1242 if not self.upstart_status('system-services'):
1243 raise error.AutoservError('Chrome failed to reach login. '
1244 'System services not running.')
1245
1246
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001247 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001248 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001249 message = 'Beginning verify for host %s board %s model %s'
1250 info = self.host_info_store.get()
1251 message %= (self.hostname, info.board, info.model)
1252 self.record('INFO', None, None, message)
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001253 self._repair_strategy.verify(self)
1254
1255
Fang Deng96667ca2013-08-01 17:46:18 -07001256 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001257 connect_timeout=None, alive_interval=None,
1258 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001259 """Override default make_ssh_command to use options tuned for Chrome OS.
1260
1261 Tuning changes:
1262 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1263 connection failure. Consistency with remote_access.sh.
1264
Samuel Tan2ce155b2015-06-23 18:24:38 -07001265 - ServerAliveInterval=900; which causes SSH to ping connection every
1266 900 seconds. In conjunction with ServerAliveCountMax ensures
1267 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001268 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001269 the test completed successfully. Later increased from 180 seconds to
1270 900 seconds to account for tests where the DUT is suspended for
1271 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001272
1273 - ServerAliveCountMax=3; consistency with remote_access.sh.
1274
1275 - ConnectAttempts=4; reduce flakiness in connection errors;
1276 consistency with remote_access.sh.
1277
1278 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1279 Host keys change with every new installation, don't waste
1280 memory/space saving them.
1281
1282 - SSH protocol forced to 2; needed for ServerAliveInterval.
1283
1284 @param user User name to use for the ssh connection.
1285 @param port Port on the target host to use for ssh connection.
1286 @param opts Additional options to the ssh command.
1287 @param hosts_file Ignored.
1288 @param connect_timeout Ignored.
1289 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001290 @param alive_count_max Ignored.
1291 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001292 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001293 options = ' '.join([opts, '-o Protocol=2'])
1294 return super(CrosHost, self).make_ssh_command(
1295 user=user, port=port, opts=options, hosts_file='/dev/null',
1296 connect_timeout=30, alive_interval=900, alive_count_max=3,
1297 connection_attempts=4)
1298
1299
Jason Abeleb6f924f2013-11-13 16:01:54 -08001300 def syslog(self, message, tag='autotest'):
1301 """Logs a message to syslog on host.
1302
1303 @param message String message to log into syslog
1304 @param tag String tag prefix for syslog
1305
1306 """
1307 self.run('logger -t "%s" "%s"' % (tag, message))
1308
1309
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001310 def _ping_check_status(self, status):
1311 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001312
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001313 @param status Check the ping status against this value.
1314 @return True iff `status` and the result of ping are the same
1315 (i.e. both True or both False).
1316
1317 """
1318 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1319 return not (status ^ (ping_val == 0))
1320
1321 def _ping_wait_for_status(self, status, timeout):
1322 """Wait for the host to have a given status (UP or DOWN).
1323
1324 Status is checked by polling. Polling will not last longer
1325 than the number of seconds in `timeout`. The polling
1326 interval will be long enough that only approximately
1327 _PING_WAIT_COUNT polling cycles will be executed, subject
1328 to a maximum interval of about one minute.
1329
1330 @param status Waiting will stop immediately if `ping` of the
1331 host returns this status.
1332 @param timeout Poll for at most this many seconds.
1333 @return True iff the host status from `ping` matched the
1334 requested status at the time of return.
1335
1336 """
1337 # _ping_check_status() takes about 1 second, hence the
1338 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001339 # FIXME: if the ping command errors then _ping_check_status()
1340 # returns instantly. If timeout is also smaller than twice
1341 # _PING_WAIT_COUNT then the while loop below forks many
1342 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1343 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1344 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001345 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1346 end_time = time.time() + timeout
1347 while time.time() <= end_time:
1348 if self._ping_check_status(status):
1349 return True
1350 if poll_interval > 0:
1351 time.sleep(poll_interval)
1352
1353 # The last thing we did was sleep(poll_interval), so it may
1354 # have been too long since the last `ping`. Check one more
1355 # time, just to be sure.
1356 return self._ping_check_status(status)
1357
1358 def ping_wait_up(self, timeout):
1359 """Wait for the host to respond to `ping`.
1360
1361 N.B. This method is not a reliable substitute for
1362 `wait_up()`, because a host that responds to ping will not
1363 necessarily respond to ssh. This method should only be used
1364 if the target DUT can be considered functional even if it
1365 can't be reached via ssh.
1366
1367 @param timeout Minimum time to allow before declaring the
1368 host to be non-responsive.
1369 @return True iff the host answered to ping before the timeout.
1370
1371 """
1372 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001373
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001374 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001375 """Wait until the host no longer responds to `ping`.
1376
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001377 This function can be used as a slightly faster version of
1378 `wait_down()`, by avoiding potentially long ssh timeouts.
1379
1380 @param timeout Minimum time to allow for the host to become
1381 non-responsive.
1382 @return True iff the host quit answering ping before the
1383 timeout.
1384
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001385 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001386 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001387
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001388 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001389 """Wait for the client to enter low-power sleep mode.
1390
1391 The test for "is asleep" can't distinguish a system that is
1392 powered off; to confirm that the unit was asleep, it is
1393 necessary to force resume, and then call
1394 `test_wait_for_resume()`.
1395
1396 This function is expected to be called from a test as part
1397 of a sequence like the following:
1398
1399 ~~~~~~~~
1400 boot_id = host.get_boot_id()
1401 # trigger sleep on the host
1402 host.test_wait_for_sleep()
1403 # trigger resume on the host
1404 host.test_wait_for_resume(boot_id)
1405 ~~~~~~~~
1406
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001407 @param sleep_timeout time limit in seconds to allow the host sleep.
1408
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001409 @exception TestFail The host did not go to sleep within
1410 the allowed time.
1411 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001412 if sleep_timeout is None:
1413 sleep_timeout = self.SLEEP_TIMEOUT
1414
1415 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001416 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001417 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001418
1419
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001420 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001421 """Wait for the client to resume from low-power sleep mode.
1422
1423 The `old_boot_id` parameter should be the value from
1424 `get_boot_id()` obtained prior to entering sleep mode. A
1425 `TestFail` exception is raised if the boot id changes.
1426
1427 See @ref test_wait_for_sleep for more on this function's
1428 usage.
1429
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001430 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001431 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001432 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001433
1434 @exception TestFail The host did not respond within the
1435 allowed time.
1436 @exception TestFail The host responded, but the boot id test
1437 indicated a reboot rather than a sleep
1438 cycle.
1439 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001440 if resume_timeout is None:
1441 resume_timeout = self.RESUME_TIMEOUT
1442
1443 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001444 raise error.TestFail(
1445 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001446 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001447 else:
1448 new_boot_id = self.get_boot_id()
1449 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001450 logging.error('client rebooted (old boot %s, new boot %s)',
1451 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001452 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001453 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001454
1455
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001456 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001457 """Wait for the client to shut down.
1458
1459 The test for "has shut down" can't distinguish a system that
1460 is merely asleep; to confirm that the unit was down, it is
1461 necessary to force boot, and then call test_wait_for_boot().
1462
1463 This function is expected to be called from a test as part
1464 of a sequence like the following:
1465
1466 ~~~~~~~~
1467 boot_id = host.get_boot_id()
1468 # trigger shutdown on the host
1469 host.test_wait_for_shutdown()
1470 # trigger boot on the host
1471 host.test_wait_for_boot(boot_id)
1472 ~~~~~~~~
1473
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001474 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001475 @exception TestFail The host did not shut down within the
1476 allowed time.
1477 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001478 if shutdown_timeout is None:
1479 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1480
1481 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001482 raise error.TestFail(
1483 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001484 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001485
1486
1487 def test_wait_for_boot(self, old_boot_id=None):
1488 """Wait for the client to boot from cold power.
1489
1490 The `old_boot_id` parameter should be the value from
1491 `get_boot_id()` obtained prior to shutting down. A
1492 `TestFail` exception is raised if the boot id does not
1493 change. The boot id test is omitted if `old_boot_id` is not
1494 specified.
1495
1496 See @ref test_wait_for_shutdown for more on this function's
1497 usage.
1498
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001499 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001500 shut down.
1501
1502 @exception TestFail The host did not respond within the
1503 allowed time.
1504 @exception TestFail The host responded, but the boot id test
1505 indicated that there was no reboot.
1506 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001507 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001508 raise error.TestFail(
1509 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001510 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001511 elif old_boot_id:
1512 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001513 logging.error('client not rebooted (boot %s)',
1514 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001515 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001516 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001517
1518
1519 @staticmethod
1520 def check_for_rpm_support(hostname):
1521 """For a given hostname, return whether or not it is powered by an RPM.
1522
Simran Basi1df55112013-09-06 11:25:09 -07001523 @param hostname: hostname to check for rpm support.
1524
Simran Basid5e5e272012-09-24 15:23:59 -07001525 @return None if this host does not follows the defined naming format
1526 for RPM powered DUT's in the lab. If it does follow the format,
1527 it returns a regular expression MatchObject instead.
1528 """
Fang Dengbaff9082015-01-06 13:46:15 -08001529 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001530
1531
1532 def has_power(self):
1533 """For this host, return whether or not it is powered by an RPM.
1534
1535 @return True if this host is in the CROS lab and follows the defined
1536 naming format.
1537 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001538 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001539
1540
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001541 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001542 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001543
1544 @param state Specifies which power state to set to DUT
1545 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001546 use. By default "RPM" or "CCD" will be used based
1547 on servo type. Valid values from
1548 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001549
1550 """
1551 ACCEPTABLE_STATES = ['ON', 'OFF']
1552
Garry Wang5e5538a2019-04-08 15:36:18 -07001553 if not power_method:
1554 power_method = self.get_default_power_method()
1555
1556 state = state.upper()
1557 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001558 raise error.TestError('State must be one of: %s.'
1559 % (ACCEPTABLE_STATES,))
1560
1561 if power_method == self.POWER_CONTROL_SERVO:
1562 logging.info('Setting servo port J10 to %s', state)
1563 self.servo.set('prtctl3_pwren', state.lower())
1564 time.sleep(self._USB_POWER_TIMEOUT)
1565 elif power_method == self.POWER_CONTROL_MANUAL:
1566 logging.info('You have %d seconds to set the AC power to %s.',
1567 self._POWER_CYCLE_TIMEOUT, state)
1568 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001569 elif power_method == self.POWER_CONTROL_CCD:
1570 servo_role = 'src' if state == 'ON' else 'snk'
1571 logging.info('servo ccd power pass through detected,'
1572 ' changing servo_role to %s.', servo_role)
1573 self.servo.set_servo_v4_role(servo_role)
1574 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001575 # Make sure we don't leave DUT with no power(servo_role=snk)
1576 # when DUT is not pingable, as we raise a exception here
1577 # that may break a power cycle in the middle.
1578 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001579 raise error.AutoservError(
1580 'DUT failed to regain network connection after %d seconds.'
1581 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001582 else:
1583 if not self.has_power():
1584 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001585 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07001586 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07001587
1588
Garry Wang5e5538a2019-04-08 15:36:18 -07001589 def power_off(self, power_method=None):
1590 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001591
1592 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001593 use. By default "RPM" or "CCD" will be used based
1594 on servo type. Valid values from
1595 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001596
1597 """
1598 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07001599
1600
Garry Wang5e5538a2019-04-08 15:36:18 -07001601 def power_on(self, power_method=None):
1602 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001603
1604 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001605 use. By default "RPM" or "CCD" will be used based
1606 on servo type. Valid values from
1607 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001608
1609 """
1610 self._set_power('ON', power_method)
1611
1612
Garry Wang5e5538a2019-04-08 15:36:18 -07001613 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001614 """Cycle power to this host by turning it OFF, then ON.
1615
1616 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001617 use. By default "RPM" or "CCD" will be used based
1618 on servo type. Valid values from
1619 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001620
1621 """
Garry Wang5e5538a2019-04-08 15:36:18 -07001622 if not power_method:
1623 power_method = self.get_default_power_method()
1624
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001625 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07001626 self.POWER_CONTROL_MANUAL,
1627 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001628 self.power_off(power_method=power_method)
1629 time.sleep(self._POWER_CYCLE_TIMEOUT)
1630 self.power_on(power_method=power_method)
1631 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08001632 self._add_rpm_changed_tag()
1633 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07001634
1635
1636 def get_platform(self):
1637 """Determine the correct platform label for this host.
1638
1639 @returns a string representing this host's platform.
1640 """
C Shapiroed87c6f2018-04-19 09:13:58 -06001641 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1642 run_method=self.run)
1643 unibuild = release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1'
1644 platform = ''
1645 if unibuild:
1646 cmd = 'mosys platform model'
1647 result = self.run(command=cmd, ignore_status=True)
1648 if result.exit_status == 0:
1649 platform = result.stdout.strip()
1650
1651 if not platform:
1652 # Look at the firmware for non-unibuild cases or if mosys fails.
C Shapiro218e8752017-09-22 11:10:57 -06001653 crossystem = utils.Crossystem(self)
1654 crossystem.init()
1655 # Extract fwid value and use the leading part as the platform id.
1656 # fwid generally follow the format of {platform}.{firmware version}
1657 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
1658 platform = crossystem.fwid().split('.')[0].lower()
1659 # Newer platforms start with 'Google_' while the older ones do not.
C Shapiroed87c6f2018-04-19 09:13:58 -06001660 platform = platform.replace('google_', '')
1661 return platform
Simran Basic6f1f7a2012-10-16 10:47:46 -07001662
1663
Hung-ying Tyanb1328032014-04-01 14:18:54 +08001664 def get_architecture(self):
1665 """Determine the correct architecture label for this host.
1666
1667 @returns a string representing this host's architecture.
1668 """
1669 crossystem = utils.Crossystem(self)
1670 crossystem.init()
1671 return crossystem.arch()
1672
1673
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001674 def get_chrome_version(self):
1675 """Gets the Chrome version number and milestone as strings.
1676
1677 Invokes "chrome --version" to get the version number and milestone.
1678
1679 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
1680 current Chrome version number as a string (in the form "W.X.Y.Z")
1681 and "milestone" is the first component of the version number
1682 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
1683 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
1684 of "chrome --version" and the milestone will be the empty string.
1685
1686 """
MK Ryu35d661e2014-09-25 17:44:10 -07001687 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001688 return utils.parse_chrome_version(version_string)
1689
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001690
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001691 def get_ec_version(self):
1692 """Get the ec version as strings.
1693
1694 @returns a string representing this host's ec version.
1695 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001696 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001697 result = self.run(command, ignore_status=True)
1698 if result.exit_status != 0:
1699 return ''
1700 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001701
1702
1703 def get_firmware_version(self):
1704 """Get the firmware version as strings.
1705
1706 @returns a string representing this host's firmware version.
1707 """
1708 crossystem = utils.Crossystem(self)
1709 crossystem.init()
1710 return crossystem.fwid()
1711
1712
1713 def get_hardware_revision(self):
1714 """Get the hardware revision as strings.
1715
1716 @returns a string representing this host's hardware revision.
1717 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001718 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001719 result = self.run(command, ignore_status=True)
1720 if result.exit_status != 0:
1721 return ''
1722 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001723
1724
1725 def get_kernel_version(self):
1726 """Get the kernel version as strings.
1727
1728 @returns a string representing this host's kernel version.
1729 """
1730 return self.run('uname -r').stdout.strip()
1731
1732
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08001733 def get_cpu_name(self):
1734 """Get the cpu name as strings.
1735
1736 @returns a string representing this host's cpu name.
1737 """
1738
1739 # Try get cpu name from device tree first
1740 if self.path_exists('/proc/device-tree/compatible'):
1741 command = ' | '.join(
1742 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
1743 'tail -1'])
1744 return self.run(command).stdout.strip().replace(',', ' ')
1745
1746 # Get cpu name from uname -p
1747 command = 'uname -p'
1748 ret = self.run(command).stdout.strip()
1749
1750 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
1751 # Try get cpu name from /proc/cpuinfo instead
1752 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
1753 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
1754 self = self.run(command).stdout.strip()
1755
1756 # Remove bloat from CPU name, for example
1757 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
1758 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
1759 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
1760 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
1761 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
1762 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
1763
1764
1765 def get_screen_resolution(self):
1766 """Get the screen(s) resolution as strings.
1767 In case of more than 1 monitor, return resolution for each monitor
1768 separate with plus sign.
1769
1770 @returns a string representing this host's screen(s) resolution.
1771 """
1772 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
1773 ret = self.run(command, ignore_status=True)
1774 # We might have Chromebox without a screen
1775 if ret.exit_status != 0:
1776 return ''
1777 return ret.stdout.strip().replace('\n', '+')
1778
1779
1780 def get_mem_total_gb(self):
1781 """Get total memory available in the system in GiB (2^20).
1782
1783 @returns an integer representing total memory
1784 """
1785 mem_total_kb = self.read_from_meminfo('MemTotal')
1786 kb_in_gb = float(2 ** 20)
1787 return int(round(mem_total_kb / kb_in_gb))
1788
1789
1790 def get_disk_size_gb(self):
1791 """Get size of disk in GB (10^9)
1792
1793 @returns an integer representing size of disk, 0 in Error Case
1794 """
1795 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
1796 result = self.run(command, ignore_status=True)
1797 if result.exit_status != 0:
1798 return 0
1799 _, _, block, _ = re.split(r' +', result.stdout.strip())
1800 byte_per_block = 1024.0
1801 disk_kb_in_gb = 1e9
1802 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
1803
1804
1805 def get_battery_size(self):
1806 """Get size of battery in Watt-hour via sysfs
1807
1808 This method assumes that battery support voltage_min_design and
1809 charge_full_design sysfs.
1810
1811 @returns a float representing Battery size, 0 if error.
1812 """
1813 # sysfs report data in micro scale
1814 battery_scale = 1e6
1815
1816 command = 'cat /sys/class/power_supply/*/voltage_min_design'
1817 result = self.run(command, ignore_status=True)
1818 if result.exit_status != 0:
1819 return 0
1820 voltage = float(result.stdout.strip()) / battery_scale
1821
1822 command = 'cat /sys/class/power_supply/*/charge_full_design'
1823 result = self.run(command, ignore_status=True)
1824 if result.exit_status != 0:
1825 return 0
1826 amphereHour = float(result.stdout.strip()) / battery_scale
1827
1828 return voltage * amphereHour
1829
1830
1831 def get_low_battery_shutdown_percent(self):
1832 """Get the percent-based low-battery shutdown threshold.
1833
1834 @returns a float representing low-battery shutdown percent, 0 if error.
1835 """
1836 ret = 0.0
1837 try:
1838 command = 'check_powerd_config --low_battery_shutdown_percent'
1839 ret = float(self.run(command).stdout)
1840 except error.CmdError:
1841 logging.debug("Can't run %s", command)
1842 except ValueError:
1843 logging.debug("Didn't get number from %s", command)
1844
1845 return ret
1846
1847
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07001848 def has_hammer(self):
1849 """Check whether DUT has hammer device or not.
1850
1851 @returns boolean whether device has hammer or not
1852 """
1853 command = 'grep Hammer /sys/bus/usb/devices/*/product'
1854 return self.run(command, ignore_status=True).exit_status == 0
1855
1856
Niranjan Kumar34618872017-05-31 12:57:09 -07001857 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07001858 """Returns True if the specified switch was provided to Chrome.
1859
1860 @param switch The chrome switch to search for.
1861 """
Niranjan Kumar34618872017-05-31 12:57:09 -07001862
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07001863 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
1864 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07001865
1866
1867 def oobe_triggers_update(self):
1868 """Returns True if this host has an OOBE flow during which
1869 it will perform an update check and perhaps an update.
1870 One example of such a flow is Hands-Off Zero-Touch Enrollment.
1871 As more such flows are developed, code handling them needs
1872 to be added here.
1873
1874 @return Boolean indicating whether this host's OOBE triggers an update.
1875 """
1876 return self.is_chrome_switch_present(
1877 '--enterprise-enable-zero-touch-enrollment=hands-off')
1878
1879
Kevin Chenga2619dc2016-03-28 11:42:08 -07001880 # TODO(kevcheng): change this to just return the board without the
1881 # 'board:' prefix and fix up all the callers. Also look into removing the
1882 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07001883 def get_board(self):
1884 """Determine the correct board label for this host.
1885
1886 @returns a string representing this host's board.
1887 """
1888 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1889 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001890 return (ds_constants.BOARD_PREFIX +
1891 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07001892
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07001893 def get_channel(self):
1894 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07001895
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07001896 @returns: a string represeting this host's build channel.
1897 (stable, dev, beta). None on fail.
1898 """
1899 return lsbrelease_utils.get_chromeos_channel(
1900 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07001901
Kevin Chenga328da62016-03-31 10:49:04 -07001902 def get_power_supply(self):
1903 """
1904 Determine what type of power supply the host has
1905
1906 @returns a string representing this host's power supply.
1907 'power:battery' when the device has a battery intended for
1908 extended use
1909 'power:AC_primary' when the device has a battery not intended
1910 for extended use (for moving the machine, etc)
1911 'power:AC_only' when the device has no battery at all.
1912 """
1913 psu = self.run(command='mosys psu type', ignore_status=True)
1914 if psu.exit_status:
1915 # The psu command for mosys is not included for all platforms. The
1916 # assumption is that the device will have a battery if the command
1917 # is not found.
1918 return 'power:battery'
1919
1920 psu_str = psu.stdout.strip()
1921 if psu_str == 'unknown':
1922 return None
1923
1924 return 'power:%s' % psu_str
1925
1926
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08001927 def has_battery(self):
1928 """Determine if DUT has a battery.
1929
1930 Returns:
1931 Boolean, False if known not to have battery, True otherwise.
1932 """
1933 rv = True
1934 power_supply = self.get_power_supply()
1935 if power_supply == 'power:battery':
1936 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
1937 board_type = self.get_board_type()
1938 if board_type in _NO_BATTERY_BOARD_TYPE:
1939 logging.warn('Do NOT believe type %s has battery. '
1940 'See debug for mosys details', board_type)
1941 psu = self.system_output('mosys -vvvv psu type',
1942 ignore_status=True)
1943 logging.debug(psu)
1944 rv = False
1945 elif power_supply == 'power:AC_only':
1946 rv = False
1947
1948 return rv
1949
1950
Kevin Chenga328da62016-03-31 10:49:04 -07001951 def get_servo(self):
1952 """Determine if the host has a servo attached.
1953
1954 If the host has a working servo attached, it should have a servo label.
1955
1956 @return: string 'servo' if the host has servo attached. Otherwise,
1957 returns None.
1958 """
1959 return 'servo' if self._servo_host else None
1960
1961
Kevin Chenga328da62016-03-31 10:49:04 -07001962 def has_internal_display(self):
1963 """Determine if the device under test is equipped with an internal
1964 display.
1965
1966 @return: 'internal_display' if one is present; None otherwise.
1967 """
1968 from autotest_lib.client.cros.graphics import graphics_utils
1969 from autotest_lib.client.common_lib import utils as common_utils
1970
1971 def __system_output(cmd):
1972 return self.run(cmd).stdout
1973
1974 def __read_file(remote_path):
1975 return self.run('cat %s' % remote_path).stdout
1976
1977 # Hijack the necessary client functions so that we can take advantage
1978 # of the client lib here.
1979 # FIXME: find a less hacky way than this
1980 original_system_output = utils.system_output
1981 original_read_file = common_utils.read_file
1982 utils.system_output = __system_output
1983 common_utils.read_file = __read_file
1984 try:
1985 return ('internal_display' if graphics_utils.has_internal_display()
1986 else None)
1987 finally:
1988 utils.system_output = original_system_output
1989 common_utils.read_file = original_read_file
1990
1991
Dan Shi85276d42014-04-08 22:11:45 -07001992 def is_boot_from_usb(self):
1993 """Check if DUT is boot from USB.
1994
1995 @return: True if DUT is boot from usb.
1996 """
1997 device = self.run('rootdev -s -d').stdout.strip()
1998 removable = int(self.run('cat /sys/block/%s/removable' %
1999 os.path.basename(device)).stdout.strip())
2000 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002001
2002
2003 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002004 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002005
2006 @param key: meminfo requested
2007
2008 @return the memory value as a string
2009
2010 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002011 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2012 logging.debug('%s', meminfo)
2013 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002014
2015
Rohit Makasana98e696f2016-06-03 18:48:10 -07002016 def get_cpu_arch(self):
2017 """Returns CPU arch of the device.
2018
2019 @return CPU architecture of the DUT.
2020 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002021 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002022 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2023 ignore_status=True).stdout:
2024 return 'x86_64'
2025 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2026 ignore_status=True).stdout:
2027 return 'arm'
2028 return 'i386'
2029
2030
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002031 def get_board_type(self):
2032 """
2033 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002034 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2035
2036 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002037 """
Danny Chan471a8d12015-08-18 14:57:41 -07002038 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2039 ignore_status=True).stdout
2040 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002041 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002042 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002043
2044
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002045 def get_arc_version(self):
2046 """Return ARC version installed on the DUT.
2047
2048 @returns ARC version as string if the CrOS build has ARC, else None.
2049 """
2050 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2051 ignore_status=True).stdout
2052 if arc_version:
2053 return arc_version.split('=')[-1].strip()
2054 return None
2055
2056
Gilad Arnolda76bef02015-09-29 13:55:15 -07002057 def get_os_type(self):
2058 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002059
2060
Kevin Chenga2619dc2016-03-28 11:42:08 -07002061 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002062 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002063 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002064
2065
2066 def get_default_power_method(self):
2067 """
2068 Get the default power method for power_on/off/cycle() methods.
2069 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2070 """
2071 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002072 self._default_power_method = self.POWER_CONTROL_RPM
Garry Wang5e5538a2019-04-08 15:36:18 -07002073 info = self.host_info_store.get()
2074 if info.attributes.get('servo_type') == self.CCD_SERVO:
Garry Wang1a004aa2019-05-16 22:56:51 -07002075 if self.servo:
2076 self._default_power_method = self.POWER_CONTROL_CCD
2077 else:
2078 logging.warn('CCD servo detected but failed to apply CCD'
2079 ' servo RPM method due to servo instance'
2080 ' was not initialized, fall back to normal'
2081 ' RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002082 return self._default_power_method