# Copyright (c) 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 datetime, logging, os, time, shutil
from autotest_lib.client.bin import test, utils
from autotest_lib.client.common_lib import error
from autotest_lib.client.common_lib import file_utils
from autotest_lib.client.common_lib import sequence_utils
from autotest_lib.client.common_lib.cros import chrome
from autotest_lib.client.cros import constants as cros_constants
from autotest_lib.client.cros.chameleon import chameleon
from autotest_lib.client.cros.chameleon import chameleon_port_finder
from autotest_lib.client.cros.chameleon import chameleon_video_capturer
from autotest_lib.client.cros.image_comparison import publisher
from autotest_lib.client.cros.video import constants
from autotest_lib.client.cros.video import frame_checksum_utils
from autotest_lib.client.cros.video import native_html5_player
from autotest_lib.client.cros.video import helper_logger
from autotest_lib.client.cros.multimedia import local_facade_factory
class video_GlitchDetection(test.test):
"""
Video playback test using image comparison.
Captures frames using chameleon and compares them to known golden frames.
If frames don't match, upload to GS for viewing later.
"""
version = 2
@helper_logger.video_log_wrapper
def run_once(self, source_path, codec, resolution, host, args,
collect_only = False):
board = utils.get_current_board()
shutil.copy2(constants.VIDEO_HTML_FILEPATH, self.bindir)
file_utils.make_leaf_dir(constants.TEST_DIR)
with chrome.Chrome(
extra_browser_args=helper_logger.chrome_vmodule_flag(),
extension_paths = [cros_constants.DISPLAY_TEST_EXTENSION],
autotest_ext=True,
init_network_controller=True) as cr:
cr.browser.platform.SetHTTPServerDirectories(self.bindir)
html_fullpath = os.path.join(self.bindir, 'video.html')
player = native_html5_player.NativeHtml5Player(
tab = cr.browser.tabs[0],
full_url = cr.browser.platform.http_server.UrlOf(html_fullpath),
video_id = 'video',
video_src_path = source_path,
event_timeout = 120)
chameleon_board = chameleon.create_chameleon_board(host.hostname,
args)
display_facade = local_facade_factory.LocalFacadeFactory(
cr).create_display_facade()
finder = chameleon_port_finder.ChameleonVideoInputFinder(
chameleon_board, display_facade)
ports = finder.find_all_ports().connected
logging.debug(finder)
if not ports:
raise error.TestError(finder)
capturer = chameleon_video_capturer.ChameleonVideoCapturer(
ports[0], display_facade)
with capturer:
player.load_video()
player.verify_video_can_play()
display_facade.move_to_display(
display_facade.get_first_external_display_id())
display_facade.set_fullscreen(True)
# HACK: Unset and reset fullscreen. There is a bug in Chrome
# that fails to move the window to a correct position.
# Resetting fullscren helps, check http://crbug.com/574284
display_facade.set_fullscreen(False)
display_facade.set_fullscreen(True)
time.sleep(5)
box = (0, 0, constants.DESIRED_WIDTH, constants.DESIRED_HEIGHT)
#TODO: mussa, Revisit once crbug/580736 is fixed
for n in xrange(constants.NUM_CAPTURE_TRIES):
logging.debug('Trying to capture frames. TRY #%d', n + 1)
raw_test_checksums = capturer.capture_only(
player, max_frame_count = constants.FCOUNT,
box = box)
raw_test_checksums = [tuple(checksum) for checksum in
raw_test_checksums]
overreach_counts = self.overreach_frame_counts(
raw_test_checksums,
constants.MAX_FRAME_REPEAT_COUNT)
if not overreach_counts: # no checksums exceeded threshold
break
player.pause()
player.seek_to(datetime.timedelta(seconds=0))
else:
msg = ('Framecount overreach detected even after %d '
'tries. Checksums: %s' % (constants.NUM_CAPTURE_TRIES,
overreach_counts))
raise error.TestFail(msg)
# produces unique checksums mapped to their occur. indices
test_checksum_indices = frame_checksum_utils.checksum_indices(
raw_test_checksums)
test_checksums = test_checksum_indices.keys()
test_indices = test_checksum_indices.values()
golden_checksums_filepath = os.path.join(
constants.TEST_DIR,
constants.GOLDEN_CHECKSUMS_FILENAME)
if collect_only:
capturer.write_images(test_indices, constants.TEST_DIR,
constants.IMAGE_FORMAT)
logging.debug("Write golden checksum file to %s",
golden_checksums_filepath)
with open(golden_checksums_filepath, "w+") as f:
for checksum in test_checksums:
f.write(' '.join([str(i) for i in checksum]) + '\n')
return
golden_checksums_remote_filepath = os.path.join(
constants.GOLDEN_CHECKSUM_REMOTE_BASE_DIR,
board,
codec + '_' + resolution,
constants.GOLDEN_CHECKSUMS_FILENAME)
file_utils.download_file(golden_checksums_remote_filepath,
golden_checksums_filepath)
golden_checksums = self.read_checksum_file(
golden_checksums_filepath)
golden_checksum_count = len(golden_checksums)
test_checksum_count = len(test_checksums)
eps = constants.MAX_DIFF_TOTAL_FCOUNT
if golden_checksum_count - test_checksum_count > eps:
msg = ('Expecting about %d checksums, received %d. '
'Allowed delta is %d') % (
golden_checksum_count,
test_checksum_count,
eps)
raise error.TestFail(msg)
# Some frames might be missing during either golden frame
# collection or during a test run. Using LCS ensures we
# ignore a few missing frames while comparing test vs golden
lcs_len = sequence_utils.lcs_length(golden_checksums,
test_checksums)
missing_frames_count = len(golden_checksums) - lcs_len
unknown_frames_count = len(test_checksums) - lcs_len
msg = ('# of matching frames : %d. # of missing frames : %d. '
'# of unknown test frames : %d. Max allowed # of '
'missing frames : %d. # of golden frames : %d. # of '
'test_checksums : %d' %(lcs_len, missing_frames_count,
unknown_frames_count,
constants.MAX_NONMATCHING_FCOUNT,
len(golden_checksums),
len(test_checksums)))
logging.debug(msg)
if (missing_frames_count + unknown_frames_count >
constants.MAX_NONMATCHING_FCOUNT):
unknown_frames = set(test_checksums) - set(golden_checksums)
store_indices = [test_checksum_indices[c] for c in
unknown_frames]
paths = capturer.write_images(store_indices,
constants.TEST_DIR,
constants.IMAGE_FORMAT)
path_publish = publisher.ImageDiffPublisher(self.resultsdir)
path_publish.publish_paths(paths, self.tagged_testname)
raise error.TestFail("Too many non-matching frames")
def overreach_frame_counts(self, checksums, max_frame_repeat_count):
"""
Checks that captured frames have not exceed the max repeat count.
@param checksums: list of frame checksums received from chameleon.
@param max_frame_repeat_count: int. max allowed count.
@return : dictionary, checksums and their counts
"""
logging.debug("Verify no frame is repeated more than %d",
max_frame_repeat_count)
counts = frame_checksum_utils.checksum_counts(checksums)
overreach_counts = {}
for k, v in counts.iteritems():
logging.debug("%s : %s", k, v)
if v > max_frame_repeat_count:
overreach_counts[k] = v
return overreach_counts
def read_checksum_file(self, path):
"""
Reads the golden checksum file. Each line in file has the format
w x y z where w x y z is a chameleon frame checksum
@param path: complete path to the golden checksum file.
@returns a list of 4-tuples (w x y z).
"""
checksums = []
with open(path) as f:
for line in f:
checksum = tuple(int(val) for val in line.split())
checksums.append(checksum)
return checksums