VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testmanager/webui/wuimain.py@ 106061

Last change on this file since 106061 was 106061, checked in by vboxsync, 4 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 72.1 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: wuimain.py 106061 2024-09-16 14:03:52Z vboxsync $
3
4"""
5Test Manager Core - WUI - The Main page.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2024 Oracle and/or its affiliates.
11
12This file is part of VirtualBox base platform packages, as
13available from https://www.virtualbox.org.
14
15This program is free software; you can redistribute it and/or
16modify it under the terms of the GNU General Public License
17as published by the Free Software Foundation, in version 3 of the
18License.
19
20This program is distributed in the hope that it will be useful, but
21WITHOUT ANY WARRANTY; without even the implied warranty of
22MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
23General Public License for more details.
24
25You should have received a copy of the GNU General Public License
26along with this program; if not, see <https://www.gnu.org/licenses>.
27
28The contents of this file may alternatively be used under the terms
29of the Common Development and Distribution License Version 1.0
30(CDDL), a copy of it is provided in the "COPYING.CDDL" file included
31in the VirtualBox distribution, in which case the provisions of the
32CDDL are applicable instead of those of the GPL.
33
34You may elect to license modified versions of this file under the
35terms and conditions of either the GPL or the CDDL or both.
36
37SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
38"""
39__version__ = "$Revision: 106061 $"
40
41# Standard Python imports.
42
43# Validation Kit imports.
44from testmanager import config;
45from testmanager.core.base import TMExceptionBase, TMTooManyRows;
46from testmanager.webui.wuibase import WuiDispatcherBase, WuiException;
47from testmanager.webui.wuicontentbase import WuiTmLink;
48from common import webutils, utils;
49
50
51
52class WuiMain(WuiDispatcherBase):
53 """
54 WUI Main page.
55
56 Note! All cylic dependency avoiance stuff goes here in the dispatcher code,
57 not in the action specific code. This keeps the uglyness in one place
58 and reduces load time dependencies in the more critical code path.
59 """
60
61 ## The name of the script.
62 ksScriptName = 'index.py'
63
64 ## @name Actions
65 ## @{
66 ksActionResultsUnGrouped = 'ResultsUnGrouped'
67 ksActionResultsGroupedBySchedGroup = 'ResultsGroupedBySchedGroup'
68 ksActionResultsGroupedByTestGroup = 'ResultsGroupedByTestGroup'
69 ksActionResultsGroupedByBuildRev = 'ResultsGroupedByBuildRev'
70 ksActionResultsGroupedByBuildCat = 'ResultsGroupedByBuildCat'
71 ksActionResultsGroupedByTestBox = 'ResultsGroupedByTestBox'
72 ksActionResultsGroupedByTestCase = 'ResultsGroupedByTestCase'
73 ksActionResultsGroupedByOS = 'ResultsGroupedByOS'
74 ksActionResultsGroupedByArch = 'ResultsGroupedByArch'
75 ksActionTestSetDetails = 'TestSetDetails';
76 ksActionTestResultDetails = ksActionTestSetDetails;
77 ksActionTestSetDetailsFromResult = 'TestSetDetailsFromResult'
78 ksActionTestResultFailureDetails = 'TestResultFailureDetails'
79 ksActionTestResultFailureAdd = 'TestResultFailureAdd'
80 ksActionTestResultFailureAddPost = 'TestResultFailureAddPost'
81 ksActionTestResultFailureEdit = 'TestResultFailureEdit'
82 ksActionTestResultFailureEditPost = 'TestResultFailureEditPost'
83 ksActionTestResultFailureDoRemove = 'TestResultFailureDoRemove'
84 ksActionViewLog = 'ViewLog'
85 ksActionGetFile = 'GetFile'
86 ksActionReportSummary = 'ReportSummary';
87 ksActionReportRate = 'ReportRate';
88 ksActionReportTestCaseFailures = 'ReportTestCaseFailures';
89 ksActionReportTestBoxFailures = 'ReportTestBoxFailures';
90 ksActionReportFailureReasons = 'ReportFailureReasons';
91 ksActionGraphWiz = 'GraphWiz';
92 ksActionVcsHistoryTooltip = 'VcsHistoryTooltip'; ##< Hardcoded in common.js.
93 ## @}
94
95 ## @name Standard report parameters
96 ## @{
97 ksParamReportPeriods = 'cPeriods';
98 ksParamReportPeriodInHours = 'cHoursPerPeriod';
99 ksParamReportSubject = 'sSubject';
100 ksParamReportSubjectIds = 'SubjectIds';
101 ## @}
102
103 ## @name Graph Wizard parameters
104 ## Common parameters: ksParamReportPeriods, ksParamReportPeriodInHours, ksParamReportSubjectIds,
105 ## ksParamReportSubject, ksParamEffectivePeriod, and ksParamEffectiveDate.
106 ## @{
107 ksParamGraphWizTestBoxIds = 'aidTestBoxes';
108 ksParamGraphWizBuildCatIds = 'aidBuildCats';
109 ksParamGraphWizTestCaseIds = 'aidTestCases';
110 ksParamGraphWizSepTestVars = 'fSepTestVars';
111 ksParamGraphWizImpl = 'enmImpl';
112 ksParamGraphWizWidth = 'cx';
113 ksParamGraphWizHeight = 'cy';
114 ksParamGraphWizDpi = 'dpi';
115 ksParamGraphWizFontSize = 'cPtFont';
116 ksParamGraphWizErrorBarY = 'fErrorBarY';
117 ksParamGraphWizMaxErrorBarY = 'cMaxErrorBarY';
118 ksParamGraphWizMaxPerGraph = 'cMaxPerGraph';
119 ksParamGraphWizXkcdStyle = 'fXkcdStyle';
120 ksParamGraphWizTabular = 'fTabular';
121 ksParamGraphWizSrcTestSetId = 'idSrcTestSet';
122 ## @}
123
124 ## @name Graph implementations values for ksParamGraphWizImpl.
125 ## @{
126 ksGraphWizImpl_Default = 'default';
127 ksGraphWizImpl_Matplotlib = 'matplotlib';
128 ksGraphWizImpl_Charts = 'charts';
129 kasGraphWizImplValid = [ ksGraphWizImpl_Default, ksGraphWizImpl_Matplotlib, ksGraphWizImpl_Charts];
130 kaasGraphWizImplCombo = [
131 ( ksGraphWizImpl_Default, 'Default' ),
132 ( ksGraphWizImpl_Matplotlib, 'Matplotlib (server)' ),
133 ( ksGraphWizImpl_Charts, 'Google Charts (client)'),
134 ];
135 ## @}
136
137 ## @name Log Viewer parameters.
138 ## @{
139 ksParamLogSetId = 'LogViewer_idTestSet';
140 ksParamLogFileId = 'LogViewer_idFile';
141 ksParamLogChunkSize = 'LogViewer_cbChunk';
142 ksParamLogChunkNo = 'LogViewer_iChunk';
143 ## @}
144
145 ## @name File getter parameters.
146 ## @{
147 ksParamGetFileSetId = 'GetFile_idTestSet';
148 ksParamGetFileId = 'GetFile_idFile';
149 ksParamGetFileDownloadIt = 'GetFile_fDownloadIt';
150 ## @}
151
152 ## @name VCS history parameters.
153 ## @{
154 ksParamVcsHistoryRepository = 'repo';
155 ksParamVcsHistoryRevision = 'rev';
156 ksParamVcsHistoryEntries = 'cEntries';
157 ## @}
158
159 ## @name Test result listing parameters.
160 ## @{
161 ## If this param is specified, then show only results for this member when results grouped by some parameter.
162 ksParamGroupMemberId = 'GroupMemberId'
163 ## Optional parameter for indicating whether to restrict the listing to failures only.
164 ksParamOnlyFailures = 'OnlyFailures';
165 ## The sheriff parameter for getting failures needing a reason or two assigned to them.
166 ksParamOnlyNeedingReason = 'OnlyNeedingReason';
167 ## Result listing sorting.
168 ksParamTestResultsSortBy = 'enmSortBy'
169 ## @}
170
171 ## Effective time period. one of the first column values in kaoResultPeriods.
172 ksParamEffectivePeriod = 'sEffectivePeriod'
173
174 ## Test result period values.
175 kaoResultPeriods = [
176 ( '1 hour', '1 hour', 1 ),
177 ( '2 hours', '2 hours', 2 ),
178 ( '3 hours', '3 hours', 3 ),
179 ( '6 hours', '6 hours', 6 ),
180 ( '12 hours', '12 hours', 12 ),
181
182 ( '1 day', '1 day', 24 ),
183 ( '2 days', '2 days', 48 ),
184 ( '3 days', '3 days', 72 ),
185
186 ( '1 week', '1 week', 168 ),
187 ( '2 weeks', '2 weeks', 336 ),
188 ( '3 weeks', '3 weeks', 504 ),
189
190 ( '1 month', '1 month', 31 * 24 ), # The approx hour count varies with the start date.
191 ( '2 months', '2 months', (31 + 31) * 24 ), # Using maximum values.
192 ( '3 months', '3 months', (31 + 30 + 31) * 24 ),
193
194 ( '6 months', '6 months', (31 + 31 + 30 + 31 + 30 + 31) * 24 ),
195
196 ( '1 year', '1 year', 365 * 24 ),
197 ];
198 ## The default test result period.
199 ksResultPeriodDefault = '6 hours';
200
201
202
203 def __init__(self, oSrvGlue):
204 WuiDispatcherBase.__init__(self, oSrvGlue, self.ksScriptName);
205 self._sTemplate = 'template.html'
206
207 #
208 # Populate the action dispatcher dictionary.
209 # Lambda is forbidden because of readability, speed and reducing number of imports.
210 #
211 self._dDispatch[self.ksActionResultsUnGrouped] = self._actionResultsUnGrouped;
212 self._dDispatch[self.ksActionResultsGroupedByTestGroup] = self._actionResultsGroupedByTestGroup;
213 self._dDispatch[self.ksActionResultsGroupedByBuildRev] = self._actionResultsGroupedByBuildRev;
214 self._dDispatch[self.ksActionResultsGroupedByBuildCat] = self._actionResultsGroupedByBuildCat;
215 self._dDispatch[self.ksActionResultsGroupedByTestBox] = self._actionResultsGroupedByTestBox;
216 self._dDispatch[self.ksActionResultsGroupedByTestCase] = self._actionResultsGroupedByTestCase;
217 self._dDispatch[self.ksActionResultsGroupedByOS] = self._actionResultsGroupedByOS;
218 self._dDispatch[self.ksActionResultsGroupedByArch] = self._actionResultsGroupedByArch;
219 self._dDispatch[self.ksActionResultsGroupedBySchedGroup] = self._actionResultsGroupedBySchedGroup;
220
221 self._dDispatch[self.ksActionTestSetDetails] = self._actionTestSetDetails;
222 self._dDispatch[self.ksActionTestSetDetailsFromResult] = self._actionTestSetDetailsFromResult;
223
224 self._dDispatch[self.ksActionTestResultFailureAdd] = self._actionTestResultFailureAdd;
225 self._dDispatch[self.ksActionTestResultFailureAddPost] = self._actionTestResultFailureAddPost;
226 self._dDispatch[self.ksActionTestResultFailureDetails] = self._actionTestResultFailureDetails;
227 self._dDispatch[self.ksActionTestResultFailureDoRemove] = self._actionTestResultFailureDoRemove;
228 self._dDispatch[self.ksActionTestResultFailureEdit] = self._actionTestResultFailureEdit;
229 self._dDispatch[self.ksActionTestResultFailureEditPost] = self._actionTestResultFailureEditPost;
230
231 self._dDispatch[self.ksActionViewLog] = self._actionViewLog;
232 self._dDispatch[self.ksActionGetFile] = self._actionGetFile;
233
234 self._dDispatch[self.ksActionReportSummary] = self._actionReportSummary;
235 self._dDispatch[self.ksActionReportRate] = self._actionReportRate;
236 self._dDispatch[self.ksActionReportTestCaseFailures] = self._actionReportTestCaseFailures;
237 self._dDispatch[self.ksActionReportFailureReasons] = self._actionReportFailureReasons;
238 self._dDispatch[self.ksActionGraphWiz] = self._actionGraphWiz;
239
240 self._dDispatch[self.ksActionVcsHistoryTooltip] = self._actionVcsHistoryTooltip;
241
242 # Legacy.
243 self._dDispatch['TestResultDetails'] = self._dDispatch[self.ksActionTestSetDetails];
244
245
246 #
247 # Popupate the menus.
248 #
249
250 # Additional URL parameters keeping for time navigation.
251 sExtraTimeNav = ''
252 dCurParams = oSrvGlue.getParameters()
253 if dCurParams is not None:
254 for sExtraParam in [ self.ksParamItemsPerPage, self.ksParamEffectiveDate, self.ksParamEffectivePeriod, ]:
255 if sExtraParam in dCurParams:
256 sExtraTimeNav += '&%s' % (webutils.encodeUrlParams({sExtraParam: dCurParams[sExtraParam]}),)
257
258 # Additional URL parameters for reports
259 sExtraReports = '';
260 if dCurParams is not None:
261 for sExtraParam in [ self.ksParamReportPeriods, self.ksParamReportPeriodInHours, self.ksParamEffectiveDate, ]:
262 if sExtraParam in dCurParams:
263 sExtraReports += '&%s' % (webutils.encodeUrlParams({sExtraParam: dCurParams[sExtraParam]}),)
264
265 # Shorthand to keep within margins.
266 sActUrlBase = self._sActionUrlBase;
267 sOnlyFailures = '&%s%s' % ( webutils.encodeUrlParams({self.ksParamOnlyFailures: True}), sExtraTimeNav, );
268 sSheriff = '&%s%s' % ( webutils.encodeUrlParams({self.ksParamOnlyNeedingReason: True}), sExtraTimeNav, );
269
270 self._aaoMenus = \
271 [
272 [
273 'Sheriff', sActUrlBase + self.ksActionResultsUnGrouped + sSheriff,
274 [
275 [ 'Grouped by', None ],
276 [ 'Ungrouped', sActUrlBase + self.ksActionResultsUnGrouped + sSheriff, False ],
277 [ 'Sched group', sActUrlBase + self.ksActionResultsGroupedBySchedGroup + sSheriff, False ],
278 [ 'Test group', sActUrlBase + self.ksActionResultsGroupedByTestGroup + sSheriff, False ],
279 [ 'Test case', sActUrlBase + self.ksActionResultsGroupedByTestCase + sSheriff, False ],
280 [ 'Testbox', sActUrlBase + self.ksActionResultsGroupedByTestBox + sSheriff, False ],
281 [ 'OS', sActUrlBase + self.ksActionResultsGroupedByOS + sSheriff, False ],
282 [ 'Architecture', sActUrlBase + self.ksActionResultsGroupedByArch + sSheriff, False ],
283 [ 'Revision', sActUrlBase + self.ksActionResultsGroupedByBuildRev + sSheriff, False ],
284 [ 'Build category', sActUrlBase + self.ksActionResultsGroupedByBuildCat + sSheriff, False ],
285 ]
286 ],
287 [
288 'Reports', sActUrlBase + self.ksActionReportSummary,
289 [
290 [ 'Summary', sActUrlBase + self.ksActionReportSummary + sExtraReports, False ],
291 [ 'Success rate', sActUrlBase + self.ksActionReportRate + sExtraReports, False ],
292 [ 'Test case failures', sActUrlBase + self.ksActionReportTestCaseFailures + sExtraReports, False ],
293 [ 'Testbox failures', sActUrlBase + self.ksActionReportTestBoxFailures + sExtraReports, False ],
294 [ 'Failure reasons', sActUrlBase + self.ksActionReportFailureReasons + sExtraReports, False ],
295 ]
296 ],
297 [
298 'Test Results', sActUrlBase + self.ksActionResultsUnGrouped + sExtraTimeNav,
299 [
300 [ 'Grouped by', None ],
301 [ 'Ungrouped', sActUrlBase + self.ksActionResultsUnGrouped + sExtraTimeNav, False ],
302 [ 'Sched group', sActUrlBase + self.ksActionResultsGroupedBySchedGroup + sExtraTimeNav, False ],
303 [ 'Test group', sActUrlBase + self.ksActionResultsGroupedByTestGroup + sExtraTimeNav, False ],
304 [ 'Test case', sActUrlBase + self.ksActionResultsGroupedByTestCase + sExtraTimeNav, False ],
305 [ 'Testbox', sActUrlBase + self.ksActionResultsGroupedByTestBox + sExtraTimeNav, False ],
306 [ 'OS', sActUrlBase + self.ksActionResultsGroupedByOS + sExtraTimeNav, False ],
307 [ 'Architecture', sActUrlBase + self.ksActionResultsGroupedByArch + sExtraTimeNav, False ],
308 [ 'Revision', sActUrlBase + self.ksActionResultsGroupedByBuildRev + sExtraTimeNav, False ],
309 [ 'Build category', sActUrlBase + self.ksActionResultsGroupedByBuildCat + sExtraTimeNav, False ],
310 ]
311 ],
312 [
313 'Test Failures', sActUrlBase + self.ksActionResultsUnGrouped + sOnlyFailures,
314 [
315 [ 'Grouped by', None ],
316 [ 'Ungrouped', sActUrlBase + self.ksActionResultsUnGrouped + sOnlyFailures, False ],
317 [ 'Sched group', sActUrlBase + self.ksActionResultsGroupedBySchedGroup + sOnlyFailures, False ],
318 [ 'Test group', sActUrlBase + self.ksActionResultsGroupedByTestGroup + sOnlyFailures, False ],
319 [ 'Test case', sActUrlBase + self.ksActionResultsGroupedByTestCase + sOnlyFailures, False ],
320 [ 'Testbox', sActUrlBase + self.ksActionResultsGroupedByTestBox + sOnlyFailures, False ],
321 [ 'OS', sActUrlBase + self.ksActionResultsGroupedByOS + sOnlyFailures, False ],
322 [ 'Architecture', sActUrlBase + self.ksActionResultsGroupedByArch + sOnlyFailures, False ],
323 [ 'Revision', sActUrlBase + self.ksActionResultsGroupedByBuildRev + sOnlyFailures, False ],
324 [ 'Build category', sActUrlBase + self.ksActionResultsGroupedByBuildCat + sOnlyFailures, False ],
325 ]
326 ],
327 [
328 '> Admin', 'admin.py?' + webutils.encodeUrlParams(self._dDbgParams), []
329 ],
330 ];
331
332
333 #
334 # Overriding parent methods.
335 #
336
337 def _generatePage(self):
338 """Override parent handler in order to change page title."""
339 if self._sPageTitle is not None:
340 self._sPageTitle = 'Test Results - ' + self._sPageTitle
341
342 return WuiDispatcherBase._generatePage(self)
343
344 def _actionDefault(self):
345 """Show the default admin page."""
346 from testmanager.webui.wuitestresult import WuiGroupedResultList;
347 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
348 self._sAction = self.ksActionResultsUnGrouped
349 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeNone,
350 TestResultLogic, TestResultFilter, WuiGroupedResultList);
351
352 def _isMenuMatch(self, sMenuUrl, sActionParam):
353 if super(WuiMain, self)._isMenuMatch(sMenuUrl, sActionParam):
354 fOnlyNeedingReason = self.getBoolParam(self.ksParamOnlyNeedingReason, fDefault = False);
355 if fOnlyNeedingReason:
356 return (sMenuUrl.find(self.ksParamOnlyNeedingReason) > 0);
357 fOnlyFailures = self.getBoolParam(self.ksParamOnlyFailures, fDefault = False);
358 return (sMenuUrl.find(self.ksParamOnlyFailures) > 0) == fOnlyFailures \
359 and sMenuUrl.find(self.ksParamOnlyNeedingReason) < 0;
360 return False;
361
362
363 #
364 # Navigation bar stuff
365 #
366
367 def _generateSortBySelector(self, dParams, sPreamble, sPostamble):
368 """
369 Generate HTML code for the sort by selector.
370 """
371 from testmanager.core.testresults import TestResultLogic;
372
373 if self.ksParamTestResultsSortBy in dParams:
374 enmResultSortBy = dParams[self.ksParamTestResultsSortBy];
375 del dParams[self.ksParamTestResultsSortBy];
376 else:
377 enmResultSortBy = TestResultLogic.ksResultsSortByRunningAndStart;
378
379 sHtmlSortBy = '<form name="TimeForm" method="GET"> Sort by\n';
380 sHtmlSortBy += sPreamble;
381 sHtmlSortBy += '\n <select name="%s" onchange="window.location=' % (self.ksParamTestResultsSortBy,);
382 sHtmlSortBy += '\'?%s&%s=\' + ' % (webutils.encodeUrlParams(dParams), self.ksParamTestResultsSortBy)
383 sHtmlSortBy += 'this.options[this.selectedIndex].value;" title="Sorting by">\n'
384
385 fSelected = False;
386 for enmCode, sTitle in TestResultLogic.kaasResultsSortByTitles:
387 if enmCode == enmResultSortBy:
388 fSelected = True;
389 sHtmlSortBy += ' <option value="%s"%s>%s</option>\n' \
390 % (enmCode, ' selected="selected"' if enmCode == enmResultSortBy else '', sTitle,);
391 assert fSelected;
392 sHtmlSortBy += ' </select>\n';
393 sHtmlSortBy += sPostamble;
394 sHtmlSortBy += '\n</form>\n'
395 return sHtmlSortBy;
396
397 def _generateStatusSelector(self, dParams, fOnlyFailures):
398 """
399 Generate HTML code for the status code selector. Currently very simple.
400 """
401 dParams[self.ksParamOnlyFailures] = not fOnlyFailures;
402 return WuiTmLink('Show all results' if fOnlyFailures else 'Only show failed tests', '', dParams,
403 fBracketed = False).toHtml();
404
405 def _generateTimeWalker(self, dParams, tsEffective, sCurPeriod):
406 """
407 Generates HTML code for walking back and forth in time.
408 """
409 # Have to do some math here. :-/
410 if tsEffective is None:
411 self._oDb.execute('SELECT CURRENT_TIMESTAMP - \'' + sCurPeriod + '\'::interval');
412 tsNext = None;
413 tsPrev = self._oDb.fetchOne()[0];
414 else:
415 self._oDb.execute('SELECT %s::TIMESTAMP - \'' + sCurPeriod + '\'::interval,\n'
416 ' %s::TIMESTAMP + \'' + sCurPeriod + '\'::interval',
417 (tsEffective, tsEffective,));
418 tsPrev, tsNext = self._oDb.fetchOne();
419
420 # Forget about page No when changing a period
421 if WuiDispatcherBase.ksParamPageNo in dParams:
422 del dParams[WuiDispatcherBase.ksParamPageNo]
423
424 # Format.
425 dParams[WuiDispatcherBase.ksParamEffectiveDate] = str(tsPrev);
426 sPrev = '<a href="?%s" title="One period earlier">&lt;&lt;</a>&nbsp;&nbsp;' \
427 % (webutils.encodeUrlParams(dParams),);
428
429 if tsNext is not None:
430 dParams[WuiDispatcherBase.ksParamEffectiveDate] = str(tsNext);
431 sNext = '&nbsp;&nbsp;<a href="?%s" title="One period later">&gt;&gt;</a>' \
432 % (webutils.encodeUrlParams(dParams),);
433 else:
434 sNext = '&nbsp;&nbsp;&gt;&gt;';
435
436 from testmanager.webui.wuicontentbase import WuiListContentBase; ## @todo move to better place.
437 return WuiListContentBase.generateTimeNavigation('top', self.getParameters(), self.getEffectiveDateParam(),
438 sPrev, sNext, False);
439
440 def _generateResultPeriodSelector(self, dParams, sCurPeriod):
441 """
442 Generate HTML code for result period selector.
443 """
444
445 if self.ksParamEffectivePeriod in dParams:
446 del dParams[self.ksParamEffectivePeriod];
447
448 # Forget about page No when changing a period
449 if WuiDispatcherBase.ksParamPageNo in dParams:
450 del dParams[WuiDispatcherBase.ksParamPageNo]
451
452 sHtmlPeriodSelector = '<form name="PeriodForm" method="GET">\n'
453 sHtmlPeriodSelector += ' Period is\n'
454 sHtmlPeriodSelector += ' <select name="%s" onchange="window.location=' % self.ksParamEffectivePeriod
455 sHtmlPeriodSelector += '\'?%s&%s=\' + ' % (webutils.encodeUrlParams(dParams), self.ksParamEffectivePeriod)
456 sHtmlPeriodSelector += 'this.options[this.selectedIndex].value;">\n'
457
458 for sPeriodValue, sPeriodCaption, _ in self.kaoResultPeriods:
459 sHtmlPeriodSelector += ' <option value="%s"%s>%s</option>\n' \
460 % (webutils.quoteUrl(sPeriodValue),
461 ' selected="selected"' if sPeriodValue == sCurPeriod else '',
462 sPeriodCaption)
463
464 sHtmlPeriodSelector += ' </select>\n' \
465 '</form>\n'
466
467 return sHtmlPeriodSelector
468
469 def _generateGroupContentSelector(self, aoGroupMembers, iCurrentMember, sAltAction):
470 """
471 Generate HTML code for group content selector.
472 """
473
474 dParams = self.getParameters()
475
476 if self.ksParamGroupMemberId in dParams:
477 del dParams[self.ksParamGroupMemberId]
478
479 if sAltAction is not None:
480 if self.ksParamAction in dParams:
481 del dParams[self.ksParamAction];
482 dParams[self.ksParamAction] = sAltAction;
483
484 sHtmlSelector = '<form name="GroupContentForm" method="GET">\n'
485 sHtmlSelector += ' <select name="%s" onchange="window.location=' % self.ksParamGroupMemberId
486 sHtmlSelector += '\'?%s&%s=\' + ' % (webutils.encodeUrlParams(dParams), self.ksParamGroupMemberId)
487 sHtmlSelector += 'this.options[this.selectedIndex].value;">\n'
488
489 sHtmlSelector += '<option value="-1">All</option>\n'
490
491 for iGroupMemberId, sGroupMemberName in aoGroupMembers:
492 if iGroupMemberId is not None:
493 sHtmlSelector += ' <option value="%s"%s>%s</option>\n' \
494 % (iGroupMemberId,
495 ' selected="selected"' if iGroupMemberId == iCurrentMember else '',
496 sGroupMemberName)
497
498 sHtmlSelector += ' </select>\n' \
499 '</form>\n'
500
501 return sHtmlSelector
502
503 def _generatePagesSelector(self, dParams, cItems, cItemsPerPage, iPage):
504 """
505 Generate HTML code for pages (1, 2, 3 ... N) selector
506 """
507
508 if WuiDispatcherBase.ksParamPageNo in dParams:
509 del dParams[WuiDispatcherBase.ksParamPageNo]
510
511 sHrefPtr = '<a href="?%s&%s=' % (webutils.encodeUrlParams(dParams).replace('%', '%%'),
512 WuiDispatcherBase.ksParamPageNo)
513 sHrefPtr += '%d">%s</a>'
514
515 cNumOfPages = (cItems + cItemsPerPage - 1) // cItemsPerPage;
516 cPagesToDisplay = 10
517 cPagesRangeStart = iPage - cPagesToDisplay // 2 \
518 if not iPage - cPagesToDisplay / 2 < 0 else 0
519 cPagesRangeEnd = cPagesRangeStart + cPagesToDisplay \
520 if not cPagesRangeStart + cPagesToDisplay > cNumOfPages else cNumOfPages
521 # Adjust pages range
522 if cNumOfPages < cPagesToDisplay:
523 cPagesRangeStart = 0
524 cPagesRangeEnd = cNumOfPages
525
526 # 1 2 3 4...
527 sHtmlPager = '&nbsp;\n'.join(sHrefPtr % (x, str(x + 1)) if x != iPage else str(x + 1)
528 for x in range(cPagesRangeStart, cPagesRangeEnd))
529 if cPagesRangeStart > 0:
530 sHtmlPager = '%s&nbsp; ... &nbsp;\n' % (sHrefPtr % (0, str(1))) + sHtmlPager
531 if cPagesRangeEnd < cNumOfPages:
532 sHtmlPager += ' ... %s\n' % (sHrefPtr % (cNumOfPages, str(cNumOfPages + 1)))
533
534 # Prev/Next (using << >> because &laquo; and &raquo are too tiny).
535 if iPage > 0:
536 dParams[WuiDispatcherBase.ksParamPageNo] = iPage - 1
537 sHtmlPager = ('<a title="Previous page" href="?%s">&lt;&lt;</a>&nbsp;&nbsp;\n'
538 % (webutils.encodeUrlParams(dParams), )) \
539 + sHtmlPager;
540 else:
541 sHtmlPager = '&lt;&lt;&nbsp;&nbsp;\n' + sHtmlPager
542
543 if iPage + 1 < cNumOfPages:
544 dParams[WuiDispatcherBase.ksParamPageNo] = iPage + 1
545 sHtmlPager += '\n&nbsp; <a title="Next page" href="?%s">&gt;&gt;</a>\n' % (webutils.encodeUrlParams(dParams),)
546 else:
547 sHtmlPager += '\n&nbsp; &gt;&gt;\n'
548
549 return sHtmlPager
550
551 def _generateItemPerPageSelector(self, dParams, cItemsPerPage):
552 """
553 Generate HTML code for items per page selector
554 Note! Modifies dParams!
555 """
556
557 from testmanager.webui.wuicontentbase import WuiListContentBase; ## @todo move to better place.
558 return WuiListContentBase.generateItemPerPageSelector('top', dParams, cItemsPerPage);
559
560 def _generateResultNavigation(self, cItems, cItemsPerPage, iPage, tsEffective, sCurPeriod, fOnlyFailures,
561 sHtmlMemberSelector):
562 """ Make custom time navigation bar for the results. """
563
564 # Generate the elements.
565 sHtmlStatusSelector = self._generateStatusSelector(self.getParameters(), fOnlyFailures);
566 sHtmlSortBySelector = self._generateSortBySelector(self.getParameters(), '', sHtmlStatusSelector);
567 sHtmlPeriodSelector = self._generateResultPeriodSelector(self.getParameters(), sCurPeriod)
568 sHtmlTimeWalker = self._generateTimeWalker(self.getParameters(), tsEffective, sCurPeriod);
569
570 if cItems > 0:
571 sHtmlPager = self._generatePagesSelector(self.getParameters(), cItems, cItemsPerPage, iPage)
572 sHtmlItemsPerPageSelector = self._generateItemPerPageSelector(self.getParameters(), cItemsPerPage)
573 else:
574 sHtmlPager = ''
575 sHtmlItemsPerPageSelector = ''
576
577 # Generate navigation bar
578 sHtml = '<table width=100%>\n' \
579 '<tr>\n' \
580 ' <td width=30%>' + sHtmlMemberSelector + '</td>\n' \
581 ' <td width=40% align=center>' + sHtmlTimeWalker + '</td>' \
582 ' <td width=30% align=right>\n' + sHtmlPeriodSelector + '</td>\n' \
583 '</tr>\n' \
584 '<tr>\n' \
585 ' <td width=30%>' + sHtmlSortBySelector + '</td>\n' \
586 ' <td width=40% align=center>\n' + sHtmlPager + '</td>\n' \
587 ' <td width=30% align=right>\n' + sHtmlItemsPerPageSelector + '</td>\n'\
588 '</tr>\n' \
589 '</table>\n'
590
591 return sHtml
592
593 def _generateReportNavigation(self, tsEffective, cHoursPerPeriod, cPeriods):
594 """ Make time navigation bar for the reports. """
595
596 # The period length selector.
597 dParams = self.getParameters();
598 if WuiMain.ksParamReportPeriodInHours in dParams:
599 del dParams[WuiMain.ksParamReportPeriodInHours];
600 sHtmlPeriodLength = '';
601 sHtmlPeriodLength += '<form name="ReportPeriodInHoursForm" method="GET">\n' \
602 ' Period length <select name="%s" onchange="window.location=\'?%s&%s=\' + ' \
603 'this.options[this.selectedIndex].value;" title="Statistics period length in hours.">\n' \
604 % (WuiMain.ksParamReportPeriodInHours,
605 webutils.encodeUrlParams(dParams),
606 WuiMain.ksParamReportPeriodInHours)
607 for cHours in [ 1, 2, 3, 4, 5, 6, 7, 8, 9, 12, 18, 24, 48, 72, 96, 120, 144, 168 ]:
608 sHtmlPeriodLength += ' <option value="%d"%s>%d hour%s</option>\n' \
609 % (cHours, 'selected="selected"' if cHours == cHoursPerPeriod else '', cHours,
610 's' if cHours > 1 else '');
611 sHtmlPeriodLength += ' </select>\n' \
612 '</form>\n'
613
614 # The period count selector.
615 dParams = self.getParameters();
616 if WuiMain.ksParamReportPeriods in dParams:
617 del dParams[WuiMain.ksParamReportPeriods];
618 sHtmlCountOfPeriods = '';
619 sHtmlCountOfPeriods += '<form name="ReportPeriodsForm" method="GET">\n' \
620 ' Periods <select name="%s" onchange="window.location=\'?%s&%s=\' + ' \
621 'this.options[this.selectedIndex].value;" title="Statistics periods to report.">\n' \
622 % (WuiMain.ksParamReportPeriods,
623 webutils.encodeUrlParams(dParams),
624 WuiMain.ksParamReportPeriods)
625 for cCurPeriods in range(2, 43):
626 sHtmlCountOfPeriods += ' <option value="%d"%s>%d</option>\n' \
627 % (cCurPeriods, 'selected="selected"' if cCurPeriods == cPeriods else '', cCurPeriods);
628 sHtmlCountOfPeriods += ' </select>\n' \
629 '</form>\n'
630
631 # The time walker.
632 sHtmlTimeWalker = self._generateTimeWalker(self.getParameters(), tsEffective, '%d hours' % (cHoursPerPeriod));
633
634 # Combine them all.
635 sHtml = '<table width=100%>\n' \
636 ' <tr>\n' \
637 ' <td width=30% align="center">\n' + sHtmlPeriodLength + '</td>\n' \
638 ' <td width=40% align="center">\n' + sHtmlTimeWalker + '</td>' \
639 ' <td width=30% align="center">\n' + sHtmlCountOfPeriods + '</td>\n' \
640 ' </tr>\n' \
641 '</table>\n';
642 return sHtml;
643
644
645 #
646 # The rest of stuff
647 #
648
649 def _actionGroupedResultsListing( #pylint: disable=too-many-locals
650 self,
651 enmResultsGroupingType,
652 oResultsLogicType,
653 oResultFilterType,
654 oResultsListContentType):
655 """
656 Override generic listing action.
657
658 oResultsLogicType implements getEntriesCount, fetchResultsForListing and more.
659 oResultFilterType is a child of ModelFilterBase.
660 oResultsListContentType is a child of WuiListContentBase.
661 """
662 from testmanager.core.testresults import TestResultLogic;
663
664 cItemsPerPage = self.getIntParam(self.ksParamItemsPerPage, iMin = 2, iMax = 9999, iDefault = 128);
665 iPage = self.getIntParam(self.ksParamPageNo, iMin = 0, iMax = 999999, iDefault = 0);
666 tsEffective = self.getEffectiveDateParam();
667 iGroupMemberId = self.getIntParam(self.ksParamGroupMemberId, iMin = -1, iMax = 999999, iDefault = -1);
668 fOnlyFailures = self.getBoolParam(self.ksParamOnlyFailures, fDefault = False);
669 fOnlyNeedingReason = self.getBoolParam(self.ksParamOnlyNeedingReason, fDefault = False);
670 enmResultSortBy = self.getStringParam(self.ksParamTestResultsSortBy,
671 asValidValues = TestResultLogic.kasResultsSortBy,
672 sDefault = TestResultLogic.ksResultsSortByRunningAndStart);
673 oFilter = oResultFilterType().initFromParams(self);
674
675 # Get testing results period and validate it
676 asValidValues = [x for (x, _, _) in self.kaoResultPeriods]
677 sCurPeriod = self.getStringParam(self.ksParamEffectivePeriod, asValidValues = asValidValues,
678 sDefault = self.ksResultPeriodDefault)
679 assert sCurPeriod != ''; # Impossible!
680
681 self._checkForUnknownParameters()
682
683 #
684 # Fetch the group members.
685 #
686 # If no grouping is selected, we'll fill the grouping combo with
687 # testboxes just to avoid having completely useless combo box.
688 #
689 oTrLogic = TestResultLogic(self._oDb);
690 sAltSelectorAction = None;
691 if enmResultsGroupingType in (TestResultLogic.ksResultsGroupingTypeNone, TestResultLogic.ksResultsGroupingTypeTestBox,):
692 aoTmp = oTrLogic.getTestBoxes(tsNow = tsEffective, sPeriod = sCurPeriod)
693 aoGroupMembers = sorted(list({(x.idTestBox, '%s (%s)' % (x.sName, str(x.ip))) for x in aoTmp }),
694 reverse = False, key = lambda asData: asData[1])
695
696 if enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeTestBox:
697 self._sPageTitle = 'Grouped by Test Box';
698 else:
699 self._sPageTitle = 'Ungrouped results';
700 sAltSelectorAction = self.ksActionResultsGroupedByTestBox;
701 aoGroupMembers.insert(0, [None, None]); # The "All" member.
702
703 elif enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeTestGroup:
704 aoTmp = oTrLogic.getTestGroups(tsNow = tsEffective, sPeriod = sCurPeriod);
705 aoGroupMembers = sorted(list({ (x.idTestGroup, x.sName ) for x in aoTmp }),
706 reverse = False, key = lambda asData: asData[1])
707 self._sPageTitle = 'Grouped by Test Group'
708
709 elif enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeBuildRev:
710 aoTmp = oTrLogic.getBuilds(tsNow = tsEffective, sPeriod = sCurPeriod)
711 aoGroupMembers = sorted(list({ (x.iRevision, '%s.%d' % (x.oCat.sBranch, x.iRevision)) for x in aoTmp }),
712 reverse = True, key = lambda asData: asData[0])
713 self._sPageTitle = 'Grouped by Build'
714
715 elif enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeBuildCat:
716 aoTmp = oTrLogic.getBuildCategories(tsNow = tsEffective, sPeriod = sCurPeriod)
717 aoGroupMembers = sorted(list({ (x.idBuildCategory,
718 '%s / %s / %s / %s' % ( x.sProduct, x.sBranch, ', '.join(x.asOsArches), x.sType) )
719 for x in aoTmp }),
720 reverse = True, key = lambda asData: asData[1]);
721 self._sPageTitle = 'Grouped by Build Category'
722
723 elif enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeTestCase:
724 aoTmp = oTrLogic.getTestCases(tsNow = tsEffective, sPeriod = sCurPeriod)
725 aoGroupMembers = sorted(list({ (x.idTestCase, '%s' % x.sName) for x in aoTmp }),
726 reverse = False, key = lambda asData: asData[1])
727 self._sPageTitle = 'Grouped by Test Case'
728
729 elif enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeOS:
730 aoTmp = oTrLogic.getOSes(tsNow = tsEffective, sPeriod = sCurPeriod)
731 aoGroupMembers = sorted(list(set(aoTmp)), reverse = False, key = lambda asData: asData[1]);
732 self._sPageTitle = 'Grouped by OS'
733
734 elif enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeArch:
735 aoTmp = oTrLogic.getArchitectures(tsNow = tsEffective, sPeriod = sCurPeriod)
736 aoGroupMembers = sorted(list(set(aoTmp)), reverse = False, key = lambda asData: asData[1]);
737 self._sPageTitle = 'Grouped by Architecture'
738
739 elif enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeSchedGroup:
740 aoTmp = oTrLogic.getSchedGroups(tsNow = tsEffective, sPeriod = sCurPeriod)
741 aoGroupMembers = sorted(list({ (x.idSchedGroup, '%s' % x.sName) for x in aoTmp }),
742 reverse = False, key = lambda asData: asData[1])
743 self._sPageTitle = 'Grouped by Scheduling Group'
744
745 else:
746 raise TMExceptionBase('Unknown grouping type')
747
748 _sPageBody = ''
749 oContent = None
750 cEntriesMax = 0
751 _dParams = self.getParameters()
752 oResultLogic = oResultsLogicType(self._oDb);
753 for idMember, sMemberName in aoGroupMembers:
754 #
755 # Count and fetch entries to be displayed.
756 #
757
758 # Skip group members that were not specified.
759 if idMember != iGroupMemberId \
760 and ( (idMember is not None and enmResultsGroupingType == TestResultLogic.ksResultsGroupingTypeNone)
761 or (iGroupMemberId > 0 and enmResultsGroupingType != TestResultLogic.ksResultsGroupingTypeNone) ):
762 continue
763
764 cEntries = oResultLogic.getEntriesCount(tsNow = tsEffective,
765 sInterval = sCurPeriod,
766 oFilter = oFilter,
767 enmResultsGroupingType = enmResultsGroupingType,
768 iResultsGroupingValue = idMember,
769 fOnlyFailures = fOnlyFailures,
770 fOnlyNeedingReason = fOnlyNeedingReason);
771 if cEntries == 0: # Do not display empty groups
772 continue
773 aoEntries = oResultLogic.fetchResultsForListing(iPage * cItemsPerPage,
774 cItemsPerPage,
775 tsNow = tsEffective,
776 sInterval = sCurPeriod,
777 oFilter = oFilter,
778 enmResultSortBy = enmResultSortBy,
779 enmResultsGroupingType = enmResultsGroupingType,
780 iResultsGroupingValue = idMember,
781 fOnlyFailures = fOnlyFailures,
782 fOnlyNeedingReason = fOnlyNeedingReason);
783 cEntriesMax = max(cEntriesMax, cEntries)
784
785 #
786 # Format them.
787 #
788 oContent = oResultsListContentType(aoEntries,
789 cEntries,
790 iPage,
791 cItemsPerPage,
792 tsEffective,
793 fnDPrint = self._oSrvGlue.dprint,
794 oDisp = self)
795
796 (_, sHtml) = oContent.show(fShowNavigation = False)
797 if sMemberName is not None:
798 _sPageBody += '<table width=100%><tr><td>'
799
800 _dParams[self.ksParamGroupMemberId] = idMember
801 sLink = WuiTmLink(sMemberName, '', _dParams, fBracketed = False).toHtml()
802
803 _sPageBody += '<h2>%s (%d)</h2></td>' % (sLink, cEntries)
804 _sPageBody += '<td><br></td>'
805 _sPageBody += '</tr></table>'
806 _sPageBody += sHtml
807 _sPageBody += '<br>'
808
809 #
810 # Complete the page by slapping navigation controls at the top and
811 # bottom of it.
812 #
813 sHtmlNavigation = self._generateResultNavigation(cEntriesMax, cItemsPerPage, iPage,
814 tsEffective, sCurPeriod, fOnlyFailures,
815 self._generateGroupContentSelector(aoGroupMembers, iGroupMemberId,
816 sAltSelectorAction));
817 if cEntriesMax > 0:
818 self._sPageBody = sHtmlNavigation + _sPageBody + sHtmlNavigation;
819 else:
820 self._sPageBody = sHtmlNavigation + '<p align="center"><i>No data to display</i></p>\n';
821
822 #
823 # Now, generate a filter control panel for the side bar.
824 #
825 if hasattr(oFilter, 'kiBranches'):
826 oFilter.aCriteria[oFilter.kiBranches].fExpanded = True;
827 if hasattr(oFilter, 'kiTestStatus'):
828 oFilter.aCriteria[oFilter.kiTestStatus].fExpanded = True;
829 self._sPageFilter = self._generateResultFilter(oFilter, oResultLogic, tsEffective, sCurPeriod,
830 enmResultsGroupingType = enmResultsGroupingType,
831 aoGroupMembers = aoGroupMembers,
832 fOnlyFailures = fOnlyFailures,
833 fOnlyNeedingReason = fOnlyNeedingReason);
834 return True;
835
836 def _generateResultFilter(self, oFilter, oResultLogic, tsNow, sPeriod, enmResultsGroupingType = None, aoGroupMembers = None,
837 fOnlyFailures = False, fOnlyNeedingReason = False):
838 """
839 Generates the result filter for the left hand side.
840 """
841 _ = enmResultsGroupingType; _ = aoGroupMembers; _ = fOnlyFailures; _ = fOnlyNeedingReason;
842 oResultLogic.fetchPossibleFilterOptions(oFilter, tsNow, sPeriod)
843
844 # Add non-filter parameters as hidden fields so we can use 'GET' and have URLs to bookmark.
845 self._dSideMenuFormAttrs['method'] = 'GET';
846 sHtml = u'';
847 for sKey, oValue in self._oSrvGlue.getParameters().items():
848 if len(sKey) > 3:
849 if hasattr(oValue, 'startswith'):
850 sHtml += u'<input type="hidden" name="%s" value="%s"/>\n' \
851 % (webutils.escapeAttr(sKey), webutils.escapeAttr(oValue),);
852 else:
853 for oSubValue in oValue:
854 sHtml += u'<input type="hidden" name="%s" value="%s"/>\n' \
855 % (webutils.escapeAttr(sKey), webutils.escapeAttr(oSubValue),);
856
857 # Generate the filter panel.
858 sHtml += u'<div id="side-filters">\n' \
859 u' <p>Filters' \
860 u' <span class="tm-side-filter-title-buttons"><input type="submit" value="Apply" />\n' \
861 u' <a href="javascript:toggleSidebarSize();" class="tm-sidebar-size-link">&#x00bb;&#x00bb;</a></span></p>\n';
862 sHtml += u' <dl>\n';
863 for oCrit in oFilter.aCriteria:
864 if oCrit.aoPossible or oCrit.sType == oCrit.ksType_Ranges:
865 if ( oCrit.oSub is None \
866 and ( oCrit.sState == oCrit.ksState_Selected \
867 or (len(oCrit.aoPossible) <= 2 and oCrit.sType != oCrit.ksType_Ranges))) \
868 or ( oCrit.oSub is not None \
869 and ( oCrit.sState == oCrit.ksState_Selected \
870 or oCrit.oSub.sState == oCrit.ksState_Selected \
871 or (len(oCrit.aoPossible) <= 2 and len(oCrit.oSub.aoPossible) <= 2))) \
872 or oCrit.fExpanded is True:
873 sClass = 'sf-collapsible';
874 sChar = '&#9660;';
875 else:
876 sClass = 'sf-expandable';
877 sChar = '&#9654;';
878
879 sHtml += u' <dt class="%s"><a href="javascript:void(0)" onclick="toggleCollapsibleDtDd(this);">%s %s</a> ' \
880 % (sClass, sChar, webutils.escapeElem(oCrit.sName),);
881 sHtml += u'<span class="tm-side-filter-dt-buttons">';
882 if oCrit.sInvVarNm is not None:
883 sHtml += u'<input id="sf-union-%s" class="tm-side-filter-union-input" ' \
884 u'name="%s" value="1" type="checkbox"%s />' \
885 u'<label for="sf-union-%s" class="tm-side-filter-union-input"></label>' \
886 % ( oCrit.sInvVarNm, oCrit.sInvVarNm, ' checked' if oCrit.fInverted else '', oCrit.sInvVarNm,);
887 sHtml += u' <input type="submit" value="Apply" />';
888 sHtml += u'</span>';
889 sHtml += u'</dt>\n' \
890 u' <dd class="%s">\n' \
891 u' <ul>\n' \
892 % (sClass);
893
894 if oCrit.sType == oCrit.ksType_Ranges:
895 assert not oCrit.oSub;
896 assert not oCrit.aoPossible;
897 asValues = [];
898 for tRange in oCrit.aoSelected:
899 if tRange[0] == tRange[1]:
900 asValues.append('%s' % (tRange[0],));
901 else:
902 asValues.append('%s-%s' % (tRange[0] if tRange[0] is not None else 'inf',
903 tRange[1] if tRange[1] is not None else 'inf'));
904 sHtml += u' <li title="%s"><input type="text" name="%s" value="%s"/></li>\n' \
905 % ( webutils.escapeAttr('comma separate list of numerical ranges'), oCrit.sVarNm,
906 ', '.join(asValues), );
907 else:
908 for oDesc in oCrit.aoPossible:
909 fChecked = oDesc.oValue in oCrit.aoSelected;
910 sHtml += u' <li%s%s><label><input type="checkbox" name="%s" value="%s"%s%s/>%s%s</label>\n' \
911 % ( ' class="side-filter-irrelevant"' if oDesc.fIrrelevant else '',
912 (' title="%s"' % (webutils.escapeAttr(oDesc.sHover,)) if oDesc.sHover is not None else ''),
913 oCrit.sVarNm,
914 oDesc.oValue,
915 ' checked' if fChecked else '',
916 ' onclick="toggleCollapsibleCheckbox(this);"' if oDesc.aoSubs is not None else '',
917 webutils.escapeElem(oDesc.sDesc),
918 '<span class="side-filter-count"> [%u]</span>' % (oDesc.cTimes) if oDesc.cTimes is not None
919 else '', );
920 if oDesc.aoSubs is not None:
921 sHtml += u' <ul class="sf-checkbox-%s">\n' % ('collapsible' if fChecked else 'expandable', );
922 for oSubDesc in oDesc.aoSubs:
923 fSubChecked = oSubDesc.oValue in oCrit.oSub.aoSelected;
924 sHtml += u' <li%s%s><label><input type="checkbox" name="%s" value="%s"%s/>%s%s</label>\n' \
925 % ( ' class="side-filter-irrelevant"' if oSubDesc.fIrrelevant else '',
926 ' title="%s"' % ( webutils.escapeAttr(oSubDesc.sHover,) if oSubDesc.sHover is not None
927 else ''),
928 oCrit.oSub.sVarNm, oSubDesc.oValue, ' checked' if fSubChecked else '',
929 webutils.escapeElem(oSubDesc.sDesc),
930 '<span class="side-filter-count"> [%u]</span>' % (oSubDesc.cTimes)
931 if oSubDesc.cTimes is not None else '', );
932
933 sHtml += u' </ul>\n';
934 sHtml += u' </li>';
935
936 sHtml += u' </ul>\n' \
937 u' </dd>\n';
938
939 sHtml += u' </dl>\n';
940 sHtml += u' <input type="submit" value="Apply"/>\n';
941 sHtml += u' <input type="reset" value="Reset"/>\n';
942 sHtml += u' <button type="button" onclick="clearForm(\'side-menu-form\');">Clear</button>\n';
943 sHtml += u'</div>\n';
944 return sHtml;
945
946 def _actionResultsUnGrouped(self):
947 """ Action wrapper. """
948 from testmanager.webui.wuitestresult import WuiGroupedResultList;
949 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
950 #return self._actionResultsListing(TestResultLogic, WuiGroupedResultList)?
951 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeNone,
952 TestResultLogic, TestResultFilter, WuiGroupedResultList);
953
954 def _actionResultsGroupedByTestGroup(self):
955 """ Action wrapper. """
956 from testmanager.webui.wuitestresult import WuiGroupedResultList;
957 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
958 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeTestGroup,
959 TestResultLogic, TestResultFilter, WuiGroupedResultList);
960
961 def _actionResultsGroupedByBuildRev(self):
962 """ Action wrapper. """
963 from testmanager.webui.wuitestresult import WuiGroupedResultList;
964 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
965 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeBuildRev,
966 TestResultLogic, TestResultFilter, WuiGroupedResultList);
967
968 def _actionResultsGroupedByBuildCat(self):
969 """ Action wrapper. """
970 from testmanager.webui.wuitestresult import WuiGroupedResultList;
971 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
972 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeBuildCat,
973 TestResultLogic, TestResultFilter, WuiGroupedResultList);
974
975 def _actionResultsGroupedByTestBox(self):
976 """ Action wrapper. """
977 from testmanager.webui.wuitestresult import WuiGroupedResultList;
978 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
979 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeTestBox,
980 TestResultLogic, TestResultFilter, WuiGroupedResultList);
981
982 def _actionResultsGroupedByTestCase(self):
983 """ Action wrapper. """
984 from testmanager.webui.wuitestresult import WuiGroupedResultList;
985 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
986 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeTestCase,
987 TestResultLogic, TestResultFilter, WuiGroupedResultList);
988
989 def _actionResultsGroupedByOS(self):
990 """ Action wrapper. """
991 from testmanager.webui.wuitestresult import WuiGroupedResultList;
992 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
993 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeOS,
994 TestResultLogic, TestResultFilter, WuiGroupedResultList);
995
996 def _actionResultsGroupedByArch(self):
997 """ Action wrapper. """
998 from testmanager.webui.wuitestresult import WuiGroupedResultList;
999 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
1000 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeArch,
1001 TestResultLogic, TestResultFilter, WuiGroupedResultList);
1002
1003 def _actionResultsGroupedBySchedGroup(self):
1004 """ Action wrapper. """
1005 from testmanager.webui.wuitestresult import WuiGroupedResultList;
1006 from testmanager.core.testresults import TestResultLogic, TestResultFilter;
1007 return self._actionGroupedResultsListing(TestResultLogic.ksResultsGroupingTypeSchedGroup,
1008 TestResultLogic, TestResultFilter, WuiGroupedResultList);
1009
1010
1011 def _actionTestSetDetailsCommon(self, idTestSet):
1012 """Show test case execution result details."""
1013 from testmanager.core.build import BuildDataEx;
1014 from testmanager.core.testbox import TestBoxData;
1015 from testmanager.core.testcase import TestCaseDataEx;
1016 from testmanager.core.testcaseargs import TestCaseArgsDataEx;
1017 from testmanager.core.testgroup import TestGroupData;
1018 from testmanager.core.testresults import TestResultLogic;
1019 from testmanager.core.testset import TestSetData;
1020 from testmanager.webui.wuitestresult import WuiTestResult;
1021
1022 self._sTemplate = 'template-details.html';
1023 self._checkForUnknownParameters()
1024
1025 oTestSetData = TestSetData().initFromDbWithId(self._oDb, idTestSet);
1026 try:
1027 (oTestResultTree, _) = TestResultLogic(self._oDb).fetchResultTree(idTestSet);
1028 except TMTooManyRows:
1029 (oTestResultTree, _) = TestResultLogic(self._oDb).fetchResultTree(idTestSet, 2);
1030 oBuildDataEx = BuildDataEx().initFromDbWithId(self._oDb, oTestSetData.idBuild, oTestSetData.tsCreated);
1031 try: oBuildValidationKitDataEx = BuildDataEx().initFromDbWithId(self._oDb, oTestSetData.idBuildTestSuite,
1032 oTestSetData.tsCreated);
1033 except: oBuildValidationKitDataEx = None;
1034 oTestBoxData = TestBoxData().initFromDbWithGenId(self._oDb, oTestSetData.idGenTestBox);
1035 oTestGroupData = TestGroupData().initFromDbWithId(self._oDb, ## @todo This bogus time wise. Bad DB design?
1036 oTestSetData.idTestGroup, oTestSetData.tsCreated);
1037 oTestCaseDataEx = TestCaseDataEx().initFromDbWithGenId(self._oDb, oTestSetData.idGenTestCase,
1038 oTestSetData.tsConfig);
1039 oTestCaseArgsDataEx = TestCaseArgsDataEx().initFromDbWithGenIdEx(self._oDb, oTestSetData.idGenTestCaseArgs,
1040 oTestSetData.tsConfig);
1041
1042 oContent = WuiTestResult(oDisp = self, fnDPrint = self._oSrvGlue.dprint);
1043 (self._sPageTitle, self._sPageBody) = oContent.showTestCaseResultDetails(oTestResultTree,
1044 oTestSetData,
1045 oBuildDataEx,
1046 oBuildValidationKitDataEx,
1047 oTestBoxData,
1048 oTestGroupData,
1049 oTestCaseDataEx,
1050 oTestCaseArgsDataEx);
1051 return True
1052
1053 def _actionTestSetDetails(self):
1054 """Show test case execution result details."""
1055 from testmanager.core.testset import TestSetData;
1056
1057 idTestSet = self.getIntParam(TestSetData.ksParam_idTestSet);
1058 return self._actionTestSetDetailsCommon(idTestSet);
1059
1060 def _actionTestSetDetailsFromResult(self):
1061 """Show test case execution result details."""
1062 from testmanager.core.testresults import TestResultData;
1063 from testmanager.core.testset import TestSetData;
1064 idTestResult = self.getIntParam(TestSetData.ksParam_idTestResult);
1065 oTestResultData = TestResultData().initFromDbWithId(self._oDb, idTestResult);
1066 return self._actionTestSetDetailsCommon(oTestResultData.idTestSet);
1067
1068
1069 def _actionTestResultFailureAdd(self):
1070 """ Pro forma. """
1071 from testmanager.core.testresultfailures import TestResultFailureData;
1072 from testmanager.webui.wuitestresultfailure import WuiTestResultFailure;
1073 return self._actionGenericFormAdd(TestResultFailureData, WuiTestResultFailure);
1074
1075 def _actionTestResultFailureAddPost(self):
1076 """Add test result failure result"""
1077 from testmanager.core.testresultfailures import TestResultFailureLogic, TestResultFailureData;
1078 from testmanager.webui.wuitestresultfailure import WuiTestResultFailure;
1079 if self.ksParamRedirectTo not in self._dParams:
1080 raise WuiException('Missing parameter ' + self.ksParamRedirectTo);
1081
1082 return self._actionGenericFormAddPost(TestResultFailureData, TestResultFailureLogic,
1083 WuiTestResultFailure, self.ksActionResultsUnGrouped);
1084
1085 def _actionTestResultFailureDoRemove(self):
1086 """ Action wrapper. """
1087 from testmanager.core.testresultfailures import TestResultFailureData, TestResultFailureLogic;
1088 return self._actionGenericDoRemove(TestResultFailureLogic, TestResultFailureData.ksParam_idTestResult,
1089 self.ksActionResultsUnGrouped);
1090
1091 def _actionTestResultFailureDetails(self):
1092 """ Pro forma. """
1093 from testmanager.core.testresultfailures import TestResultFailureLogic, TestResultFailureData;
1094 from testmanager.webui.wuitestresultfailure import WuiTestResultFailure;
1095 return self._actionGenericFormDetails(TestResultFailureData, TestResultFailureLogic,
1096 WuiTestResultFailure, 'idTestResult');
1097
1098 def _actionTestResultFailureEdit(self):
1099 """ Pro forma. """
1100 from testmanager.core.testresultfailures import TestResultFailureData;
1101 from testmanager.webui.wuitestresultfailure import WuiTestResultFailure;
1102 return self._actionGenericFormEdit(TestResultFailureData, WuiTestResultFailure,
1103 TestResultFailureData.ksParam_idTestResult);
1104
1105 def _actionTestResultFailureEditPost(self):
1106 """Edit test result failure result"""
1107 from testmanager.core.testresultfailures import TestResultFailureLogic, TestResultFailureData;
1108 from testmanager.webui.wuitestresultfailure import WuiTestResultFailure;
1109 return self._actionGenericFormEditPost(TestResultFailureData, TestResultFailureLogic,
1110 WuiTestResultFailure, self.ksActionResultsUnGrouped);
1111
1112 def _actionViewLog(self):
1113 """
1114 Log viewer action.
1115 """
1116 from testmanager.core.testresults import TestResultLogic, TestResultFileDataEx;
1117 from testmanager.core.testset import TestSetData, TestSetLogic;
1118 from testmanager.webui.wuilogviewer import WuiLogViewer;
1119
1120 self._sTemplate = 'template-details.html'; ## @todo create new template (background color, etc)
1121 idTestSet = self.getIntParam(self.ksParamLogSetId, iMin = 1);
1122 idLogFile = self.getIntParam(self.ksParamLogFileId, iMin = 0, iDefault = 0);
1123 cbChunk = self.getIntParam(self.ksParamLogChunkSize, iMin = 256, iMax = 16777216, iDefault = 1024*1024);
1124 iChunk = self.getIntParam(self.ksParamLogChunkNo, iMin = 0,
1125 iMax = config.g_kcMbMaxMainLog * 1048576 / cbChunk, iDefault = 0);
1126 self._checkForUnknownParameters();
1127
1128 oTestSet = TestSetData().initFromDbWithId(self._oDb, idTestSet);
1129 if idLogFile == 0:
1130 oTestFile = TestResultFileDataEx().initFakeMainLog(oTestSet);
1131 aoTimestamps = TestResultLogic(self._oDb).fetchTimestampsForLogViewer(idTestSet);
1132 else:
1133 oTestFile = TestSetLogic(self._oDb).getFile(idTestSet, idLogFile);
1134 aoTimestamps = [];
1135 if oTestFile.sMime not in [ 'text/plain',]:
1136 raise WuiException('The log view does not display files of type: %s' % (oTestFile.sMime,));
1137
1138 oContent = WuiLogViewer(oTestSet, oTestFile, cbChunk, iChunk, aoTimestamps,
1139 oDisp = self, fnDPrint = self._oSrvGlue.dprint);
1140 (self._sPageTitle, self._sPageBody) = oContent.show();
1141 return True;
1142
1143 def _actionGetFile(self):
1144 """
1145 Get file action.
1146 """
1147 from testmanager.core.testset import TestSetData, TestSetLogic;
1148 from testmanager.core.testresults import TestResultFileDataEx;
1149
1150 idTestSet = self.getIntParam(self.ksParamGetFileSetId, iMin = 1);
1151 idFile = self.getIntParam(self.ksParamGetFileId, iMin = 0, iDefault = 0);
1152 fDownloadIt = self.getBoolParam(self.ksParamGetFileDownloadIt, fDefault = True);
1153 self._checkForUnknownParameters();
1154
1155 #
1156 # Get the file info and open it.
1157 #
1158 oTestSet = TestSetData().initFromDbWithId(self._oDb, idTestSet);
1159 if idFile == 0:
1160 oTestFile = TestResultFileDataEx().initFakeMainLog(oTestSet);
1161 else:
1162 oTestFile = TestSetLogic(self._oDb).getFile(idTestSet, idFile);
1163
1164 (oFile, oSizeOrError, _) = oTestSet.openFile(oTestFile.sFile, 'rb');
1165 if oFile is None:
1166 raise Exception(oSizeOrError);
1167
1168 #
1169 # Send the file.
1170 #
1171 self._oSrvGlue.setHeaderField('Content-Type', oTestFile.getMimeWithEncoding());
1172 if fDownloadIt:
1173 self._oSrvGlue.setHeaderField('Content-Disposition', 'attachment; filename="TestSet-%d-%s"'
1174 % (idTestSet, oTestFile.sFile,));
1175 while True:
1176 abChunk = oFile.read(262144);
1177 if not abChunk:
1178 break;
1179 self._oSrvGlue.writeRaw(abChunk);
1180 return self.ksDispatchRcAllDone;
1181
1182 def _actionGenericReport(self, oModelType, oFilterType, oReportType):
1183 """
1184 Generic report action.
1185 oReportType is a child of WuiReportContentBase.
1186 oFilterType is a child of ModelFilterBase.
1187 oModelType is a child of ReportModelBase.
1188 """
1189 from testmanager.core.report import ReportModelBase;
1190
1191 tsEffective = self.getEffectiveDateParam();
1192 cPeriods = self.getIntParam(self.ksParamReportPeriods, iMin = 2, iMax = 99, iDefault = 7);
1193 cHoursPerPeriod = self.getIntParam(self.ksParamReportPeriodInHours, iMin = 1, iMax = 168, iDefault = 24);
1194 sSubject = self.getStringParam(self.ksParamReportSubject, ReportModelBase.kasSubjects,
1195 ReportModelBase.ksSubEverything);
1196 if sSubject == ReportModelBase.ksSubEverything:
1197 aidSubjects = self.getListOfIntParams(self.ksParamReportSubjectIds, aiDefaults = []);
1198 else:
1199 aidSubjects = self.getListOfIntParams(self.ksParamReportSubjectIds, iMin = 1);
1200 if aidSubjects is None:
1201 raise WuiException('Missing parameter %s' % (self.ksParamReportSubjectIds,));
1202
1203 aiSortColumnsDup = self.getListOfIntParams(self.ksParamSortColumns,
1204 iMin = -getattr(oReportType, 'kcMaxSortColumns', cPeriods) + 1,
1205 iMax = getattr(oReportType, 'kcMaxSortColumns', cPeriods), aiDefaults = []);
1206 aiSortColumns = [];
1207 for iSortColumn in aiSortColumnsDup:
1208 if iSortColumn not in aiSortColumns:
1209 aiSortColumns.append(iSortColumn);
1210
1211 oFilter = oFilterType().initFromParams(self);
1212 self._checkForUnknownParameters();
1213
1214 dParams = \
1215 {
1216 self.ksParamEffectiveDate: tsEffective,
1217 self.ksParamReportPeriods: cPeriods,
1218 self.ksParamReportPeriodInHours: cHoursPerPeriod,
1219 self.ksParamReportSubject: sSubject,
1220 self.ksParamReportSubjectIds: aidSubjects,
1221 };
1222 ## @todo oFilter.
1223
1224 oModel = oModelType(self._oDb, tsEffective, cPeriods, cHoursPerPeriod, sSubject, aidSubjects, oFilter);
1225 oContent = oReportType(oModel, dParams, fSubReport = False, aiSortColumns = aiSortColumns,
1226 fnDPrint = self._oSrvGlue.dprint, oDisp = self);
1227 (self._sPageTitle, self._sPageBody) = oContent.show();
1228 sNavi = self._generateReportNavigation(tsEffective, cHoursPerPeriod, cPeriods);
1229 self._sPageBody = sNavi + self._sPageBody;
1230
1231 if hasattr(oFilter, 'kiBranches'):
1232 oFilter.aCriteria[oFilter.kiBranches].fExpanded = True;
1233 self._sPageFilter = self._generateResultFilter(oFilter, oModel, tsEffective, '%s hours' % (cHoursPerPeriod * cPeriods,));
1234 return True;
1235
1236 def _actionReportSummary(self):
1237 """ Action wrapper. """
1238 from testmanager.core.report import ReportLazyModel, ReportFilter;
1239 from testmanager.webui.wuireport import WuiReportSummary;
1240 return self._actionGenericReport(ReportLazyModel, ReportFilter, WuiReportSummary);
1241
1242 def _actionReportRate(self):
1243 """ Action wrapper. """
1244 from testmanager.core.report import ReportLazyModel, ReportFilter;
1245 from testmanager.webui.wuireport import WuiReportSuccessRate;
1246 return self._actionGenericReport(ReportLazyModel, ReportFilter, WuiReportSuccessRate);
1247
1248 def _actionReportTestCaseFailures(self):
1249 """ Action wrapper. """
1250 from testmanager.core.report import ReportLazyModel, ReportFilter;
1251 from testmanager.webui.wuireport import WuiReportTestCaseFailures;
1252 return self._actionGenericReport(ReportLazyModel, ReportFilter, WuiReportTestCaseFailures);
1253
1254 def _actionReportFailureReasons(self):
1255 """ Action wrapper. """
1256 from testmanager.core.report import ReportLazyModel, ReportFilter;
1257 from testmanager.webui.wuireport import WuiReportFailureReasons;
1258 return self._actionGenericReport(ReportLazyModel, ReportFilter, WuiReportFailureReasons);
1259
1260 def _actionGraphWiz(self):
1261 """
1262 Graph wizard action.
1263 """
1264 from testmanager.core.report import ReportModelBase, ReportGraphModel;
1265 from testmanager.webui.wuigraphwiz import WuiGraphWiz;
1266 self._sTemplate = 'template-graphwiz.html';
1267
1268 tsEffective = self.getEffectiveDateParam();
1269 cPeriods = self.getIntParam(self.ksParamReportPeriods, iMin = 1, iMax = 1, iDefault = 1); # Not needed yet.
1270 sTmp = self.getStringParam(self.ksParamReportPeriodInHours, sDefault = '3 weeks');
1271 (cHoursPerPeriod, sError) = utils.parseIntervalHours(sTmp);
1272 if sError is not None: raise WuiException(sError);
1273 asSubjectIds = self.getListOfStrParams(self.ksParamReportSubjectIds);
1274 sSubject = self.getStringParam(self.ksParamReportSubject, [ReportModelBase.ksSubEverything],
1275 ReportModelBase.ksSubEverything); # dummy
1276 aidTestBoxes = self.getListOfIntParams(self.ksParamGraphWizTestBoxIds, iMin = 1, aiDefaults = []);
1277 aidBuildCats = self.getListOfIntParams(self.ksParamGraphWizBuildCatIds, iMin = 1, aiDefaults = []);
1278 aidTestCases = self.getListOfIntParams(self.ksParamGraphWizTestCaseIds, iMin = 1, aiDefaults = []);
1279 fSepTestVars = self.getBoolParam(self.ksParamGraphWizSepTestVars, fDefault = False);
1280
1281 enmGraphImpl = self.getStringParam(self.ksParamGraphWizImpl, asValidValues = self.kasGraphWizImplValid,
1282 sDefault = self.ksGraphWizImpl_Default);
1283 cx = self.getIntParam(self.ksParamGraphWizWidth, iMin = 128, iMax = 8192, iDefault = 1280);
1284 cy = self.getIntParam(self.ksParamGraphWizHeight, iMin = 128, iMax = 8192, iDefault = int(cx * 5 / 16) );
1285 cDotsPerInch = self.getIntParam(self.ksParamGraphWizDpi, iMin = 64, iMax = 512, iDefault = 96);
1286 cPtFont = self.getIntParam(self.ksParamGraphWizFontSize, iMin = 6, iMax = 32, iDefault = 8);
1287 fErrorBarY = self.getBoolParam(self.ksParamGraphWizErrorBarY, fDefault = False);
1288 cMaxErrorBarY = self.getIntParam(self.ksParamGraphWizMaxErrorBarY, iMin = 8, iMax = 9999999, iDefault = 18);
1289 cMaxPerGraph = self.getIntParam(self.ksParamGraphWizMaxPerGraph, iMin = 1, iMax = 24, iDefault = 8);
1290 fXkcdStyle = self.getBoolParam(self.ksParamGraphWizXkcdStyle, fDefault = False);
1291 fTabular = self.getBoolParam(self.ksParamGraphWizTabular, fDefault = False);
1292 idSrcTestSet = self.getIntParam(self.ksParamGraphWizSrcTestSetId, iDefault = None);
1293 self._checkForUnknownParameters();
1294
1295 dParams = \
1296 {
1297 self.ksParamEffectiveDate: tsEffective,
1298 self.ksParamReportPeriods: cPeriods,
1299 self.ksParamReportPeriodInHours: cHoursPerPeriod,
1300 self.ksParamReportSubject: sSubject,
1301 self.ksParamReportSubjectIds: asSubjectIds,
1302 self.ksParamGraphWizTestBoxIds: aidTestBoxes,
1303 self.ksParamGraphWizBuildCatIds: aidBuildCats,
1304 self.ksParamGraphWizTestCaseIds: aidTestCases,
1305 self.ksParamGraphWizSepTestVars: fSepTestVars,
1306
1307 self.ksParamGraphWizImpl: enmGraphImpl,
1308 self.ksParamGraphWizWidth: cx,
1309 self.ksParamGraphWizHeight: cy,
1310 self.ksParamGraphWizDpi: cDotsPerInch,
1311 self.ksParamGraphWizFontSize: cPtFont,
1312 self.ksParamGraphWizErrorBarY: fErrorBarY,
1313 self.ksParamGraphWizMaxErrorBarY: cMaxErrorBarY,
1314 self.ksParamGraphWizMaxPerGraph: cMaxPerGraph,
1315 self.ksParamGraphWizXkcdStyle: fXkcdStyle,
1316 self.ksParamGraphWizTabular: fTabular,
1317 self.ksParamGraphWizSrcTestSetId: idSrcTestSet,
1318 };
1319
1320 oModel = ReportGraphModel(self._oDb, tsEffective, cPeriods, cHoursPerPeriod, sSubject, asSubjectIds,
1321 aidTestBoxes, aidBuildCats, aidTestCases, fSepTestVars);
1322 oContent = WuiGraphWiz(oModel, dParams, fSubReport = False, fnDPrint = self._oSrvGlue.dprint, oDisp = self);
1323 (self._sPageTitle, self._sPageBody) = oContent.show();
1324 return True;
1325
1326 def _actionVcsHistoryTooltip(self):
1327 """
1328 Version control system history.
1329 """
1330 from testmanager.webui.wuivcshistory import WuiVcsHistoryTooltip;
1331 from testmanager.core.vcsrevisions import VcsRevisionLogic;
1332
1333 self._sTemplate = 'template-tooltip.html';
1334 iRevision = self.getIntParam(self.ksParamVcsHistoryRevision, iMin = 0, iMax = 999999999);
1335 sRepository = self.getStringParam(self.ksParamVcsHistoryRepository);
1336 cEntries = self.getIntParam(self.ksParamVcsHistoryEntries, iMin = 1, iMax = 1024, iDefault = 8);
1337 self._checkForUnknownParameters();
1338
1339 aoEntries = VcsRevisionLogic(self._oDb).fetchTimeline(sRepository, iRevision, cEntries);
1340 oContent = WuiVcsHistoryTooltip(aoEntries, sRepository, iRevision, cEntries,
1341 fnDPrint = self._oSrvGlue.dprint, oDisp = self);
1342 (self._sPageTitle, self._sPageBody) = oContent.show();
1343 return True;
1344
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette