midori/waf-modules/wafadmin/Tools/unittestw.py
2013-10-23 20:26:27 -07:00

211 lines
6.8 KiB
Python

#! /usr/bin/env python
# encoding: utf-8
import os,sys
import Build,TaskGen,Utils,Options,Logs,Task
from TaskGen import before,after,feature
from Constants import*
class unit_test(object):
def __init__(self):
self.returncode_ok=0
self.num_tests_ok=0
self.num_tests_failed=0
self.num_tests_err=0
self.total_num_tests=0
self.max_label_length=0
self.unit_tests=Utils.ordered_dict()
self.unit_test_results={}
self.unit_test_erroneous={}
self.change_to_testfile_dir=False
self.want_to_see_test_output=False
self.want_to_see_test_error=False
self.run_if_waf_does='check'
def run(self):
self.num_tests_ok=0
self.num_tests_failed=0
self.num_tests_err=0
self.total_num_tests=0
self.max_label_length=0
self.unit_tests=Utils.ordered_dict()
self.unit_test_results={}
self.unit_test_erroneous={}
ld_library_path=[]
if not Options.commands[self.run_if_waf_does]:return
for obj in Build.bld.all_task_gen:
try:
link_task=obj.link_task
except AttributeError:
pass
else:
lib_path=link_task.outputs[0].parent.abspath(obj.env)
if lib_path not in ld_library_path:
ld_library_path.append(lib_path)
unit_test=getattr(obj,'unit_test','')
if unit_test and'cprogram'in obj.features:
try:
output=obj.path
filename=os.path.join(output.abspath(obj.env),obj.target)
srcdir=output.abspath()
label=os.path.join(output.bldpath(obj.env),obj.target)
self.max_label_length=max(self.max_label_length,len(label))
self.unit_tests[label]=(filename,srcdir)
except KeyError:
pass
self.total_num_tests=len(self.unit_tests)
Utils.pprint('GREEN','Running the unit tests')
count=0
result=1
for label in self.unit_tests.allkeys:
file_and_src=self.unit_tests[label]
filename=file_and_src[0]
srcdir=file_and_src[1]
count+=1
line=Build.bld.progress_line(count,self.total_num_tests,Logs.colors.GREEN,Logs.colors.NORMAL)
if Options.options.progress_bar and line:
sys.stderr.write(line)
sys.stderr.flush()
try:
kwargs={}
kwargs['env']=os.environ.copy()
if self.change_to_testfile_dir:
kwargs['cwd']=srcdir
if not self.want_to_see_test_output:
kwargs['stdout']=Utils.pproc.PIPE
if not self.want_to_see_test_error:
kwargs['stderr']=Utils.pproc.PIPE
if ld_library_path:
v=kwargs['env']
def add_path(dct,path,var):
dct[var]=os.pathsep.join(Utils.to_list(path)+[os.environ.get(var,'')])
if sys.platform=='win32':
add_path(v,ld_library_path,'PATH')
elif sys.platform=='darwin':
add_path(v,ld_library_path,'DYLD_LIBRARY_PATH')
add_path(v,ld_library_path,'LD_LIBRARY_PATH')
else:
add_path(v,ld_library_path,'LD_LIBRARY_PATH')
pp=Utils.pproc.Popen(filename,**kwargs)
(out,err)=pp.communicate()
result=int(pp.returncode==self.returncode_ok)
if result:
self.num_tests_ok+=1
else:
self.num_tests_failed+=1
self.unit_test_results[label]=result
self.unit_test_erroneous[label]=0
except OSError:
self.unit_test_erroneous[label]=1
self.num_tests_err+=1
except KeyboardInterrupt:
pass
if Options.options.progress_bar:sys.stdout.write(Logs.colors.cursor_on)
def print_results(self):
if not Options.commands[self.run_if_waf_does]:return
p=Utils.pprint
if self.total_num_tests==0:
p('YELLOW','No unit tests present')
return
for label in self.unit_tests.allkeys:
filename=self.unit_tests[label]
err=0
result=0
try:err=self.unit_test_erroneous[label]
except KeyError:pass
try:result=self.unit_test_results[label]
except KeyError:pass
n=self.max_label_length-len(label)
if err:n+=4
elif result:n+=7
else:n+=3
line='%s %s'%(label,'.'*n)
if err:p('RED','%sERROR'%line)
elif result:p('GREEN','%sOK'%line)
else:p('YELLOW','%sFAILED'%line)
percentage_ok=float(self.num_tests_ok)/float(self.total_num_tests)*100.0
percentage_failed=float(self.num_tests_failed)/float(self.total_num_tests)*100.0
percentage_erroneous=float(self.num_tests_err)/float(self.total_num_tests)*100.0
p('NORMAL','''
Successful tests: %i (%.1f%%)
Failed tests: %i (%.1f%%)
Erroneous tests: %i (%.1f%%)
Total number of tests: %i
'''%(self.num_tests_ok,percentage_ok,self.num_tests_failed,percentage_failed,self.num_tests_err,percentage_erroneous,self.total_num_tests))
p('GREEN','Unit tests finished')
import threading
testlock=threading.Lock()
def set_options(opt):
opt.add_option('--alltests',action='store_true',default=True,help='Exec all unit tests',dest='all_tests')
def make_test(self):
if not'cprogram'in self.features:
Logs.error('test cannot be executed %s'%self)
return
self.default_install_path=None
self.create_task('utest',self.link_task.outputs)
def exec_test(self):
status=0
variant=self.env.variant()
filename=self.inputs[0].abspath(self.env)
self.ut_exec=getattr(self,'ut_exec',[filename])
if getattr(self.generator,'ut_fun',None):
self.generator.ut_fun(self)
try:
fu=getattr(self.generator.bld,'all_test_paths')
except AttributeError:
fu=os.environ.copy()
self.generator.bld.all_test_paths=fu
lst=[]
for obj in self.generator.bld.all_task_gen:
link_task=getattr(obj,'link_task',None)
if link_task and link_task.env.variant()==variant:
lst.append(link_task.outputs[0].parent.abspath(obj.env))
def add_path(dct,path,var):
dct[var]=os.pathsep.join(Utils.to_list(path)+[os.environ.get(var,'')])
if sys.platform=='win32':
add_path(fu,lst,'PATH')
elif sys.platform=='darwin':
add_path(fu,lst,'DYLD_LIBRARY_PATH')
add_path(fu,lst,'LD_LIBRARY_PATH')
else:
add_path(fu,lst,'LD_LIBRARY_PATH')
cwd=getattr(self.generator,'ut_cwd','')or self.inputs[0].parent.abspath(self.env)
proc=Utils.pproc.Popen(self.ut_exec,cwd=cwd,env=fu,stderr=Utils.pproc.PIPE,stdout=Utils.pproc.PIPE)
(stdout,stderr)=proc.communicate()
tup=(filename,proc.returncode,stdout,stderr)
self.generator.utest_result=tup
testlock.acquire()
try:
bld=self.generator.bld
Logs.debug("ut: %r",tup)
try:
bld.utest_results.append(tup)
except AttributeError:
bld.utest_results=[tup]
finally:
testlock.release()
cls=Task.task_type_from_func('utest',func=exec_test,color='PINK',ext_in='.bin')
old=cls.runnable_status
def test_status(self):
ret=old(self)
if ret==SKIP_ME and getattr(Options.options,'all_tests',False):
return RUN_ME
return ret
cls.runnable_status=test_status
cls.quiet=1
def summary(bld):
lst=getattr(bld,'utest_results',[])
if lst:
Utils.pprint('CYAN','execution summary')
total=len(lst)
tfail=len([x for x in lst if x[1]])
Utils.pprint('CYAN',' tests that pass %d/%d'%(total-tfail,total))
for(f,code,out,err)in lst:
if not code:
Utils.pprint('CYAN',' %s'%f)
Utils.pprint('CYAN',' tests that fail %d/%d'%(tfail,total))
for(f,code,out,err)in lst:
if code:
Utils.pprint('CYAN',' %s'%f)
feature('test')(make_test)
after('apply_link','vars_target_cprogram')(make_test)