blob: cc6401b5031405154b73f9cbf07e44082b714d41 [file] [log] [blame]
Dennis Jeffreydffc0bd2013-05-03 13:24:31 -07001# Copyright (c) 2013 The Chromium OS Authors. All rights reserved.
2# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
Achuith Bhandarkarf3469ec2016-06-08 16:58:57 -07005import logging, os, re
Achuith Bhandarkara2df47b2013-10-09 21:23:15 -07006
Nicolas Boichat451e4662016-07-16 07:51:01 +08007from autotest_lib.client.common_lib.cros import arc_util
Don Garrett11b13972015-09-11 17:37:53 +00008from autotest_lib.client.cros import constants
Achuith Bhandarkar88a2bab2015-07-09 17:36:25 -07009from autotest_lib.client.bin import utils
Prathmesh Prabhuc3961df2015-07-21 10:44:54 -070010from telemetry.core import cros_interface, exceptions, util
Achuith Bhandarkarb058a8f2015-07-02 01:29:05 -070011from telemetry.internal.browser import browser_finder, browser_options
12from telemetry.internal.browser import extension_to_load
Achuith Bhandarkar704134d2015-03-05 17:31:57 -080013
14Error = exceptions.Error
Dennis Jeffreydffc0bd2013-05-03 13:24:31 -070015
16
Victor Hsieh43651ac2016-04-20 12:52:21 -070017# Cached result of whether ARC is available on current device.
18_arc_available = None
19
20
Nicolas Norvez56f9b232016-06-23 17:58:59 -070021def is_arc_available():
Victor Hsieh000240b2016-04-11 16:04:31 -070022 """Returns true if ARC is available on current device."""
Victor Hsieh43651ac2016-04-20 12:52:21 -070023 global _arc_available
24 if _arc_available is not None:
25 return _arc_available
26
27 def _check_lsb_release():
28 lsb_release = '/etc/lsb-release'
29 if not os.path.exists(lsb_release):
30 return False
31 with open(lsb_release) as f:
32 for line in f:
33 if line.startswith('CHROMEOS_ARC_VERSION='):
34 return True
35 return False
36
37 _arc_available = _check_lsb_release()
38 return _arc_available
39
40
Achuith Bhandarkarf3469ec2016-06-08 16:58:57 -070041def NormalizeEmail(username):
42 """Remove dots from username. Add @gmail.com if necessary.
43
44 TODO(achuith): Get rid of this when crbug.com/358427 is fixed.
45
46 @param username: username/email to be scrubbed.
47 """
48 parts = re.split('@', username)
49 parts[0] = re.sub('\.', '', parts[0])
50
51 if len(parts) == 1:
52 parts.append('gmail.com')
53 return '@'.join(parts)
54
55
Achuith Bhandarkared498932013-07-16 17:01:40 -070056class Chrome(object):
57 """Wrapper for creating a telemetry browser instance with extensions."""
Achuith Bhandarkar86c46812013-07-15 17:13:39 -070058
59
Achuith Bhandarkared498932013-07-16 17:01:40 -070060 BROWSER_TYPE_LOGIN = 'system'
61 BROWSER_TYPE_GUEST = 'system-guest'
Achuith Bhandarkar86c46812013-07-15 17:13:39 -070062
63
Achuith Bhandarkarb0e99072013-11-18 16:15:03 -080064 def __init__(self, logged_in=True, extension_paths=[], autotest_ext=False,
Achuith Bhandarkarb7ef5e52014-03-20 14:18:45 -070065 is_component=True, num_tries=3, extra_browser_args=None,
Achuith Bhandarkar882a8ab2014-08-26 15:51:20 -070066 clear_enterprise_policy=True, dont_override_profile=False,
Achuith Bhandarkara2bceaf2014-11-14 12:10:16 -080067 disable_gaia_services=True, disable_default_apps = True,
68 auto_login=True, gaia_login=False,
Victor Hsieh000240b2016-04-11 16:04:31 -070069 username=None, password=None, gaia_id=None,
F#md6d56ba2016-06-10 14:40:52 -070070 arc_mode=None, disable_arc_opt_in=True):
Dean Liaob12e2ee2013-11-19 16:49:12 +080071 """
Achuith Bhandarkar944405a2013-11-21 14:47:48 -080072 Constructor of telemetry wrapper.
73
74 @param logged_in: Regular user (True) or guest user (False).
75 @param extension_paths: path of unpacked extension to install.
76 @param autotest_ext: Load a component extension with privileges to
77 invoke chrome.autotestPrivate.
78 @param is_component: Whether extensions should be loaded as component
79 extensions.
Achuith Bhandarkarb7ef5e52014-03-20 14:18:45 -070080 @param num_tries: Number of attempts to log in.
Dean Liaob12e2ee2013-11-19 16:49:12 +080081 @param extra_browser_args: Additional argument(s) to pass to the
Achuith Bhandarkar3875e0c2014-03-20 14:17:31 -070082 browser. It can be a string or a list.
Achuith Bhandarkar5d6c26c2014-04-25 11:19:38 -070083 @param clear_enterprise_policy: Clear enterprise policy before
84 logging in.
Achuith Bhandarkar882a8ab2014-08-26 15:51:20 -070085 @param dont_override_profile: Don't delete cryptohome before login.
86 Telemetry will output a warning with this
87 option.
Achuith Bhandarkar3c654f32014-09-29 06:33:35 -070088 @param disable_gaia_services: For enterprise autotests, this option may
89 be used to enable policy fetch.
Achuith Bhandarkara2bceaf2014-11-14 12:10:16 -080090 @param disable_default_apps: For tests that exercise default apps.
Achuith Bhandarkar1cce9dc2014-02-05 14:23:34 -080091 @param auto_login: Does not login automatically if this is False.
Achuith Bhandarkar3875e0c2014-03-20 14:17:31 -070092 Useful if you need to examine oobe.
93 @param gaia_login: Logs in to real gaia.
Achuith Bhandarkar1cce9dc2014-02-05 14:23:34 -080094 @param username: Log in using this username instead of the default.
Alexander Alekseevd9682862015-10-01 22:42:49 -070095 @param password: Log in using this password instead of the default.
96 @param gaia_id: Log in using this gaia_id instead of the default.
Victor Hsieh43651ac2016-04-20 12:52:21 -070097 @param arc_mode: How ARC instance should be started. Default is to not
98 start.
F#md6d56ba2016-06-10 14:40:52 -070099 @param disable_arc_opt_in: For opt in flow autotest. This option is used
100 to disable the arc opt in flow.
Dean Liaob12e2ee2013-11-19 16:49:12 +0800101 """
Achuith Bhandarkara2df47b2013-10-09 21:23:15 -0700102 self._autotest_ext_path = None
103 if autotest_ext:
104 self._autotest_ext_path = os.path.join(os.path.dirname(__file__),
105 'autotest_private_ext')
106 extension_paths.append(self._autotest_ext_path)
107
Achuith Bhandarkar7fb57f72013-08-29 15:29:06 -0700108 finder_options = browser_options.BrowserFinderOptions()
Nicolas Norvez56f9b232016-06-23 17:58:59 -0700109 if is_arc_available() and arc_util.should_start_arc(arc_mode):
F#md6d56ba2016-06-10 14:40:52 -0700110 if disable_arc_opt_in:
111 finder_options.browser_options.AppendExtraBrowserArgs(
112 arc_util.get_extra_chrome_flags())
Victor Hsieh43651ac2016-04-20 12:52:21 -0700113 logged_in = True
114
Achuith Bhandarkared498932013-07-16 17:01:40 -0700115 self._browser_type = (self.BROWSER_TYPE_LOGIN
116 if logged_in else self.BROWSER_TYPE_GUEST)
Achuith Bhandarkarf130c402013-09-11 14:39:22 -0700117 finder_options.browser_type = self.browser_type
Dean Liaob12e2ee2013-11-19 16:49:12 +0800118 if extra_browser_args:
119 finder_options.browser_options.AppendExtraBrowserArgs(
120 extra_browser_args)
Achuith Bhandarkarf130c402013-09-11 14:39:22 -0700121
Achuith Bhandarkar3e5051d2013-10-15 15:20:12 -0700122 # finder options must be set before parse_args(), browser options must
123 # be set before Create().
Todd Brocheb6f4812014-04-29 16:04:28 -0700124 # TODO(crbug.com/360890) Below MUST be '2' so that it doesn't inhibit
125 # autotest debug logs
126 finder_options.verbosity = 2
Achuith Bhandarkardafc9a52013-09-24 15:26:33 +0200127 finder_options.CreateParser().parse_args(args=[])
Achuith Bhandarkara2df47b2013-10-09 21:23:15 -0700128 b_options = finder_options.browser_options
129 b_options.disable_component_extensions_with_background_pages = False
130 b_options.create_browser_with_oobe = True
Achuith Bhandarkar5d6c26c2014-04-25 11:19:38 -0700131 b_options.clear_enterprise_policy = clear_enterprise_policy
Achuith Bhandarkar882a8ab2014-08-26 15:51:20 -0700132 b_options.dont_override_profile = dont_override_profile
Achuith Bhandarkar3c654f32014-09-29 06:33:35 -0700133 b_options.disable_gaia_services = disable_gaia_services
Achuith Bhandarkara2bceaf2014-11-14 12:10:16 -0800134 b_options.disable_default_apps = disable_default_apps
135 b_options.disable_component_extensions_with_background_pages = disable_default_apps
Achuith Bhandarkar1cce9dc2014-02-05 14:23:34 -0800136
137 b_options.auto_login = auto_login
Achuith Bhandarkar3875e0c2014-03-20 14:17:31 -0700138 b_options.gaia_login = gaia_login
F#md6d56ba2016-06-10 14:40:52 -0700139
Nicolas Norvez56f9b232016-06-23 17:58:59 -0700140 if is_arc_available() and not disable_arc_opt_in:
F#md6d56ba2016-06-10 14:40:52 -0700141 arc_util.set_browser_options_for_opt_in(b_options)
142
Achuith Bhandarkar1cce9dc2014-02-05 14:23:34 -0800143 self.username = b_options.username if username is None else username
144 self.password = b_options.password if password is None else password
Achuith Bhandarkarf3469ec2016-06-08 16:58:57 -0700145 self.username = NormalizeEmail(self.username)
Achuith Bhandarkar1cce9dc2014-02-05 14:23:34 -0800146 b_options.username = self.username
147 b_options.password = self.password
Achuith Bhandarkar6f4accc2016-08-23 14:03:44 -0700148 self.gaia_id = b_options.gaia_id if gaia_id is None else gaia_id
149 b_options.gaia_id = self.gaia_id
Achuith Bhandarkara2df47b2013-10-09 21:23:15 -0700150
Shuhei Takahashi8462b882016-05-31 17:27:32 +0900151 self.arc_mode = arc_mode
152
Achuith Bhandarkar531a62a2016-04-13 11:09:43 -0700153 if logged_in:
Achuith Bhandarkar6f4accc2016-08-23 14:03:44 -0700154 extensions_to_load = b_options.extensions_to_load
155 for path in extension_paths:
156 extension = extension_to_load.ExtensionToLoad(
157 path, self.browser_type, is_component=is_component)
158 extensions_to_load.append(extension)
159 self._extensions_to_load = extensions_to_load
Achuith Bhandarkar531a62a2016-04-13 11:09:43 -0700160
David James54a830e2015-05-06 17:21:47 -0700161 # Turn on collection of Chrome coredumps via creation of a magic file.
162 # (Without this, Chrome coredumps are trashed.)
Don Garrett11b13972015-09-11 17:37:53 +0000163 open(constants.CHROME_CORE_MAGIC_FILE, 'w').close()
David James54a830e2015-05-06 17:21:47 -0700164
Achuith Bhandarkarb0e99072013-11-18 16:15:03 -0800165 for i in range(num_tries):
166 try:
167 browser_to_create = browser_finder.FindBrowser(finder_options)
khmela7cf9dd2016-09-20 10:50:23 -0700168 self.network_controller = \
Achuith Bhandarkar08edc202016-08-23 14:21:55 -0700169 browser_to_create.platform.network_controller
170 # TODO(achuith): Remove if condition after catapult:#2584 has
171 # landed and PFQ has rolled. crbug.com/639730.
khmela7cf9dd2016-09-20 10:50:23 -0700172 if hasattr(self.network_controller, 'InitializeIfNeeded'):
173 self.network_controller.InitializeIfNeeded()
Achuith Bhandarkara2bceaf2014-11-14 12:10:16 -0800174 self._browser = browser_to_create.Create(finder_options)
Nicolas Norvez56f9b232016-06-23 17:58:59 -0700175 if is_arc_available():
F#md6d56ba2016-06-10 14:40:52 -0700176 if not disable_arc_opt_in:
177 arc_util.opt_in(self.browser)
Ricky Zhoucf350ee2016-06-13 15:47:33 -0700178 arc_util.post_processing_after_browser(self)
Achuith Bhandarkarb0e99072013-11-18 16:15:03 -0800179 break
khmela7cf9dd2016-09-20 10:50:23 -0700180 except exceptions.LoginException as e:
Achuith Bhandarkarf800edf2013-12-06 13:37:52 -0800181 logging.error('Timed out logging in, tries=%d, error=%s',
182 i, repr(e))
Achuith Bhandarkarb0e99072013-11-18 16:15:03 -0800183 if i == num_tries-1:
184 raise
Achuith Bhandarkared498932013-07-16 17:01:40 -0700185
186
187 def __enter__(self):
188 return self
189
190
191 def __exit__(self, *args):
Derek Basehore41acbf82014-07-11 18:34:30 -0700192 self.close()
Achuith Bhandarkared498932013-07-16 17:01:40 -0700193
194
195 @property
196 def browser(self):
197 """Returns a telemetry browser instance."""
198 return self._browser
199
200
201 def get_extension(self, extension_path):
202 """Fetches a telemetry extension instance given the extension path."""
203 for ext in self._extensions_to_load:
204 if extension_path == ext.path:
205 return self.browser.extensions[ext]
206 return None
207
208
209 @property
Achuith Bhandarkara2df47b2013-10-09 21:23:15 -0700210 def autotest_ext(self):
211 """Returns the autotest extension."""
212 return self.get_extension(self._autotest_ext_path)
213
214
215 @property
216 def login_status(self):
217 """Returns login status."""
218 ext = self.autotest_ext
219 if not ext:
220 return None
221
222 ext.ExecuteJavaScript('''
223 window.__login_status = null;
224 chrome.autotestPrivate.loginStatus(function(s) {
225 window.__login_status = s;
226 });
227 ''')
228 return ext.EvaluateJavaScript('window.__login_status')
229
230
Victor Hsiehfc3417e2016-03-25 16:49:59 -0700231 def get_visible_notifications(self):
232 """Returns an array of visible notifications of Chrome.
233
234 For specific type of each notification, please refer to Chromium's
235 chrome/common/extensions/api/autotest_private.idl.
236 """
237 ext = self.autotest_ext
238 if not ext:
239 return None
240
241 ext.ExecuteJavaScript('''
242 window.__items = null;
243 chrome.autotestPrivate.getVisibleNotifications(function(items) {
244 window.__items = items;
245 });
246 ''')
247 if ext.EvaluateJavaScript('window.__items') is None:
248 return None
249 return ext.EvaluateJavaScript('window.__items')
250
251
Achuith Bhandarkara2df47b2013-10-09 21:23:15 -0700252 @property
Achuith Bhandarkared498932013-07-16 17:01:40 -0700253 def browser_type(self):
254 """Returns the browser_type."""
255 return self._browser_type
Achuith Bhandarkar49c72d92013-07-25 11:10:10 -0700256
257
Achuith Bhandarkar16ee9772015-01-23 17:18:18 -0800258 @staticmethod
259 def did_browser_crash(func):
Achuith Bhandarkar7fb57f72013-08-29 15:29:06 -0700260 """Runs func, returns True if the browser crashed, False otherwise.
261
262 @param func: function to run.
263
264 """
Achuith Bhandarkar5abf60b2013-08-01 12:35:53 -0700265 try:
266 func()
khmela7cf9dd2016-09-20 10:50:23 -0700267 except Error:
Achuith Bhandarkar5abf60b2013-08-01 12:35:53 -0700268 return True
269 return False
Derek Basehore41acbf82014-07-11 18:34:30 -0700270
271
Achuith Bhandarkar88a2bab2015-07-09 17:36:25 -0700272 @staticmethod
273 def wait_for_browser_restart(func):
274 """Runs func, and waits for a browser restart.
275
276 @param func: function to run.
277
278 """
279 _cri = cros_interface.CrOSInterface()
280 pid = _cri.GetChromePid()
281 Chrome.did_browser_crash(func)
282 utils.poll_for_condition(lambda: pid != _cri.GetChromePid(), timeout=60)
283
284
Achuith Bhandarkar16ee9772015-01-23 17:18:18 -0800285 def wait_for_browser_to_come_up(self):
286 """Waits for the browser to come up. This should only be called after a
287 browser crash.
288 """
289 def _BrowserReady(cr):
290 tabs = [] # Wrapper for pass by reference.
291 if self.did_browser_crash(
292 lambda: tabs.append(cr.browser.tabs.New())):
293 return False
294 try:
295 tabs[0].Close()
Achuith Bhandarkarfab82d12015-02-26 11:05:35 -0800296 except:
Achuith Bhandarkar16ee9772015-01-23 17:18:18 -0800297 # crbug.com/350941
298 logging.error('Timed out closing tab')
299 return True
300 util.WaitFor(lambda: _BrowserReady(self), timeout=10)
301
302
Derek Basehore41acbf82014-07-11 18:34:30 -0700303 def close(self):
khmela7cf9dd2016-09-20 10:50:23 -0700304 try:
305 if hasattr(self.network_controller, 'Close'):
306 self.network_controller.Close()
307 logging.info('Network controller is closed')
308 except Error as e:
309 logging.error('Failed to close network controller, error=%s', e)
310
Derek Basehore41acbf82014-07-11 18:34:30 -0700311 """Closes the browser."""
Hidehiko Abe202ed322016-04-14 15:08:43 +0900312 try:
Nicolas Norvez56f9b232016-06-23 17:58:59 -0700313 if is_arc_available():
Shuhei Takahashi8462b882016-05-31 17:27:32 +0900314 arc_util.pre_processing_before_close(self)
Hidehiko Abe202ed322016-04-14 15:08:43 +0900315 finally:
316 self._browser.Close()