# Copyright 2014 The Chromium OS 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 import random import time from autotest_lib.client.bin import test from autotest_lib.client.common_lib.cros import chrome from autotest_lib.client.common_lib import error from autotest_lib.client.cros.audio import cras_utils _STRESS_ITERATIONS = 100 # Total number of iterations. _MAX_OPENED_TAB = 20 # Max number of tabs open and playing audio. _RETAIN_TAB = 5 # In case we hit the _MAX_OPENED_TAB limit, # close all except last 5 tabs. _MAX_TABS_TO_OPEN = 10 # Max number of tabs can be opened in one iteration. _CRASH_PATH = '/var/spool/crash' class audio_ActiveStreamStress(test.test): """Verifies the active audio streams.""" version = 1 _active_stream_count = 0 _existing_cras_reports = [] _cr = None # TODO(rohitbm): add more(including video) file types and download them from gs://. _streams = ('audio.mp3', 'audio.wav', 'audio.m4a') _stream_index = 0 _tab_count = 0 def run_once(self): # Collect existing cras crash reports. self._existing_cras_reports = self.collect_cras_crash() with chrome.Chrome(init_network_controller=True) as self._cr: self._cr.browser.platform.SetHTTPServerDirectories(self.bindir) self.push_new_stream(self._cr.browser.tabs[0]) # TODO(rohitbm): decide whether to perform verification on each # open/close or at end of the iteration. self.verify_active_streams() push_count = 0 pop_count = 0 # Stress test logic: # Test runs for n number of iterations. For one iteration, # a = random(10) tabs(streams) are created and # b = random(a) tabs are closed. If the next iteration finds that, # total number of opened tabs are more than _MAX_OPENED_TAB, # test will close (total opened tabs - 5) tabs. # This will balance number of opened tabs and will allow to close # tabs in a control manner. for count in xrange(1, _STRESS_ITERATIONS): if self._tab_count > _MAX_OPENED_TAB: for i in xrange(1, (self._tab_count - _RETAIN_TAB)): pop_count += 1 self.pop_stream() logging.info('Total streams closed: %d', pop_count) random_tab = random.randint(1, 10) for i in xrange(1, random_tab): push_count += 1 self.push_new_stream(self._cr.browser.tabs.New()) logging.info('Total new streams created: %d', push_count) time.sleep(5) # Delay for active streams to play. for i in xrange(1, random.randint(1, random_tab)): pop_count += 1 self.pop_stream() logging.info('Total streams closed: %d', pop_count) def get_stream_index(self): if self._stream_index == len(self._streams): # Reset the stream index if the index reached to the end. self._stream_index = 0 return self._stream_index def push_new_stream(self, tab): """Starts next audio stream from self._streams list. @param tab: tab to open an audio stream. """ self._tab_count += 1 tab.Navigate(self._cr.browser.platform.http_server.UrlOf( os.path.join(self.bindir, self._streams[self.get_stream_index()]))) tab.ExecuteJavaScript( "document.getElementsByTagName('video')[0].loop=true") # TODO(rohitbm): add playback verification. self._stream_index += 1 self._active_stream_count += 1 time.sleep(1) # Adding a delay so cras can update the active count. self.verify_active_streams() def pop_stream(self): """Turns off the first available stream by closing the first tab.""" if len(self._cr.browser.tabs) > 0: self._cr.browser.tabs[0].Close() self._tab_count -= 1 self._active_stream_count -= 1 time.sleep(1) # Adding delay so cras can update the active count. self.verify_active_streams() def verify_active_streams(self): """Verifies test active audio streams with cras active streams.""" cras_stream_count = cras_utils.get_active_stream_count() if self._active_stream_count != cras_stream_count: cras_crash_reports = self.collect_cras_crash() new_reports = list(set(cras_crash_reports) - set(self._existing_cras_reports)) error_msg = ('Active stream count: %d is not matching with ' 'cras active stream count: %d. ' 'Number of cras crashes %d : %s' % (self._active_stream_count, cras_stream_count, len(new_reports), new_reports)) raise error.TestError(error_msg) def collect_cras_crash(self): """Check for cras crashes. @return a list of cras crash reports found. """ crash_reports = [] if not os.path.isdir(_CRASH_PATH): logging.debug('No cras crash detected!') else: cras_reports = os.listdir(_CRASH_PATH) crash_reports = [report for report in cras_reports if report.startswith('cras')] return crash_reports