| Index: tools/perf/page_sets/memory_test.py
|
| diff --git a/tools/perf/page_sets/memory_test.py b/tools/perf/page_sets/memory_test.py
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..37b131719dace130de63954a55cb26d6e278950b
|
| --- /dev/null
|
| +++ b/tools/perf/page_sets/memory_test.py
|
| @@ -0,0 +1,369 @@
|
| +# Copyright 2014 The Chromium Authors. All rights reserved.
|
| +# Use of this source code is governed by a BSD-style license that can be
|
| +# found in the LICENSE file.
|
| +import logging
|
| +import os
|
| +
|
| +from page_sets.login_helpers import dropbox_login
|
| +from page_sets.login_helpers import google_login
|
| +
|
| +from telemetry.page import page as page_module
|
| +from telemetry.page import shared_page_state
|
| +from telemetry import story
|
| +
|
| +DUMP_WAIT_TIME = 10
|
| +DUMP_COUNT = 10
|
| +
|
| +class MemoryTestPage(page_module.Page):
|
| + def __init__(self, story_set, group, url, page_spec):
|
| + name = 'load:%s:%s' % (group, page_spec.name)
|
| + super(MemoryTestPage, self).__init__(
|
| + page_set=story_set,
|
| + name=name,
|
| + url=url,
|
| + shared_page_state_class=shared_page_state.SharedPageState,
|
| + credentials_path='data/credentials.json')
|
| + self._page_spec = page_spec
|
| +
|
| + @property
|
| + def platform(self):
|
| + return self.story_set.PLATFORM
|
| +
|
| + def _dumpWithInteraction(self, action_runner, text, do_gc=False):
|
| + with action_runner.CreateInteraction(text):
|
| + action_runner.Wait(3)
|
| + if do_gc:
|
| + action_runner.ForceGarbageCollection()
|
| + action_runner.SimulateMemoryPressureNotification('critical')
|
| + action_runner.Wait(3)
|
| + if not action_runner.tab.browser.DumpMemory():
|
| + logging.error('Unable to get a memory dump for %s.', self.name)
|
| +
|
| + def _RunPageInteractions(self, action_runner):
|
| + action_runner.Wait(60)
|
| + if not action_runner.tab.browser.DumpMemory():
|
| + logging.error('Unable to get a memory dump for %s.', self.name)
|
| +
|
| + def _RunPageAndMoveToBackground(self, action_runner):
|
| + action_runner.Wait(60)
|
| + # Force V8 GC.
|
| + action_runner.ForceGarbageCollection()
|
| + action_runner.Wait(10)
|
| + # Dump before moving the page to background.
|
| + if not action_runner.tab.browser.DumpMemory():
|
| + logging.error('Unable to get a memory dump for %s.', self.name)
|
| + action_runner.Wait(3)
|
| + # Open blank tab (Move the page to background).
|
| + action_runner.tab.browser.tabs.New()
|
| + # Wait until purge-and-suspend
|
| + # Needs "--purge-and-suspend-time=X and X < 20"
|
| + action_runner.Wait(120)
|
| + if not action_runner.tab.browser.DumpMemory():
|
| + logging.error('Unable to get a memory dump for %s.', self.name)
|
| +
|
| + def _RunPageAndMoveToBackgroundWithNoDump(self, action_runner):
|
| + action_runner.Wait(60)
|
| + action_runner.ForceGarbageCollection()
|
| + action_runner.Wait(10)
|
| + action_runner.tab.browser.tabs.New()
|
| + action_runner.Wait(540)
|
| +
|
| +
|
| + def RunPageInteractions(self, action_runner):
|
| + action_runner.tab.WaitForDocumentReadyStateToBeComplete()
|
| + if self._page_spec.post_load_hook:
|
| + self._page_spec.post_load_hook(self.platform, action_runner)
|
| + self._RunPageAndMoveToBackground(action_runner)
|
| + #self._RunPageAndMoveToBackgroundWithNoDump(action_runner)
|
| +
|
| + def RunNavigateSteps(self, action_runner):
|
| + if self._page_spec.login_hook:
|
| + self._page_spec.login_hook(action_runner, self.credentials_path)
|
| + super(MemoryTestPage, self).RunNavigateSteps(action_runner)
|
| +
|
| +
|
| +def _LogIntoGoogleAccount(action_runner, credentials_path):
|
| + google_login.LoginGoogleAccount(action_runner, 'googletest', credentials_path)
|
| +
|
| +
|
| +def _LogIntoGoogleAccountAndSetUpGmailSession(action_runner, credentials_path):
|
| + _LogIntoGoogleAccount(action_runner, credentials_path)
|
| +
|
| + # Navigating to https://mail.google.com immediately leads to an infinite
|
| + # redirection loop due to a bug in WPR (see
|
| + # https://github.com/chromium/web-page-replay/issues/70). We therefore first
|
| + # navigate to a sub-URL to set up the session and hit the resulting
|
| + # redirection loop. Afterwards, we can safely navigate to
|
| + # https://mail.google.com.
|
| + action_runner.Navigate(
|
| + 'https://mail.google.com/mail/mu/mp/872/trigger_redirection_loop')
|
| + action_runner.tab.WaitForDocumentReadyStateToBeComplete()
|
| +
|
| +
|
| +def _LogIntoDropboxAccount(action_runner, credentials_path):
|
| + dropbox_login.LoginAccount(action_runner, 'dropbox', credentials_path)
|
| +
|
| +
|
| +def _CloseInboxInterstitialAndWaitUntilGmailReady(platform, action_runner):
|
| + if platform == 'desktop':
|
| + # Wait until the UI loads.
|
| + action_runner.WaitForJavaScriptCondition(
|
| + 'document.getElementById("loading").style.display === "none"')
|
| + elif platform == 'mobile':
|
| + # Close the "Get Inbox by Gmail" interstitial.
|
| + action_runner.WaitForJavaScriptCondition(
|
| + 'document.querySelector("#isppromo a") !== null')
|
| + action_runner.ExecuteJavaScript(
|
| + 'document.querySelector("#isppromo a").click()')
|
| + # Wait until the UI loads.
|
| + action_runner.WaitForJavaScriptCondition(
|
| + 'document.getElementById("apploadingdiv").style.height === "0px"')
|
| +
|
| +
|
| +def _WaitUntilBubblesReady(_, action_runner):
|
| + # The #logo element is removed right before the main menu is displayed.
|
| + action_runner.WaitForJavaScriptCondition(
|
| + 'document.getElementById("logo") === null')
|
| +
|
| +
|
| +def _WaitUntilSpyChaseReady(_, action_runner):
|
| + # The background of the game canvas is set when the "Tap screen to play"
|
| + # caption is displayed.
|
| + action_runner.WaitForJavaScriptCondition(
|
| + 'document.querySelector("#game canvas").style.background !== ""')
|
| +
|
| +
|
| +def _WaitUntilFlickrReady(_, action_runner):
|
| + # Wait until the 'Recently tagged' view loads.
|
| + action_runner.WaitForJavaScriptCondition('''
|
| + document.querySelector(
|
| + '.search-photos-everyone-trending-view .photo-list-view') !== null''')
|
| +
|
| +
|
| +class _PageSpec(object):
|
| +
|
| + def __init__(self, name, url, login_hook=None, post_load_hook=None):
|
| + assert isinstance(url, (dict, str))
|
| + self._name = name
|
| + self._url = url
|
| + self._login_hook = login_hook
|
| + self._post_load_hook = post_load_hook
|
| +
|
| + @property
|
| + def name(self):
|
| + return self._name
|
| +
|
| + @property
|
| + def url(self):
|
| + return self._url
|
| +
|
| + @property
|
| + def login_hook(self):
|
| + return self._login_hook
|
| +
|
| + @property
|
| + def post_load_hook(self):
|
| + return self._post_load_hook
|
| +
|
| +
|
| +_SINGLE_PAGE_SPECS = {
|
| + # Search and e-commerce.
|
| + 'search': [
|
| + _PageSpec(
|
| + name='google',
|
| + url='https://www.google.com/#hl=en&q=science'),
|
| + _PageSpec(
|
| + name='baidu',
|
| + url='https://www.baidu.com/s?word=google'),
|
| + _PageSpec(
|
| + name='yahoo',
|
| + url='https://search.yahoo.com/search;_ylt=?p=google'),
|
| + _PageSpec(
|
| + name='amazon',
|
| + url='https://www.amazon.com/s/?field-keywords=nexus'),
|
| + # "ali_trackid" in the URL suppresses "Download app" interstitial.
|
| + _PageSpec(
|
| + name='taobao',
|
| + url={'desktop': 'https://world.taobao.com/',
|
| + 'mobile': 'http://m.intl.taobao.com/?ali_trackid'}),
|
| + _PageSpec(
|
| + name='yandex',
|
| + url='https://yandex.ru/touchsearch?text=science'),
|
| + # Redirects to the "http://" version.
|
| + _PageSpec(
|
| + name='ebay',
|
| + url='https://www.ebay.com/sch/i.html?_nkw=headphones'),
|
| + ],
|
| +
|
| + # Social networks.
|
| + 'social': [
|
| + # Using Facebook login often causes "404 Not Found" with WPR.
|
| + _PageSpec(
|
| + name='facebook',
|
| + url='https://www.facebook.com/rihanna'),
|
| + _PageSpec(
|
| + name='twitter',
|
| + url='https://www.twitter.com/justinbieber?skip_interstitial=true'),
|
| + # Due to the deterministic date injected by WPR (February 2008), the
|
| + # cookie set by https://vk.com immediately expires, so the page keeps
|
| + # refreshing indefinitely on mobile
|
| + # (see https://github.com/chromium/web-page-replay/issues/71).
|
| + _PageSpec(
|
| + name='vk',
|
| + url={'desktop': 'https://vk.com/sbeatles'}),
|
| + _PageSpec(
|
| + name='instagram',
|
| + url='https://www.instagram.com/selenagomez/'),
|
| + _PageSpec(
|
| + name='pinterest',
|
| + url='https://uk.pinterest.com/categories/popular/'),
|
| + # Redirects to the "http://" version.
|
| + _PageSpec(
|
| + name='tumblr',
|
| + url='https://50thousand.tumblr.com/'),
|
| + ],
|
| +
|
| + # News, discussion and knowledge portals and blogs.
|
| + 'news': [
|
| + # Redirects to the "http://" version.
|
| + _PageSpec(
|
| + name='bbc',
|
| + url='https://www.bbc.co.uk/news/world-asia-china-36189636'),
|
| + # Using "https://" shows "Your connection is not private".
|
| + _PageSpec(
|
| + name='cnn',
|
| + url=(
|
| + 'http://edition.cnn.com/2016/05/02/health/three-habitable-planets-earth-dwarf-star/index.html')),
|
| + _PageSpec(
|
| + name='reddit',
|
| + url={'desktop': (
|
| + 'https://www.reddit.com/r/AskReddit/comments/4hi90e/whats_your_best_wedding_horror_story/'),
|
| + 'mobile': (
|
| + 'https://m.reddit.com/r/AskReddit/comments/4hi90e/whats_your_best_wedding_horror_story/')}),
|
| + # Using "https://" hangs and shows "This site can't be reached".
|
| + _PageSpec(
|
| + name='qq',
|
| + url='http://news.qq.com/a/20160503/003186.htm'),
|
| + # Using "https://" leads to missing images and scripts on mobile (due
|
| + # to mixed content). The desktop page
|
| + # (http://news.sohu.com/20160503/n447433356.shtml) almost always fails
|
| + # to completely load due to
|
| + # https://github.com/chromium/web-page-replay/issues/74.
|
| + _PageSpec(
|
| + name='sohu',
|
| + url={'mobile': 'http://m.sohu.com/n/447433356/'}),
|
| + _PageSpec(
|
| + name='wikipedia',
|
| + url='https://en.wikipedia.org/wiki/Science'),
|
| + ],
|
| +
|
| + # Audio and video.
|
| + 'media': [
|
| + # No way to disable autoplay on desktop.
|
| + _PageSpec(
|
| + name='youtube',
|
| + url='https://www.youtube.com/watch?v=QGfhS1hfTWw&autoplay=false'),
|
| + # The side panel with related videos doesn't show on desktop due to
|
| + # https://github.com/chromium/web-page-replay/issues/74.
|
| + _PageSpec(
|
| + name='dailymotion',
|
| + url=(
|
| + 'https://www.dailymotion.com/video/x489k7d_street-performer-shows-off-slinky-skills_fun?autoplay=false')),
|
| + _PageSpec(
|
| + name='google_images',
|
| + url='https://www.google.co.uk/search?tbm=isch&q=love'),
|
| + # No way to disable autoplay on desktop. Album artwork doesn't load due
|
| + # to https://github.com/chromium/web-page-replay/issues/73.
|
| + _PageSpec(
|
| + name='soundcloud',
|
| + url='https://soundcloud.com/lifeofdesiigner/desiigner-panda'),
|
| + _PageSpec(
|
| + name='9gag',
|
| + url='https://www.9gag.com/'),
|
| + _PageSpec(
|
| + name='flickr',
|
| + url='https://www.flickr.com/photos/tags/farm',
|
| + post_load_hook=_WaitUntilFlickrReady),
|
| + ],
|
| +
|
| + # Online tools (documents, emails, storage, ...).
|
| + 'tools': [
|
| + _PageSpec(
|
| + name='docs',
|
| + url=(
|
| + 'https://docs.google.com/document/d/1GvzDP-tTLmJ0myRhUAfTYWs3ZUFilUICg8psNHyccwQ/edit?usp=sharing')),
|
| + _PageSpec(
|
| + name='gmail',
|
| + url='https://mail.google.com/mail/',
|
| + login_hook=_LogIntoGoogleAccountAndSetUpGmailSession,
|
| + post_load_hook=_CloseInboxInterstitialAndWaitUntilGmailReady),
|
| + _PageSpec(
|
| + name='maps',
|
| + url='https://www.google.com/maps/place/London,+UK/'),
|
| + _PageSpec(
|
| + name='stackoverflow',
|
| + url=(
|
| + 'https://stackoverflow.com/questions/36827659/compiling-an-application-for-use-in-highly-radioactive-environments')),
|
| + _PageSpec(
|
| + name='dropbox',
|
| + url='https://www.dropbox.com',
|
| + login_hook=_LogIntoDropboxAccount),
|
| + _PageSpec(
|
| + name='weather',
|
| + url='https://weather.com/en-GB/weather/today/l/USCA0286:1:US'),
|
| + _PageSpec(
|
| + name='drive',
|
| + url='https://drive.google.com/drive/my-drive',
|
| + login_hook=_LogIntoGoogleAccount),
|
| + ],
|
| +
|
| + # In-browser games (HTML5 and Flash).
|
| + 'games': [
|
| + _PageSpec(
|
| + name='bubbles',
|
| + url=(
|
| + 'https://games.cdn.famobi.com/html5games/s/smarty-bubbles/v010/?fg_domain=play.famobi.com&fg_uid=d8f24956-dc91-4902-9096-a46cb1353b6f&fg_pid=4638e320-4444-4514-81c4-d80a8c662371&fg_beat=620'),
|
| + post_load_hook=_WaitUntilBubblesReady),
|
| + # Using "https://" hangs and shows "This site can't be reached".
|
| + _PageSpec(
|
| + name='lazors',
|
| + url='http://www8.games.mobi/games/html5/lazors/lazors.html'),
|
| + # Using "https://" shows "Your connection is not private".
|
| + _PageSpec(
|
| + name='spychase',
|
| + url='http://playstar.mobi/games/spychase/index.php',
|
| + post_load_hook=_WaitUntilSpyChaseReady),
|
| + # Desktop only (requires Flash). Using "https://" causes
|
| + # "404 Not Found" during WPR recording.
|
| + _PageSpec(
|
| + name='miniclip',
|
| + url={'desktop': 'http://www.miniclip.com/games/en/'}),
|
| + # Desktop only (requires Flash).
|
| + _PageSpec(
|
| + name='alphabetty',
|
| + url={'desktop': 'https://king.com/play/alphabetty'}),
|
| + ],
|
| +}
|
| +
|
| +class MemoryTestPageSet(story.StorySet):
|
| +
|
| + PLATFORM = NotImplemented
|
| +
|
| + def __init__(self):
|
| + super(MemoryTestPageSet, self).__init__()
|
| +
|
| + param = os.getenv("MEMORY_TEST_PARAM")
|
| + name = os.getenv("MEMORY_TEST_NAME")
|
| +
|
| + for group_name, page_specs in iter(_SINGLE_PAGE_SPECS.items()):
|
| + if param is not None and group_name != param:
|
| + continue
|
| + for page_spec in page_specs:
|
| + if name is not None and page_spec.name != name:
|
| + continue
|
| + url = page_spec.url
|
| + if isinstance(url, dict):
|
| + url = url.get(self.PLATFORM)
|
| + if url is None:
|
| + continue # URL not supported on the platform.
|
| + self.AddStory(MemoryTestPage(self, group_name, url, page_spec))
|
|
|