[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