[Piglit] [PATCH] framework: Replace custom serialization format with json
Chad Versace
chad at chad-versace.us
Wed Jul 13 15:22:57 PDT 2011
The results file produced by piglit-run.py contains a serialized
TestrunResult, and the serialization format was horridly homebrew. This
commit replaces that insanity with json.
Benefits:
- Net loss of 113 lines of code (ignoring comments and empty lines).
- By using the json module in the Python standard library, serializing
and unserializing is nearly as simple as `json.dump(object, file)`
and `json.load(object, file)`.
- By using a format that is easy to manipulate, it is now simple to
extend Piglit to allow users to embed custom data into the results
file.
As a side effect, the summary file is no longer needed, so it is no longer
produced.
CC: Kenneth Graunke <kenneth at whitecape.org>
CC: Paul Berry <stereotype441 at gmail.com>
Signed-off-by: Chad Versace <chad at chad-versace.us>
---
framework/core.py | 308 +++++++++++++++++-----------------------------
framework/exectest.py | 2 +-
framework/summary.py | 8 +-
piglit-create-summary.py | 76 -----------
piglit-merge-results.py | 4 +-
piglit-run.py | 54 ++++++---
piglit-summary-html.py | 18 ++-
7 files changed, 172 insertions(+), 298 deletions(-)
delete mode 100755 piglit-create-summary.py
diff --git a/framework/core.py b/framework/core.py
index ee56852..f91e92d 100644
--- a/framework/core.py
+++ b/framework/core.py
@@ -24,6 +24,7 @@
# Piglit core
import errno
+import json
import os
import platform
import stat
@@ -73,13 +74,6 @@ def checkDir(dirname, failifexists):
if e.errno != errno.EEXIST:
raise
-# Encode a string
-def encode(text):
- return text.encode("string_escape")
-
-def decode(text):
- return text.decode("string_escape")
-
if 'PIGLIT_BUILD_DIR' in os.environ:
testBinDir = os.environ['PIGLIT_BUILD_DIR'] + '/bin/'
else:
@@ -91,176 +85,85 @@ else:
#############################################################################
class TestResult(dict):
- def __init__(self, *args):
- dict.__init__(self)
-
- assert(len(args) == 0 or len(args) == 2)
-
- if len(args) == 2:
- for k in args[0]:
- self.__setattr__(k, args[0][k])
-
- self.update(args[1])
-
- def __repr__(self):
- attrnames = set(dir(self)) - set(dir(self.__class__()))
- return '%(class)s(%(dir)s,%(dict)s)' % {
- 'class': self.__class__.__name__,
- 'dir': dict([(k, self.__getattribute__(k)) for k in attrnames]),
- 'dict': dict.__repr__(self)
- }
-
- def allTestResults(self, name):
- return {name: self}
-
- def write(self, file, path):
- result = StringIO()
- print >> result, "@test: " + encode(path)
- for k in self:
- v = self[k]
- if type(v) == list:
- print >> result, k + "!"
- for s in v:
- print >> result, " " + encode(str(s))
- print >> result, "!"
- else:
- print >> result, k + ": " + encode(str(v))
- print >> result, "!"
- file.write(result.getvalue())
+ pass
class GroupResult(dict):
- def __init__(self, *args):
- dict.__init__(self)
+ def get_subgroup(self, path, create=True):
+ '''
+ Retrieve subgroup specified by path
- assert(len(args) == 0 or len(args) == 2)
+ For example, ``self.get_subgroup('a/b/c')`` will attempt to
+ return ``self['a']['b']['c']``. If any subgroup along ``path``
+ does not exist, then it will be created if ``create`` is true;
+ otherwise, ``None`` is returned.
+ '''
+ group = self
+ for subname in path.split('/'):
+ if subname not in group:
+ if create:
+ group[subname] = GroupResult()
+ else:
+ return None
+ group = group[subname]
+ assert(isinstance(group, GroupResult))
+ return group
- if len(args) == 2:
- for k in args[0]:
- self.__setattr__(k, args[0][k])
+ @staticmethod
+ def make_tree(tests):
+ '''
+ Convert a flat dict of test results to a hierarchical tree
- self.update(args[1])
+ ``tests`` is a dict whose items have form ``(path, TestResult)``,
+ where path is a string with form ``group1/group2/.../test_name``.
- def __repr__(self):
- attrnames = set(dir(self)) - set(dir(self.__class__()))
- return '%(class)s(%(dir)s,%(dict)s)' % {
- 'class': self.__class__.__name__,
- 'dir': dict([(k, self.__getattribute__(k)) for k in attrnames]),
- 'dict': dict.__repr__(self)
- }
+ Return a tree whose leaves are the values of ``tests`` and
+ whose nodes, which have type ``GroupResult``, reflect the
+ paths in ``tests``.
+ '''
+ root = GroupResult()
- def allTestResults(self, groupName):
- collection = {}
- for name, sub in self.items():
- subfullname = name
- if len(groupName) > 0:
- subfullname = groupName + '/' + subfullname
- collection.update(sub.allTestResults(subfullname))
- return collection
+ for (path, result) in tests.items():
+ group_path = os.path.dirname(path)
+ test_name = os.path.basename(path)
- def write(self, file, groupName):
- for name, sub in self.items():
- subfullname = name
- if len(groupName) > 0:
- subfullname = groupName + '/' + subfullname
- sub.write(file, subfullname)
+ group = root.get_subgroup(group_path)
+ group[test_name] = TestResult(result)
+ return root
class TestrunResult:
- def __init__(self, *args):
- self.name = ''
- self.globalkeys = ['name', 'href', 'glxinfo', 'lspci', 'time']
- self.results = GroupResult()
-
- def allTestResults(self):
- '''Return a dictionary containing (name: TestResult) mappings.
- Note that writing to this dictionary has no effect.'''
- return self.results.allTestResults('')
+ def __init__(self):
+ self.serialized_keys = [
+ 'name',
+ 'tests',
+ 'glxinfo',
+ 'lspci',
+ 'time_elapsed',
+ ]
+ self.name = None
+ self.glxinfo = None
+ self.lspci = None
+ self.tests = {}
def write(self, file):
- for key in self.globalkeys:
- if key in self.__dict__:
- print >>file, "%s: %s" % (key, encode(self.__dict__[key]))
-
- self.results.write(file,'')
+ # Serialize only the keys in serialized_keys.
+ keys = set(self.__dict__.keys()).intersection(self.serialized_keys)
+ raw_dict = dict([(k, self.__dict__[k]) for k in keys])
+ json.dump(raw_dict, file, indent=4)
def parseFile(self, file):
- def arrayparser(a):
- def cb(line):
- if line == '!':
- del stack[-1]
- else:
- a.append(decode(line[1:]))
- return cb
-
- def dictparser(d):
- def cb(line):
- if line == '!':
- del stack[-1]
- return
-
- colon = line.find(':')
- if colon < 0:
- excl = line.find('!')
- if excl < 0:
- raise Exception("Line %(linenr)d: Bad format" % locals())
-
- key = line[:excl]
- d[key] = []
- stack.append(arrayparser(d[key]))
- return
-
- key = line[:colon]
- value = decode(line[colon+2:])
- d[key] = value
- return cb
-
- def toplevel(line):
- colon = line.find(':')
- if colon < 0:
- raise Exception("Line %(linenr)d: Bad format" % locals())
-
- key = line[:colon]
- value = decode(line[colon+2:])
- if key in self.globalkeys:
- self.__dict__[key] = value
- elif key == '@test':
- comp = value.split('/')
- group = self.results
- for name in comp[:-1]:
- if name not in group:
- group[name] = GroupResult()
- group = group[name]
+ raw_dict = json.load(file)
- result = TestResult()
- group[comp[-1]] = result
-
- stack.append(dictparser(result))
- else:
- raise Exception("Line %d: Unknown key %s" % (linenr, key))
-
- stack = [toplevel]
- linenr = 1
- for line in file:
- if line[-1] == '\n':
- stack[-1](line[0:-1])
- linenr = linenr + 1
-
- def parseDir(self, path, PreferSummary):
- main = None
- filelist = [path + '/main', path + '/summary']
- if PreferSummary:
- filelist[:0] = [path + '/summary']
- for filename in filelist:
- try:
- main = open(filename, 'U')
- break
- except:
- pass
- if not main:
- raise Exception("Failed to open %(path)s" % locals())
- self.parseFile(main)
- main.close()
+ # Check that only expected keys were unserialized.
+ for key in raw_dict:
+ if key not in self.serialized_keys:
+ raise Exception('unexpected key in results file: ' + str(key))
+
+ self.__dict__.update(raw_dict)
+ # Replace each raw dict in self.tests with a TestResult.
+ for (path, result) in self.tests.items():
+ self.tests[path] = TestResult(result)
#############################################################################
##### Generic Test classes
@@ -268,7 +171,6 @@ class TestrunResult:
class Environment:
def __init__(self):
- self.file = sys.stdout
self.execute = True
self.filter = []
self.exclude_filter = []
@@ -283,11 +185,12 @@ class Environment:
return stderr+stdout
def collectData(self):
+ result = {}
if platform.system() != 'Windows':
- self.file.write("glxinfo:", '@@@' + encode(self.run('glxinfo')), "\n")
+ result['glxinfo'] = self.run('glxinfo')
if platform.system() == 'Linux':
- self.file.write("lspci:", '@@@' + encode(self.run('lspci')), "\n")
-
+ result['lspci'] = self.run('lspci')
+ return result
class Test:
ignoreErrors = []
@@ -304,13 +207,26 @@ class Test:
def run(self):
raise NotImplementedError
- def doRun(self, env, path):
+ def doRun(self, env, path, testrun):
+ '''
+ Schedule test to be run
+
+ :path:
+ Fully qualified test name as a string. For example,
+ ``spec/glsl-1.30/preprocessor/compiler/keywords/void.frag``.
+
+ :testrun:
+ A TestrunResult object that accumulates test results.
+ After this test has executed, the test's ``TestResult`` is
+ assigned to ``testrun.tests[path]``
+ '''
+ args = (env, path, testrun)
if self.runConcurrent:
- ConcurrentTestPool().put(self.__doRunWork, args = (env, path,))
+ ConcurrentTestPool().put(self.__doRunWork, args=args)
else:
- self.__doRunWork(env, path)
+ self.__doRunWork(*args)
- def __doRunWork(self, env, path):
+ def __doRunWork(self, env, path, testrun):
# Exclude tests that don't match the filter regexp
if len(env.filter) > 0:
if not True in map(lambda f: f.search(path) != None, env.filter):
@@ -336,18 +252,18 @@ class Test:
if 'result' not in result:
result['result'] = 'fail'
if not isinstance(result, TestResult):
- result = TestResult({}, result)
+ result = TestResult(result)
result['result'] = 'warn'
result['note'] = 'Result not returned as an instance of TestResult'
except:
result = TestResult()
result['result'] = 'fail'
result['exception'] = str(sys.exc_info()[0]) + str(sys.exc_info()[1])
- result['traceback'] = '@@@' + "".join(traceback.format_tb(sys.exc_info()[2]))
+ result['traceback'] = "".join(traceback.format_tb(sys.exc_info()[2]))
status(result['result'])
- result.write(env.file, path)
+ testrun.tests[path] = result
if Test.sleep:
time.sleep(Test.sleep)
else:
@@ -379,12 +295,17 @@ class Test:
class Group(dict):
- def doRun(self, env, path):
+ def doRun(self, env, path, testrun):
+ '''
+ Schedule all tests in group for execution.
+
+ See ``Test.doRun``.
+ '''
for sub in sorted(self):
spath = sub
if len(path) > 0:
spath = path + '/' + spath
- self[sub].doRun(env, spath)
+ self[sub].doRun(env, spath, testrun)
class TestProfile:
@@ -392,12 +313,14 @@ class TestProfile:
self.tests = Group()
self.sleep = 0
- def run(self, env):
- time_start = time.time()
- self.tests.doRun(env, '')
+ def run(self, env, testrun):
+ '''
+ Schedule all tests in profile for execution.
+
+ See ``Test.doRun``.
+ '''
+ self.tests.doRun(env, '', testrun)
ConcurrentTestPool().join()
- time_end = time.time()
- env.file.write("time:", time_end-time_start, "\n")
def remove_test(self, test_path):
"""Remove a fully qualified test from the profile.
@@ -428,24 +351,19 @@ def loadTestProfile(filename):
traceback.print_exc()
raise Exception('Could not read tests profile')
-def loadTestResults(path, PreferSummary=False):
+def loadTestResults(path):
+ if os.path.isdir(path):
+ filepath = os.path.join(path, 'main')
+ else:
+ filepath = path
+
+ testrun = TestrunResult()
try:
- mode = os.stat(path)[stat.ST_MODE]
- testrun = TestrunResult()
- if stat.S_ISDIR(mode):
- testrun.parseDir(path, PreferSummary)
- else:
- file = open(path, 'r')
+ with open(filepath, 'r') as file:
testrun.parseFile(file)
- file.close()
-
- if len(testrun.name) == 0:
- if path[-1] == '/':
- testrun.name = os.path.basename(path[0:-1])
- else:
- testrun.name = os.path.basename(path)
-
- return testrun
- except:
+ except OSError:
traceback.print_exc()
raise Exception('Could not read tests results')
+
+ assert(testrun.name is not None)
+ return testrun
diff --git a/framework/exectest.py b/framework/exectest.py
index c80c156..e41cd21 100644
--- a/framework/exectest.py
+++ b/framework/exectest.py
@@ -94,7 +94,7 @@ class PlainExecTest(Test):
self.handleErr(results, err)
- results['info'] = "@@@Returncode: %d\n\nErrors:\n%s\n\nOutput:\n%s" % (proc.returncode, err, out)
+ results['info'] = "Returncode: %d\n\nErrors:\n%s\n\nOutput:\n%s" % (proc.returncode, err, out)
results['returncode'] = proc.returncode
results['command'] = ' '.join(self.command)
else:
diff --git a/framework/summary.py b/framework/summary.py
index c69fd2f..cac0254 100644
--- a/framework/summary.py
+++ b/framework/summary.py
@@ -156,7 +156,7 @@ results is an array of GroupResult instances, one per testrun
childresults
)
else:
- childresults = [r.get(name, core.TestResult({}, { 'result': 'skip' }))
+ childresults = [r.get(name, core.TestResult({ 'result': 'skip' }))
for r in self.results]
self.children[name] = TestSummary(
@@ -188,8 +188,12 @@ class Summary:
"""\
testruns is an array of TestrunResult instances
"""
+ groups = [
+ core.GroupResult.make_tree(testrun.tests)
+ for testrun in testruns
+ ]
self.testruns = testruns
- self.root = GroupSummary(self, '', 'All', [tr.results for tr in testruns])
+ self.root = GroupSummary(self, '', 'All', groups)
def allTests(self):
"""\
diff --git a/piglit-create-summary.py b/piglit-create-summary.py
deleted file mode 100755
index 914d09e..0000000
--- a/piglit-create-summary.py
+++ /dev/null
@@ -1,76 +0,0 @@
-#!/usr/bin/env python
-#
-# Permission is hereby granted, free of charge, to any person
-# obtaining a copy of this software and associated documentation
-# files (the "Software"), to deal in the Software without
-# restriction, including without limitation the rights to use,
-# copy, modify, merge, publish, distribute, sublicense, and/or
-# sell copies of the Software, and to permit persons to whom the
-# Software is furnished to do so, subject to the following
-# conditions:
-#
-# This permission notice shall be included in all copies or
-# substantial portions of the Software.
-#
-# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
-# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
-# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
-# PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHOR(S) BE
-# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
-# AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
-# OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
-# DEALINGS IN THE SOFTWARE.
-
-
-from getopt import getopt, GetoptError
-import sys
-
-import framework.core as core
-
-
-
-#############################################################################
-##### Main program
-#############################################################################
-def usage():
- USAGE = """\
-Usage: %(progName)s [options] [main results file]
-
-Options:
- -h, --help Show this message
-
-Example:
- %(progName)s results/main > results/summary
-"""
- print USAGE % {'progName': sys.argv[0]}
- sys.exit(1)
-
-def main():
- env = core.Environment()
-
- try:
- options, args = getopt(sys.argv[1:], "h", [ "help" ])
- except GetoptError:
- usage()
-
- OptionName = ''
-
- for name, value in options:
- if name in ('-h', '--help'):
- usage()
-
- if len(args) != 1:
- usage()
-
- resultsFilename = args[0]
-
- results = core.loadTestResults(resultsFilename)
- for testname, result in results.allTestResults().items():
- if 'info' in result:
- if len(result['info']) > 4096:
- result['info'] = result['info'][0:4096]
- results.write(sys.stdout)
-
-
-if __name__ == "__main__":
- main()
diff --git a/piglit-merge-results.py b/piglit-merge-results.py
index 4095e48..eff8d5c 100755
--- a/piglit-merge-results.py
+++ b/piglit-merge-results.py
@@ -68,8 +68,8 @@ def main():
for resultsDir in args:
results = core.loadTestResults(resultsDir)
- for testname, result in results.allTestResults().items():
- combined.results[testname] = result
+ for testname, result in results.tests.items():
+ combined.tests[testname] = result
combined.write(sys.stdout)
diff --git a/piglit-run.py b/piglit-run.py
index 1e6d515..d5e85f8 100755
--- a/piglit-run.py
+++ b/piglit-run.py
@@ -23,8 +23,12 @@
from getopt import getopt, GetoptError
+import json
+import os.path as path
import re
import sys, os
+import time
+import traceback
import framework.core as core
from framework.threads import synchronized_self
@@ -120,27 +124,43 @@ def main():
core.checkDir(resultsDir, False)
+ results = core.TestrunResult()
+
+ # Set results.name
+ if OptionName is '':
+ results.name = path.basename(resultsDir)
+ else:
+ results.name = OptionName
+
+ results.__dict__.update(env.collectData())
+
profile = core.loadTestProfile(profileFilename)
- env.file = SyncFileWriter(resultsDir + '/main')
- env.file.writeLine("name: %(name)s" % { 'name': core.encode(OptionName) })
- env.collectData()
- profile.run(env)
- env.file.close()
-
- print "Writing summary file..."
- results = core.loadTestResults(resultsDir)
- for testname, result in results.allTestResults().items():
- if 'info' in result:
- if len(result['info']) > 4096:
- result['info'] = result['info'][0:4096]
- file = open(resultsDir + '/summary', "w")
- results.write(file)
- file.close()
+ time_start = time.time()
+
+ try:
+ profile.run(env, results)
+ except Exception as e:
+ if isinstance(e, KeyboardInterrupt):
+ # When the user interrupts the test run, he may still
+ # want the partial test results. So ignore
+ # KeyboardInterruption and proceed to writing the
+ # result files.
+ pass
+ else:
+ traceback.print_exc()
+ sys.exit(1)
+
+ time_end = time.time()
+ results.time_elapsed = time_end - time_start
+
+ result_filepath = os.path.join(resultsDir, 'main')
+ print("Writing results file...")
+ with open(result_filepath, 'w') as f:
+ results.write(f)
print
print 'Thank you for running Piglit!'
- print 'Summary for submission has been written to ' + resultsDir + '/summary'
-
+ print 'Results have been written to ' + result_filepath
if __name__ == "__main__":
main()
diff --git a/piglit-summary-html.py b/piglit-summary-html.py
index 4e94237..d00e11e 100755
--- a/piglit-summary-html.py
+++ b/piglit-summary-html.py
@@ -88,8 +88,8 @@ def buildDetailValue(detail):
items = items + ResultListItem % { 'detail': buildDetailValue(d) }
return ResultList % { 'items': items }
- elif type(detail) == str and detail[0:3] == '@@@':
- return ResultMString % { 'detail': cgi.escape(detail[3:]) }
+ elif isinstance(detail, str) or isinstance(detail, unicode):
+ return ResultMString % { 'detail': cgi.escape(detail) }
return cgi.escape(str(detail))
@@ -97,7 +97,9 @@ def buildDetailValue(detail):
def buildDetails(testResult):
details = []
for name in testResult:
- if type(name) != str or name == 'result':
+ assert(isinstance(name, basestring))
+
+ if name == 'result':
continue
value = buildDetailValue(testResult[name])
@@ -133,7 +135,13 @@ def writeResultHtml(test, testResult, filename):
writefile(filename, Result % locals())
def writeTestrunHtml(testrun, filename):
- detaildict = dict(filter(lambda item: item[0] in testrun.globalkeys, testrun.__dict__.items()))
+ detail_keys = [
+ key
+ for key in testrun.__dict__.keys()
+ if key in testrun.serialized_keys
+ if key != 'tests'
+ ]
+ detaildict = dict([(k, testrun.__dict__[k]) for k in detail_keys])
details = buildDetails(detaildict)
name = testrun.name
codename = testrun.codename
@@ -281,7 +289,7 @@ def parse_listfile(filename):
return eval(code)
def loadresult(descr, OptionPreferSummary):
- result = core.loadTestResults(descr[0], OptionPreferSummary)
+ result = core.loadTestResults(descr[0])
if len(descr) > 1:
result.__dict__.update(descr[1])
return result
--
1.7.6
More information about the Piglit
mailing list