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