README: add Vulkan into the generic description
[piglit.git] / framework / results.py
blob19a287f6902652bcd6823fafc14a49c94a550ea9
1 # coding=utf-8
2 # Permission is hereby granted, free of charge, to any person
3 # obtaining a copy of this software and associated documentation
4 # files (the "Software"), to deal in the Software without
5 # restriction, including without limitation the rights to use,
6 # copy, modify, merge, publish, distribute, sublicense, and/or
7 # sell copies of the Software, and to permit persons to whom the
8 # Software is furnished to do so, subject to the following
9 # conditions:
11 # This permission notice shall be included in all copies or
12 # substantial portions of the Software.
14 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
15 # KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
16 # WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
17 # PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHOR(S) BE
18 # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
19 # AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
20 # OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21 # DEALINGS IN THE SOFTWARE.
23 """ Module for results generation """
25 import collections
26 import copy
27 import datetime
29 from framework import status, exceptions, grouptools
31 __all__ = [
32 'TestrunResult',
33 'TestResult',
37 class Subtests(collections.abc.MutableMapping):
38 """A dict-like object that stores Statuses as values."""
39 def __init__(self, dict_=None):
40 self.__container = collections.OrderedDict()
42 if dict_ is not None:
43 self.update(dict_)
45 def __setitem__(self, name, value):
46 self.__container[name.lower()] = status.status_lookup(value)
48 def __getitem__(self, name):
49 return self.__container[name.lower()]
51 def __delitem__(self, name):
52 del self.__container[name.lower()]
54 def __iter__(self):
55 return iter(self.__container)
57 def __len__(self):
58 return len(self.__container)
60 def __repr__(self):
61 return repr(self.__container)
63 def to_json(self):
64 res = dict(self)
65 res['__type__'] = 'Subtests'
66 return res
68 @classmethod
69 def from_dict(cls, dict_):
70 if '__type__' in dict_:
71 del dict_['__type__']
73 res = cls(dict_)
75 return res
78 class StringDescriptor(object): # pylint: disable=too-few-public-methods
79 """A Shared data descriptor class for TestResult.
81 This provides a property that can be passed a str or unicode, but always
82 returns a unicode object.
84 """
85 def __init__(self, name, default=''):
86 assert isinstance(default, str)
87 self.__name = name
88 self.__default = default
90 def __get__(self, instance, cls):
91 return getattr(instance, self.__name, self.__default)
93 def __set__(self, instance, value):
94 if isinstance(value, bytes):
95 setattr(instance, self.__name, value.decode('utf-8', 'replace'))
96 elif isinstance(value, str):
97 setattr(instance, self.__name, value)
98 else:
99 raise TypeError('{} attribute must be a unicode or bytes instance, '
100 'but was {}.'.format(self.__name, type(value)))
102 def __delete__(self, instance):
103 raise NotImplementedError
106 class TimeAttribute(object):
107 """Attribute of TestResult for time.
109 This attribute provides a couple of nice helpers. It stores the start and
110 end time and provides methods for getting the total and delta of the times.
113 __slots__ = ['start', 'end']
115 def __init__(self, start=0.0, end=0.0):
116 self.start = start
117 self.end = end
119 @property
120 def total(self):
121 return self.end - self.start
123 @property
124 def delta(self):
125 return str(datetime.timedelta(seconds=self.total))
127 def to_json(self):
128 return {
129 'start': self.start,
130 'end': self.end,
131 '__type__': 'TimeAttribute',
134 @classmethod
135 def from_dict(cls, dict_):
136 dict_ = copy.copy(dict_)
138 if '__type__' in dict_:
139 del dict_['__type__']
140 return cls(**dict_)
143 class TestResult(object):
144 """An object representing the result of a single test."""
145 __slots__ = ['returncode', '_err', '_out', 'time', 'command', 'traceback',
146 'environment', 'subtests', 'dmesg', '__result', 'images',
147 'exception', 'pid']
148 err = StringDescriptor('_err')
149 out = StringDescriptor('_out')
151 def __init__(self, result=None):
152 self.returncode = None
153 self.time = TimeAttribute()
154 self.command = str()
155 self.environment = str()
156 self.subtests = Subtests()
157 self.dmesg = str()
158 self.images = None
159 self.traceback = None
160 self.exception = None
161 self.pid = []
162 if result:
163 self.result = result
164 else:
165 self.__result = status.NOTRUN
167 @property
168 def result(self):
169 """Return the result of the test.
171 If there are subtests return the "worst" value of those subtests. If
172 there are not return the stored value of the test. There is an
173 exception to this rule, and that's if the status is crash; since this
174 status is set by the framework, and can be generated even when some or
175 all unit tests pass.
178 if self.subtests and self.__result != status.CRASH:
179 return max(self.subtests.values())
180 return self.__result
182 @property
183 def raw_result(self):
184 """Get the result of the test without taking subtests into account."""
185 return self.__result
187 @result.setter
188 def result(self, new):
189 try:
190 self.__result = status.status_lookup(new)
191 except exceptions.PiglitInternalError as e:
192 raise exceptions.PiglitFatalError(str(e))
194 def to_json(self):
195 """Return the TestResult as a json serializable object."""
196 obj = {
197 '__type__': 'TestResult',
198 'command': self.command,
199 'environment': self.environment,
200 'err': self.err,
201 'out': self.out,
202 'result': self.result,
203 'returncode': self.returncode,
204 'subtests': self.subtests.to_json(),
205 'time': self.time.to_json(),
206 'exception': self.exception,
207 'traceback': self.traceback,
208 'dmesg': self.dmesg,
209 'images': self.images,
210 'pid': self.pid,
212 return obj
214 @classmethod
215 def from_dict(cls, dict_):
216 """Load an already generated result in dictionary form.
218 This is used as an alternate constructor which converts an existing
219 dictionary into a TestResult object. It converts a key 'result' into a
220 status.Status object
223 # pylint will say that assining to inst.out or inst.err is a non-slot
224 # because self.err and self.out are descriptors, methods that act like
225 # variables. Just silence pylint
226 # pylint: disable=assigning-non-slot
227 inst = cls()
229 for each in ['returncode', 'command', 'exception', 'environment',
230 'traceback', 'dmesg', 'images', 'pid', 'result']:
231 if each in dict_:
232 setattr(inst, each, dict_[each])
234 # Set special instances
235 if 'subtests' in dict_:
236 inst.subtests = Subtests.from_dict(dict_['subtests'])
237 if 'time' in dict_:
238 inst.time = TimeAttribute.from_dict(dict_['time'])
240 # out and err must be set manually to avoid replacing the setter
241 if 'out' in dict_:
242 inst.out = dict_['out']
243 if 'err' in dict_:
244 inst.err = dict_['err']
246 return inst
248 def update(self, dict_):
249 """Update the results and subtests fields from a piglit test.
251 Native piglit tests output their data as valid json, and piglit uses
252 the json module to parse this data. This method consumes that raw
253 dictionary data and updates itself.
256 if 'result' in dict_:
257 self.result = dict_['result']
258 if 'images' in dict_:
259 self.images = dict_['images']
260 elif 'subtest' in dict_:
261 self.subtests.update(dict_['subtest'])
264 class Totals(dict):
265 def __init__(self, *args, **kwargs):
266 super(Totals, self).__init__(*args, **kwargs)
267 for each in status.ALL:
268 each = str(each)
269 if each not in self:
270 self[each] = 0
272 def __bool__(self):
273 # Since totals are prepopulated, calling 'if not <Totals instance>'
274 # will always result in True, this will cause it to return True only if
275 # one of the values is not zero
276 for each in self.values():
277 if each != 0:
278 return True
279 return False
281 def to_json(self):
282 """Convert totals to a json object."""
283 result = copy.copy(self)
284 result['__type__'] = 'Totals'
285 return result
287 @classmethod
288 def from_dict(cls, dict_):
289 """Convert a dictionary into a Totals object."""
290 tots = cls(dict_)
291 if '__type__' in tots:
292 del tots['__type__']
293 return tots
296 class TestrunResult(object):
297 """The result of a single piglit run."""
298 def __init__(self):
299 self.info = {}
300 self.options = {}
301 self.time_elapsed = TimeAttribute()
302 self.tests = collections.OrderedDict()
303 self.totals = collections.defaultdict(Totals)
305 def get_result(self, key):
306 """Get the result of a test or subtest.
308 If neither a test nor a subtest instance exist, then raise the original
309 KeyError generated from looking up <key> in the tests attribute. It is
310 the job of the caller to handle this error.
312 Arguments:
313 key -- the key name of the test to return
316 try:
317 return self.tests[key].result
318 except KeyError as e:
319 name, test = grouptools.splitname(key)
320 try:
321 return self.tests[name].subtests[test]
322 except KeyError:
323 raise e
325 def calculate_group_totals(self):
326 """Calculate the number of passes, fails, etc at each level."""
327 for name, result in self.tests.items():
328 # If there are subtests treat the test as if it is a group instead
329 # of a test.
330 if result.subtests:
331 for res in result.subtests.values():
332 res = str(res)
333 temp = name
335 self.totals[temp][res] += 1
336 while temp:
337 temp = grouptools.groupname(temp)
338 self.totals[temp][res] += 1
339 self.totals['root'][res] += 1
340 else:
341 res = str(result.result)
342 while name:
343 name = grouptools.groupname(name)
344 self.totals[name][res] += 1
345 self.totals['root'][res] += 1
347 def to_json(self):
348 if not self.totals:
349 self.calculate_group_totals()
350 rep = copy.copy(self.__dict__)
351 rep['tests'] = collections.OrderedDict((k, t.to_json())
352 for k, t in self.tests.items())
353 rep['__type__'] = 'TestrunResult'
354 return rep
356 @classmethod
357 def from_dict(cls, dict_, _no_totals=False):
358 """Convert a dictionary into a TestrunResult.
360 This method is meant to be used for loading results from json or
361 similar formats
363 _no_totals is not meant to be used externally, it allows us to control
364 the generation of totals when loading old results formats.
367 res = cls()
368 for name in ['name', 'info', 'options', 'results_version']:
369 value = dict_.get(name)
370 if value:
371 setattr(res, name, value)
373 # Since this is used to load partial metadata when writing final test
374 # results there is no guarantee that this will have a "time_elapsed"
375 # key
376 if 'time_elapsed' in dict_:
377 setattr(res, 'time_elapsed',
378 TimeAttribute.from_dict(dict_['time_elapsed']))
379 res.tests = collections.OrderedDict((n, TestResult.from_dict(t))
380 for n, t in dict_['tests'].items())
382 if not 'totals' in dict_ and not _no_totals:
383 res.calculate_group_totals()
384 else:
385 res.totals = {n: Totals.from_dict(t) for n, t in
386 dict_['totals'].items()}
388 return res