#!/usr/bin/env python## run-tests.py - Run a set of tests on Mercurial## Copyright 2006 Matt Mackall <mpm@selenic.com>## This software may be used and distributed according to the terms of the# GNU General Public License version 2 or any later version.# Modifying this script is tricky because it has many modes:# - serial (default) vs parallel (-jN, N > 1)# - no coverage (default) vs coverage (-c, -C, -s)# - temp install (default) vs specific hg script (--with-hg, --local)# - tests are a mix of shell scripts and Python scripts## If you change this script, it is recommended that you ensure you# haven't broken it by running it in various modes with a representative# sample of test scripts. For example:## 1) serial, no coverage, temp install:# ./run-tests.py test-s*# 2) serial, no coverage, local hg:# ./run-tests.py --local test-s*# 3) serial, coverage, temp install:# ./run-tests.py -c test-s*# 4) serial, coverage, local hg:# ./run-tests.py -c --local test-s* # unsupported# 5) parallel, no coverage, temp install:# ./run-tests.py -j2 test-s*# 6) parallel, no coverage, local hg:# ./run-tests.py -j2 --local test-s*# 7) parallel, coverage, temp install:# ./run-tests.py -j2 -c test-s* # currently broken# 8) parallel, coverage, local install:# ./run-tests.py -j2 -c --local test-s* # unsupported (and broken)# 9) parallel, custom tmp dir:# ./run-tests.py -j2 --tmpdir /tmp/myhgtests## (You could use any subset of the tests: test-s* happens to match# enough that it's worth doing parallel runs, few enough that it# completes fairly quickly, includes both shell and Python scripts, and# includes some scripts that run daemon processes.)fromdistutilsimportversionimportdifflibimporterrnoimportoptparseimportosimportshutilimportsubprocessimportsignalimportsysimporttempfileimporttimeimportrandomimportreimportthreadingimportkilldaemonsaskillmodimportQueueasqueueprocesslock=threading.Lock()# subprocess._cleanup can race with any Popen.wait or Popen.poll on py24# http://bugs.python.org/issue1731717 for details. We shouldn't be producing# zombies but it's pretty harmless even if we do.ifsys.version_info<(2,5):subprocess._cleanup=lambda:Noneclosefds=os.name=='posix'defPopen4(cmd,wd,timeout,env=None):processlock.acquire()p=subprocess.Popen(cmd,shell=True,bufsize=-1,cwd=wd,env=env,close_fds=closefds,stdin=subprocess.PIPE,stdout=subprocess.PIPE,stderr=subprocess.STDOUT)processlock.release()p.fromchild=p.stdoutp.tochild=p.stdinp.childerr=p.stderrp.timeout=Falseiftimeout:deft():start=time.time()whiletime.time()-start<timeoutandp.returncodeisNone:time.sleep(.1)p.timeout=Trueifp.returncodeisNone:terminate(p)threading.Thread(target=t).start()returnp# reserved exit code to skip test (used by hghave)SKIPPED_STATUS=80SKIPPED_PREFIX='skipped: 'FAILED_PREFIX='hghave check failed: 'PYTHON=sys.executable.replace('\\','/')IMPL_PATH='PYTHONPATH'if'java'insys.platform:IMPL_PATH='JYTHONPATH'requiredtools=[os.path.basename(sys.executable),"diff","grep","unzip","gunzip","bunzip2","sed"]createdfiles=[]defaults={'jobs':('HGTEST_JOBS',1),'timeout':('HGTEST_TIMEOUT',180),'port':('HGTEST_PORT',20059),'shell':('HGTEST_SHELL','sh'),}defparselistfiles(files,listtype,warn=True):entries=dict()forfilenameinfiles:try:path=os.path.expanduser(os.path.expandvars(filename))f=open(path,"r")exceptIOError,err:iferr.errno!=errno.ENOENT:raiseifwarn:print"warning: no such %s file: %s"%(listtype,filename)continueforlineinf.readlines():line=line.split('#',1)[0].strip()ifline:entries[line]=filenamef.close()returnentriesdefgetparser():parser=optparse.OptionParser("%prog [options] [tests]")# keep these sortedparser.add_option("--blacklist",action="append",help="skip tests listed in the specified blacklist file")parser.add_option("--whitelist",action="append",help="always run tests listed in the specified whitelist file")parser.add_option("--changed",type="string",help="run tests that are changed in parent rev or working directory")parser.add_option("-C","--annotate",action="store_true",help="output files annotated with coverage")parser.add_option("-c","--cover",action="store_true",help="print a test coverage report")parser.add_option("-d","--debug",action="store_true",help="debug mode: write output of test scripts to console"" rather than capturing and diff'ing it (disables timeout)")parser.add_option("-f","--first",action="store_true",help="exit on the first test failure")parser.add_option("-H","--htmlcov",action="store_true",help="create an HTML report of the coverage of the files")parser.add_option("-i","--interactive",action="store_true",help="prompt to accept changed output")parser.add_option("-j","--jobs",type="int",help="number of jobs to run in parallel"" (default: $%s or %d)"%defaults['jobs'])parser.add_option("--keep-tmpdir",action="store_true",help="keep temporary directory after running tests")parser.add_option("-k","--keywords",help="run tests matching keywords")parser.add_option("-l","--local",action="store_true",help="shortcut for --with-hg=<testdir>/../hg")parser.add_option("--loop",action="store_true",help="loop tests repeatedly")parser.add_option("-n","--nodiff",action="store_true",help="skip showing test changes")parser.add_option("-p","--port",type="int",help="port on which servers should listen"" (default: $%s or %d)"%defaults['port'])parser.add_option("--compiler",type="string",help="compiler to build with")parser.add_option("--pure",action="store_true",help="use pure Python code instead of C extensions")parser.add_option("-R","--restart",action="store_true",help="restart at last error")parser.add_option("-r","--retest",action="store_true",help="retest failed tests")parser.add_option("-S","--noskips",action="store_true",help="don't report skip tests verbosely")parser.add_option("--shell",type="string",help="shell to use (default: $%s or %s)"%defaults['shell'])parser.add_option("-t","--timeout",type="int",help="kill errant tests after TIMEOUT seconds"" (default: $%s or %d)"%defaults['timeout'])parser.add_option("--time",action="store_true",help="time how long each test takes")parser.add_option("--tmpdir",type="string",help="run tests in the given temporary directory"" (implies --keep-tmpdir)")parser.add_option("-v","--verbose",action="store_true",help="output verbose messages")parser.add_option("--view",type="string",help="external diff viewer")parser.add_option("--with-hg",type="string",metavar="HG",help="test using specified hg script rather than a ""temporary installation")parser.add_option("-3","--py3k-warnings",action="store_true",help="enable Py3k warnings on Python 2.6+")parser.add_option('--extra-config-opt',action="append",help='set the given config opt in the test hgrc')parser.add_option('--random',action="store_true",help='run tests in random order')foroption,(envvar,default)indefaults.items():defaults[option]=type(default)(os.environ.get(envvar,default))parser.set_defaults(**defaults)returnparserdefparseargs(args,parser):(options,args)=parser.parse_args(args)# jython is always pureif'java'insys.platformor'__pypy__'insys.modules:options.pure=Trueifoptions.with_hg:options.with_hg=os.path.expanduser(options.with_hg)ifnot(os.path.isfile(options.with_hg)andos.access(options.with_hg,os.X_OK)):parser.error('--with-hg must specify an executable hg script')ifnotos.path.basename(options.with_hg)=='hg':sys.stderr.write('warning: --with-hg should specify an hg script\n')ifoptions.local:testdir=os.path.dirname(os.path.realpath(sys.argv[0]))hgbin=os.path.join(os.path.dirname(testdir),'hg')ifos.name!='nt'andnotos.access(hgbin,os.X_OK):parser.error('--local specified, but %r not found or not executable'%hgbin)options.with_hg=hgbinoptions.anycoverage=options.coveroroptions.annotateoroptions.htmlcovifoptions.anycoverage:try:importcoveragecovver=version.StrictVersion(coverage.__version__).versionifcovver<(3,3):parser.error('coverage options require coverage 3.3 or later')exceptImportError:parser.error('coverage options now require the coverage package')ifoptions.anycoverageandoptions.local:# this needs some path mangling somewhere, I guessparser.error("sorry, coverage options do not work when --local ""is specified")globalverboseifoptions.verbose:verbose=''ifoptions.tmpdir:options.tmpdir=os.path.expanduser(options.tmpdir)ifoptions.jobs<1:parser.error('--jobs must be positive')ifoptions.interactiveandoptions.debug:parser.error("-i/--interactive and -d/--debug are incompatible")ifoptions.debug:ifoptions.timeout!=defaults['timeout']:sys.stderr.write('warning: --timeout option ignored with --debug\n')options.timeout=0ifoptions.py3k_warnings:ifsys.version_info[:2]<(2,6)orsys.version_info[:2]>=(3,0):parser.error('--py3k-warnings can only be used on Python 2.6+')ifoptions.blacklist:options.blacklist=parselistfiles(options.blacklist,'blacklist')ifoptions.whitelist:options.whitelisted=parselistfiles(options.whitelist,'whitelist')else:options.whitelisted={}return(options,args)defrename(src,dst):"""Like os.rename(), trade atomicity and opened files friendliness for existing destination support. """shutil.copy(src,dst)os.remove(src)defparsehghaveoutput(lines):'''Parse hghave log lines. Return tuple of lists (missing, failed): * the missing/unknown features * the features for which existence check failed'''missing=[]failed=[]forlineinlines:ifline.startswith(SKIPPED_PREFIX):line=line.splitlines()[0]missing.append(line[len(SKIPPED_PREFIX):])elifline.startswith(FAILED_PREFIX):line=line.splitlines()[0]failed.append(line[len(FAILED_PREFIX):])returnmissing,faileddefshowdiff(expected,output,ref,err):printforlineindifflib.unified_diff(expected,output,ref,err):sys.stdout.write(line)verbose=Falsedefvlog(*msg):ifverboseisnotFalse:iolock.acquire()ifverbose:printverbose,forminmsg:printm,printsys.stdout.flush()iolock.release()deflog(*msg):iolock.acquire()ifverbose:printverbose,forminmsg:printm,printsys.stdout.flush()iolock.release()deffindprogram(program):"""Search PATH for a executable program"""forpinos.environ.get('PATH',os.defpath).split(os.pathsep):name=os.path.join(p,program)ifos.name=='nt'oros.access(name,os.X_OK):returnnamereturnNonedefcreatehgrc(path,options):# create a fresh hgrchgrc=open(path,'w')hgrc.write('[ui]\n')hgrc.write('slash = True\n')hgrc.write('interactive = False\n')hgrc.write('[defaults]\n')hgrc.write('backout = -d "0 0"\n')hgrc.write('commit = -d "0 0"\n')hgrc.write('shelve = --date "0 0"\n')hgrc.write('tag = -d "0 0"\n')ifoptions.extra_config_opt:foroptinoptions.extra_config_opt:section,key=opt.split('.',1)assert'='inkey,('extra config opt %s must ''have an = for assignment'%opt)hgrc.write('[%s]\n%s\n'%(section,key))hgrc.close()defcreateenv(options,testtmp,threadtmp,port):env=os.environ.copy()env['TESTTMP']=testtmpenv['HOME']=testtmpenv["HGPORT"]=str(port)env["HGPORT1"]=str(port+1)env["HGPORT2"]=str(port+2)env["HGRCPATH"]=os.path.join(threadtmp,'.hgrc')env["DAEMON_PIDS"]=os.path.join(threadtmp,'daemon.pids')env["HGEDITOR"]=sys.executable+' -c "import sys; sys.exit(0)"'env["HGMERGE"]="internal:merge"env["HGUSER"]="test"env["HGENCODING"]="ascii"env["HGENCODINGMODE"]="strict"# Reset some environment variables to well-known values so that# the tests produce repeatable output.env['LANG']=env['LC_ALL']=env['LANGUAGE']='C'env['TZ']='GMT'env["EMAIL"]="Foo Bar <foo.bar@example.com>"env['COLUMNS']='80'env['TERM']='xterm'forkin('HG HGPROF CDPATH GREP_OPTIONS http_proxy no_proxy '+'NO_PROXY').split():ifkinenv:delenv[k]# unset env related to hooksforkinenv.keys():ifk.startswith('HG_'):delenv[k]returnenvdefchecktools():# Before we go any further, check for pre-requisite tools# stuff from coreutils (cat, rm, etc) are not testedforpinrequiredtools:ifos.name=='nt'andnotp.endswith('.exe'):p+='.exe'found=findprogram(p)iffound:vlog("# Found prerequisite",p,"at",found)else:print"WARNING: Did not find prerequisite tool: "+pdefterminate(proc):"""Terminate subprocess (with fallback for Python versions < 2.6)"""vlog('# Terminating process %d'%proc.pid)try:getattr(proc,'terminate',lambda:os.kill(proc.pid,signal.SIGTERM))()exceptOSError:passdefkilldaemons(pidfile):returnkillmod.killdaemons(pidfile,tryhard=False,remove=True,logfn=vlog)defcleanup(options):ifnotoptions.keep_tmpdir:vlog("# Cleaning up HGTMP",HGTMP)shutil.rmtree(HGTMP,True)forfincreatedfiles:try:os.remove(f)exceptOSError:passdefusecorrectpython():# some tests run python interpreter. they must use same# interpreter we use or bad things will happen.pyexename=sys.platform=='win32'and'python.exe'or'python'ifgetattr(os,'symlink',None):vlog("# Making python executable in test path a symlink to '%s'"%sys.executable)mypython=os.path.join(TMPBINDIR,pyexename)try:ifos.readlink(mypython)==sys.executable:returnos.unlink(mypython)exceptOSError,err:iferr.errno!=errno.ENOENT:raiseiffindprogram(pyexename)!=sys.executable:try:os.symlink(sys.executable,mypython)createdfiles.append(mypython)exceptOSError,err:# child processes may race, which is harmlessiferr.errno!=errno.EEXIST:raiseelse:exedir,exename=os.path.split(sys.executable)vlog("# Modifying search path to find %s as %s in '%s'"%(exename,pyexename,exedir))path=os.environ['PATH'].split(os.pathsep)whileexedirinpath:path.remove(exedir)os.environ['PATH']=os.pathsep.join([exedir]+path)ifnotfindprogram(pyexename):print"WARNING: Cannot find %s in search path"%pyexenamedefinstallhg(options):vlog("# Performing temporary installation of HG")installerrs=os.path.join("tests","install.err")compiler=''ifoptions.compiler:compiler='--compiler '+options.compilerpure=options.pureand"--pure"or""py3=''ifsys.version_info[0]==3:py3='--c2to3'# Run installer in hg rootscript=os.path.realpath(sys.argv[0])hgroot=os.path.dirname(os.path.dirname(script))os.chdir(hgroot)nohome='--home=""'ifos.name=='nt':# The --home="" trick works only on OS where os.sep == '/'# because of a distutils convert_path() fast-path. Avoid it at# least on Windows for now, deal with .pydistutils.cfg bugs# when they happen.nohome=''cmd=('%(exe)s setup.py %(py3)s%(pure)s clean --all'' build %(compiler)s --build-base="%(base)s"'' install --force --prefix="%(prefix)s" --install-lib="%(libdir)s"'' --install-scripts="%(bindir)s" %(nohome)s >%(logfile)s 2>&1'%{'exe':sys.executable,'py3':py3,'pure':pure,'compiler':compiler,'base':os.path.join(HGTMP,"build"),'prefix':INST,'libdir':PYTHONDIR,'bindir':BINDIR,'nohome':nohome,'logfile':installerrs})vlog("# Running",cmd)ifos.system(cmd)==0:ifnotoptions.verbose:os.remove(installerrs)else:f=open(installerrs)forlineinf:printline,f.close()sys.exit(1)os.chdir(TESTDIR)usecorrectpython()ifoptions.py3k_warningsandnotoptions.anycoverage:vlog("# Updating hg command to enable Py3k Warnings switch")f=open(os.path.join(BINDIR,'hg'),'r')lines=[line.rstrip()forlineinf]lines[0]+=' -3'f.close()f=open(os.path.join(BINDIR,'hg'),'w')forlineinlines:f.write(line+'\n')f.close()hgbat=os.path.join(BINDIR,'hg.bat')ifos.path.isfile(hgbat):# hg.bat expects to be put in bin/scripts while run-tests.py# installation layout put it in bin/ directly. Fix itf=open(hgbat,'rb')data=f.read()f.close()if'"%~dp0..\python" "%~dp0hg" %*'indata:data=data.replace('"%~dp0..\python" "%~dp0hg" %*','"%~dp0python" "%~dp0hg" %*')f=open(hgbat,'wb')f.write(data)f.close()else:print'WARNING: cannot fix hg.bat reference to python.exe'ifoptions.anycoverage:custom=os.path.join(TESTDIR,'sitecustomize.py')target=os.path.join(PYTHONDIR,'sitecustomize.py')vlog('# Installing coverage trigger to %s'%target)shutil.copyfile(custom,target)rc=os.path.join(TESTDIR,'.coveragerc')vlog('# Installing coverage rc to %s'%rc)os.environ['COVERAGE_PROCESS_START']=rcfn=os.path.join(INST,'..','.coverage')os.environ['COVERAGE_FILE']=fndefoutputtimes(options):vlog('# Producing time report')times.sort(key=lambdat:(t[1],t[0]),reverse=True)cols='%7.3f%s'print'\n%-7s%s'%('Time','Test')fortest,timetakenintimes:printcols%(timetaken,test)defoutputcoverage(options):vlog('# Producing coverage report')os.chdir(PYTHONDIR)defcovrun(*args):cmd='coverage %s'%' '.join(args)vlog('# Running: %s'%cmd)os.system(cmd)covrun('-c')omit=','.join(os.path.join(x,'*')forxin[BINDIR,TESTDIR])covrun('-i','-r','"--omit=%s"'%omit)# reportifoptions.htmlcov:htmldir=os.path.join(TESTDIR,'htmlcov')covrun('-i','-b','"--directory=%s"'%htmldir,'"--omit=%s"'%omit)ifoptions.annotate:adir=os.path.join(TESTDIR,'annotated')ifnotos.path.isdir(adir):os.mkdir(adir)covrun('-i','-a','"--directory=%s"'%adir,'"--omit=%s"'%omit)defpytest(test,wd,options,replacements,env):py3kswitch=options.py3k_warningsand' -3'or''cmd='%s%s "%s"'%(PYTHON,py3kswitch,test)vlog("# Running",cmd)ifos.name=='nt':replacements.append((r'\r\n','\n'))returnrun(cmd,wd,options,replacements,env)needescape=re.compile(r'[\x00-\x08\x0b-\x1f\x7f-\xff]').searchescapesub=re.compile(r'[\x00-\x08\x0b-\x1f\\\x7f-\xff]').subescapemap=dict((chr(i),r'\x%02x'%i)foriinrange(256))escapemap.update({'\\':'\\\\','\r':r'\r'})defescapef(m):returnescapemap[m.group(0)]defstringescape(s):returnescapesub(escapef,s)defrematch(el,l):try:# use \Z to ensure that the regex matches to the end of the stringifos.name=='nt':returnre.match(el+r'\r?\n\Z',l)returnre.match(el+r'\n\Z',l)exceptre.error:# el is an invalid regexreturnFalsedefglobmatch(el,l):# The only supported special characters are * and ? plus / which also# matches \ on windows. Escaping of these caracters is supported.ifel+'\n'==l:ifos.altsep:# matching on "/" is not needed for this linereturn'-glob'returnTruei,n=0,len(el)res=''whilei<n:c=el[i]i+=1ifc=='\\'andel[i]in'*?\\/':res+=el[i-1:i+1]i+=1elifc=='*':res+='.*'elifc=='?':res+='.'elifc=='/'andos.altsep:res+='[/\\\\]'else:res+=re.escape(c)returnrematch(res,l)deflinematch(el,l):ifel==l:# perfect match (fast)returnTrueifel:ifel.endswith(" (esc)\n"):el=el[:-7].decode('string-escape')+'\n'ifel==loros.name=='nt'andel[:-1]+'\r\n'==l:returnTrueifel.endswith(" (re)\n"):returnrematch(el[:-6],l)ifel.endswith(" (glob)\n"):returnglobmatch(el[:-8],l)ifos.altsepandl.replace('\\','/')==el:return'+glob'returnFalsedeftsttest(test,wd,options,replacements,env):# We generate a shell script which outputs unique markers to line# up script results with our source. These markers include input# line number and the last return codesalt="SALT"+str(time.time())defaddsalt(line,inpython):ifinpython:script.append('%s%d 0\n'%(salt,line))else:script.append('echo %s%s $?\n'%(salt,line))# After we run the shell script, we re-unify the script output# with non-active parts of the source, with synchronization by our# SALT line number markers. The after table contains the# non-active components, ordered by line numberafter={}pos=prepos=-1# Expected shellscript outputexpected={}# We keep track of whether or not we're in a Python block so we# can generate the surrounding doctest magicinpython=False# True or False when in a true or false conditional sectionskipping=Nonedefhghave(reqs):# TODO: do something smarter when all other uses of hghave is gonetdir=TESTDIR.replace('\\','/')proc=Popen4('%s -c "%s/hghave %s"'%(options.shell,tdir,' '.join(reqs)),wd,0)stdout,stderr=proc.communicate()ret=proc.wait()ifwifexited(ret):ret=os.WEXITSTATUS(ret)ifret==2:printstdoutsys.exit(1)returnret==0f=open(test)t=f.readlines()f.close()script=[]ifoptions.debug:script.append('set -x\n')ifos.getenv('MSYSTEM'):script.append('alias pwd="pwd -W"\n')n=0forn,linenumerate(t):ifnotl.endswith('\n'):l+='\n'ifl.startswith('#if'):ifskippingisnotNone:after.setdefault(pos,[]).append(' !!! nested #if\n')skipping=nothghave(l.split()[1:])after.setdefault(pos,[]).append(l)elifl.startswith('#else'):ifskippingisNone:after.setdefault(pos,[]).append(' !!! missing #if\n')skipping=notskippingafter.setdefault(pos,[]).append(l)elifl.startswith('#endif'):ifskippingisNone:after.setdefault(pos,[]).append(' !!! missing #if\n')skipping=Noneafter.setdefault(pos,[]).append(l)elifskipping:after.setdefault(pos,[]).append(l)elifl.startswith(' >>> '):# python inlinesafter.setdefault(pos,[]).append(l)prepos=pospos=nifnotinpython:# we've just entered a Python block, add the headerinpython=Trueaddsalt(prepos,False)# make sure we report the exit codescript.append('%s -m heredoctest <<EOF\n'%PYTHON)addsalt(n,True)script.append(l[2:])elifl.startswith(' ... '):# python inlinesafter.setdefault(prepos,[]).append(l)script.append(l[2:])elifl.startswith(' $ '):# commandsifinpython:script.append("EOF\n")inpython=Falseafter.setdefault(pos,[]).append(l)prepos=pospos=naddsalt(n,False)cmd=l[4:].split()iflen(cmd)==2andcmd[0]=='cd':l=' $ cd %s || exit 1\n'%cmd[1]script.append(l[4:])elifl.startswith(' > '):# continuationsafter.setdefault(prepos,[]).append(l)script.append(l[4:])elifl.startswith(' '):# results# queue up a list of expected resultsexpected.setdefault(pos,[]).append(l[2:])else:ifinpython:script.append("EOF\n")inpython=False# non-command/result - queue up for merged outputafter.setdefault(pos,[]).append(l)ifinpython:script.append("EOF\n")ifskippingisnotNone:after.setdefault(pos,[]).append(' !!! missing #endif\n')addsalt(n+1,False)# Write out the script and execute itname=wd+'.sh'f=open(name,'w')forlinscript:f.write(l)f.close()cmd='%s "%s"'%(options.shell,name)vlog("# Running",cmd)exitcode,output=run(cmd,wd,options,replacements,env)# do not merge output if skipped, return hghave message instead# similarly, with --debug, output is Noneifexitcode==SKIPPED_STATUSoroutputisNone:returnexitcode,output# Merge the script output back into a unified testwarnonly=1# 1: not yet, 2: yes, 3: for sure notifexitcode!=0:# failure has been reportedwarnonly=3# set to "for sure not"pos=-1postout=[]forlinoutput:lout,lcmd=l,Noneifsaltinl:lout,lcmd=l.split(salt,1)iflout:ifnotlout.endswith('\n'):lout+=' (no-eol)\n'# find the expected output at the current positionel=Noneifposinexpectedandexpected[pos]:el=expected[pos].pop(0)r=linematch(el,lout)ifisinstance(r,str):ifr=='+glob':lout=el[:-1]+' (glob)\n'r=''# warn only this lineelifr=='-glob':lout=''.join(el.rsplit(' (glob)',1))r=''# warn only this lineelse:log('\ninfo, unknown linematch result: %r\n'%r)r=Falseifr:postout.append(" "+el)else:ifneedescape(lout):lout=stringescape(lout.rstrip('\n'))+" (esc)\n"postout.append(" "+lout)# let diff deal with itifr!='':# if line failedwarnonly=3# set to "for sure not"elifwarnonly==1:# is "not yet" (and line is warn only)warnonly=2# set to "yes" do warniflcmd:# add on last return coderet=int(lcmd.split()[1])ifret!=0:postout.append(" [%s]\n"%ret)ifposinafter:# merge in non-active test bitspostout+=after.pop(pos)pos=int(lcmd.split()[0])ifposinafter:postout+=after.pop(pos)ifwarnonly==2:exitcode=False# set exitcode to warnedreturnexitcode,postoutwifexited=getattr(os,"WIFEXITED",lambdax:False)defrun(cmd,wd,options,replacements,env):"""Run command in a sub-process, capturing the output (stdout and stderr). Return a tuple (exitcode, output). output is None in debug mode."""# TODO: Use subprocess.Popen if we're running on Python 2.4ifoptions.debug:proc=subprocess.Popen(cmd,shell=True,cwd=wd,env=env)ret=proc.wait()return(ret,None)proc=Popen4(cmd,wd,options.timeout,env)defcleanup():terminate(proc)ret=proc.wait()ifret==0:ret=signal.SIGTERM<<8killdaemons(env['DAEMON_PIDS'])returnretoutput=''proc.tochild.close()try:output=proc.fromchild.read()exceptKeyboardInterrupt:vlog('# Handling keyboard interrupt')cleanup()raiseret=proc.wait()ifwifexited(ret):ret=os.WEXITSTATUS(ret)ifproc.timeout:ret='timeout'ifret:killdaemons(env['DAEMON_PIDS'])ifabort:raiseKeyboardInterrupt()fors,rinreplacements:output=re.sub(s,r,output)returnret,output.splitlines(True)defrunone(options,test,count):'''returns a result element: (code, test, msg)'''defskip(msg):ifoptions.verbose:log("\nSkipping %s: %s"%(testpath,msg))return's',test,msgdeffail(msg,ret):warned=retisFalseifnotoptions.nodiff:log("\n%s: %s%s"%(warnedand'Warning'or'ERROR',test,msg))if(notretandoptions.interactiveandos.path.exists(testpath+".err")):iolock.acquire()print"Accept this change? [n] ",answer=sys.stdin.readline().strip()iolock.release()ifanswer.lower()in"y yes".split():iftest.endswith(".t"):rename(testpath+".err",testpath)else:rename(testpath+".err",testpath+".out")return'.',test,''returnwarnedand'~'or'!',test,msgdefsuccess():return'.',test,''defignore(msg):return'i',test,msgdefdescribe(ret):ifret<0:return'killed by signal %d'%-retreturn'returned error code %d'%rettestpath=os.path.join(TESTDIR,test)err=os.path.join(TESTDIR,test+".err")lctest=test.lower()ifnotos.path.exists(testpath):returnskip("doesn't exist")ifnot(options.whitelistedandtestinoptions.whitelisted):ifoptions.blacklistandtestinoptions.blacklist:returnskip("blacklisted")ifoptions.retestandnotos.path.exists(test+".err"):returnignore("not retesting")ifoptions.keywords:fp=open(test)t=fp.read().lower()+test.lower()fp.close()forkinoptions.keywords.lower().split():ifkint:breakelse:returnignore("doesn't match keyword")ifnotos.path.basename(lctest).startswith("test-"):returnskip("not a test file")forext,func,outintesttypes:iflctest.endswith(ext):runner=funcref=os.path.join(TESTDIR,test+out)breakelse:returnskip("unknown test type")vlog("# Test",test)ifos.path.exists(err):os.remove(err)# Remove any previous output files# Make a tmp subdirectory to work inthreadtmp=os.path.join(HGTMP,"child%d"%count)testtmp=os.path.join(threadtmp,os.path.basename(test))os.mkdir(threadtmp)os.mkdir(testtmp)port=options.port+count*3replacements=[(r':%s\b'%port,':$HGPORT'),(r':%s\b'%(port+1),':$HGPORT1'),(r':%s\b'%(port+2),':$HGPORT2'),]ifos.name=='nt':replacements.append((''.join(c.isalpha()and'[%s%s]'%(c.lower(),c.upper())orcin'/\\'andr'[/\\]'orc.isdigit()andcor'\\'+cforcintesttmp),'$TESTTMP'))else:replacements.append((re.escape(testtmp),'$TESTTMP'))env=createenv(options,testtmp,threadtmp,port)createhgrc(env['HGRCPATH'],options)starttime=time.time()try:ret,out=runner(testpath,testtmp,options,replacements,env)exceptKeyboardInterrupt:endtime=time.time()log('INTERRUPTED: %s (after %d seconds)'%(test,endtime-starttime))raiseendtime=time.time()times.append((test,endtime-starttime))vlog("# Ret was:",ret)killdaemons(env['DAEMON_PIDS'])skipped=(ret==SKIPPED_STATUS)# If we're not in --debug mode and reference output file exists,# check test output against it.ifoptions.debug:refout=None# to match "out is None"elifos.path.exists(ref):f=open(ref,"r")refout=f.read().splitlines(True)f.close()else:refout=[]if(ret!=0orout!=refout)andnotskippedandnotoptions.debug:# Save errors to a file for diagnosisf=open(err,"wb")forlineinout:f.write(line)f.close()ifskipped:ifoutisNone:# debug mode: nothing to parsemissing=['unknown']failed=Noneelse:missing,failed=parsehghaveoutput(out)ifnotmissing:missing=['irrelevant']iffailed:result=fail("hghave failed checking for %s"%failed[-1],ret)skipped=Falseelse:result=skip(missing[-1])elifret=='timeout':result=fail("timed out",ret)elifout!=refout:ifnotoptions.nodiff:iolock.acquire()ifoptions.view:os.system("%s%s%s"%(options.view,ref,err))else:showdiff(refout,out,ref,err)iolock.release()ifret:result=fail("output changed and "+describe(ret),ret)else:result=fail("output changed",ret)elifret:result=fail(describe(ret),ret)else:result=success()ifnotoptions.verbose:iolock.acquire()sys.stdout.write(result[0])sys.stdout.flush()iolock.release()ifnotoptions.keep_tmpdir:shutil.rmtree(threadtmp,True)returnresult_hgpath=Nonedef_gethgpath():"""Return the path to the mercurial package that is actually found by the current Python interpreter."""global_hgpathif_hgpathisnotNone:return_hgpathcmd='%s -c "import mercurial; print (mercurial.__path__[0])"'pipe=os.popen(cmd%PYTHON)try:_hgpath=pipe.read().strip()finally:pipe.close()return_hgpathdef_checkhglib(verb):"""Ensure that the 'mercurial' package imported by python is the one we expect it to be. If not, print a warning to stderr."""expecthg=os.path.join(PYTHONDIR,'mercurial')actualhg=_gethgpath()ifos.path.abspath(actualhg)!=os.path.abspath(expecthg):sys.stderr.write('warning: %s with unexpected mercurial lib: %s\n'' (expected %s)\n'%(verb,actualhg,expecthg))results={'.':[],'!':[],'~':[],'s':[],'i':[]}times=[]iolock=threading.Lock()abort=Falsedefscheduletests(options,tests):jobs=options.jobsdone=queue.Queue()running=0count=0globalabortdefjob(test,count):try:done.put(runone(options,test,count))exceptKeyboardInterrupt:passexcept:# re-raisesdone.put(('!',test,'run-test raised an error, see traceback'))raisetry:whiletestsorrunning:ifnotdone.empty()orrunning==jobsornottests:try:code,test,msg=done.get(True,1)results[code].append((test,msg))ifoptions.firstandcodenotin'.si':breakexceptqueue.Empty:continuerunning-=1iftestsandnotrunning==jobs:test=tests.pop(0)ifoptions.loop:tests.append(test)t=threading.Thread(target=job,name=test,args=(test,count))t.start()running+=1count+=1exceptKeyboardInterrupt:abort=Truedefruntests(options,tests):try:ifINST:installhg(options)_checkhglib("Testing")else:usecorrectpython()ifoptions.restart:orig=list(tests)whiletests:ifos.path.exists(tests[0]+".err"):breaktests.pop(0)ifnottests:print"running all tests"tests=origscheduletests(options,tests)failed=len(results['!'])warned=len(results['~'])tested=len(results['.'])+failed+warnedskipped=len(results['s'])ignored=len(results['i'])printifnotoptions.noskips:forsinresults['s']:print"Skipped %s: %s"%sforsinresults['~']:print"Warned %s: %s"%sforsinresults['!']:print"Failed %s: %s"%s_checkhglib("Tested")print"# Ran %d tests, %d skipped, %d warned, %d failed."%(tested,skipped+ignored,warned,failed)ifresults['!']:print'python hash seed:',os.environ['PYTHONHASHSEED']ifoptions.time:outputtimes(options)ifoptions.anycoverage:outputcoverage(options)exceptKeyboardInterrupt:failed=Trueprint"\ninterrupted!"iffailed:return1ifwarned:return80testtypes=[('.py',pytest,'.out'),('.t',tsttest,'')]defmain(args,parser=None):parser=parserorgetparser()(options,args)=parseargs(args,parser)os.umask(022)checktools()ifnotargs:ifoptions.changed:proc=Popen4('hg st --rev "%s" -man0 .'%options.changed,None,0)stdout,stderr=proc.communicate()args=stdout.strip('\0').split('\0')else:args=os.listdir(".")tests=[tfortinargsifos.path.basename(t).startswith("test-")and(t.endswith(".py")ort.endswith(".t"))]ifoptions.random:random.shuffle(tests)else:# keywords for slow testsslow='svn gendoc check-code-hg'.split()defsortkey(f):# run largest tests first, as they tend to take the longesttry:val=-os.stat(f).st_sizeexceptOSError,e:ife.errno!=errno.ENOENT:raisereturn-1e9# file does not exist, tell earlyforkwinslow:ifkwinf:val*=10returnvaltests.sort(key=sortkey)if'PYTHONHASHSEED'notinos.environ:# use a random python hash seed all the time# we do the randomness ourself to know what seed is usedos.environ['PYTHONHASHSEED']=str(random.getrandbits(32))globalTESTDIR,HGTMP,INST,BINDIR,TMPBINDIR,PYTHONDIR,COVERAGE_FILETESTDIR=os.environ["TESTDIR"]=os.getcwd()ifoptions.tmpdir:options.keep_tmpdir=Truetmpdir=options.tmpdirifos.path.exists(tmpdir):# Meaning of tmpdir has changed since 1.3: we used to create# HGTMP inside tmpdir; now HGTMP is tmpdir. So fail if# tmpdir already exists.print"error: temp dir %r already exists"%tmpdirreturn1# Automatically removing tmpdir sounds convenient, but could# really annoy anyone in the habit of using "--tmpdir=/tmp"# or "--tmpdir=$HOME".#vlog("# Removing temp dir", tmpdir)#shutil.rmtree(tmpdir)os.makedirs(tmpdir)else:d=Noneifos.name=='nt':# without this, we get the default temp dir location, but# in all lowercase, which causes troubles with paths (issue3490)d=os.getenv('TMP')tmpdir=tempfile.mkdtemp('','hgtests.',d)HGTMP=os.environ['HGTMP']=os.path.realpath(tmpdir)ifoptions.with_hg:INST=NoneBINDIR=os.path.dirname(os.path.realpath(options.with_hg))TMPBINDIR=os.path.join(HGTMP,'install','bin')os.makedirs(TMPBINDIR)# This looks redundant with how Python initializes sys.path from# the location of the script being executed. Needed because the# "hg" specified by --with-hg is not the only Python script# executed in the test suite that needs to import 'mercurial'# ... which means it's not really redundant at all.PYTHONDIR=BINDIRelse:INST=os.path.join(HGTMP,"install")BINDIR=os.environ["BINDIR"]=os.path.join(INST,"bin")TMPBINDIR=BINDIRPYTHONDIR=os.path.join(INST,"lib","python")os.environ["BINDIR"]=BINDIRos.environ["PYTHON"]=PYTHONpath=[BINDIR]+os.environ["PATH"].split(os.pathsep)ifTMPBINDIR!=BINDIR:path=[TMPBINDIR]+pathos.environ["PATH"]=os.pathsep.join(path)# Include TESTDIR in PYTHONPATH so that out-of-tree extensions# can run .../tests/run-tests.py test-foo where test-foo# adds an extension to HGRC. Also include run-test.py directory to import# modules like heredoctest.pypath=[PYTHONDIR,TESTDIR,os.path.abspath(os.path.dirname(__file__))]# We have to augment PYTHONPATH, rather than simply replacing# it, in case external libraries are only available via current# PYTHONPATH. (In particular, the Subversion bindings on OS X# are in /opt/subversion.)oldpypath=os.environ.get(IMPL_PATH)ifoldpypath:pypath.append(oldpypath)os.environ[IMPL_PATH]=os.pathsep.join(pypath)COVERAGE_FILE=os.path.join(TESTDIR,".coverage")vlog("# Using TESTDIR",TESTDIR)vlog("# Using HGTMP",HGTMP)vlog("# Using PATH",os.environ["PATH"])vlog("# Using",IMPL_PATH,os.environ[IMPL_PATH])try:returnruntests(options,tests)or0finally:time.sleep(.1)cleanup(options)if__name__=='__main__':sys.exit(main(sys.argv[1:]))