VirtualBox

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

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

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.9 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: wuihlpgraphmatplotlib.py 106061 2024-09-16 14:03:52Z vboxsync $
3
4"""
5Test Manager Web-UI - Graph Helpers - Implemented using matplotlib.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2024 Oracle and/or its affiliates.
11
12This file is part of VirtualBox base platform packages, as
13available from https://www.virtualbox.org.
14
15This program is free software; you can redistribute it and/or
16modify it under the terms of the GNU General Public License
17as published by the Free Software Foundation, in version 3 of the
18License.
19
20This program is distributed in the hope that it will be useful, but
21WITHOUT ANY WARRANTY; without even the implied warranty of
22MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
23General Public License for more details.
24
25You should have received a copy of the GNU General Public License
26along with this program; if not, see <https://www.gnu.org/licenses>.
27
28The contents of this file may alternatively be used under the terms
29of the Common Development and Distribution License Version 1.0
30(CDDL), a copy of it is provided in the "COPYING.CDDL" file included
31in the VirtualBox distribution, in which case the provisions of the
32CDDL are applicable instead of those of the GPL.
33
34You may elect to license modified versions of this file under the
35terms and conditions of either the GPL or the CDDL or both.
36
37SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
38"""
39__version__ = "$Revision: 106061 $"
40
41# Standard Python Import and extensions installed on the system.
42import re;
43import sys;
44if sys.version_info[0] >= 3:
45 from io import StringIO as StringIO; # pylint: disable=import-error,no-name-in-module,useless-import-alias
46else:
47 from StringIO import StringIO as StringIO; # pylint: disable=import-error,no-name-in-module,useless-import-alias
48
49import matplotlib; # pylint: disable=import-error
50matplotlib.use('Agg'); # Force backend.
51import matplotlib.pyplot; # pylint: disable=import-error
52from numpy import arange as numpy_arange; # pylint: disable=no-name-in-module,import-error,wrong-import-order
53
54# Validation Kit imports.
55from testmanager.webui.wuihlpgraphbase import WuiHlpGraphBase;
56
57
58class 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
102class 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
197class 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
270class 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
279class 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
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