1# Copyright 2015 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
5
6import datetime, time
7
8from autotest_lib.client.bin import utils
9from autotest_lib.client.common_lib import error
10from autotest_lib.client.cros.video import method_logger
11
12
13class VideoPlayer(object):
14    """
15    Provides interface to interact with and control video playback via js.
16
17    Specific players such as VimeoPlayer will inherit from this class and
18    customize its behavior.
19
20    """
21
22
23    def __init__(self, tab, full_url, video_id, video_src_path='',
24                 event_timeout=5, polling_wait_time=1):
25        """
26        @param tab: object, tab to interact with the tab in the browser.
27        @param full_url: string, full url pointing to html file to load.
28        @param video_src_path: path, complete path to video used for test.
29        @param video_id: string, name of the video_id element in the html file.
30        @param time_out_for_events_s: integer, how long to wait for an event
31                                      before timing out
32        @param time_btwn_polling_s: integer, how long to wait between one call
33                                    to check a condition and the next.
34
35        """
36        self.tab = tab
37        self.full_url = full_url
38        self.video_id = video_id
39        self.video_src_path = video_src_path
40        self.event_timeout = event_timeout
41        self.polling_wait_time = polling_wait_time
42        self.tab.Navigate(self.full_url)
43
44
45    @method_logger.log
46    def load_video(self, wait_for_canplay=True):
47        """
48        Loads video into browser.
49        @param wait_for_canplay: video will be verified before play
50
51        """
52        self.tab.WaitForDocumentReadyStateToBeComplete()
53        self.wait_for_script_ready()
54        time.sleep(2)
55        self.inject_source_file()
56        if wait_for_canplay:
57             self.wait_for_video_ready()
58
59
60    def inject_source_file(self):
61        """
62        Injects source file into html file if needed.
63
64        Created for subclasses that need it
65
66        """
67        pass
68
69
70    @method_logger.log
71    def wait_for_script_ready(self):
72        """
73        Wait for Javascript variables and functions to be defined.
74
75        """
76        exception_msg = 'Script did not ready in time.'
77
78        self._wait_for_event(self.is_javascript_ready, exception_msg)
79
80
81    @method_logger.log
82    def wait_for_video_ready(self):
83        """
84        Waits for video to signal that is ready.
85
86        Each class that inherits from this will define its is_video_ready
87        function.
88
89        """
90        exception_msg = 'Video did not signal ready in time.'
91
92        self._wait_for_event(self.is_video_ready, exception_msg)
93
94
95    @method_logger.log
96    def verify_video_can_play(self, duration=0):
97        """
98        Plays video and ensures that reported video current time is > 0.
99
100        @param duration: duration to play a video
101        @raises: error.TestError if current time is not > 0 after time > 0s
102
103        """
104        exception_msg = 'Expected current time >%ds.' %duration
105
106        self.play()
107
108        # check that video is playing
109        self._wait_for_event(
110                  lambda : self.currentTime() > duration, exception_msg)
111
112        self.pause()
113
114        # seek back to the beginning of video
115        self.seek_to(datetime.timedelta(seconds=0))
116
117
118    @method_logger.log
119    def seek_to(self, timestamp):
120        """
121        Uses javascript to set currentTime property of video to desired time.
122
123        @param timestamp: timedelta, instance of time to navigate video to.
124
125        """
126        self.seek_to(timestamp)
127
128
129    @method_logger.log
130    def wait_for_video_to_seek(self):
131        """
132        Waits for video's currentTime to equal the time it was seeked to.
133
134        """
135        exception_msg = 'Video did not complete seeking in time.'
136
137        self._wait_for_event(self.has_video_finished_seeking, exception_msg)
138
139        # it usually takes a little while before new frame renders, so wait
140        time.sleep(1)
141
142
143    @method_logger.log
144    def _wait_for_event(self, predicate_function, exception_msg):
145        """
146         Helper method to wait for a desired condition.
147
148         @param predicate_function: object, function which returns true when
149                                    desired condition is achieved.
150         @param exception_msg: string, an exception message to show when desired
151                               condition is not achieved in allowed time.
152
153        """
154        fullmsg = exception_msg + ' Waited for %ss' % self.event_timeout
155
156        utils.poll_for_condition(predicate_function,
157                                 error.TestError(fullmsg),
158                                 self.event_timeout,
159                                 self.polling_wait_time)
160