VirtualBox

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

Last change on this file since 64319 was 62484, checked in by vboxsync, 9 years ago

(C) 2016

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