1 | # -*- coding: utf-8 -*-
|
---|
2 | # $Id: wuihlpgraphmatplotlib.py 82968 2020-02-04 10:35:17Z vboxsync $
|
---|
3 |
|
---|
4 | """
|
---|
5 | Test Manager Web-UI - Graph Helpers - Implemented using matplotlib.
|
---|
6 | """
|
---|
7 |
|
---|
8 | __copyright__ = \
|
---|
9 | """
|
---|
10 | Copyright (C) 2012-2020 Oracle Corporation
|
---|
11 |
|
---|
12 | This file is part of VirtualBox Open Source Edition (OSE), as
|
---|
13 | available from http://www.virtualbox.org. This file is free software;
|
---|
14 | you can redistribute it and/or modify it under the terms of the GNU
|
---|
15 | General Public License (GPL) as published by the Free Software
|
---|
16 | Foundation, in version 2 as it comes in the "COPYING" file of the
|
---|
17 | VirtualBox OSE distribution. VirtualBox OSE is distributed in the
|
---|
18 | hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
|
---|
19 |
|
---|
20 | The contents of this file may alternatively be used under the terms
|
---|
21 | of the Common Development and Distribution License Version 1.0
|
---|
22 | (CDDL) only, as it comes in the "COPYING.CDDL" file of the
|
---|
23 | VirtualBox OSE distribution, in which case the provisions of the
|
---|
24 | CDDL are applicable instead of those of the GPL.
|
---|
25 |
|
---|
26 | You may elect to license modified versions of this file under the
|
---|
27 | terms and conditions of either the GPL or the CDDL or both.
|
---|
28 | """
|
---|
29 | __version__ = "$Revision: 82968 $"
|
---|
30 |
|
---|
31 | # Standard Python Import and extensions installed on the system.
|
---|
32 | import re;
|
---|
33 | import sys;
|
---|
34 | if sys.version_info[0] >= 3:
|
---|
35 | from io import StringIO as StringIO; # pylint: disable=import-error,no-name-in-module,useless-import-alias
|
---|
36 | else:
|
---|
37 | from StringIO import StringIO as StringIO; # pylint: disable=import-error,no-name-in-module,useless-import-alias
|
---|
38 |
|
---|
39 | import matplotlib; # pylint: disable=import-error
|
---|
40 | matplotlib.use('Agg'); # Force backend.
|
---|
41 | import matplotlib.pyplot; # pylint: disable=import-error
|
---|
42 | from numpy import arange as numpy_arange; # pylint: disable=no-name-in-module,import-error,wrong-import-order
|
---|
43 |
|
---|
44 | # Validation Kit imports.
|
---|
45 | from testmanager.webui.wuihlpgraphbase import WuiHlpGraphBase;
|
---|
46 |
|
---|
47 |
|
---|
48 | class WuiHlpGraphMatplotlibBase(WuiHlpGraphBase):
|
---|
49 | """ Base class for the matplotlib graphs. """
|
---|
50 |
|
---|
51 | def __init__(self, sId, oData, oDisp):
|
---|
52 | WuiHlpGraphBase.__init__(self, sId, oData, oDisp);
|
---|
53 | self._fXkcdStyle = True;
|
---|
54 |
|
---|
55 | def setXkcdStyle(self, fEnabled = True):
|
---|
56 | """ Enables xkcd style graphs for implementations that supports it. """
|
---|
57 | self._fXkcdStyle = fEnabled;
|
---|
58 | return True;
|
---|
59 |
|
---|
60 | def _createFigure(self):
|
---|
61 | """
|
---|
62 | Wrapper around matplotlib.pyplot.figure that feeds the figure the
|
---|
63 | basic graph configuration.
|
---|
64 | """
|
---|
65 | if self._fXkcdStyle and matplotlib.__version__ > '1.2.9':
|
---|
66 | matplotlib.pyplot.xkcd(); # pylint: disable=no-member
|
---|
67 | matplotlib.rcParams.update({'font.size': self._cPtFont});
|
---|
68 |
|
---|
69 | oFigure = matplotlib.pyplot.figure(figsize = (float(self._cxGraph) / self._cDpiGraph,
|
---|
70 | float(self._cyGraph) / self._cDpiGraph),
|
---|
71 | dpi = self._cDpiGraph);
|
---|
72 | return oFigure;
|
---|
73 |
|
---|
74 | def _produceSvg(self, oFigure, fTightLayout = True):
|
---|
75 | """ Creates an SVG string from the given figure. """
|
---|
76 | oOutput = StringIO();
|
---|
77 | if fTightLayout:
|
---|
78 | oFigure.tight_layout();
|
---|
79 | oFigure.savefig(oOutput, format = 'svg');
|
---|
80 |
|
---|
81 | if self._oDisp and self._oDisp.isBrowserGecko('20100101'):
|
---|
82 | # This browser will stretch images to fit if no size or width is given.
|
---|
83 | sSubstitute = r'\1 \3 reserveAspectRatio="xMidYMin meet"';
|
---|
84 | else:
|
---|
85 | # Chrome and IE likes to have the sizes as well as the viewBox.
|
---|
86 | sSubstitute = r'\1 \3 reserveAspectRatio="xMidYMin meet" \2 \4';
|
---|
87 | return re.sub(r'(<svg) (height="\d+pt") (version="\d+.\d+" viewBox="\d+ \d+ \d+ \d+") (width="\d+pt")',
|
---|
88 | sSubstitute,
|
---|
89 | oOutput.getvalue().decode('utf8'),
|
---|
90 | count = 1);
|
---|
91 |
|
---|
92 | class WuiHlpBarGraph(WuiHlpGraphMatplotlibBase):
|
---|
93 | """
|
---|
94 | Bar graph.
|
---|
95 | """
|
---|
96 |
|
---|
97 | def __init__(self, sId, oData, oDisp = None):
|
---|
98 | WuiHlpGraphMatplotlibBase.__init__(self, sId, oData, oDisp);
|
---|
99 | self.fpMax = None;
|
---|
100 | self.fpMin = 0.0;
|
---|
101 | self.cxBarWidth = None;
|
---|
102 |
|
---|
103 | def setRangeMax(self, fpMax):
|
---|
104 | """ Sets the max range."""
|
---|
105 | self.fpMax = float(fpMax);
|
---|
106 | return None;
|
---|
107 |
|
---|
108 | def invertYDirection(self):
|
---|
109 | """ Inverts the direction of the Y-axis direction. """
|
---|
110 | ## @todo self.fYInverted = True;
|
---|
111 | return None;
|
---|
112 |
|
---|
113 | def renderGraph(self): # pylint: disable=too-many-locals
|
---|
114 | aoTable = self._oData.aoTable;
|
---|
115 |
|
---|
116 | #
|
---|
117 | # Extract/structure the required data.
|
---|
118 | #
|
---|
119 | aoSeries = list();
|
---|
120 | for j in range(len(aoTable[1].aoValues)):
|
---|
121 | aoSeries.append(list());
|
---|
122 | asNames = list();
|
---|
123 | oXRange = numpy_arange(self._oData.getGroupCount());
|
---|
124 | fpMin = self.fpMin;
|
---|
125 | fpMax = self.fpMax;
|
---|
126 | if self.fpMax is None:
|
---|
127 | fpMax = float(aoTable[1].aoValues[0]);
|
---|
128 |
|
---|
129 | for i in range(1, len(aoTable)):
|
---|
130 | asNames.append(aoTable[i].sName);
|
---|
131 | for j in range(len(aoTable[i].aoValues)):
|
---|
132 | fpValue = float(aoTable[i].aoValues[j]);
|
---|
133 | aoSeries[j].append(fpValue);
|
---|
134 | if fpValue < fpMin:
|
---|
135 | fpMin = fpValue;
|
---|
136 | if fpValue > fpMax:
|
---|
137 | fpMax = fpValue;
|
---|
138 |
|
---|
139 | fpMid = fpMin + (fpMax - fpMin) / 2.0;
|
---|
140 |
|
---|
141 | if self.cxBarWidth is None:
|
---|
142 | self.cxBarWidth = 1.0 / (len(aoTable[0].asValues) + 1.1);
|
---|
143 |
|
---|
144 | # Render the PNG.
|
---|
145 | oFigure = self._createFigure();
|
---|
146 | oSubPlot = oFigure.add_subplot(1, 1, 1);
|
---|
147 |
|
---|
148 | aoBars = list();
|
---|
149 | for i, _ in enumerate(aoSeries):
|
---|
150 | sColor = self.calcSeriesColor(i);
|
---|
151 | aoBars.append(oSubPlot.bar(oXRange + self.cxBarWidth * i,
|
---|
152 | aoSeries[i],
|
---|
153 | self.cxBarWidth,
|
---|
154 | color = sColor,
|
---|
155 | align = 'edge'));
|
---|
156 |
|
---|
157 | #oSubPlot.set_title('Title')
|
---|
158 | #oSubPlot.set_xlabel('X-axis')
|
---|
159 | #oSubPlot.set_xticks(oXRange + self.cxBarWidth);
|
---|
160 | oSubPlot.set_xticks(oXRange);
|
---|
161 | oLegend = oSubPlot.legend(aoTable[0].asValues, loc = 'best', fancybox = True);
|
---|
162 | oLegend.get_frame().set_alpha(0.5);
|
---|
163 | oSubPlot.set_xticklabels(asNames, ha = "left");
|
---|
164 | #oSubPlot.set_ylabel('Y-axis')
|
---|
165 | oSubPlot.set_yticks(numpy_arange(fpMin, fpMax + (fpMax - fpMin) / 10 * 0, fpMax / 10));
|
---|
166 | oSubPlot.grid(True);
|
---|
167 | fpPadding = (fpMax - fpMin) * 0.02;
|
---|
168 | for i, _ in enumerate(aoBars):
|
---|
169 | aoRects = aoBars[i]
|
---|
170 | for j, _ in enumerate(aoRects):
|
---|
171 | oRect = aoRects[j];
|
---|
172 | fpValue = float(aoTable[j + 1].aoValues[i]);
|
---|
173 | if fpValue <= fpMid:
|
---|
174 | oSubPlot.text(oRect.get_x() + oRect.get_width() / 2.0,
|
---|
175 | oRect.get_height() + fpPadding,
|
---|
176 | aoTable[j + 1].asValues[i],
|
---|
177 | ha = 'center', va = 'bottom', rotation = 'vertical', alpha = 0.6, fontsize = 'small');
|
---|
178 | else:
|
---|
179 | oSubPlot.text(oRect.get_x() + oRect.get_width() / 2.0,
|
---|
180 | oRect.get_height() - fpPadding,
|
---|
181 | aoTable[j + 1].asValues[i],
|
---|
182 | ha = 'center', va = 'top', rotation = 'vertical', alpha = 0.6, fontsize = 'small');
|
---|
183 |
|
---|
184 | return self._produceSvg(oFigure);
|
---|
185 |
|
---|
186 |
|
---|
187 |
|
---|
188 |
|
---|
189 | class WuiHlpLineGraph(WuiHlpGraphMatplotlibBase):
|
---|
190 | """
|
---|
191 | Line graph.
|
---|
192 | """
|
---|
193 |
|
---|
194 | def __init__(self, sId, oData, oDisp = None, fErrorBarY = False):
|
---|
195 | # oData must be a WuiHlpGraphDataTableEx like object.
|
---|
196 | WuiHlpGraphMatplotlibBase.__init__(self, sId, oData, oDisp);
|
---|
197 | self._cMaxErrorBars = 12;
|
---|
198 | self._fErrorBarY = fErrorBarY;
|
---|
199 |
|
---|
200 | def setErrorBarY(self, fEnable):
|
---|
201 | """ Enables or Disables error bars, making this work like a line graph. """
|
---|
202 | self._fErrorBarY = fEnable;
|
---|
203 | return True;
|
---|
204 |
|
---|
205 | def renderGraph(self): # pylint: disable=too-many-locals
|
---|
206 | aoSeries = self._oData.aoSeries;
|
---|
207 |
|
---|
208 | oFigure = self._createFigure();
|
---|
209 | oSubPlot = oFigure.add_subplot(1, 1, 1);
|
---|
210 | if self._oData.sYUnit is not None:
|
---|
211 | oSubPlot.set_ylabel(self._oData.sYUnit);
|
---|
212 | if self._oData.sXUnit is not None:
|
---|
213 | oSubPlot.set_xlabel(self._oData.sXUnit);
|
---|
214 |
|
---|
215 | cSeriesNames = 0;
|
---|
216 | cYMin = 1000;
|
---|
217 | cYMax = 0;
|
---|
218 | for iSeries, oSeries in enumerate(aoSeries):
|
---|
219 | sColor = self.calcSeriesColor(iSeries);
|
---|
220 | cYMin = min(cYMin, min(oSeries.aoYValues));
|
---|
221 | cYMax = max(cYMax, max(oSeries.aoYValues));
|
---|
222 | if not self._fErrorBarY:
|
---|
223 | oSubPlot.errorbar(oSeries.aoXValues, oSeries.aoYValues, color = sColor);
|
---|
224 | elif len(oSeries.aoXValues) > self._cMaxErrorBars:
|
---|
225 | if matplotlib.__version__ < '1.3.0':
|
---|
226 | oSubPlot.errorbar(oSeries.aoXValues, oSeries.aoYValues, color = sColor);
|
---|
227 | else:
|
---|
228 | oSubPlot.errorbar(oSeries.aoXValues, oSeries.aoYValues,
|
---|
229 | yerr = [oSeries.aoYErrorBarBelow, oSeries.aoYErrorBarAbove],
|
---|
230 | errorevery = len(oSeries.aoXValues) / self._cMaxErrorBars,
|
---|
231 | color = sColor );
|
---|
232 | else:
|
---|
233 | oSubPlot.errorbar(oSeries.aoXValues, oSeries.aoYValues,
|
---|
234 | yerr = [oSeries.aoYErrorBarBelow, oSeries.aoYErrorBarAbove],
|
---|
235 | color = sColor);
|
---|
236 | cSeriesNames += oSeries.sName is not None;
|
---|
237 |
|
---|
238 | if cYMin != 0 or cYMax != 0:
|
---|
239 | oSubPlot.set_ylim(bottom = 0);
|
---|
240 |
|
---|
241 | if cSeriesNames > 0:
|
---|
242 | oLegend = oSubPlot.legend([oSeries.sName for oSeries in aoSeries], loc = 'best', fancybox = True);
|
---|
243 | oLegend.get_frame().set_alpha(0.5);
|
---|
244 |
|
---|
245 | if self._sTitle is not None:
|
---|
246 | oSubPlot.set_title(self._sTitle);
|
---|
247 |
|
---|
248 | if self._cxGraph >= 256:
|
---|
249 | oSubPlot.minorticks_on();
|
---|
250 | oSubPlot.grid(True, 'major', axis = 'both');
|
---|
251 | oSubPlot.grid(True, 'both', axis = 'x');
|
---|
252 |
|
---|
253 | if True: # pylint: disable=using-constant-test
|
---|
254 | # oSubPlot.axis('off');
|
---|
255 | #oSubPlot.grid(True, 'major', axis = 'none');
|
---|
256 | #oSubPlot.grid(True, 'both', axis = 'none');
|
---|
257 | matplotlib.pyplot.setp(oSubPlot, xticks = [], yticks = []);
|
---|
258 |
|
---|
259 | return self._produceSvg(oFigure);
|
---|
260 |
|
---|
261 |
|
---|
262 | class WuiHlpLineGraphErrorbarY(WuiHlpLineGraph):
|
---|
263 | """
|
---|
264 | Line graph with an errorbar for the Y axis.
|
---|
265 | """
|
---|
266 |
|
---|
267 | def __init__(self, sId, oData, oDisp = None):
|
---|
268 | WuiHlpLineGraph.__init__(self, sId, oData, fErrorBarY = True);
|
---|
269 |
|
---|
270 |
|
---|
271 | class WuiHlpMiniSuccessRateGraph(WuiHlpGraphMatplotlibBase):
|
---|
272 | """
|
---|
273 | Mini rate graph.
|
---|
274 | """
|
---|
275 |
|
---|
276 | def __init__(self, sId, oData, oDisp = None):
|
---|
277 | """
|
---|
278 | oData must be a WuiHlpGraphDataTableEx like object, but only aoSeries,
|
---|
279 | aoSeries[].aoXValues, and aoSeries[].aoYValues will be used. The
|
---|
280 | values are expected to be a percentage, i.e. values between 0 and 100.
|
---|
281 | """
|
---|
282 | WuiHlpGraphMatplotlibBase.__init__(self, sId, oData, oDisp);
|
---|
283 | self.setFontSize(6);
|
---|
284 |
|
---|
285 | def renderGraph(self): # pylint: disable=too-many-locals
|
---|
286 | assert len(self._oData.aoSeries) == 1;
|
---|
287 | oSeries = self._oData.aoSeries[0];
|
---|
288 |
|
---|
289 | # hacking
|
---|
290 | #self.setWidth(512);
|
---|
291 | #self.setHeight(128);
|
---|
292 | # end
|
---|
293 |
|
---|
294 | oFigure = self._createFigure();
|
---|
295 | from mpl_toolkits.axes_grid.axislines import SubplotZero; # pylint: disable=import-error
|
---|
296 | oAxis = SubplotZero(oFigure, 111);
|
---|
297 | oFigure.add_subplot(oAxis);
|
---|
298 |
|
---|
299 | # Disable all the normal axis.
|
---|
300 | oAxis.axis['right'].set_visible(False)
|
---|
301 | oAxis.axis['top'].set_visible(False)
|
---|
302 | oAxis.axis['bottom'].set_visible(False)
|
---|
303 | oAxis.axis['left'].set_visible(False)
|
---|
304 |
|
---|
305 | # Use the zero axis instead.
|
---|
306 | oAxis.axis['yzero'].set_axisline_style('-|>');
|
---|
307 | oAxis.axis['yzero'].set_visible(True);
|
---|
308 | oAxis.axis['xzero'].set_axisline_style('-|>');
|
---|
309 | oAxis.axis['xzero'].set_visible(True);
|
---|
310 |
|
---|
311 | if oSeries.aoYValues[-1] == 100:
|
---|
312 | sColor = 'green';
|
---|
313 | elif oSeries.aoYValues[-1] > 75:
|
---|
314 | sColor = 'yellow';
|
---|
315 | else:
|
---|
316 | sColor = 'red';
|
---|
317 | oAxis.plot(oSeries.aoXValues, oSeries.aoYValues, '.-', color = sColor, linewidth = 3);
|
---|
318 | oAxis.fill_between(oSeries.aoXValues, oSeries.aoYValues, facecolor = sColor, alpha = 0.5)
|
---|
319 |
|
---|
320 | oAxis.set_xlim(left = -0.01);
|
---|
321 | oAxis.set_xticklabels([]);
|
---|
322 | oAxis.set_xmargin(1);
|
---|
323 |
|
---|
324 | oAxis.set_ylim(bottom = 0, top = 100);
|
---|
325 | oAxis.set_yticks([0, 50, 100]);
|
---|
326 | oAxis.set_ylabel('%');
|
---|
327 | #oAxis.set_yticklabels([]);
|
---|
328 | oAxis.set_yticklabels(['', '%', '']);
|
---|
329 |
|
---|
330 | return self._produceSvg(oFigure, False);
|
---|
331 |
|
---|