4
4
"""Tests that get run automatically on a merge."""
7
import os, os.path, errno
11
10
from StringIO import StringIO
12
from threading import Thread
15
class NonBlockingReader(Thread):
19
def __init__(self,file):
24
self.result = self.file.read()
27
if self.result is None:
28
raise RuntimeError("read() called before run()")
32
if self.result is None:
33
raise RuntimeError("readlines() called before run()")
34
return self.result.splitlines()
12
from subprocess import Popen, PIPE
13
from signal import SIGKILL, SIGTERM
14
from select import select
16
# Die and kill the kids if no output for 10 minutes. Tune this if if your
17
# slow arsed machine needs it. The main use for this is to keep the pqm
18
# queue flowing without having to give it a lifeless enema.
38
22
"""Call test.py with whatever arguments this script was run with.
91
75
# Drop the template database if it exists - the Makefile does this
92
76
# too, but we can explicity check for errors here
93
77
con = psycopg.connect('dbname=template1')
78
con.set_isolation_level(0)
96
cur.execute('end transaction; drop database launchpad_ftest_template')
81
cur.execute('drop database launchpad_ftest_template')
97
82
except psycopg.ProgrammingError, x:
98
83
if 'does not exist' not in str(x):
86
select count(*) from pg_stat_activity
87
where datname in ('launchpad_dev',
88
'launchpad_ftest_template', 'launchpad_ftest')
90
existing_connections = cur.fetchone()[0]
91
if existing_connections > 0:
92
print 'Cannot rebuild database. There are %d open connections.' % (
149
146
print 'Running tests.'
150
cmd = 'cd %s; %s test.py %s < /dev/null' % (
151
here, sys.executable, ' '.join(sys.argv[1:])
154
proc = popen2.Popen3(cmd, True)
155
stdin, out, err = proc.tochild, proc.fromchild, proc.childerr
157
# Use non-blocking reader threads to cope with differing expectations
158
# from the proess of when to consume data from out and error.
159
errthread = NonBlockingReader(err)
160
outthread = NonBlockingReader(out)
165
exitcode = proc.wait()
166
test_ok = (os.WIFEXITED(exitcode) and os.WEXITSTATUS(exitcode) == 0)
168
errlines = errthread.readlines()
169
dataout = outthread.read()
148
cmd = [sys.executable, 'test.py'] + sys.argv[1:]
150
# This would be simpler if we set stderr=STDOUT to combine the streams
151
proc = Popen(cmd, stdin=PIPE, stdout=PIPE, stderr=PIPE)
154
out = [] # stdout from tests
155
err = [] # stderr from tests
156
open_readers = set([proc.stderr, proc.stdout])
158
rlist, wlist, xlist = select(open_readers, [], [], TIMEOUT)
161
if proc.poll() is None:
163
print 'Tests hung - no output for %d seconds. Killing.' % TIMEOUT
164
killem(proc.pid, SIGTERM)
166
if proc.poll() is None:
167
print 'Not dead yet! - slaughtering mercilessly'
168
killem(proc.pid, SIGKILL)
171
if proc.stdout in rlist:
172
out.append(os.read(proc.stdout.fileno(), 1024))
174
open_readers.remove(proc.stdout)
175
if proc.stderr in rlist:
176
err.append(os.read(proc.stderr.fileno(), 1024))
178
open_readers.remove(proc.stderr)
180
test_ok = (proc.wait() == 0)
172
for line in errlines:
186
for line in err.split('\n'):
173
187
if re.match('^Ran\s\d+\stest(s)?\sin\s[\d\.]+s$', line):
177
191
print '---- test stdout ----'
179
193
print '---- end test stdout ----'
181
195
print '---- test stderr ----'
182
print '\n'.join(errlines)
183
197
print '---- end test stderr ----'
200
def killem(pid, signal):
201
"""Kill the process group leader identified by pid and other group members
203
Note that test.py sets its process to a process group leader.
206
os.killpg(os.getpgid(pid), signal)
208
if x.errno == errno.ESRCH:
186
213
if __name__ == '__main__':