VirtualBox

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

Last change on this file since 85582 was 82968, checked in by vboxsync, 5 years 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.7 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: wuihlpgraphmatplotlib.py 82968 2020-02-04 10:35:17Z vboxsync $
3
4"""
5Test Manager Web-UI - Graph Helpers - Implemented using matplotlib.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2020 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: 82968 $"
30
31# Standard Python Import and extensions installed on the system.
32import re;
33import sys;
34if sys.version_info[0] >= 3:
35 from io import StringIO as StringIO; # pylint: disable=import-error,no-name-in-module,useless-import-alias
36else:
37 from StringIO import StringIO as StringIO; # pylint: disable=import-error,no-name-in-module,useless-import-alias
38
39import matplotlib; # pylint: disable=import-error
40matplotlib.use('Agg'); # Force backend.
41import matplotlib.pyplot; # pylint: disable=import-error
42from numpy import arange as numpy_arange; # pylint: disable=no-name-in-module,import-error,wrong-import-order
43
44# Validation Kit imports.
45from testmanager.webui.wuihlpgraphbase import WuiHlpGraphBase;
46
47
48class 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
92class 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
189class 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
262class 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
271class 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
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