summaryrefslogtreecommitdiffstats
path: root/meta/lib/oeqa
diff options
context:
space:
mode:
authorMarkus Lehtonen <markus.lehtonen@linux.intel.com>2017-03-24 16:17:25 +0200
committerRichard Purdie <richard.purdie@linuxfoundation.org>2017-03-27 08:15:06 +0100
commit7132f540419af42e09acdd00a1fa62a44408d68d (patch)
treebbe6154afe1adebbb576873331511631e105623e /meta/lib/oeqa
parent867c2dcbd76b7cfadf03e5237b19d5cbd892dd0d (diff)
downloadpoky-7132f540419af42e09acdd00a1fa62a44408d68d.tar.gz
oe-build-perf-test: pack all buildstat in one file
Write out all buildstats into one big json file, instead of using multiple per-measurement files. Individual buildstats will be indexed using "<test_name>.<measurement_name>" as the key. Also, changes the per-testcase working directories into temporary directories that will be removed after test execution as there are no more per-testcase data files to store permanently. [YOCTO #10582] (From OE-Core rev: a7f2e8915db379021f3409ca640de5d3b054a830) Signed-off-by: Markus Lehtonen <markus.lehtonen@linux.intel.com> Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
Diffstat (limited to 'meta/lib/oeqa')
-rw-r--r--meta/lib/oeqa/buildperf/base.py42
-rw-r--r--meta/lib/oeqa/buildperf/test_basic.py24
2 files changed, 35 insertions, 31 deletions
diff --git a/meta/lib/oeqa/buildperf/base.py b/meta/lib/oeqa/buildperf/base.py
index dd473a0bdc..ffe42dc87b 100644
--- a/meta/lib/oeqa/buildperf/base.py
+++ b/meta/lib/oeqa/buildperf/base.py
@@ -16,6 +16,7 @@ import os
16import re 16import re
17import resource 17import resource
18import socket 18import socket
19import shutil
19import time 20import time
20import unittest 21import unittest
21import xml.etree.ElementTree as ET 22import xml.etree.ElementTree as ET
@@ -127,7 +128,6 @@ class BuildPerfTestResult(unittest.TextTestResult):
127 def startTest(self, test): 128 def startTest(self, test):
128 """Pre-test hook""" 129 """Pre-test hook"""
129 test.base_dir = self.out_dir 130 test.base_dir = self.out_dir
130 os.mkdir(test.out_dir)
131 log.info("Executing test %s: %s", test.name, test.shortDescription()) 131 log.info("Executing test %s: %s", test.name, test.shortDescription())
132 self.stream.write(datetime.now().strftime("[%Y-%m-%d %H:%M:%S] ")) 132 self.stream.write(datetime.now().strftime("[%Y-%m-%d %H:%M:%S] "))
133 super(BuildPerfTestResult, self).startTest(test) 133 super(BuildPerfTestResult, self).startTest(test)
@@ -150,6 +150,16 @@ class BuildPerfTestResult(unittest.TextTestResult):
150 return sorted(compound, key=lambda info: info[1].start_time) 150 return sorted(compound, key=lambda info: info[1].start_time)
151 151
152 152
153 def write_buildstats_json(self):
154 """Write buildstats file"""
155 buildstats = OrderedDict()
156 for _, test, _ in self.all_results():
157 for key, val in test.buildstats.items():
158 buildstats[test.name + '.' + key] = val
159 with open(os.path.join(self.out_dir, 'buildstats.json'), 'w') as fobj:
160 json.dump(buildstats, fobj, cls=ResultsJsonEncoder)
161
162
153 def write_results_json(self): 163 def write_results_json(self):
154 """Write test results into a json-formatted file""" 164 """Write test results into a json-formatted file"""
155 results = OrderedDict([('tester_host', self.hostname), 165 results = OrderedDict([('tester_host', self.hostname),
@@ -221,8 +231,6 @@ class BuildPerfTestResult(unittest.TextTestResult):
221 ET.SubElement(measurement, 'time', 231 ET.SubElement(measurement, 'time',
222 timestamp=vals['start_time'].isoformat()).text = \ 232 timestamp=vals['start_time'].isoformat()).text = \
223 str(vals['elapsed_time'].total_seconds()) 233 str(vals['elapsed_time'].total_seconds())
224 if 'buildstats_file' in vals:
225 ET.SubElement(measurement, 'buildstats_file').text = vals['buildstats_file']
226 attrib = dict((k, str(v)) for k, v in vals['iostat'].items()) 234 attrib = dict((k, str(v)) for k, v in vals['iostat'].items())
227 ET.SubElement(measurement, 'iostat', attrib=attrib) 235 ET.SubElement(measurement, 'iostat', attrib=attrib)
228 attrib = dict((k, str(v)) for k, v in vals['rusage'].items()) 236 attrib = dict((k, str(v)) for k, v in vals['rusage'].items())
@@ -238,7 +246,6 @@ class BuildPerfTestResult(unittest.TextTestResult):
238 dom_doc = minidom.parseString(ET.tostring(top, 'utf-8')) 246 dom_doc = minidom.parseString(ET.tostring(top, 'utf-8'))
239 with open(os.path.join(self.out_dir, 'results.xml'), 'w') as fobj: 247 with open(os.path.join(self.out_dir, 'results.xml'), 'w') as fobj:
240 dom_doc.writexml(fobj, addindent=' ', newl='\n', encoding='utf-8') 248 dom_doc.writexml(fobj, addindent=' ', newl='\n', encoding='utf-8')
241 return
242 249
243 250
244class BuildPerfTestCase(unittest.TestCase): 251class BuildPerfTestCase(unittest.TestCase):
@@ -254,6 +261,7 @@ class BuildPerfTestCase(unittest.TestCase):
254 self.start_time = None 261 self.start_time = None
255 self.elapsed_time = None 262 self.elapsed_time = None
256 self.measurements = OrderedDict() 263 self.measurements = OrderedDict()
264 self.buildstats = OrderedDict()
257 # self.err is supposed to be a tuple from sys.exc_info() 265 # self.err is supposed to be a tuple from sys.exc_info()
258 self.err = None 266 self.err = None
259 self.bb_vars = get_bb_vars() 267 self.bb_vars = get_bb_vars()
@@ -263,17 +271,24 @@ class BuildPerfTestCase(unittest.TestCase):
263 self.sizes = [] 271 self.sizes = []
264 272
265 @property 273 @property
266 def out_dir(self): 274 def tmp_dir(self):
267 return os.path.join(self.base_dir, self.name) 275 return os.path.join(self.base_dir, self.name + '.tmp')
268 276
269 def shortDescription(self): 277 def shortDescription(self):
270 return super(BuildPerfTestCase, self).shortDescription() or "" 278 return super(BuildPerfTestCase, self).shortDescription() or ""
271 279
272 def setUp(self): 280 def setUp(self):
273 """Set-up fixture for each test""" 281 """Set-up fixture for each test"""
282 if not os.path.isdir(self.tmp_dir):
283 os.mkdir(self.tmp_dir)
274 if self.build_target: 284 if self.build_target:
275 self.run_cmd(['bitbake', self.build_target, '-c', 'fetchall']) 285 self.run_cmd(['bitbake', self.build_target, '-c', 'fetchall'])
276 286
287 def tearDown(self):
288 """Tear-down fixture for each test"""
289 if os.path.isdir(self.tmp_dir):
290 shutil.rmtree(self.tmp_dir)
291
277 def run(self, *args, **kwargs): 292 def run(self, *args, **kwargs):
278 """Run test""" 293 """Run test"""
279 self.start_time = datetime.now() 294 self.start_time = datetime.now()
@@ -349,9 +364,7 @@ class BuildPerfTestCase(unittest.TestCase):
349 ('rusage', data['rusage']), 364 ('rusage', data['rusage']),
350 ('iostat', data['iostat'])]) 365 ('iostat', data['iostat'])])
351 if save_bs: 366 if save_bs:
352 bs_file = self.save_buildstats(legend) 367 self.save_buildstats(name)
353 measurement['values']['buildstats_file'] = \
354 os.path.relpath(bs_file, self.base_dir)
355 368
356 self._append_measurement(measurement) 369 self._append_measurement(measurement)
357 370
@@ -379,7 +392,7 @@ class BuildPerfTestCase(unittest.TestCase):
379 # Append to 'sizes' array for globalres log 392 # Append to 'sizes' array for globalres log
380 self.sizes.append(str(size)) 393 self.sizes.append(str(size))
381 394
382 def save_buildstats(self, label=None): 395 def save_buildstats(self, measurement_name):
383 """Save buildstats""" 396 """Save buildstats"""
384 def split_nevr(nevr): 397 def split_nevr(nevr):
385 """Split name and version information from recipe "nevr" string""" 398 """Split name and version information from recipe "nevr" string"""
@@ -451,14 +464,7 @@ class BuildPerfTestCase(unittest.TestCase):
451 task)) 464 task))
452 buildstats.append(recipe_bs) 465 buildstats.append(recipe_bs)
453 466
454 # Write buildstats into json file 467 self.buildstats[measurement_name] = buildstats
455 postfix = '.' + str_to_fn(label) if label else ''
456 postfix += '.json'
457 outfile = os.path.join(self.out_dir, 'buildstats' + postfix)
458 with open(outfile, 'w') as fobj:
459 json.dump(buildstats, fobj, indent=4, sort_keys=True,
460 cls=ResultsJsonEncoder)
461 return outfile
462 468
463 def rm_tmp(self): 469 def rm_tmp(self):
464 """Cleanup temporary/intermediate files and directories""" 470 """Cleanup temporary/intermediate files and directories"""
diff --git a/meta/lib/oeqa/buildperf/test_basic.py b/meta/lib/oeqa/buildperf/test_basic.py
index 1333407a65..a9e4a5b731 100644
--- a/meta/lib/oeqa/buildperf/test_basic.py
+++ b/meta/lib/oeqa/buildperf/test_basic.py
@@ -51,21 +51,19 @@ class Test1P3(BuildPerfTestCase):
51 51
52 def test13(self): 52 def test13(self):
53 """Build core-image-sato with rm_work enabled""" 53 """Build core-image-sato with rm_work enabled"""
54 postfile = os.path.join(self.out_dir, 'postfile.conf') 54 postfile = os.path.join(self.tmp_dir, 'postfile.conf')
55 with open(postfile, 'w') as fobj: 55 with open(postfile, 'w') as fobj:
56 fobj.write('INHERIT += "rm_work"\n') 56 fobj.write('INHERIT += "rm_work"\n')
57 try: 57
58 self.rm_tmp() 58 self.rm_tmp()
59 self.rm_sstate() 59 self.rm_sstate()
60 self.rm_cache() 60 self.rm_cache()
61 self.sync() 61 self.sync()
62 cmd = ['bitbake', '-R', postfile, self.build_target] 62 cmd = ['bitbake', '-R', postfile, self.build_target]
63 self.measure_cmd_resources(cmd, 'build', 63 self.measure_cmd_resources(cmd, 'build',
64 'bitbake' + self.build_target, 64 'bitbake' + self.build_target,
65 save_bs=True) 65 save_bs=True)
66 self.measure_disk_usage(self.bb_vars['TMPDIR'], 'tmpdir', 'tmpdir') 66 self.measure_disk_usage(self.bb_vars['TMPDIR'], 'tmpdir', 'tmpdir')
67 finally:
68 os.unlink(postfile)
69 67
70 68
71class Test2(BuildPerfTestCase): 69class Test2(BuildPerfTestCase):