VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testmanager/webui/wuireport.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: 26.8 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: wuireport.py 62484 2016-07-22 18:35:33Z vboxsync $
3
4"""
5Test Manager WUI - Reports.
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
32# Validation Kit imports.
33from common import webutils;
34from testmanager.webui.wuicontentbase import WuiContentBase, WuiTmLink, WuiSvnLinkWithTooltip;
35from testmanager.webui.wuihlpgraph import WuiHlpGraphDataTable, WuiHlpBarGraph;
36from testmanager.webui.wuitestresult import WuiTestSetLink;
37from testmanager.webui.wuiadmintestcase import WuiTestCaseDetailsLink;
38from testmanager.webui.wuiadmintestbox import WuiTestBoxDetailsLink;
39from testmanager.core.report import ReportModelBase;
40
41
42class WuiReportSummaryLink(WuiTmLink):
43 """ Generic report summary link. """
44
45 def __init__(self, sSubject, aIdSubjects, sName = WuiContentBase.ksShortReportLink,
46 tsNow = None, cPeriods = None, cHoursPerPeriod = None, fBracketed = False, dExtraParams = None):
47 from testmanager.webui.wuimain import WuiMain;
48 dParams = {
49 WuiMain.ksParamAction: WuiMain.ksActionReportSummary,
50 WuiMain.ksParamReportSubject: sSubject,
51 WuiMain.ksParamReportSubjectIds: aIdSubjects,
52 };
53 if dExtraParams is not None:
54 dParams.update(dExtraParams);
55 if tsNow is not None:
56 dParams[WuiMain.ksParamEffectiveDate] = tsNow;
57 if cPeriods is not None:
58 dParams[WuiMain.ksParamReportPeriods] = cPeriods;
59 if cPeriods is not None:
60 dParams[WuiMain.ksParamReportPeriodInHours] = cHoursPerPeriod;
61 WuiTmLink.__init__(self, sName, WuiMain.ksScriptName, dParams, fBracketed = fBracketed);
62
63
64class WuiReportBase(WuiContentBase):
65 """
66 Base class for the reports.
67 """
68
69 def __init__(self, oModel, dParams, fSubReport = False, fnDPrint = None, oDisp = None):
70 WuiContentBase.__init__(self, fnDPrint = fnDPrint, oDisp = oDisp);
71 self._oModel = oModel;
72 self._dParams = dParams;
73 self._fSubReport = fSubReport;
74 self._sTitle = None;
75
76 # Additional URL parameters for reports
77 self._dExtraParams = {};
78 dCurParams = None if oDisp is None else oDisp.getParameters();
79 if dCurParams is not None:
80 from testmanager.webui.wuimain import WuiMain;
81 for sExtraParam in [ WuiMain.ksParamReportPeriods, WuiMain.ksParamReportPeriodInHours,
82 WuiMain.ksParamEffectiveDate, ]:
83 if sExtraParam in dCurParams:
84 self._dExtraParams[sExtraParam] = dCurParams[sExtraParam];
85
86
87 def generateNavigator(self, sWhere):
88 """
89 Generates the navigator (manipulate _dParams).
90 Returns HTML.
91 """
92 assert sWhere == 'top' or sWhere == 'bottom';
93
94 return '';
95
96 def generateReportBody(self):
97 """
98 This is overridden by the child class to generate the report.
99 Returns HTML.
100 """
101 return '<h3>Must override generateReportBody!</h3>';
102
103 def show(self):
104 """
105 Generate the report.
106 Returns (sTitle, HTML).
107 """
108
109 sTitle = self._sTitle if self._sTitle is not None else type(self).__name__;
110 sReport = self.generateReportBody();
111 if not self._fSubReport:
112 sReport = self.generateNavigator('top') + sReport + self.generateNavigator('bottom');
113 sTitle = self._oModel.sSubject + ' - ' + sTitle; ## @todo add subject to title in a proper way!
114
115 sReport += '\n\n<!-- HEYYOU: sSubject=%s aidSubjects=%s -->\n\n' % (self._oModel.sSubject, self._oModel.aidSubjects);
116 return (sTitle, sReport);
117
118
119class WuiReportSuccessRate(WuiReportBase):
120 """
121 Generates a report displaying the success rate over time.
122 """
123
124 def generateReportBody(self):
125 self._sTitle = 'Success rate';
126
127 adPeriods = self._oModel.getSuccessRates();
128
129 sReport = '';
130
131 oTable = WuiHlpGraphDataTable('Period', [ 'Succeeded', 'Skipped', 'Failed' ]);
132
133 #for i in range(len(adPeriods) - 1, -1, -1):
134 for i, dStatuses in enumerate(adPeriods):
135 cSuccess = dStatuses[ReportModelBase.ksTestStatus_Success] + dStatuses[ReportModelBase.ksTestStatus_Skipped];
136 cTotal = cSuccess + dStatuses[ReportModelBase.ksTestStatus_Failure];
137 sPeriod = self._oModel.getPeriodDesc(i);
138 if cTotal > 0:
139 iPctSuccess = dStatuses[ReportModelBase.ksTestStatus_Success] * 100 / cTotal;
140 iPctSkipped = dStatuses[ReportModelBase.ksTestStatus_Skipped] * 100 / cTotal;
141 iPctFailure = dStatuses[ReportModelBase.ksTestStatus_Failure] * 100 / cTotal;
142 oTable.addRow(sPeriod, [ iPctSuccess, iPctSkipped, iPctFailure ],
143 [ '%s%% (%d)' % (iPctSuccess, dStatuses[ReportModelBase.ksTestStatus_Success]),
144 '%s%% (%d)' % (iPctSkipped, dStatuses[ReportModelBase.ksTestStatus_Skipped]),
145 '%s%% (%d)' % (iPctFailure, dStatuses[ReportModelBase.ksTestStatus_Failure]), ]);
146 else:
147 oTable.addRow(sPeriod, [ 0, 0, 0 ], [ '0%', '0%', '0%' ]);
148
149 cTotalNow = adPeriods[0][ReportModelBase.ksTestStatus_Success];
150 cTotalNow += adPeriods[0][ReportModelBase.ksTestStatus_Skipped];
151 cSuccessNow = cTotalNow;
152 cTotalNow += adPeriods[0][ReportModelBase.ksTestStatus_Failure];
153 sReport += '<p>Current success rate: ';
154 if cTotalNow > 0:
155 sReport += '%s%% (thereof %s%% skipped)</p>\n' \
156 % ( cSuccessNow * 100 / cTotalNow, adPeriods[0][ReportModelBase.ksTestStatus_Skipped] * 100 / cTotalNow);
157 else:
158 sReport += 'N/A</p>\n'
159
160 oGraph = WuiHlpBarGraph('success-rate', oTable, self._oDisp);
161 oGraph.setRangeMax(100);
162 sReport += oGraph.renderGraph();
163
164 return sReport;
165
166
167class WuiReportFailuresBase(WuiReportBase):
168 """
169 Common parent of WuiReportFailureReasons and WuiReportTestCaseFailures.
170 """
171
172 def _splitSeriesIntoMultipleGraphs(self, aidSorted, cMaxSeriesPerGraph = 8):
173 """
174 Splits the ID array into one or more arrays, making sure we don't
175 have too many series per graph.
176 Returns array of ID arrays.
177 """
178 if len(aidSorted) <= cMaxSeriesPerGraph + 2:
179 return [aidSorted,];
180 cGraphs = len(aidSorted) / cMaxSeriesPerGraph + (len(aidSorted) % cMaxSeriesPerGraph != 0);
181 cPerGraph = len(aidSorted) / cGraphs + (len(aidSorted) % cGraphs != 0);
182
183 aaoRet = [];
184 cLeft = len(aidSorted);
185 iSrc = 0;
186 while cLeft > 0:
187 cThis = cPerGraph;
188 if cLeft <= cPerGraph + 2:
189 cThis = cLeft;
190 elif cLeft <= cPerGraph * 2 + 4:
191 cThis = cLeft / 2;
192 aaoRet.append(aidSorted[iSrc : iSrc + cThis]);
193 iSrc += cThis;
194 cLeft -= cThis;
195 return aaoRet;
196
197 def _formatEdgeOccurenceSubject(self, oTransient):
198 """
199 Worker for _formatEdgeOccurence that child classes overrides to format
200 their type of subject data in the best possible way.
201 """
202 _ = oTransient;
203 assert False;
204 return '';
205
206 def _formatEdgeOccurence(self, oTransient):
207 """
208 Helper for formatting the transients.
209 oTransient is of type ReportFailureReasonTransient or ReportTestCaseFailureTransient.
210 """
211 sHtml = u'<li>';
212 if oTransient.fEnter: sHtml += 'Since ';
213 else: sHtml += 'Until ';
214 sHtml += WuiSvnLinkWithTooltip(oTransient.iRevision, oTransient.sRepository, fBracketed = 'False').toHtml();
215 sHtml += u', %s: ' % (WuiTestSetLink(oTransient.idTestSet, self.formatTsShort(oTransient.tsDone),
216 fBracketed = False).toHtml(), )
217 sHtml += self._formatEdgeOccurenceSubject(oTransient);
218 sHtml += u'</li>\n';
219 return sHtml;
220
221 def _generateTransitionList(self, oSet):
222 """
223 Generates the enter and leave lists.
224 """
225 # Skip this if we're looking at builds.
226 if self._oModel.sSubject in [self._oModel.ksSubBuild,] and len(self._oModel.aidSubjects) in [1, 2]:
227 return u'';
228
229 sHtml = u'<h4>Movements:</h4>\n' \
230 u'<ul>\n';
231 if len(oSet.aoEnterInfo) == 0 and len(oSet.aoLeaveInfo) == 0:
232 sHtml += u'<li>No changes</li>\n';
233 else:
234 for oTransient in oSet.aoEnterInfo:
235 sHtml += self._formatEdgeOccurence(oTransient);
236 for oTransient in oSet.aoLeaveInfo:
237 sHtml += self._formatEdgeOccurence(oTransient);
238 sHtml += u'</ul>\n';
239
240 return sHtml;
241
242
243 def _formatSeriesNameColumnHeadersForTable(self):
244 """ Formats the series name column for the HTML table. """
245 return '<th>Subject Name</th>';
246
247 def _formatSeriesNameForTable(self, oSet, idKey):
248 """ Formats the series name for the HTML table. """
249 _ = oSet;
250 return '<td>%d</td>' % (idKey,);
251
252 def _formatRowValueForTable(self, oRow, oPeriod, cColsPerSeries):
253 """ Formats a row value for the HTML table. """
254 _ = oPeriod;
255 if oRow is None:
256 return u'<td colspan="%d"> </td>' % (cColsPerSeries,);
257 if cColsPerSeries == 2:
258 return u'<td align="right">%u%%</td><td align="center">%u / %u</td>' \
259 % (oRow.cHits * 100 / oRow.cTotal, oRow.cHits, oRow.cTotal);
260 return u'<td align="center">%u</td>' % (oRow.cHits,);
261
262 def _formatSeriesTotalForTable(self, oSet, idKey, cColsPerSeries):
263 """ Formats the totals cell for a data series in the HTML table. """
264 dcTotalPerId = getattr(oSet, 'dcTotalPerId', None);
265 if cColsPerSeries == 2:
266 return u'<td align="right">%u%%</td><td align="center">%u/%u</td>' \
267 % (oSet.dcHitsPerId[idKey] * 100 / dcTotalPerId[idKey], oSet.dcHitsPerId[idKey], dcTotalPerId[idKey]);
268 return u'<td align="center">%u</td>' % (oSet.dcHitsPerId[idKey],);
269
270 def _generateTableForSet(self, oSet, aidSorted = None, iSortColumn = 0,
271 fWithTotals = True, cColsPerSeries = None):
272 """
273 Turns the set into a table.
274
275 Returns raw html.
276 """
277 sHtml = u'<table class="tmtbl-report-set" width="100%%">\n';
278 if cColsPerSeries is None:
279 cColsPerSeries = 2 if hasattr(oSet, 'dcTotalPerId') else 1;
280
281 # Header row.
282 sHtml += u' <tr><thead><th>#</th>';
283 sHtml += self._formatSeriesNameColumnHeadersForTable();
284 for iPeriod, oPeriod in enumerate(reversed(oSet.aoPeriods)):
285 sHtml += u'<th colspan="%d">%s%s</th>' % ( cColsPerSeries, webutils.escapeElem(oPeriod.sDesc),
286 '&#x25bc;' if iPeriod == iSortColumn else '');
287 if fWithTotals:
288 sHtml += u'<th colspan="%d">Total%s</th>' % (cColsPerSeries,
289 '&#x25bc;' if iSortColumn == len(oSet.aoPeriods) else '');
290 sHtml += u'</thead></td>\n';
291
292 # Each data series.
293 if aidSorted is None:
294 aidSorted = oSet.dSubjects.keys();
295 sHtml += u' <tbody>\n';
296 for iRow, idKey in enumerate(aidSorted):
297 sHtml += u' <tr class="%s">' % ('tmodd' if iRow & 1 else 'tmeven',);
298 sHtml += u'<td align="left">#%u</td>' % (iRow + 1,);
299 sHtml += self._formatSeriesNameForTable(oSet, idKey);
300 for oPeriod in reversed(oSet.aoPeriods):
301 oRow = oPeriod.dRowsById.get(idKey, None);
302 sHtml += self._formatRowValueForTable(oRow, oPeriod, cColsPerSeries);
303 if fWithTotals:
304 sHtml += self._formatSeriesTotalForTable(oSet, idKey, cColsPerSeries);
305 sHtml += u' </tr>\n';
306 sHtml += u' </tbody>\n';
307 sHtml += u'</table>\n';
308 return sHtml;
309
310
311class WuiReportFailuresWithTotalBase(WuiReportFailuresBase):
312 """
313 For ReportPeriodSetWithTotalBase.
314 """
315
316 def _formatSeriedNameForGraph(self, oSubject):
317 """
318 Format the subject name for the graph.
319 """
320 return str(oSubject);
321
322 def _getSortedIds(self, oSet, fByTotal = None):
323 """
324 Get default sorted subject IDs and which column.
325 """
326
327 if fByTotal is None:
328 fByTotal = oSet.cMaxTotal < 10;
329
330 if fByTotal is True:
331 # Sort the total.
332 aidSortedRaw = sorted(oSet.dSubjects,
333 key = lambda idKey: oSet.dcHitsPerId[idKey] * 10000 / oSet.dcTotalPerId[idKey],
334 reverse = True);
335 iColumn = len(oSet.aoPeriods);
336 else:
337 # Sort by NOW column.
338 dTmp = {};
339 for idKey in oSet.dSubjects:
340 oRow = oSet.aoPeriods[-1].dRowsById.get(idKey, None);
341 if oRow is None: dTmp[idKey] = 0;
342 else: dTmp[idKey] = oRow.cHits * 10000 / max(1, oRow.cTotal);
343 aidSortedRaw = sorted(dTmp, key = lambda idKey: dTmp[idKey], reverse = True);
344 iColumn = 0;
345 return (aidSortedRaw, iColumn);
346
347 def _generateGraph(self, oSet, sIdBase, aidSortedRaw):
348 """
349 Generates graph.
350 """
351 sHtml = u'';
352 fGenerateGraph = len(aidSortedRaw) <= 6 and len(aidSortedRaw) > 0; ## Make this configurable.
353 if fGenerateGraph:
354 # Figure the graph width for all of them.
355 uPctMax = max(oSet.uMaxPct, oSet.cMaxHits * 100 / oSet.cMaxTotal);
356 uPctMax = max(uPctMax + 2, 10);
357
358 for _, aidSorted in enumerate(self._splitSeriesIntoMultipleGraphs(aidSortedRaw, 8)):
359 asNames = [];
360 for idKey in aidSorted:
361 oSubject = oSet.dSubjects[idKey];
362 asNames.append(self._formatSeriedNameForGraph(oSubject));
363
364 oTable = WuiHlpGraphDataTable('Period', asNames);
365
366 for _, oPeriod in enumerate(reversed(oSet.aoPeriods)):
367 aiValues = [];
368 asValues = [];
369
370 for idKey in aidSorted:
371 oRow = oPeriod.dRowsById.get(idKey, None);
372 if oRow is not None:
373 uPct = oRow.cHits * 100 / oRow.cTotal;
374 aiValues.append(uPct);
375 asValues.append('%u%% (%u/%u)' % (uPct, oRow.cHits, oRow.cTotal));
376 else:
377 aiValues.append(0);
378 asValues.append('0');
379
380 oTable.addRow(oPeriod.sDesc, aiValues, asValues);
381
382 if True: # pylint: disable=W0125
383 aiValues = [];
384 asValues = [];
385 for idKey in aidSorted:
386 uPct = oSet.dcHitsPerId[idKey] * 100 / oSet.dcTotalPerId[idKey];
387 aiValues.append(uPct);
388 asValues.append('%u%% (%u/%u)' % (uPct, oSet.dcHitsPerId[idKey], oSet.dcTotalPerId[idKey]));
389 oTable.addRow('Totals', aiValues, asValues);
390
391 oGraph = WuiHlpBarGraph(sIdBase, oTable, self._oDisp);
392 oGraph.setRangeMax(uPctMax);
393 sHtml += '<br>\n';
394 sHtml += oGraph.renderGraph();
395 return sHtml;
396
397
398
399class WuiReportFailureReasons(WuiReportFailuresBase):
400 """
401 Generates a report displaying the failure reasons over time.
402 """
403
404 def _formatEdgeOccurenceSubject(self, oTransient):
405 return u'%s / %s' % ( webutils.escapeElem(oTransient.oReason.oCategory.sShort),
406 webutils.escapeElem(oTransient.oReason.sShort),);
407
408 def _formatSeriesNameColumnHeadersForTable(self):
409 return '<th>Failure Reason</th>';
410
411 def _formatSeriesNameForTable(self, oSet, idKey):
412 oReason = oSet.dSubjects[idKey];
413 sHtml = u'<td>';
414 sHtml += u'%s / %s' % ( webutils.escapeElem(oReason.oCategory.sShort), webutils.escapeElem(oReason.sShort),);
415 sHtml += u'</td>';
416 return sHtml;
417
418
419 def generateReportBody(self):
420 self._sTitle = 'Failure reasons';
421
422 #
423 # Get the data and sort the data series in descending order of badness.
424 #
425 oSet = self._oModel.getFailureReasons();
426 aidSortedRaw = sorted(oSet.dSubjects, key = lambda idReason: oSet.dcHitsPerId[idReason], reverse = True);
427
428 #
429 # Generate table and transition list. These are the most useful ones with the current graph machinery.
430 #
431 sHtml = self._generateTableForSet(oSet, aidSortedRaw, len(oSet.aoPeriods));
432 sHtml += self._generateTransitionList(oSet);
433
434 #
435 # Check if most of the stuff is without any assign reason, if so, skip
436 # that part of the graph so it doesn't offset the interesting bits.
437 #
438 fIncludeWithoutReason = True;
439 for oPeriod in reversed(oSet.aoPeriods):
440 if oPeriod.cWithoutReason > oSet.cMaxHits * 4:
441 fIncludeWithoutReason = False;
442 sHtml += '<p>Warning: Many failures without assigned reason!</p>\n';
443 break;
444
445 #
446 # Generate the graph.
447 #
448 fGenerateGraph = len(aidSortedRaw) <= 9 and len(aidSortedRaw) > 0; ## Make this configurable.
449 if fGenerateGraph:
450 aidSorted = aidSortedRaw;
451
452 asNames = [];
453 for idReason in aidSorted:
454 oReason = oSet.dSubjects[idReason];
455 asNames.append('%s / %s' % (oReason.oCategory.sShort, oReason.sShort,) )
456 if fIncludeWithoutReason:
457 asNames.append('No reason');
458
459 oTable = WuiHlpGraphDataTable('Period', asNames);
460
461 cMax = oSet.cMaxHits;
462 for _, oPeriod in enumerate(reversed(oSet.aoPeriods)):
463 aiValues = [];
464
465 for idReason in aidSorted:
466 oRow = oPeriod.dRowsById.get(idReason, None);
467 iValue = oRow.cHits if oRow is not None else 0;
468 aiValues.append(iValue);
469
470 if fIncludeWithoutReason:
471 aiValues.append(oPeriod.cWithoutReason);
472 if oPeriod.cWithoutReason > cMax:
473 cMax = oPeriod.cWithoutReason;
474
475 oTable.addRow(oPeriod.sDesc, aiValues);
476
477 oGraph = WuiHlpBarGraph('failure-reason', oTable, self._oDisp);
478 oGraph.setRangeMax(max(cMax + 1, 3));
479 sHtml += oGraph.renderGraph();
480 return sHtml;
481
482
483class WuiReportTestCaseFailures(WuiReportFailuresWithTotalBase):
484 """
485 Generates a report displaying the failure reasons over time.
486 """
487
488 def _formatEdgeOccurenceSubject(self, oTransient):
489 sHtml = u'%s ' % ( webutils.escapeElem(oTransient.oSubject.sName),);
490 sHtml += WuiTestCaseDetailsLink(oTransient.oSubject.idTestCase, fBracketed = False).toHtml();
491 return sHtml;
492
493 def _formatSeriesNameColumnHeadersForTable(self):
494 return '<th>Test Case</th>';
495
496 def _formatSeriesNameForTable(self, oSet, idKey):
497 oTestCase = oSet.dSubjects[idKey];
498 sHtml = u'<td>';
499 sHtml += WuiReportSummaryLink(ReportModelBase.ksSubTestCase, oTestCase.idTestCase, sName = oTestCase.sName,
500 dExtraParams = self._dExtraParams).toHtml();
501 sHtml += u' ';
502 sHtml += WuiTestCaseDetailsLink(oTestCase.idTestCase).toHtml();
503 sHtml += u'</td>';
504 return sHtml;
505
506 def _formatSeriedNameForGraph(self, oSubject):
507 return oSubject.sName;
508
509 def generateReportBody(self):
510 self._sTitle = 'Test Case Failures';
511 oSet = self._oModel.getTestCaseFailures();
512 (aidSortedRaw, iSortColumn) = self._getSortedIds(oSet);
513
514 sHtml = self._generateTableForSet(oSet, aidSortedRaw, iSortColumn);
515 sHtml += self._generateTransitionList(oSet);
516 sHtml += self._generateGraph(oSet, 'testcase-graph', aidSortedRaw);
517 return sHtml;
518
519
520class WuiReportTestCaseArgsFailures(WuiReportFailuresWithTotalBase):
521 """
522 Generates a report displaying the failure reasons over time.
523 """
524
525 @staticmethod
526 def _formatName(oTestCaseArgs):
527 """ Internal helper for formatting the testcase name. """
528 if oTestCaseArgs.sSubName is not None and len(oTestCaseArgs.sSubName) > 0:
529 sName = u'%s / %s' % ( oTestCaseArgs.oTestCase.sName, oTestCaseArgs.sSubName, );
530 else:
531 sName = u'%s / #%u' % ( oTestCaseArgs.oTestCase.sName, oTestCaseArgs.idTestCaseArgs, );
532 return sName;
533
534 def _formatEdgeOccurenceSubject(self, oTransient):
535 sHtml = u'%s ' % ( webutils.escapeElem(self._formatName(oTransient.oSubject)),);
536 sHtml += WuiTestCaseDetailsLink(oTransient.oSubject.idTestCase, fBracketed = False).toHtml();
537 return sHtml;
538
539 def _formatSeriesNameColumnHeadersForTable(self):
540 return '<th>Test Case / Variation</th>';
541
542 def _formatSeriesNameForTable(self, oSet, idKey):
543 oTestCaseArgs = oSet.dSubjects[idKey];
544 sHtml = u'<td>';
545 sHtml += WuiReportSummaryLink(ReportModelBase.ksSubTestCaseArgs, oTestCaseArgs.idTestCaseArgs,
546 sName = self._formatName(oTestCaseArgs), dExtraParams = self._dExtraParams).toHtml();
547 sHtml += u' ';
548 sHtml += WuiTestCaseDetailsLink(oTestCaseArgs.idTestCase).toHtml();
549 sHtml += u'</td>';
550 return sHtml;
551
552 def _formatSeriedNameForGraph(self, oSubject):
553 return self._formatName(oSubject);
554
555 def generateReportBody(self):
556 self._sTitle = 'Test Case Variation Failures';
557 oSet = self._oModel.getTestCaseVariationFailures();
558 (aidSortedRaw, iSortColumn) = self._getSortedIds(oSet);
559
560 sHtml = self._generateTableForSet(oSet, aidSortedRaw, iSortColumn);
561 sHtml += self._generateTransitionList(oSet);
562 sHtml += self._generateGraph(oSet, 'testcasearg-graph', aidSortedRaw);
563 return sHtml;
564
565
566
567class WuiReportTestBoxFailures(WuiReportFailuresWithTotalBase):
568 """
569 Generates a report displaying the failure reasons over time.
570 """
571
572 def _formatEdgeOccurenceSubject(self, oTransient):
573 sHtml = u'%s ' % ( webutils.escapeElem(oTransient.oSubject.sName),);
574 sHtml += WuiTestBoxDetailsLink(oTransient.oSubject.idTestBox, fBracketed = False).toHtml();
575 return sHtml;
576
577 def _formatSeriesNameColumnHeadersForTable(self):
578 return '<th colspan="5">Test Box</th>';
579
580 def _formatSeriesNameForTable(self, oSet, idKey):
581 oTestBox = oSet.dSubjects[idKey];
582 sHtml = u'<td>';
583 sHtml += WuiReportSummaryLink(ReportModelBase.ksSubTestBox, oTestBox.idTestBox, sName = oTestBox.sName,
584 dExtraParams = self._dExtraParams).toHtml();
585 sHtml += u' ';
586 sHtml += WuiTestBoxDetailsLink(oTestBox.idTestBox).toHtml();
587 sHtml += u'</td>';
588 sOsAndVer = '%s %s' % (oTestBox.sOs, oTestBox.sOsVersion.strip(),);
589 if len(sOsAndVer) < 22:
590 sHtml += u'<td>%s</td>' % (webutils.escapeElem(sOsAndVer),);
591 else: # wonder if td.title works..
592 sHtml += u'<td title="%s" width="1%%" style="white-space:nowrap;">%s...</td>' \
593 % (webutils.escapeAttr(sOsAndVer), webutils.escapeElem(sOsAndVer[:20]));
594 sHtml += u'<td>%s</td>' % (webutils.escapeElem(oTestBox.getArchBitString()),);
595 sHtml += u'<td>%s</td>' % (webutils.escapeElem(oTestBox.getPrettyCpuVendor()),);
596 sHtml += u'<td>%s' % (oTestBox.getPrettyCpuVersion(),);
597 if oTestBox.fCpuNestedPaging: sHtml += u', np';
598 elif oTestBox.fCpuHwVirt: sHtml += u', hw';
599 else: sHtml += u', raw';
600 if oTestBox.fCpu64BitGuest: sHtml += u', 64';
601 sHtml += u'</td>';
602 return sHtml;
603
604 def _formatSeriedNameForGraph(self, oSubject):
605 return oSubject.sName;
606
607 def generateReportBody(self):
608 self._sTitle = 'Test Box Failures';
609 oSet = self._oModel.getTestBoxFailures();
610 (aidSortedRaw, iSortColumn) = self._getSortedIds(oSet);
611
612 sHtml = self._generateTableForSet(oSet, aidSortedRaw, iSortColumn);
613 sHtml += self._generateTransitionList(oSet);
614 sHtml += self._generateGraph(oSet, 'testbox-graph', aidSortedRaw);
615 return sHtml;
616
617
618class WuiReportSummary(WuiReportBase):
619 """
620 Summary report.
621 """
622
623 def generateReportBody(self):
624 self._sTitle = 'Summary';
625 sHtml = '<p>This will display several reports and listings useful to get an overview of %s (id=%s).</p>' \
626 % (self._oModel.sSubject, self._oModel.aidSubjects,);
627
628 aoReports = [];
629
630 aoReports.append(WuiReportSuccessRate( self._oModel, self._dParams, fSubReport = True,
631 fnDPrint = self._fnDPrint, oDisp = self._oDisp));
632 aoReports.append(WuiReportTestCaseFailures(self._oModel, self._dParams, fSubReport = True,
633 fnDPrint = self._fnDPrint, oDisp = self._oDisp));
634 if self._oModel.sSubject == ReportModelBase.ksSubTestCase:
635 aoReports.append(WuiReportTestCaseArgsFailures(self._oModel, self._dParams, fSubReport = True,
636 fnDPrint = self._fnDPrint, oDisp = self._oDisp));
637 aoReports.append(WuiReportTestBoxFailures( self._oModel, self._dParams, fSubReport = True,
638 fnDPrint = self._fnDPrint, oDisp = self._oDisp));
639 aoReports.append(WuiReportFailureReasons( self._oModel, self._dParams, fSubReport = True,
640 fnDPrint = self._fnDPrint, oDisp = self._oDisp));
641
642 for oReport in aoReports:
643 (sTitle, sContent) = oReport.show();
644 sHtml += '<br>'; # drop this layout hack
645 sHtml += '<div>';
646 sHtml += '<h3>%s</h3>\n' % (webutils.escapeElem(sTitle),);
647 sHtml += sContent;
648 sHtml += '</div>';
649
650 return sHtml;
651
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