1 # Copyright 2014 The Chromium 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.
5 from telemetry
import decorators
6 from telemetry
.internal
.results
import page_test_results
7 from telemetry
.page
import page
as page_module
8 from telemetry
.testing
import options_for_unittests
9 from telemetry
.testing
import page_test_test_case
10 from telemetry
.timeline
import model
as model_module
11 from telemetry
.timeline
import slice as slice_data
12 from telemetry
.util
import wpr_modes
14 from measurements
import task_execution_time
17 class TestTaskExecutionTimePage(page_module
.Page
):
19 def __init__(self
, page_set
, base_dir
):
20 super(TestTaskExecutionTimePage
, self
).__init
__(
21 'file://blank.html', page_set
, base_dir
)
23 def RunPageInteractions(self
, action_runner
):
24 with action_runner
.CreateGestureInteraction('ScrollAction'):
25 action_runner
.ScrollPage()
28 # Disable for accessing private API of task_execution_time.
29 # pylint: disable=protected-access
30 class TaskExecutionTimeUnitTest(page_test_test_case
.PageTestTestCase
):
33 self
._options
= options_for_unittests
.GetCopy()
34 self
._options
.browser_options
.wpr_mode
= wpr_modes
.WPR_OFF
35 self
._first
_thread
_name
= (
36 task_execution_time
.TaskExecutionTime
._RENDERER
_THREADS
[0])
37 self
._measurement
= None
40 @decorators.Enabled('android')
41 def testSomeResultsReturnedFromDummyPage(self
):
42 self
._GenerateDataForEmptyPageSet
()
44 results
= self
.RunMeasurement(self
._measurement
,
46 options
=self
._options
)
48 self
.assertGreater(len(results
.all_page_specific_values
), 0)
50 # http://crbug.com/466994
52 def testSlicesConformToRequiredNamingConventionsUsingDummyPage(self
):
53 """This test ensures the presence of required keywords.
55 Some arbitrary keywords are required to generate the names of the top 10
56 tasks. The code has a weak dependancy on 'src_func', 'class' and 'line'
57 existing; if they exist in a slice's args they are used to generate a
58 name, if they don't exists the code falls back to using the name of the
59 slice, which is less clear.
61 If the code has been refactored and these keywords no longer exist
62 the code that relies on them in task_execution_time.py should be
63 updated to use the appropriate technique for assertaining this data
64 (and this test changed in the same way).
66 self
._GenerateDataForEmptyPageSet
()
68 self
.RunMeasurement(self
._measurement
,
70 options
=self
._options
)
72 required_keywords
= {'src_func': 0, 'class': 0, 'line': 0}
74 # Check all slices and count the uses of the required keywords.
75 for thread
in self
._measurement
._renderer
_process
.threads
.itervalues():
76 for slice_info
in thread
.IterAllSlices():
77 _CheckSliceForKeywords(slice_info
, required_keywords
)
79 # Confirm that all required keywords have at least one instance.
80 for use_counts
in required_keywords
.itervalues():
81 self
.assertGreater(use_counts
, 0)
83 def testMockedResultsCorrectlyReturned(self
):
84 data
= self
._GenerateResultsFromMockedData
()
86 # Confirm we get back 4 results (3 tasks and a section-use %).
87 self
.assertEqual(len(data
.results
.all_page_specific_values
), 4)
89 # Check that the 3 tasks we added exist in the resulting output
91 task_prefix
= 'process 1:%s:' % (self
._first
_thread
_name
)
92 slow_result
= self
._findResultFromName
(task_prefix
+ 'slow', data
)
93 self
.assertEqual(slow_result
.value
, 1000)
95 medium_result
= self
._findResultFromName
(task_prefix
+ 'medium', data
)
96 self
.assertEqual(medium_result
.value
, 500)
98 fast_result
= self
._findResultFromName
(task_prefix
+ 'fast', data
)
99 self
.assertEqual(fast_result
.value
, 1)
101 def testNonIdlePercentagesAreCorrect(self
):
102 data
= self
._GenerateResultsFromMockedData
()
104 # Confirm that 100% of tasks are in the normal section.
105 percentage_result
= self
._findResultFromName
(
106 'process 1:%s:Section_%s' % (
107 self
._first
_thread
_name
,
108 task_execution_time
.TaskExecutionTime
.NORMAL_SECTION
),
110 self
.assertEqual(percentage_result
.value
, 100)
112 def testIdleTasksAreReported(self
):
113 data
= self
._GenerateResultsFromMockedIdleData
()
115 # The 'slow_sub_slice' should be inside the Idle section and therefore
116 # removed from the results.
117 for result
in data
.results
.all_page_specific_values
:
118 if 'slow_sub_slice' in result
.name
:
119 self
.fail('Tasks within idle section should not be reported')
121 # The 'not_idle' slice should not have the IDLE_SECTION added to its name
123 for result
in data
.results
.all_page_specific_values
:
124 if 'not_idle' in result
.name
:
126 task_execution_time
.TaskExecutionTime
.IDLE_SECTION
130 self
.fail('Task was incorrectly marked as Idle')
132 def testIdlePercentagesAreCorrect(self
):
133 data
= self
._GenerateResultsFromMockedIdleData
()
135 # Check the percentage section usage is correctly calculated.
136 # Total = 1000 (idle) + 250 (normal), so normal = (250 * 100) / 1250 = 20%.
137 normal_percentage_result
= self
._findResultFromName
(
138 'process 1:%s:Section_%s' % (
139 self
._first
_thread
_name
,
140 task_execution_time
.TaskExecutionTime
.NORMAL_SECTION
),
142 self
.assertEqual(normal_percentage_result
.value
, 20)
143 # Check the percentage section usage is correctly calculated.
144 idle_percentage_result
= self
._findResultFromName
(
145 'process 1:%s:Section_%s' % (
146 self
._first
_thread
_name
,
147 task_execution_time
.TaskExecutionTime
.IDLE_SECTION
),
149 self
.assertEqual(idle_percentage_result
.value
, 80)
151 def testTopNTasksAreCorrectlyReported(self
):
152 data
= self
._GenerateDataForEmptyPageSet
()
154 # Add too many increasing-durtation tasks and confirm we only get the
155 # slowest _NUMBER_OF_RESULTS_TO_DISPLAY tasks reported back.
158 for duration
in xrange(
159 task_execution_time
.TaskExecutionTime
._NUMBER
_OF
_RESULTS
_TO
_DISPLAY
+
161 data
.AddSlice('task' + str(duration
), 0, duration
)
163 # Run the code we are testing.
164 self
._measurement
._AddResults
(data
.results
)
166 # Check that the last (i.e. biggest) _NUMBER_OF_RESULTS_TO_DISPLAY get
167 # returned in the results.
168 for duration
in xrange(
171 task_execution_time
.TaskExecutionTime
._NUMBER
_OF
_RESULTS
_TO
_DISPLAY
):
172 self
._findResultFromName
(
173 'process 1:%s:task%s' % (self
._first
_thread
_name
, str(duration
)),
176 def _findResultFromName(self
, name
, data
):
177 for result
in data
.results
.all_page_specific_values
:
178 if result
.name
== name
:
180 self
.fail('Expected result "%s" missing.' % (name
))
182 def _GenerateResultsFromMockedData(self
):
183 data
= self
._GenerateDataForEmptyPageSet
()
185 data
.AddSlice('fast', 0, 1)
186 data
.AddSlice('medium', 0, 500)
187 data
.AddSlice('slow', 0, 1000)
189 # Run the code we are testing and return results.
190 self
._measurement
._AddResults
(data
.results
)
193 def _GenerateResultsFromMockedIdleData(self
):
194 data
= self
._GenerateDataForEmptyPageSet
()
196 # Make a slice that looks like an idle task parent.
198 slow_slice_duration
= 1000
199 fast_slice_duration
= 250
200 parent_slice
= data
.AddSlice(
201 task_execution_time
.TaskExecutionTime
.IDLE_SECTION_TRIGGER
,
204 # Add a sub-slice, this should be reported back as occuring in idle time.
205 sub_slice
= slice_data
.Slice(
214 parent_slice
.sub_slices
.append(sub_slice
)
216 # Add a non-idle task.
217 data
.AddSlice('not_idle', slice_start_time
, fast_slice_duration
)
219 # Run the code we are testing.
220 self
._measurement
._AddResults
(data
.results
)
224 def _GenerateDataForEmptyPageSet(self
):
225 self
._measurement
= task_execution_time
.TaskExecutionTime()
226 self
._page
_set
= self
.CreateEmptyPageSet()
227 page
= TestTaskExecutionTimePage(self
._page
_set
, self
._page
_set
.base_dir
)
228 self
._page
_set
.AddStory(page
)
230 # Get the name of a thread used by task_execution_time metric and set up
231 # some dummy execution data pretending to be from that thread & process.
232 data
= TaskExecutionTestData(self
._first
_thread
_name
)
233 self
._measurement
._renderer
_process
= data
._renderer
_process
235 # Pretend we are about to run the tests to silence lower level asserts.
236 data
.results
.WillRunPage(page
)
241 def _CheckSliceForKeywords(slice_info
, required_keywords
):
242 for argument
in slice_info
.args
:
243 if argument
in required_keywords
:
244 required_keywords
[argument
] += 1
245 # recurse into our sub-slices.
246 for sub_slice
in slice_info
.sub_slices
:
247 _CheckSliceForKeywords(sub_slice
, required_keywords
)
250 class TaskExecutionTestData(object):
252 def __init__(self
, thread_name
):
253 self
._model
= model_module
.TimelineModel()
254 self
._renderer
_process
= self
._model
.GetOrCreateProcess(1)
255 self
._renderer
_thread
= self
._renderer
_process
.GetOrCreateThread(2)
256 self
._renderer
_thread
.name
= thread_name
257 self
._results
= page_test_results
.PageTestResults()
263 def AddSlice(self
, name
, timestamp
, duration
):
264 new_slice
= slice_data
.Slice(
273 self
._renderer
_thread
.all_slices
.append(new_slice
)