##// END OF EJS Templates
run-tests: fix `HGTESTEXTRAEXTENSIONS` with py3...
Matt Harbison -
r46707:af3a6900 default
parent child Browse files
Show More
@@ -1,3819 +1,3819 b''
1 #!/usr/bin/env python3
1 #!/usr/bin/env python3
2 #
2 #
3 # run-tests.py - Run a set of tests on Mercurial
3 # run-tests.py - Run a set of tests on Mercurial
4 #
4 #
5 # Copyright 2006 Matt Mackall <mpm@selenic.com>
5 # Copyright 2006 Matt Mackall <mpm@selenic.com>
6 #
6 #
7 # This software may be used and distributed according to the terms of the
7 # This software may be used and distributed according to the terms of the
8 # GNU General Public License version 2 or any later version.
8 # GNU General Public License version 2 or any later version.
9
9
10 # Modifying this script is tricky because it has many modes:
10 # Modifying this script is tricky because it has many modes:
11 # - serial (default) vs parallel (-jN, N > 1)
11 # - serial (default) vs parallel (-jN, N > 1)
12 # - no coverage (default) vs coverage (-c, -C, -s)
12 # - no coverage (default) vs coverage (-c, -C, -s)
13 # - temp install (default) vs specific hg script (--with-hg, --local)
13 # - temp install (default) vs specific hg script (--with-hg, --local)
14 # - tests are a mix of shell scripts and Python scripts
14 # - tests are a mix of shell scripts and Python scripts
15 #
15 #
16 # If you change this script, it is recommended that you ensure you
16 # If you change this script, it is recommended that you ensure you
17 # haven't broken it by running it in various modes with a representative
17 # haven't broken it by running it in various modes with a representative
18 # sample of test scripts. For example:
18 # sample of test scripts. For example:
19 #
19 #
20 # 1) serial, no coverage, temp install:
20 # 1) serial, no coverage, temp install:
21 # ./run-tests.py test-s*
21 # ./run-tests.py test-s*
22 # 2) serial, no coverage, local hg:
22 # 2) serial, no coverage, local hg:
23 # ./run-tests.py --local test-s*
23 # ./run-tests.py --local test-s*
24 # 3) serial, coverage, temp install:
24 # 3) serial, coverage, temp install:
25 # ./run-tests.py -c test-s*
25 # ./run-tests.py -c test-s*
26 # 4) serial, coverage, local hg:
26 # 4) serial, coverage, local hg:
27 # ./run-tests.py -c --local test-s* # unsupported
27 # ./run-tests.py -c --local test-s* # unsupported
28 # 5) parallel, no coverage, temp install:
28 # 5) parallel, no coverage, temp install:
29 # ./run-tests.py -j2 test-s*
29 # ./run-tests.py -j2 test-s*
30 # 6) parallel, no coverage, local hg:
30 # 6) parallel, no coverage, local hg:
31 # ./run-tests.py -j2 --local test-s*
31 # ./run-tests.py -j2 --local test-s*
32 # 7) parallel, coverage, temp install:
32 # 7) parallel, coverage, temp install:
33 # ./run-tests.py -j2 -c test-s* # currently broken
33 # ./run-tests.py -j2 -c test-s* # currently broken
34 # 8) parallel, coverage, local install:
34 # 8) parallel, coverage, local install:
35 # ./run-tests.py -j2 -c --local test-s* # unsupported (and broken)
35 # ./run-tests.py -j2 -c --local test-s* # unsupported (and broken)
36 # 9) parallel, custom tmp dir:
36 # 9) parallel, custom tmp dir:
37 # ./run-tests.py -j2 --tmpdir /tmp/myhgtests
37 # ./run-tests.py -j2 --tmpdir /tmp/myhgtests
38 # 10) parallel, pure, tests that call run-tests:
38 # 10) parallel, pure, tests that call run-tests:
39 # ./run-tests.py --pure `grep -l run-tests.py *.t`
39 # ./run-tests.py --pure `grep -l run-tests.py *.t`
40 #
40 #
41 # (You could use any subset of the tests: test-s* happens to match
41 # (You could use any subset of the tests: test-s* happens to match
42 # enough that it's worth doing parallel runs, few enough that it
42 # enough that it's worth doing parallel runs, few enough that it
43 # completes fairly quickly, includes both shell and Python scripts, and
43 # completes fairly quickly, includes both shell and Python scripts, and
44 # includes some scripts that run daemon processes.)
44 # includes some scripts that run daemon processes.)
45
45
46 from __future__ import absolute_import, print_function
46 from __future__ import absolute_import, print_function
47
47
48 import argparse
48 import argparse
49 import collections
49 import collections
50 import contextlib
50 import contextlib
51 import difflib
51 import difflib
52 import distutils.version as version
52 import distutils.version as version
53 import errno
53 import errno
54 import json
54 import json
55 import multiprocessing
55 import multiprocessing
56 import os
56 import os
57 import platform
57 import platform
58 import random
58 import random
59 import re
59 import re
60 import shutil
60 import shutil
61 import signal
61 import signal
62 import socket
62 import socket
63 import subprocess
63 import subprocess
64 import sys
64 import sys
65 import sysconfig
65 import sysconfig
66 import tempfile
66 import tempfile
67 import threading
67 import threading
68 import time
68 import time
69 import unittest
69 import unittest
70 import uuid
70 import uuid
71 import xml.dom.minidom as minidom
71 import xml.dom.minidom as minidom
72
72
73 try:
73 try:
74 import Queue as queue
74 import Queue as queue
75 except ImportError:
75 except ImportError:
76 import queue
76 import queue
77
77
78 try:
78 try:
79 import shlex
79 import shlex
80
80
81 shellquote = shlex.quote
81 shellquote = shlex.quote
82 except (ImportError, AttributeError):
82 except (ImportError, AttributeError):
83 import pipes
83 import pipes
84
84
85 shellquote = pipes.quote
85 shellquote = pipes.quote
86
86
87 processlock = threading.Lock()
87 processlock = threading.Lock()
88
88
89 pygmentspresent = False
89 pygmentspresent = False
90 # ANSI color is unsupported prior to Windows 10
90 # ANSI color is unsupported prior to Windows 10
91 if os.name != 'nt':
91 if os.name != 'nt':
92 try: # is pygments installed
92 try: # is pygments installed
93 import pygments
93 import pygments
94 import pygments.lexers as lexers
94 import pygments.lexers as lexers
95 import pygments.lexer as lexer
95 import pygments.lexer as lexer
96 import pygments.formatters as formatters
96 import pygments.formatters as formatters
97 import pygments.token as token
97 import pygments.token as token
98 import pygments.style as style
98 import pygments.style as style
99
99
100 pygmentspresent = True
100 pygmentspresent = True
101 difflexer = lexers.DiffLexer()
101 difflexer = lexers.DiffLexer()
102 terminal256formatter = formatters.Terminal256Formatter()
102 terminal256formatter = formatters.Terminal256Formatter()
103 except ImportError:
103 except ImportError:
104 pass
104 pass
105
105
106 if pygmentspresent:
106 if pygmentspresent:
107
107
108 class TestRunnerStyle(style.Style):
108 class TestRunnerStyle(style.Style):
109 default_style = ""
109 default_style = ""
110 skipped = token.string_to_tokentype("Token.Generic.Skipped")
110 skipped = token.string_to_tokentype("Token.Generic.Skipped")
111 failed = token.string_to_tokentype("Token.Generic.Failed")
111 failed = token.string_to_tokentype("Token.Generic.Failed")
112 skippedname = token.string_to_tokentype("Token.Generic.SName")
112 skippedname = token.string_to_tokentype("Token.Generic.SName")
113 failedname = token.string_to_tokentype("Token.Generic.FName")
113 failedname = token.string_to_tokentype("Token.Generic.FName")
114 styles = {
114 styles = {
115 skipped: '#e5e5e5',
115 skipped: '#e5e5e5',
116 skippedname: '#00ffff',
116 skippedname: '#00ffff',
117 failed: '#7f0000',
117 failed: '#7f0000',
118 failedname: '#ff0000',
118 failedname: '#ff0000',
119 }
119 }
120
120
121 class TestRunnerLexer(lexer.RegexLexer):
121 class TestRunnerLexer(lexer.RegexLexer):
122 testpattern = r'[\w-]+\.(t|py)(#[a-zA-Z0-9_\-\.]+)?'
122 testpattern = r'[\w-]+\.(t|py)(#[a-zA-Z0-9_\-\.]+)?'
123 tokens = {
123 tokens = {
124 'root': [
124 'root': [
125 (r'^Skipped', token.Generic.Skipped, 'skipped'),
125 (r'^Skipped', token.Generic.Skipped, 'skipped'),
126 (r'^Failed ', token.Generic.Failed, 'failed'),
126 (r'^Failed ', token.Generic.Failed, 'failed'),
127 (r'^ERROR: ', token.Generic.Failed, 'failed'),
127 (r'^ERROR: ', token.Generic.Failed, 'failed'),
128 ],
128 ],
129 'skipped': [
129 'skipped': [
130 (testpattern, token.Generic.SName),
130 (testpattern, token.Generic.SName),
131 (r':.*', token.Generic.Skipped),
131 (r':.*', token.Generic.Skipped),
132 ],
132 ],
133 'failed': [
133 'failed': [
134 (testpattern, token.Generic.FName),
134 (testpattern, token.Generic.FName),
135 (r'(:| ).*', token.Generic.Failed),
135 (r'(:| ).*', token.Generic.Failed),
136 ],
136 ],
137 }
137 }
138
138
139 runnerformatter = formatters.Terminal256Formatter(style=TestRunnerStyle)
139 runnerformatter = formatters.Terminal256Formatter(style=TestRunnerStyle)
140 runnerlexer = TestRunnerLexer()
140 runnerlexer = TestRunnerLexer()
141
141
142 origenviron = os.environ.copy()
142 origenviron = os.environ.copy()
143
143
144 if sys.version_info > (3, 5, 0):
144 if sys.version_info > (3, 5, 0):
145 PYTHON3 = True
145 PYTHON3 = True
146 xrange = range # we use xrange in one place, and we'd rather not use range
146 xrange = range # we use xrange in one place, and we'd rather not use range
147
147
148 def _sys2bytes(p):
148 def _sys2bytes(p):
149 if p is None:
149 if p is None:
150 return p
150 return p
151 return p.encode('utf-8')
151 return p.encode('utf-8')
152
152
153 def _bytes2sys(p):
153 def _bytes2sys(p):
154 if p is None:
154 if p is None:
155 return p
155 return p
156 return p.decode('utf-8')
156 return p.decode('utf-8')
157
157
158 osenvironb = getattr(os, 'environb', None)
158 osenvironb = getattr(os, 'environb', None)
159 if osenvironb is None:
159 if osenvironb is None:
160 # Windows lacks os.environb, for instance. A proxy over the real thing
160 # Windows lacks os.environb, for instance. A proxy over the real thing
161 # instead of a copy allows the environment to be updated via bytes on
161 # instead of a copy allows the environment to be updated via bytes on
162 # all platforms.
162 # all platforms.
163 class environbytes(object):
163 class environbytes(object):
164 def __init__(self, strenv):
164 def __init__(self, strenv):
165 self.__len__ = strenv.__len__
165 self.__len__ = strenv.__len__
166 self.clear = strenv.clear
166 self.clear = strenv.clear
167 self._strenv = strenv
167 self._strenv = strenv
168
168
169 def __getitem__(self, k):
169 def __getitem__(self, k):
170 v = self._strenv.__getitem__(_bytes2sys(k))
170 v = self._strenv.__getitem__(_bytes2sys(k))
171 return _sys2bytes(v)
171 return _sys2bytes(v)
172
172
173 def __setitem__(self, k, v):
173 def __setitem__(self, k, v):
174 self._strenv.__setitem__(_bytes2sys(k), _bytes2sys(v))
174 self._strenv.__setitem__(_bytes2sys(k), _bytes2sys(v))
175
175
176 def __delitem__(self, k):
176 def __delitem__(self, k):
177 self._strenv.__delitem__(_bytes2sys(k))
177 self._strenv.__delitem__(_bytes2sys(k))
178
178
179 def __contains__(self, k):
179 def __contains__(self, k):
180 return self._strenv.__contains__(_bytes2sys(k))
180 return self._strenv.__contains__(_bytes2sys(k))
181
181
182 def __iter__(self):
182 def __iter__(self):
183 return iter([_sys2bytes(k) for k in iter(self._strenv)])
183 return iter([_sys2bytes(k) for k in iter(self._strenv)])
184
184
185 def get(self, k, default=None):
185 def get(self, k, default=None):
186 v = self._strenv.get(_bytes2sys(k), _bytes2sys(default))
186 v = self._strenv.get(_bytes2sys(k), _bytes2sys(default))
187 return _sys2bytes(v)
187 return _sys2bytes(v)
188
188
189 def pop(self, k, default=None):
189 def pop(self, k, default=None):
190 v = self._strenv.pop(_bytes2sys(k), _bytes2sys(default))
190 v = self._strenv.pop(_bytes2sys(k), _bytes2sys(default))
191 return _sys2bytes(v)
191 return _sys2bytes(v)
192
192
193 osenvironb = environbytes(os.environ)
193 osenvironb = environbytes(os.environ)
194
194
195 getcwdb = getattr(os, 'getcwdb')
195 getcwdb = getattr(os, 'getcwdb')
196 if not getcwdb or os.name == 'nt':
196 if not getcwdb or os.name == 'nt':
197 getcwdb = lambda: _sys2bytes(os.getcwd())
197 getcwdb = lambda: _sys2bytes(os.getcwd())
198
198
199 elif sys.version_info >= (3, 0, 0):
199 elif sys.version_info >= (3, 0, 0):
200 print(
200 print(
201 '%s is only supported on Python 3.5+ and 2.7, not %s'
201 '%s is only supported on Python 3.5+ and 2.7, not %s'
202 % (sys.argv[0], '.'.join(str(v) for v in sys.version_info[:3]))
202 % (sys.argv[0], '.'.join(str(v) for v in sys.version_info[:3]))
203 )
203 )
204 sys.exit(70) # EX_SOFTWARE from `man 3 sysexit`
204 sys.exit(70) # EX_SOFTWARE from `man 3 sysexit`
205 else:
205 else:
206 PYTHON3 = False
206 PYTHON3 = False
207
207
208 # In python 2.x, path operations are generally done using
208 # In python 2.x, path operations are generally done using
209 # bytestrings by default, so we don't have to do any extra
209 # bytestrings by default, so we don't have to do any extra
210 # fiddling there. We define the wrapper functions anyway just to
210 # fiddling there. We define the wrapper functions anyway just to
211 # help keep code consistent between platforms.
211 # help keep code consistent between platforms.
212 def _sys2bytes(p):
212 def _sys2bytes(p):
213 return p
213 return p
214
214
215 _bytes2sys = _sys2bytes
215 _bytes2sys = _sys2bytes
216 osenvironb = os.environ
216 osenvironb = os.environ
217 getcwdb = os.getcwd
217 getcwdb = os.getcwd
218
218
219 # For Windows support
219 # For Windows support
220 wifexited = getattr(os, "WIFEXITED", lambda x: False)
220 wifexited = getattr(os, "WIFEXITED", lambda x: False)
221
221
222 # Whether to use IPv6
222 # Whether to use IPv6
223 def checksocketfamily(name, port=20058):
223 def checksocketfamily(name, port=20058):
224 """return true if we can listen on localhost using family=name
224 """return true if we can listen on localhost using family=name
225
225
226 name should be either 'AF_INET', or 'AF_INET6'.
226 name should be either 'AF_INET', or 'AF_INET6'.
227 port being used is okay - EADDRINUSE is considered as successful.
227 port being used is okay - EADDRINUSE is considered as successful.
228 """
228 """
229 family = getattr(socket, name, None)
229 family = getattr(socket, name, None)
230 if family is None:
230 if family is None:
231 return False
231 return False
232 try:
232 try:
233 s = socket.socket(family, socket.SOCK_STREAM)
233 s = socket.socket(family, socket.SOCK_STREAM)
234 s.bind(('localhost', port))
234 s.bind(('localhost', port))
235 s.close()
235 s.close()
236 return True
236 return True
237 except socket.error as exc:
237 except socket.error as exc:
238 if exc.errno == errno.EADDRINUSE:
238 if exc.errno == errno.EADDRINUSE:
239 return True
239 return True
240 elif exc.errno in (errno.EADDRNOTAVAIL, errno.EPROTONOSUPPORT):
240 elif exc.errno in (errno.EADDRNOTAVAIL, errno.EPROTONOSUPPORT):
241 return False
241 return False
242 else:
242 else:
243 raise
243 raise
244 else:
244 else:
245 return False
245 return False
246
246
247
247
248 # useipv6 will be set by parseargs
248 # useipv6 will be set by parseargs
249 useipv6 = None
249 useipv6 = None
250
250
251
251
252 def checkportisavailable(port):
252 def checkportisavailable(port):
253 """return true if a port seems free to bind on localhost"""
253 """return true if a port seems free to bind on localhost"""
254 if useipv6:
254 if useipv6:
255 family = socket.AF_INET6
255 family = socket.AF_INET6
256 else:
256 else:
257 family = socket.AF_INET
257 family = socket.AF_INET
258 try:
258 try:
259 with contextlib.closing(socket.socket(family, socket.SOCK_STREAM)) as s:
259 with contextlib.closing(socket.socket(family, socket.SOCK_STREAM)) as s:
260 s.bind(('localhost', port))
260 s.bind(('localhost', port))
261 return True
261 return True
262 except socket.error as exc:
262 except socket.error as exc:
263 if exc.errno not in (
263 if exc.errno not in (
264 errno.EADDRINUSE,
264 errno.EADDRINUSE,
265 errno.EADDRNOTAVAIL,
265 errno.EADDRNOTAVAIL,
266 errno.EPROTONOSUPPORT,
266 errno.EPROTONOSUPPORT,
267 ):
267 ):
268 raise
268 raise
269 return False
269 return False
270
270
271
271
272 closefds = os.name == 'posix'
272 closefds = os.name == 'posix'
273
273
274
274
275 def Popen4(cmd, wd, timeout, env=None):
275 def Popen4(cmd, wd, timeout, env=None):
276 processlock.acquire()
276 processlock.acquire()
277 p = subprocess.Popen(
277 p = subprocess.Popen(
278 _bytes2sys(cmd),
278 _bytes2sys(cmd),
279 shell=True,
279 shell=True,
280 bufsize=-1,
280 bufsize=-1,
281 cwd=_bytes2sys(wd),
281 cwd=_bytes2sys(wd),
282 env=env,
282 env=env,
283 close_fds=closefds,
283 close_fds=closefds,
284 stdin=subprocess.PIPE,
284 stdin=subprocess.PIPE,
285 stdout=subprocess.PIPE,
285 stdout=subprocess.PIPE,
286 stderr=subprocess.STDOUT,
286 stderr=subprocess.STDOUT,
287 )
287 )
288 processlock.release()
288 processlock.release()
289
289
290 p.fromchild = p.stdout
290 p.fromchild = p.stdout
291 p.tochild = p.stdin
291 p.tochild = p.stdin
292 p.childerr = p.stderr
292 p.childerr = p.stderr
293
293
294 p.timeout = False
294 p.timeout = False
295 if timeout:
295 if timeout:
296
296
297 def t():
297 def t():
298 start = time.time()
298 start = time.time()
299 while time.time() - start < timeout and p.returncode is None:
299 while time.time() - start < timeout and p.returncode is None:
300 time.sleep(0.1)
300 time.sleep(0.1)
301 p.timeout = True
301 p.timeout = True
302 if p.returncode is None:
302 if p.returncode is None:
303 terminate(p)
303 terminate(p)
304
304
305 threading.Thread(target=t).start()
305 threading.Thread(target=t).start()
306
306
307 return p
307 return p
308
308
309
309
310 if sys.executable:
310 if sys.executable:
311 sysexecutable = sys.executable
311 sysexecutable = sys.executable
312 elif os.environ.get('PYTHONEXECUTABLE'):
312 elif os.environ.get('PYTHONEXECUTABLE'):
313 sysexecutable = os.environ['PYTHONEXECUTABLE']
313 sysexecutable = os.environ['PYTHONEXECUTABLE']
314 elif os.environ.get('PYTHON'):
314 elif os.environ.get('PYTHON'):
315 sysexecutable = os.environ['PYTHON']
315 sysexecutable = os.environ['PYTHON']
316 else:
316 else:
317 raise AssertionError('Could not find Python interpreter')
317 raise AssertionError('Could not find Python interpreter')
318
318
319 PYTHON = _sys2bytes(sysexecutable.replace('\\', '/'))
319 PYTHON = _sys2bytes(sysexecutable.replace('\\', '/'))
320 IMPL_PATH = b'PYTHONPATH'
320 IMPL_PATH = b'PYTHONPATH'
321 if 'java' in sys.platform:
321 if 'java' in sys.platform:
322 IMPL_PATH = b'JYTHONPATH'
322 IMPL_PATH = b'JYTHONPATH'
323
323
324 default_defaults = {
324 default_defaults = {
325 'jobs': ('HGTEST_JOBS', multiprocessing.cpu_count()),
325 'jobs': ('HGTEST_JOBS', multiprocessing.cpu_count()),
326 'timeout': ('HGTEST_TIMEOUT', 180),
326 'timeout': ('HGTEST_TIMEOUT', 180),
327 'slowtimeout': ('HGTEST_SLOWTIMEOUT', 1500),
327 'slowtimeout': ('HGTEST_SLOWTIMEOUT', 1500),
328 'port': ('HGTEST_PORT', 20059),
328 'port': ('HGTEST_PORT', 20059),
329 'shell': ('HGTEST_SHELL', 'sh'),
329 'shell': ('HGTEST_SHELL', 'sh'),
330 }
330 }
331
331
332 defaults = default_defaults.copy()
332 defaults = default_defaults.copy()
333
333
334
334
335 def canonpath(path):
335 def canonpath(path):
336 return os.path.realpath(os.path.expanduser(path))
336 return os.path.realpath(os.path.expanduser(path))
337
337
338
338
339 def parselistfiles(files, listtype, warn=True):
339 def parselistfiles(files, listtype, warn=True):
340 entries = dict()
340 entries = dict()
341 for filename in files:
341 for filename in files:
342 try:
342 try:
343 path = os.path.expanduser(os.path.expandvars(filename))
343 path = os.path.expanduser(os.path.expandvars(filename))
344 f = open(path, "rb")
344 f = open(path, "rb")
345 except IOError as err:
345 except IOError as err:
346 if err.errno != errno.ENOENT:
346 if err.errno != errno.ENOENT:
347 raise
347 raise
348 if warn:
348 if warn:
349 print("warning: no such %s file: %s" % (listtype, filename))
349 print("warning: no such %s file: %s" % (listtype, filename))
350 continue
350 continue
351
351
352 for line in f.readlines():
352 for line in f.readlines():
353 line = line.split(b'#', 1)[0].strip()
353 line = line.split(b'#', 1)[0].strip()
354 if line:
354 if line:
355 entries[line] = filename
355 entries[line] = filename
356
356
357 f.close()
357 f.close()
358 return entries
358 return entries
359
359
360
360
361 def parsettestcases(path):
361 def parsettestcases(path):
362 """read a .t test file, return a set of test case names
362 """read a .t test file, return a set of test case names
363
363
364 If path does not exist, return an empty set.
364 If path does not exist, return an empty set.
365 """
365 """
366 cases = []
366 cases = []
367 try:
367 try:
368 with open(path, 'rb') as f:
368 with open(path, 'rb') as f:
369 for l in f:
369 for l in f:
370 if l.startswith(b'#testcases '):
370 if l.startswith(b'#testcases '):
371 cases.append(sorted(l[11:].split()))
371 cases.append(sorted(l[11:].split()))
372 except IOError as ex:
372 except IOError as ex:
373 if ex.errno != errno.ENOENT:
373 if ex.errno != errno.ENOENT:
374 raise
374 raise
375 return cases
375 return cases
376
376
377
377
378 def getparser():
378 def getparser():
379 """Obtain the OptionParser used by the CLI."""
379 """Obtain the OptionParser used by the CLI."""
380 parser = argparse.ArgumentParser(usage='%(prog)s [options] [tests]')
380 parser = argparse.ArgumentParser(usage='%(prog)s [options] [tests]')
381
381
382 selection = parser.add_argument_group('Test Selection')
382 selection = parser.add_argument_group('Test Selection')
383 selection.add_argument(
383 selection.add_argument(
384 '--allow-slow-tests',
384 '--allow-slow-tests',
385 action='store_true',
385 action='store_true',
386 help='allow extremely slow tests',
386 help='allow extremely slow tests',
387 )
387 )
388 selection.add_argument(
388 selection.add_argument(
389 "--blacklist",
389 "--blacklist",
390 action="append",
390 action="append",
391 help="skip tests listed in the specified blacklist file",
391 help="skip tests listed in the specified blacklist file",
392 )
392 )
393 selection.add_argument(
393 selection.add_argument(
394 "--changed",
394 "--changed",
395 help="run tests that are changed in parent rev or working directory",
395 help="run tests that are changed in parent rev or working directory",
396 )
396 )
397 selection.add_argument(
397 selection.add_argument(
398 "-k", "--keywords", help="run tests matching keywords"
398 "-k", "--keywords", help="run tests matching keywords"
399 )
399 )
400 selection.add_argument(
400 selection.add_argument(
401 "-r", "--retest", action="store_true", help="retest failed tests"
401 "-r", "--retest", action="store_true", help="retest failed tests"
402 )
402 )
403 selection.add_argument(
403 selection.add_argument(
404 "--test-list",
404 "--test-list",
405 action="append",
405 action="append",
406 help="read tests to run from the specified file",
406 help="read tests to run from the specified file",
407 )
407 )
408 selection.add_argument(
408 selection.add_argument(
409 "--whitelist",
409 "--whitelist",
410 action="append",
410 action="append",
411 help="always run tests listed in the specified whitelist file",
411 help="always run tests listed in the specified whitelist file",
412 )
412 )
413 selection.add_argument(
413 selection.add_argument(
414 'tests', metavar='TESTS', nargs='*', help='Tests to run'
414 'tests', metavar='TESTS', nargs='*', help='Tests to run'
415 )
415 )
416
416
417 harness = parser.add_argument_group('Test Harness Behavior')
417 harness = parser.add_argument_group('Test Harness Behavior')
418 harness.add_argument(
418 harness.add_argument(
419 '--bisect-repo',
419 '--bisect-repo',
420 metavar='bisect_repo',
420 metavar='bisect_repo',
421 help=(
421 help=(
422 "Path of a repo to bisect. Use together with " "--known-good-rev"
422 "Path of a repo to bisect. Use together with " "--known-good-rev"
423 ),
423 ),
424 )
424 )
425 harness.add_argument(
425 harness.add_argument(
426 "-d",
426 "-d",
427 "--debug",
427 "--debug",
428 action="store_true",
428 action="store_true",
429 help="debug mode: write output of test scripts to console"
429 help="debug mode: write output of test scripts to console"
430 " rather than capturing and diffing it (disables timeout)",
430 " rather than capturing and diffing it (disables timeout)",
431 )
431 )
432 harness.add_argument(
432 harness.add_argument(
433 "-f",
433 "-f",
434 "--first",
434 "--first",
435 action="store_true",
435 action="store_true",
436 help="exit on the first test failure",
436 help="exit on the first test failure",
437 )
437 )
438 harness.add_argument(
438 harness.add_argument(
439 "-i",
439 "-i",
440 "--interactive",
440 "--interactive",
441 action="store_true",
441 action="store_true",
442 help="prompt to accept changed output",
442 help="prompt to accept changed output",
443 )
443 )
444 harness.add_argument(
444 harness.add_argument(
445 "-j",
445 "-j",
446 "--jobs",
446 "--jobs",
447 type=int,
447 type=int,
448 help="number of jobs to run in parallel"
448 help="number of jobs to run in parallel"
449 " (default: $%s or %d)" % defaults['jobs'],
449 " (default: $%s or %d)" % defaults['jobs'],
450 )
450 )
451 harness.add_argument(
451 harness.add_argument(
452 "--keep-tmpdir",
452 "--keep-tmpdir",
453 action="store_true",
453 action="store_true",
454 help="keep temporary directory after running tests",
454 help="keep temporary directory after running tests",
455 )
455 )
456 harness.add_argument(
456 harness.add_argument(
457 '--known-good-rev',
457 '--known-good-rev',
458 metavar="known_good_rev",
458 metavar="known_good_rev",
459 help=(
459 help=(
460 "Automatically bisect any failures using this "
460 "Automatically bisect any failures using this "
461 "revision as a known-good revision."
461 "revision as a known-good revision."
462 ),
462 ),
463 )
463 )
464 harness.add_argument(
464 harness.add_argument(
465 "--list-tests",
465 "--list-tests",
466 action="store_true",
466 action="store_true",
467 help="list tests instead of running them",
467 help="list tests instead of running them",
468 )
468 )
469 harness.add_argument(
469 harness.add_argument(
470 "--loop", action="store_true", help="loop tests repeatedly"
470 "--loop", action="store_true", help="loop tests repeatedly"
471 )
471 )
472 harness.add_argument(
472 harness.add_argument(
473 '--random', action="store_true", help='run tests in random order'
473 '--random', action="store_true", help='run tests in random order'
474 )
474 )
475 harness.add_argument(
475 harness.add_argument(
476 '--order-by-runtime',
476 '--order-by-runtime',
477 action="store_true",
477 action="store_true",
478 help='run slowest tests first, according to .testtimes',
478 help='run slowest tests first, according to .testtimes',
479 )
479 )
480 harness.add_argument(
480 harness.add_argument(
481 "-p",
481 "-p",
482 "--port",
482 "--port",
483 type=int,
483 type=int,
484 help="port on which servers should listen"
484 help="port on which servers should listen"
485 " (default: $%s or %d)" % defaults['port'],
485 " (default: $%s or %d)" % defaults['port'],
486 )
486 )
487 harness.add_argument(
487 harness.add_argument(
488 '--profile-runner',
488 '--profile-runner',
489 action='store_true',
489 action='store_true',
490 help='run statprof on run-tests',
490 help='run statprof on run-tests',
491 )
491 )
492 harness.add_argument(
492 harness.add_argument(
493 "-R", "--restart", action="store_true", help="restart at last error"
493 "-R", "--restart", action="store_true", help="restart at last error"
494 )
494 )
495 harness.add_argument(
495 harness.add_argument(
496 "--runs-per-test",
496 "--runs-per-test",
497 type=int,
497 type=int,
498 dest="runs_per_test",
498 dest="runs_per_test",
499 help="run each test N times (default=1)",
499 help="run each test N times (default=1)",
500 default=1,
500 default=1,
501 )
501 )
502 harness.add_argument(
502 harness.add_argument(
503 "--shell", help="shell to use (default: $%s or %s)" % defaults['shell']
503 "--shell", help="shell to use (default: $%s or %s)" % defaults['shell']
504 )
504 )
505 harness.add_argument(
505 harness.add_argument(
506 '--showchannels', action='store_true', help='show scheduling channels'
506 '--showchannels', action='store_true', help='show scheduling channels'
507 )
507 )
508 harness.add_argument(
508 harness.add_argument(
509 "--slowtimeout",
509 "--slowtimeout",
510 type=int,
510 type=int,
511 help="kill errant slow tests after SLOWTIMEOUT seconds"
511 help="kill errant slow tests after SLOWTIMEOUT seconds"
512 " (default: $%s or %d)" % defaults['slowtimeout'],
512 " (default: $%s or %d)" % defaults['slowtimeout'],
513 )
513 )
514 harness.add_argument(
514 harness.add_argument(
515 "-t",
515 "-t",
516 "--timeout",
516 "--timeout",
517 type=int,
517 type=int,
518 help="kill errant tests after TIMEOUT seconds"
518 help="kill errant tests after TIMEOUT seconds"
519 " (default: $%s or %d)" % defaults['timeout'],
519 " (default: $%s or %d)" % defaults['timeout'],
520 )
520 )
521 harness.add_argument(
521 harness.add_argument(
522 "--tmpdir",
522 "--tmpdir",
523 help="run tests in the given temporary directory"
523 help="run tests in the given temporary directory"
524 " (implies --keep-tmpdir)",
524 " (implies --keep-tmpdir)",
525 )
525 )
526 harness.add_argument(
526 harness.add_argument(
527 "-v", "--verbose", action="store_true", help="output verbose messages"
527 "-v", "--verbose", action="store_true", help="output verbose messages"
528 )
528 )
529
529
530 hgconf = parser.add_argument_group('Mercurial Configuration')
530 hgconf = parser.add_argument_group('Mercurial Configuration')
531 hgconf.add_argument(
531 hgconf.add_argument(
532 "--chg",
532 "--chg",
533 action="store_true",
533 action="store_true",
534 help="install and use chg wrapper in place of hg",
534 help="install and use chg wrapper in place of hg",
535 )
535 )
536 hgconf.add_argument(
536 hgconf.add_argument(
537 "--chg-debug",
537 "--chg-debug",
538 action="store_true",
538 action="store_true",
539 help="show chg debug logs",
539 help="show chg debug logs",
540 )
540 )
541 hgconf.add_argument("--compiler", help="compiler to build with")
541 hgconf.add_argument("--compiler", help="compiler to build with")
542 hgconf.add_argument(
542 hgconf.add_argument(
543 '--extra-config-opt',
543 '--extra-config-opt',
544 action="append",
544 action="append",
545 default=[],
545 default=[],
546 help='set the given config opt in the test hgrc',
546 help='set the given config opt in the test hgrc',
547 )
547 )
548 hgconf.add_argument(
548 hgconf.add_argument(
549 "-l",
549 "-l",
550 "--local",
550 "--local",
551 action="store_true",
551 action="store_true",
552 help="shortcut for --with-hg=<testdir>/../hg, "
552 help="shortcut for --with-hg=<testdir>/../hg, "
553 "and --with-chg=<testdir>/../contrib/chg/chg if --chg is set",
553 "and --with-chg=<testdir>/../contrib/chg/chg if --chg is set",
554 )
554 )
555 hgconf.add_argument(
555 hgconf.add_argument(
556 "--ipv6",
556 "--ipv6",
557 action="store_true",
557 action="store_true",
558 help="prefer IPv6 to IPv4 for network related tests",
558 help="prefer IPv6 to IPv4 for network related tests",
559 )
559 )
560 hgconf.add_argument(
560 hgconf.add_argument(
561 "--pure",
561 "--pure",
562 action="store_true",
562 action="store_true",
563 help="use pure Python code instead of C extensions",
563 help="use pure Python code instead of C extensions",
564 )
564 )
565 hgconf.add_argument(
565 hgconf.add_argument(
566 "--rust",
566 "--rust",
567 action="store_true",
567 action="store_true",
568 help="use Rust code alongside C extensions",
568 help="use Rust code alongside C extensions",
569 )
569 )
570 hgconf.add_argument(
570 hgconf.add_argument(
571 "--no-rust",
571 "--no-rust",
572 action="store_true",
572 action="store_true",
573 help="do not use Rust code even if compiled",
573 help="do not use Rust code even if compiled",
574 )
574 )
575 hgconf.add_argument(
575 hgconf.add_argument(
576 "--with-chg",
576 "--with-chg",
577 metavar="CHG",
577 metavar="CHG",
578 help="use specified chg wrapper in place of hg",
578 help="use specified chg wrapper in place of hg",
579 )
579 )
580 hgconf.add_argument(
580 hgconf.add_argument(
581 "--with-hg",
581 "--with-hg",
582 metavar="HG",
582 metavar="HG",
583 help="test using specified hg script rather than a "
583 help="test using specified hg script rather than a "
584 "temporary installation",
584 "temporary installation",
585 )
585 )
586
586
587 reporting = parser.add_argument_group('Results Reporting')
587 reporting = parser.add_argument_group('Results Reporting')
588 reporting.add_argument(
588 reporting.add_argument(
589 "-C",
589 "-C",
590 "--annotate",
590 "--annotate",
591 action="store_true",
591 action="store_true",
592 help="output files annotated with coverage",
592 help="output files annotated with coverage",
593 )
593 )
594 reporting.add_argument(
594 reporting.add_argument(
595 "--color",
595 "--color",
596 choices=["always", "auto", "never"],
596 choices=["always", "auto", "never"],
597 default=os.environ.get('HGRUNTESTSCOLOR', 'auto'),
597 default=os.environ.get('HGRUNTESTSCOLOR', 'auto'),
598 help="colorisation: always|auto|never (default: auto)",
598 help="colorisation: always|auto|never (default: auto)",
599 )
599 )
600 reporting.add_argument(
600 reporting.add_argument(
601 "-c",
601 "-c",
602 "--cover",
602 "--cover",
603 action="store_true",
603 action="store_true",
604 help="print a test coverage report",
604 help="print a test coverage report",
605 )
605 )
606 reporting.add_argument(
606 reporting.add_argument(
607 '--exceptions',
607 '--exceptions',
608 action='store_true',
608 action='store_true',
609 help='log all exceptions and generate an exception report',
609 help='log all exceptions and generate an exception report',
610 )
610 )
611 reporting.add_argument(
611 reporting.add_argument(
612 "-H",
612 "-H",
613 "--htmlcov",
613 "--htmlcov",
614 action="store_true",
614 action="store_true",
615 help="create an HTML report of the coverage of the files",
615 help="create an HTML report of the coverage of the files",
616 )
616 )
617 reporting.add_argument(
617 reporting.add_argument(
618 "--json",
618 "--json",
619 action="store_true",
619 action="store_true",
620 help="store test result data in 'report.json' file",
620 help="store test result data in 'report.json' file",
621 )
621 )
622 reporting.add_argument(
622 reporting.add_argument(
623 "--outputdir",
623 "--outputdir",
624 help="directory to write error logs to (default=test directory)",
624 help="directory to write error logs to (default=test directory)",
625 )
625 )
626 reporting.add_argument(
626 reporting.add_argument(
627 "-n", "--nodiff", action="store_true", help="skip showing test changes"
627 "-n", "--nodiff", action="store_true", help="skip showing test changes"
628 )
628 )
629 reporting.add_argument(
629 reporting.add_argument(
630 "-S",
630 "-S",
631 "--noskips",
631 "--noskips",
632 action="store_true",
632 action="store_true",
633 help="don't report skip tests verbosely",
633 help="don't report skip tests verbosely",
634 )
634 )
635 reporting.add_argument(
635 reporting.add_argument(
636 "--time", action="store_true", help="time how long each test takes"
636 "--time", action="store_true", help="time how long each test takes"
637 )
637 )
638 reporting.add_argument("--view", help="external diff viewer")
638 reporting.add_argument("--view", help="external diff viewer")
639 reporting.add_argument(
639 reporting.add_argument(
640 "--xunit", help="record xunit results at specified path"
640 "--xunit", help="record xunit results at specified path"
641 )
641 )
642
642
643 for option, (envvar, default) in defaults.items():
643 for option, (envvar, default) in defaults.items():
644 defaults[option] = type(default)(os.environ.get(envvar, default))
644 defaults[option] = type(default)(os.environ.get(envvar, default))
645 parser.set_defaults(**defaults)
645 parser.set_defaults(**defaults)
646
646
647 return parser
647 return parser
648
648
649
649
650 def parseargs(args, parser):
650 def parseargs(args, parser):
651 """Parse arguments with our OptionParser and validate results."""
651 """Parse arguments with our OptionParser and validate results."""
652 options = parser.parse_args(args)
652 options = parser.parse_args(args)
653
653
654 # jython is always pure
654 # jython is always pure
655 if 'java' in sys.platform or '__pypy__' in sys.modules:
655 if 'java' in sys.platform or '__pypy__' in sys.modules:
656 options.pure = True
656 options.pure = True
657
657
658 if platform.python_implementation() != 'CPython' and options.rust:
658 if platform.python_implementation() != 'CPython' and options.rust:
659 parser.error('Rust extensions are only available with CPython')
659 parser.error('Rust extensions are only available with CPython')
660
660
661 if options.pure and options.rust:
661 if options.pure and options.rust:
662 parser.error('--rust cannot be used with --pure')
662 parser.error('--rust cannot be used with --pure')
663
663
664 if options.rust and options.no_rust:
664 if options.rust and options.no_rust:
665 parser.error('--rust cannot be used with --no-rust')
665 parser.error('--rust cannot be used with --no-rust')
666
666
667 if options.local:
667 if options.local:
668 if options.with_hg or options.with_chg:
668 if options.with_hg or options.with_chg:
669 parser.error('--local cannot be used with --with-hg or --with-chg')
669 parser.error('--local cannot be used with --with-hg or --with-chg')
670 testdir = os.path.dirname(_sys2bytes(canonpath(sys.argv[0])))
670 testdir = os.path.dirname(_sys2bytes(canonpath(sys.argv[0])))
671 reporootdir = os.path.dirname(testdir)
671 reporootdir = os.path.dirname(testdir)
672 pathandattrs = [(b'hg', 'with_hg')]
672 pathandattrs = [(b'hg', 'with_hg')]
673 if options.chg:
673 if options.chg:
674 pathandattrs.append((b'contrib/chg/chg', 'with_chg'))
674 pathandattrs.append((b'contrib/chg/chg', 'with_chg'))
675 for relpath, attr in pathandattrs:
675 for relpath, attr in pathandattrs:
676 binpath = os.path.join(reporootdir, relpath)
676 binpath = os.path.join(reporootdir, relpath)
677 if os.name != 'nt' and not os.access(binpath, os.X_OK):
677 if os.name != 'nt' and not os.access(binpath, os.X_OK):
678 parser.error(
678 parser.error(
679 '--local specified, but %r not found or '
679 '--local specified, but %r not found or '
680 'not executable' % binpath
680 'not executable' % binpath
681 )
681 )
682 setattr(options, attr, _bytes2sys(binpath))
682 setattr(options, attr, _bytes2sys(binpath))
683
683
684 if options.with_hg:
684 if options.with_hg:
685 options.with_hg = canonpath(_sys2bytes(options.with_hg))
685 options.with_hg = canonpath(_sys2bytes(options.with_hg))
686 if not (
686 if not (
687 os.path.isfile(options.with_hg)
687 os.path.isfile(options.with_hg)
688 and os.access(options.with_hg, os.X_OK)
688 and os.access(options.with_hg, os.X_OK)
689 ):
689 ):
690 parser.error('--with-hg must specify an executable hg script')
690 parser.error('--with-hg must specify an executable hg script')
691 if os.path.basename(options.with_hg) not in [b'hg', b'hg.exe']:
691 if os.path.basename(options.with_hg) not in [b'hg', b'hg.exe']:
692 sys.stderr.write('warning: --with-hg should specify an hg script\n')
692 sys.stderr.write('warning: --with-hg should specify an hg script\n')
693 sys.stderr.flush()
693 sys.stderr.flush()
694
694
695 if (options.chg or options.with_chg) and os.name == 'nt':
695 if (options.chg or options.with_chg) and os.name == 'nt':
696 parser.error('chg does not work on %s' % os.name)
696 parser.error('chg does not work on %s' % os.name)
697 if options.with_chg:
697 if options.with_chg:
698 options.chg = False # no installation to temporary location
698 options.chg = False # no installation to temporary location
699 options.with_chg = canonpath(_sys2bytes(options.with_chg))
699 options.with_chg = canonpath(_sys2bytes(options.with_chg))
700 if not (
700 if not (
701 os.path.isfile(options.with_chg)
701 os.path.isfile(options.with_chg)
702 and os.access(options.with_chg, os.X_OK)
702 and os.access(options.with_chg, os.X_OK)
703 ):
703 ):
704 parser.error('--with-chg must specify a chg executable')
704 parser.error('--with-chg must specify a chg executable')
705 if options.chg and options.with_hg:
705 if options.chg and options.with_hg:
706 # chg shares installation location with hg
706 # chg shares installation location with hg
707 parser.error(
707 parser.error(
708 '--chg does not work when --with-hg is specified '
708 '--chg does not work when --with-hg is specified '
709 '(use --with-chg instead)'
709 '(use --with-chg instead)'
710 )
710 )
711
711
712 if options.color == 'always' and not pygmentspresent:
712 if options.color == 'always' and not pygmentspresent:
713 sys.stderr.write(
713 sys.stderr.write(
714 'warning: --color=always ignored because '
714 'warning: --color=always ignored because '
715 'pygments is not installed\n'
715 'pygments is not installed\n'
716 )
716 )
717
717
718 if options.bisect_repo and not options.known_good_rev:
718 if options.bisect_repo and not options.known_good_rev:
719 parser.error("--bisect-repo cannot be used without --known-good-rev")
719 parser.error("--bisect-repo cannot be used without --known-good-rev")
720
720
721 global useipv6
721 global useipv6
722 if options.ipv6:
722 if options.ipv6:
723 useipv6 = checksocketfamily('AF_INET6')
723 useipv6 = checksocketfamily('AF_INET6')
724 else:
724 else:
725 # only use IPv6 if IPv4 is unavailable and IPv6 is available
725 # only use IPv6 if IPv4 is unavailable and IPv6 is available
726 useipv6 = (not checksocketfamily('AF_INET')) and checksocketfamily(
726 useipv6 = (not checksocketfamily('AF_INET')) and checksocketfamily(
727 'AF_INET6'
727 'AF_INET6'
728 )
728 )
729
729
730 options.anycoverage = options.cover or options.annotate or options.htmlcov
730 options.anycoverage = options.cover or options.annotate or options.htmlcov
731 if options.anycoverage:
731 if options.anycoverage:
732 try:
732 try:
733 import coverage
733 import coverage
734
734
735 covver = version.StrictVersion(coverage.__version__).version
735 covver = version.StrictVersion(coverage.__version__).version
736 if covver < (3, 3):
736 if covver < (3, 3):
737 parser.error('coverage options require coverage 3.3 or later')
737 parser.error('coverage options require coverage 3.3 or later')
738 except ImportError:
738 except ImportError:
739 parser.error('coverage options now require the coverage package')
739 parser.error('coverage options now require the coverage package')
740
740
741 if options.anycoverage and options.local:
741 if options.anycoverage and options.local:
742 # this needs some path mangling somewhere, I guess
742 # this needs some path mangling somewhere, I guess
743 parser.error(
743 parser.error(
744 "sorry, coverage options do not work when --local " "is specified"
744 "sorry, coverage options do not work when --local " "is specified"
745 )
745 )
746
746
747 if options.anycoverage and options.with_hg:
747 if options.anycoverage and options.with_hg:
748 parser.error(
748 parser.error(
749 "sorry, coverage options do not work when --with-hg " "is specified"
749 "sorry, coverage options do not work when --with-hg " "is specified"
750 )
750 )
751
751
752 global verbose
752 global verbose
753 if options.verbose:
753 if options.verbose:
754 verbose = ''
754 verbose = ''
755
755
756 if options.tmpdir:
756 if options.tmpdir:
757 options.tmpdir = canonpath(options.tmpdir)
757 options.tmpdir = canonpath(options.tmpdir)
758
758
759 if options.jobs < 1:
759 if options.jobs < 1:
760 parser.error('--jobs must be positive')
760 parser.error('--jobs must be positive')
761 if options.interactive and options.debug:
761 if options.interactive and options.debug:
762 parser.error("-i/--interactive and -d/--debug are incompatible")
762 parser.error("-i/--interactive and -d/--debug are incompatible")
763 if options.debug:
763 if options.debug:
764 if options.timeout != defaults['timeout']:
764 if options.timeout != defaults['timeout']:
765 sys.stderr.write('warning: --timeout option ignored with --debug\n')
765 sys.stderr.write('warning: --timeout option ignored with --debug\n')
766 if options.slowtimeout != defaults['slowtimeout']:
766 if options.slowtimeout != defaults['slowtimeout']:
767 sys.stderr.write(
767 sys.stderr.write(
768 'warning: --slowtimeout option ignored with --debug\n'
768 'warning: --slowtimeout option ignored with --debug\n'
769 )
769 )
770 options.timeout = 0
770 options.timeout = 0
771 options.slowtimeout = 0
771 options.slowtimeout = 0
772
772
773 if options.blacklist:
773 if options.blacklist:
774 options.blacklist = parselistfiles(options.blacklist, 'blacklist')
774 options.blacklist = parselistfiles(options.blacklist, 'blacklist')
775 if options.whitelist:
775 if options.whitelist:
776 options.whitelisted = parselistfiles(options.whitelist, 'whitelist')
776 options.whitelisted = parselistfiles(options.whitelist, 'whitelist')
777 else:
777 else:
778 options.whitelisted = {}
778 options.whitelisted = {}
779
779
780 if options.showchannels:
780 if options.showchannels:
781 options.nodiff = True
781 options.nodiff = True
782
782
783 return options
783 return options
784
784
785
785
786 def rename(src, dst):
786 def rename(src, dst):
787 """Like os.rename(), trade atomicity and opened files friendliness
787 """Like os.rename(), trade atomicity and opened files friendliness
788 for existing destination support.
788 for existing destination support.
789 """
789 """
790 shutil.copy(src, dst)
790 shutil.copy(src, dst)
791 os.remove(src)
791 os.remove(src)
792
792
793
793
794 def makecleanable(path):
794 def makecleanable(path):
795 """Try to fix directory permission recursively so that the entire tree
795 """Try to fix directory permission recursively so that the entire tree
796 can be deleted"""
796 can be deleted"""
797 for dirpath, dirnames, _filenames in os.walk(path, topdown=True):
797 for dirpath, dirnames, _filenames in os.walk(path, topdown=True):
798 for d in dirnames:
798 for d in dirnames:
799 p = os.path.join(dirpath, d)
799 p = os.path.join(dirpath, d)
800 try:
800 try:
801 os.chmod(p, os.stat(p).st_mode & 0o777 | 0o700) # chmod u+rwx
801 os.chmod(p, os.stat(p).st_mode & 0o777 | 0o700) # chmod u+rwx
802 except OSError:
802 except OSError:
803 pass
803 pass
804
804
805
805
806 _unified_diff = difflib.unified_diff
806 _unified_diff = difflib.unified_diff
807 if PYTHON3:
807 if PYTHON3:
808 import functools
808 import functools
809
809
810 _unified_diff = functools.partial(difflib.diff_bytes, difflib.unified_diff)
810 _unified_diff = functools.partial(difflib.diff_bytes, difflib.unified_diff)
811
811
812
812
813 def getdiff(expected, output, ref, err):
813 def getdiff(expected, output, ref, err):
814 servefail = False
814 servefail = False
815 lines = []
815 lines = []
816 for line in _unified_diff(expected, output, ref, err):
816 for line in _unified_diff(expected, output, ref, err):
817 if line.startswith(b'+++') or line.startswith(b'---'):
817 if line.startswith(b'+++') or line.startswith(b'---'):
818 line = line.replace(b'\\', b'/')
818 line = line.replace(b'\\', b'/')
819 if line.endswith(b' \n'):
819 if line.endswith(b' \n'):
820 line = line[:-2] + b'\n'
820 line = line[:-2] + b'\n'
821 lines.append(line)
821 lines.append(line)
822 if not servefail and line.startswith(
822 if not servefail and line.startswith(
823 b'+ abort: child process failed to start'
823 b'+ abort: child process failed to start'
824 ):
824 ):
825 servefail = True
825 servefail = True
826
826
827 return servefail, lines
827 return servefail, lines
828
828
829
829
830 verbose = False
830 verbose = False
831
831
832
832
833 def vlog(*msg):
833 def vlog(*msg):
834 """Log only when in verbose mode."""
834 """Log only when in verbose mode."""
835 if verbose is False:
835 if verbose is False:
836 return
836 return
837
837
838 return log(*msg)
838 return log(*msg)
839
839
840
840
841 # Bytes that break XML even in a CDATA block: control characters 0-31
841 # Bytes that break XML even in a CDATA block: control characters 0-31
842 # sans \t, \n and \r
842 # sans \t, \n and \r
843 CDATA_EVIL = re.compile(br"[\000-\010\013\014\016-\037]")
843 CDATA_EVIL = re.compile(br"[\000-\010\013\014\016-\037]")
844
844
845 # Match feature conditionalized output lines in the form, capturing the feature
845 # Match feature conditionalized output lines in the form, capturing the feature
846 # list in group 2, and the preceeding line output in group 1:
846 # list in group 2, and the preceeding line output in group 1:
847 #
847 #
848 # output..output (feature !)\n
848 # output..output (feature !)\n
849 optline = re.compile(br'(.*) \((.+?) !\)\n$')
849 optline = re.compile(br'(.*) \((.+?) !\)\n$')
850
850
851
851
852 def cdatasafe(data):
852 def cdatasafe(data):
853 """Make a string safe to include in a CDATA block.
853 """Make a string safe to include in a CDATA block.
854
854
855 Certain control characters are illegal in a CDATA block, and
855 Certain control characters are illegal in a CDATA block, and
856 there's no way to include a ]]> in a CDATA either. This function
856 there's no way to include a ]]> in a CDATA either. This function
857 replaces illegal bytes with ? and adds a space between the ]] so
857 replaces illegal bytes with ? and adds a space between the ]] so
858 that it won't break the CDATA block.
858 that it won't break the CDATA block.
859 """
859 """
860 return CDATA_EVIL.sub(b'?', data).replace(b']]>', b'] ]>')
860 return CDATA_EVIL.sub(b'?', data).replace(b']]>', b'] ]>')
861
861
862
862
863 def log(*msg):
863 def log(*msg):
864 """Log something to stdout.
864 """Log something to stdout.
865
865
866 Arguments are strings to print.
866 Arguments are strings to print.
867 """
867 """
868 with iolock:
868 with iolock:
869 if verbose:
869 if verbose:
870 print(verbose, end=' ')
870 print(verbose, end=' ')
871 for m in msg:
871 for m in msg:
872 print(m, end=' ')
872 print(m, end=' ')
873 print()
873 print()
874 sys.stdout.flush()
874 sys.stdout.flush()
875
875
876
876
877 def highlightdiff(line, color):
877 def highlightdiff(line, color):
878 if not color:
878 if not color:
879 return line
879 return line
880 assert pygmentspresent
880 assert pygmentspresent
881 return pygments.highlight(
881 return pygments.highlight(
882 line.decode('latin1'), difflexer, terminal256formatter
882 line.decode('latin1'), difflexer, terminal256formatter
883 ).encode('latin1')
883 ).encode('latin1')
884
884
885
885
886 def highlightmsg(msg, color):
886 def highlightmsg(msg, color):
887 if not color:
887 if not color:
888 return msg
888 return msg
889 assert pygmentspresent
889 assert pygmentspresent
890 return pygments.highlight(msg, runnerlexer, runnerformatter)
890 return pygments.highlight(msg, runnerlexer, runnerformatter)
891
891
892
892
893 def terminate(proc):
893 def terminate(proc):
894 """Terminate subprocess"""
894 """Terminate subprocess"""
895 vlog('# Terminating process %d' % proc.pid)
895 vlog('# Terminating process %d' % proc.pid)
896 try:
896 try:
897 proc.terminate()
897 proc.terminate()
898 except OSError:
898 except OSError:
899 pass
899 pass
900
900
901
901
902 def killdaemons(pidfile):
902 def killdaemons(pidfile):
903 import killdaemons as killmod
903 import killdaemons as killmod
904
904
905 return killmod.killdaemons(pidfile, tryhard=False, remove=True, logfn=vlog)
905 return killmod.killdaemons(pidfile, tryhard=False, remove=True, logfn=vlog)
906
906
907
907
908 class Test(unittest.TestCase):
908 class Test(unittest.TestCase):
909 """Encapsulates a single, runnable test.
909 """Encapsulates a single, runnable test.
910
910
911 While this class conforms to the unittest.TestCase API, it differs in that
911 While this class conforms to the unittest.TestCase API, it differs in that
912 instances need to be instantiated manually. (Typically, unittest.TestCase
912 instances need to be instantiated manually. (Typically, unittest.TestCase
913 classes are instantiated automatically by scanning modules.)
913 classes are instantiated automatically by scanning modules.)
914 """
914 """
915
915
916 # Status code reserved for skipped tests (used by hghave).
916 # Status code reserved for skipped tests (used by hghave).
917 SKIPPED_STATUS = 80
917 SKIPPED_STATUS = 80
918
918
919 def __init__(
919 def __init__(
920 self,
920 self,
921 path,
921 path,
922 outputdir,
922 outputdir,
923 tmpdir,
923 tmpdir,
924 keeptmpdir=False,
924 keeptmpdir=False,
925 debug=False,
925 debug=False,
926 first=False,
926 first=False,
927 timeout=None,
927 timeout=None,
928 startport=None,
928 startport=None,
929 extraconfigopts=None,
929 extraconfigopts=None,
930 shell=None,
930 shell=None,
931 hgcommand=None,
931 hgcommand=None,
932 slowtimeout=None,
932 slowtimeout=None,
933 usechg=False,
933 usechg=False,
934 chgdebug=False,
934 chgdebug=False,
935 useipv6=False,
935 useipv6=False,
936 ):
936 ):
937 """Create a test from parameters.
937 """Create a test from parameters.
938
938
939 path is the full path to the file defining the test.
939 path is the full path to the file defining the test.
940
940
941 tmpdir is the main temporary directory to use for this test.
941 tmpdir is the main temporary directory to use for this test.
942
942
943 keeptmpdir determines whether to keep the test's temporary directory
943 keeptmpdir determines whether to keep the test's temporary directory
944 after execution. It defaults to removal (False).
944 after execution. It defaults to removal (False).
945
945
946 debug mode will make the test execute verbosely, with unfiltered
946 debug mode will make the test execute verbosely, with unfiltered
947 output.
947 output.
948
948
949 timeout controls the maximum run time of the test. It is ignored when
949 timeout controls the maximum run time of the test. It is ignored when
950 debug is True. See slowtimeout for tests with #require slow.
950 debug is True. See slowtimeout for tests with #require slow.
951
951
952 slowtimeout overrides timeout if the test has #require slow.
952 slowtimeout overrides timeout if the test has #require slow.
953
953
954 startport controls the starting port number to use for this test. Each
954 startport controls the starting port number to use for this test. Each
955 test will reserve 3 port numbers for execution. It is the caller's
955 test will reserve 3 port numbers for execution. It is the caller's
956 responsibility to allocate a non-overlapping port range to Test
956 responsibility to allocate a non-overlapping port range to Test
957 instances.
957 instances.
958
958
959 extraconfigopts is an iterable of extra hgrc config options. Values
959 extraconfigopts is an iterable of extra hgrc config options. Values
960 must have the form "key=value" (something understood by hgrc). Values
960 must have the form "key=value" (something understood by hgrc). Values
961 of the form "foo.key=value" will result in "[foo] key=value".
961 of the form "foo.key=value" will result in "[foo] key=value".
962
962
963 shell is the shell to execute tests in.
963 shell is the shell to execute tests in.
964 """
964 """
965 if timeout is None:
965 if timeout is None:
966 timeout = defaults['timeout']
966 timeout = defaults['timeout']
967 if startport is None:
967 if startport is None:
968 startport = defaults['port']
968 startport = defaults['port']
969 if slowtimeout is None:
969 if slowtimeout is None:
970 slowtimeout = defaults['slowtimeout']
970 slowtimeout = defaults['slowtimeout']
971 self.path = path
971 self.path = path
972 self.relpath = os.path.relpath(path)
972 self.relpath = os.path.relpath(path)
973 self.bname = os.path.basename(path)
973 self.bname = os.path.basename(path)
974 self.name = _bytes2sys(self.bname)
974 self.name = _bytes2sys(self.bname)
975 self._testdir = os.path.dirname(path)
975 self._testdir = os.path.dirname(path)
976 self._outputdir = outputdir
976 self._outputdir = outputdir
977 self._tmpname = os.path.basename(path)
977 self._tmpname = os.path.basename(path)
978 self.errpath = os.path.join(self._outputdir, b'%s.err' % self.bname)
978 self.errpath = os.path.join(self._outputdir, b'%s.err' % self.bname)
979
979
980 self._threadtmp = tmpdir
980 self._threadtmp = tmpdir
981 self._keeptmpdir = keeptmpdir
981 self._keeptmpdir = keeptmpdir
982 self._debug = debug
982 self._debug = debug
983 self._first = first
983 self._first = first
984 self._timeout = timeout
984 self._timeout = timeout
985 self._slowtimeout = slowtimeout
985 self._slowtimeout = slowtimeout
986 self._startport = startport
986 self._startport = startport
987 self._extraconfigopts = extraconfigopts or []
987 self._extraconfigopts = extraconfigopts or []
988 self._shell = _sys2bytes(shell)
988 self._shell = _sys2bytes(shell)
989 self._hgcommand = hgcommand or b'hg'
989 self._hgcommand = hgcommand or b'hg'
990 self._usechg = usechg
990 self._usechg = usechg
991 self._chgdebug = chgdebug
991 self._chgdebug = chgdebug
992 self._useipv6 = useipv6
992 self._useipv6 = useipv6
993
993
994 self._aborted = False
994 self._aborted = False
995 self._daemonpids = []
995 self._daemonpids = []
996 self._finished = None
996 self._finished = None
997 self._ret = None
997 self._ret = None
998 self._out = None
998 self._out = None
999 self._skipped = None
999 self._skipped = None
1000 self._testtmp = None
1000 self._testtmp = None
1001 self._chgsockdir = None
1001 self._chgsockdir = None
1002
1002
1003 self._refout = self.readrefout()
1003 self._refout = self.readrefout()
1004
1004
1005 def readrefout(self):
1005 def readrefout(self):
1006 """read reference output"""
1006 """read reference output"""
1007 # If we're not in --debug mode and reference output file exists,
1007 # If we're not in --debug mode and reference output file exists,
1008 # check test output against it.
1008 # check test output against it.
1009 if self._debug:
1009 if self._debug:
1010 return None # to match "out is None"
1010 return None # to match "out is None"
1011 elif os.path.exists(self.refpath):
1011 elif os.path.exists(self.refpath):
1012 with open(self.refpath, 'rb') as f:
1012 with open(self.refpath, 'rb') as f:
1013 return f.read().splitlines(True)
1013 return f.read().splitlines(True)
1014 else:
1014 else:
1015 return []
1015 return []
1016
1016
1017 # needed to get base class __repr__ running
1017 # needed to get base class __repr__ running
1018 @property
1018 @property
1019 def _testMethodName(self):
1019 def _testMethodName(self):
1020 return self.name
1020 return self.name
1021
1021
1022 def __str__(self):
1022 def __str__(self):
1023 return self.name
1023 return self.name
1024
1024
1025 def shortDescription(self):
1025 def shortDescription(self):
1026 return self.name
1026 return self.name
1027
1027
1028 def setUp(self):
1028 def setUp(self):
1029 """Tasks to perform before run()."""
1029 """Tasks to perform before run()."""
1030 self._finished = False
1030 self._finished = False
1031 self._ret = None
1031 self._ret = None
1032 self._out = None
1032 self._out = None
1033 self._skipped = None
1033 self._skipped = None
1034
1034
1035 try:
1035 try:
1036 os.mkdir(self._threadtmp)
1036 os.mkdir(self._threadtmp)
1037 except OSError as e:
1037 except OSError as e:
1038 if e.errno != errno.EEXIST:
1038 if e.errno != errno.EEXIST:
1039 raise
1039 raise
1040
1040
1041 name = self._tmpname
1041 name = self._tmpname
1042 self._testtmp = os.path.join(self._threadtmp, name)
1042 self._testtmp = os.path.join(self._threadtmp, name)
1043 os.mkdir(self._testtmp)
1043 os.mkdir(self._testtmp)
1044
1044
1045 # Remove any previous output files.
1045 # Remove any previous output files.
1046 if os.path.exists(self.errpath):
1046 if os.path.exists(self.errpath):
1047 try:
1047 try:
1048 os.remove(self.errpath)
1048 os.remove(self.errpath)
1049 except OSError as e:
1049 except OSError as e:
1050 # We might have raced another test to clean up a .err
1050 # We might have raced another test to clean up a .err
1051 # file, so ignore ENOENT when removing a previous .err
1051 # file, so ignore ENOENT when removing a previous .err
1052 # file.
1052 # file.
1053 if e.errno != errno.ENOENT:
1053 if e.errno != errno.ENOENT:
1054 raise
1054 raise
1055
1055
1056 if self._usechg:
1056 if self._usechg:
1057 self._chgsockdir = os.path.join(
1057 self._chgsockdir = os.path.join(
1058 self._threadtmp, b'%s.chgsock' % name
1058 self._threadtmp, b'%s.chgsock' % name
1059 )
1059 )
1060 os.mkdir(self._chgsockdir)
1060 os.mkdir(self._chgsockdir)
1061
1061
1062 def run(self, result):
1062 def run(self, result):
1063 """Run this test and report results against a TestResult instance."""
1063 """Run this test and report results against a TestResult instance."""
1064 # This function is extremely similar to unittest.TestCase.run(). Once
1064 # This function is extremely similar to unittest.TestCase.run(). Once
1065 # we require Python 2.7 (or at least its version of unittest), this
1065 # we require Python 2.7 (or at least its version of unittest), this
1066 # function can largely go away.
1066 # function can largely go away.
1067 self._result = result
1067 self._result = result
1068 result.startTest(self)
1068 result.startTest(self)
1069 try:
1069 try:
1070 try:
1070 try:
1071 self.setUp()
1071 self.setUp()
1072 except (KeyboardInterrupt, SystemExit):
1072 except (KeyboardInterrupt, SystemExit):
1073 self._aborted = True
1073 self._aborted = True
1074 raise
1074 raise
1075 except Exception:
1075 except Exception:
1076 result.addError(self, sys.exc_info())
1076 result.addError(self, sys.exc_info())
1077 return
1077 return
1078
1078
1079 success = False
1079 success = False
1080 try:
1080 try:
1081 self.runTest()
1081 self.runTest()
1082 except KeyboardInterrupt:
1082 except KeyboardInterrupt:
1083 self._aborted = True
1083 self._aborted = True
1084 raise
1084 raise
1085 except unittest.SkipTest as e:
1085 except unittest.SkipTest as e:
1086 result.addSkip(self, str(e))
1086 result.addSkip(self, str(e))
1087 # The base class will have already counted this as a
1087 # The base class will have already counted this as a
1088 # test we "ran", but we want to exclude skipped tests
1088 # test we "ran", but we want to exclude skipped tests
1089 # from those we count towards those run.
1089 # from those we count towards those run.
1090 result.testsRun -= 1
1090 result.testsRun -= 1
1091 except self.failureException as e:
1091 except self.failureException as e:
1092 # This differs from unittest in that we don't capture
1092 # This differs from unittest in that we don't capture
1093 # the stack trace. This is for historical reasons and
1093 # the stack trace. This is for historical reasons and
1094 # this decision could be revisited in the future,
1094 # this decision could be revisited in the future,
1095 # especially for PythonTest instances.
1095 # especially for PythonTest instances.
1096 if result.addFailure(self, str(e)):
1096 if result.addFailure(self, str(e)):
1097 success = True
1097 success = True
1098 except Exception:
1098 except Exception:
1099 result.addError(self, sys.exc_info())
1099 result.addError(self, sys.exc_info())
1100 else:
1100 else:
1101 success = True
1101 success = True
1102
1102
1103 try:
1103 try:
1104 self.tearDown()
1104 self.tearDown()
1105 except (KeyboardInterrupt, SystemExit):
1105 except (KeyboardInterrupt, SystemExit):
1106 self._aborted = True
1106 self._aborted = True
1107 raise
1107 raise
1108 except Exception:
1108 except Exception:
1109 result.addError(self, sys.exc_info())
1109 result.addError(self, sys.exc_info())
1110 success = False
1110 success = False
1111
1111
1112 if success:
1112 if success:
1113 result.addSuccess(self)
1113 result.addSuccess(self)
1114 finally:
1114 finally:
1115 result.stopTest(self, interrupted=self._aborted)
1115 result.stopTest(self, interrupted=self._aborted)
1116
1116
1117 def runTest(self):
1117 def runTest(self):
1118 """Run this test instance.
1118 """Run this test instance.
1119
1119
1120 This will return a tuple describing the result of the test.
1120 This will return a tuple describing the result of the test.
1121 """
1121 """
1122 env = self._getenv()
1122 env = self._getenv()
1123 self._genrestoreenv(env)
1123 self._genrestoreenv(env)
1124 self._daemonpids.append(env['DAEMON_PIDS'])
1124 self._daemonpids.append(env['DAEMON_PIDS'])
1125 self._createhgrc(env['HGRCPATH'])
1125 self._createhgrc(env['HGRCPATH'])
1126
1126
1127 vlog('# Test', self.name)
1127 vlog('# Test', self.name)
1128
1128
1129 ret, out = self._run(env)
1129 ret, out = self._run(env)
1130 self._finished = True
1130 self._finished = True
1131 self._ret = ret
1131 self._ret = ret
1132 self._out = out
1132 self._out = out
1133
1133
1134 def describe(ret):
1134 def describe(ret):
1135 if ret < 0:
1135 if ret < 0:
1136 return 'killed by signal: %d' % -ret
1136 return 'killed by signal: %d' % -ret
1137 return 'returned error code %d' % ret
1137 return 'returned error code %d' % ret
1138
1138
1139 self._skipped = False
1139 self._skipped = False
1140
1140
1141 if ret == self.SKIPPED_STATUS:
1141 if ret == self.SKIPPED_STATUS:
1142 if out is None: # Debug mode, nothing to parse.
1142 if out is None: # Debug mode, nothing to parse.
1143 missing = ['unknown']
1143 missing = ['unknown']
1144 failed = None
1144 failed = None
1145 else:
1145 else:
1146 missing, failed = TTest.parsehghaveoutput(out)
1146 missing, failed = TTest.parsehghaveoutput(out)
1147
1147
1148 if not missing:
1148 if not missing:
1149 missing = ['skipped']
1149 missing = ['skipped']
1150
1150
1151 if failed:
1151 if failed:
1152 self.fail('hg have failed checking for %s' % failed[-1])
1152 self.fail('hg have failed checking for %s' % failed[-1])
1153 else:
1153 else:
1154 self._skipped = True
1154 self._skipped = True
1155 raise unittest.SkipTest(missing[-1])
1155 raise unittest.SkipTest(missing[-1])
1156 elif ret == 'timeout':
1156 elif ret == 'timeout':
1157 self.fail('timed out')
1157 self.fail('timed out')
1158 elif ret is False:
1158 elif ret is False:
1159 self.fail('no result code from test')
1159 self.fail('no result code from test')
1160 elif out != self._refout:
1160 elif out != self._refout:
1161 # Diff generation may rely on written .err file.
1161 # Diff generation may rely on written .err file.
1162 if (
1162 if (
1163 (ret != 0 or out != self._refout)
1163 (ret != 0 or out != self._refout)
1164 and not self._skipped
1164 and not self._skipped
1165 and not self._debug
1165 and not self._debug
1166 ):
1166 ):
1167 with open(self.errpath, 'wb') as f:
1167 with open(self.errpath, 'wb') as f:
1168 for line in out:
1168 for line in out:
1169 f.write(line)
1169 f.write(line)
1170
1170
1171 # The result object handles diff calculation for us.
1171 # The result object handles diff calculation for us.
1172 with firstlock:
1172 with firstlock:
1173 if self._result.addOutputMismatch(self, ret, out, self._refout):
1173 if self._result.addOutputMismatch(self, ret, out, self._refout):
1174 # change was accepted, skip failing
1174 # change was accepted, skip failing
1175 return
1175 return
1176 if self._first:
1176 if self._first:
1177 global firsterror
1177 global firsterror
1178 firsterror = True
1178 firsterror = True
1179
1179
1180 if ret:
1180 if ret:
1181 msg = 'output changed and ' + describe(ret)
1181 msg = 'output changed and ' + describe(ret)
1182 else:
1182 else:
1183 msg = 'output changed'
1183 msg = 'output changed'
1184
1184
1185 self.fail(msg)
1185 self.fail(msg)
1186 elif ret:
1186 elif ret:
1187 self.fail(describe(ret))
1187 self.fail(describe(ret))
1188
1188
1189 def tearDown(self):
1189 def tearDown(self):
1190 """Tasks to perform after run()."""
1190 """Tasks to perform after run()."""
1191 for entry in self._daemonpids:
1191 for entry in self._daemonpids:
1192 killdaemons(entry)
1192 killdaemons(entry)
1193 self._daemonpids = []
1193 self._daemonpids = []
1194
1194
1195 if self._keeptmpdir:
1195 if self._keeptmpdir:
1196 log(
1196 log(
1197 '\nKeeping testtmp dir: %s\nKeeping threadtmp dir: %s'
1197 '\nKeeping testtmp dir: %s\nKeeping threadtmp dir: %s'
1198 % (
1198 % (
1199 _bytes2sys(self._testtmp),
1199 _bytes2sys(self._testtmp),
1200 _bytes2sys(self._threadtmp),
1200 _bytes2sys(self._threadtmp),
1201 )
1201 )
1202 )
1202 )
1203 else:
1203 else:
1204 try:
1204 try:
1205 shutil.rmtree(self._testtmp)
1205 shutil.rmtree(self._testtmp)
1206 except OSError:
1206 except OSError:
1207 # unreadable directory may be left in $TESTTMP; fix permission
1207 # unreadable directory may be left in $TESTTMP; fix permission
1208 # and try again
1208 # and try again
1209 makecleanable(self._testtmp)
1209 makecleanable(self._testtmp)
1210 shutil.rmtree(self._testtmp, True)
1210 shutil.rmtree(self._testtmp, True)
1211 shutil.rmtree(self._threadtmp, True)
1211 shutil.rmtree(self._threadtmp, True)
1212
1212
1213 if self._usechg:
1213 if self._usechg:
1214 # chgservers will stop automatically after they find the socket
1214 # chgservers will stop automatically after they find the socket
1215 # files are deleted
1215 # files are deleted
1216 shutil.rmtree(self._chgsockdir, True)
1216 shutil.rmtree(self._chgsockdir, True)
1217
1217
1218 if (
1218 if (
1219 (self._ret != 0 or self._out != self._refout)
1219 (self._ret != 0 or self._out != self._refout)
1220 and not self._skipped
1220 and not self._skipped
1221 and not self._debug
1221 and not self._debug
1222 and self._out
1222 and self._out
1223 ):
1223 ):
1224 with open(self.errpath, 'wb') as f:
1224 with open(self.errpath, 'wb') as f:
1225 for line in self._out:
1225 for line in self._out:
1226 f.write(line)
1226 f.write(line)
1227
1227
1228 vlog("# Ret was:", self._ret, '(%s)' % self.name)
1228 vlog("# Ret was:", self._ret, '(%s)' % self.name)
1229
1229
1230 def _run(self, env):
1230 def _run(self, env):
1231 # This should be implemented in child classes to run tests.
1231 # This should be implemented in child classes to run tests.
1232 raise unittest.SkipTest('unknown test type')
1232 raise unittest.SkipTest('unknown test type')
1233
1233
1234 def abort(self):
1234 def abort(self):
1235 """Terminate execution of this test."""
1235 """Terminate execution of this test."""
1236 self._aborted = True
1236 self._aborted = True
1237
1237
1238 def _portmap(self, i):
1238 def _portmap(self, i):
1239 offset = b'' if i == 0 else b'%d' % i
1239 offset = b'' if i == 0 else b'%d' % i
1240 return (br':%d\b' % (self._startport + i), b':$HGPORT%s' % offset)
1240 return (br':%d\b' % (self._startport + i), b':$HGPORT%s' % offset)
1241
1241
1242 def _getreplacements(self):
1242 def _getreplacements(self):
1243 """Obtain a mapping of text replacements to apply to test output.
1243 """Obtain a mapping of text replacements to apply to test output.
1244
1244
1245 Test output needs to be normalized so it can be compared to expected
1245 Test output needs to be normalized so it can be compared to expected
1246 output. This function defines how some of that normalization will
1246 output. This function defines how some of that normalization will
1247 occur.
1247 occur.
1248 """
1248 """
1249 r = [
1249 r = [
1250 # This list should be parallel to defineport in _getenv
1250 # This list should be parallel to defineport in _getenv
1251 self._portmap(0),
1251 self._portmap(0),
1252 self._portmap(1),
1252 self._portmap(1),
1253 self._portmap(2),
1253 self._portmap(2),
1254 (br'([^0-9])%s' % re.escape(self._localip()), br'\1$LOCALIP'),
1254 (br'([^0-9])%s' % re.escape(self._localip()), br'\1$LOCALIP'),
1255 (br'\bHG_TXNID=TXN:[a-f0-9]{40}\b', br'HG_TXNID=TXN:$ID$'),
1255 (br'\bHG_TXNID=TXN:[a-f0-9]{40}\b', br'HG_TXNID=TXN:$ID$'),
1256 ]
1256 ]
1257 r.append((self._escapepath(self._testtmp), b'$TESTTMP'))
1257 r.append((self._escapepath(self._testtmp), b'$TESTTMP'))
1258
1258
1259 replacementfile = os.path.join(self._testdir, b'common-pattern.py')
1259 replacementfile = os.path.join(self._testdir, b'common-pattern.py')
1260
1260
1261 if os.path.exists(replacementfile):
1261 if os.path.exists(replacementfile):
1262 data = {}
1262 data = {}
1263 with open(replacementfile, mode='rb') as source:
1263 with open(replacementfile, mode='rb') as source:
1264 # the intermediate 'compile' step help with debugging
1264 # the intermediate 'compile' step help with debugging
1265 code = compile(source.read(), replacementfile, 'exec')
1265 code = compile(source.read(), replacementfile, 'exec')
1266 exec(code, data)
1266 exec(code, data)
1267 for value in data.get('substitutions', ()):
1267 for value in data.get('substitutions', ()):
1268 if len(value) != 2:
1268 if len(value) != 2:
1269 msg = 'malformatted substitution in %s: %r'
1269 msg = 'malformatted substitution in %s: %r'
1270 msg %= (replacementfile, value)
1270 msg %= (replacementfile, value)
1271 raise ValueError(msg)
1271 raise ValueError(msg)
1272 r.append(value)
1272 r.append(value)
1273 return r
1273 return r
1274
1274
1275 def _escapepath(self, p):
1275 def _escapepath(self, p):
1276 if os.name == 'nt':
1276 if os.name == 'nt':
1277 return b''.join(
1277 return b''.join(
1278 c.isalpha()
1278 c.isalpha()
1279 and b'[%s%s]' % (c.lower(), c.upper())
1279 and b'[%s%s]' % (c.lower(), c.upper())
1280 or c in b'/\\'
1280 or c in b'/\\'
1281 and br'[/\\]'
1281 and br'[/\\]'
1282 or c.isdigit()
1282 or c.isdigit()
1283 and c
1283 and c
1284 or b'\\' + c
1284 or b'\\' + c
1285 for c in [p[i : i + 1] for i in range(len(p))]
1285 for c in [p[i : i + 1] for i in range(len(p))]
1286 )
1286 )
1287 else:
1287 else:
1288 return re.escape(p)
1288 return re.escape(p)
1289
1289
1290 def _localip(self):
1290 def _localip(self):
1291 if self._useipv6:
1291 if self._useipv6:
1292 return b'::1'
1292 return b'::1'
1293 else:
1293 else:
1294 return b'127.0.0.1'
1294 return b'127.0.0.1'
1295
1295
1296 def _genrestoreenv(self, testenv):
1296 def _genrestoreenv(self, testenv):
1297 """Generate a script that can be used by tests to restore the original
1297 """Generate a script that can be used by tests to restore the original
1298 environment."""
1298 environment."""
1299 # Put the restoreenv script inside self._threadtmp
1299 # Put the restoreenv script inside self._threadtmp
1300 scriptpath = os.path.join(self._threadtmp, b'restoreenv.sh')
1300 scriptpath = os.path.join(self._threadtmp, b'restoreenv.sh')
1301 testenv['HGTEST_RESTOREENV'] = _bytes2sys(scriptpath)
1301 testenv['HGTEST_RESTOREENV'] = _bytes2sys(scriptpath)
1302
1302
1303 # Only restore environment variable names that the shell allows
1303 # Only restore environment variable names that the shell allows
1304 # us to export.
1304 # us to export.
1305 name_regex = re.compile('^[a-zA-Z][a-zA-Z0-9_]*$')
1305 name_regex = re.compile('^[a-zA-Z][a-zA-Z0-9_]*$')
1306
1306
1307 # Do not restore these variables; otherwise tests would fail.
1307 # Do not restore these variables; otherwise tests would fail.
1308 reqnames = {'PYTHON', 'TESTDIR', 'TESTTMP'}
1308 reqnames = {'PYTHON', 'TESTDIR', 'TESTTMP'}
1309
1309
1310 with open(scriptpath, 'w') as envf:
1310 with open(scriptpath, 'w') as envf:
1311 for name, value in origenviron.items():
1311 for name, value in origenviron.items():
1312 if not name_regex.match(name):
1312 if not name_regex.match(name):
1313 # Skip environment variables with unusual names not
1313 # Skip environment variables with unusual names not
1314 # allowed by most shells.
1314 # allowed by most shells.
1315 continue
1315 continue
1316 if name in reqnames:
1316 if name in reqnames:
1317 continue
1317 continue
1318 envf.write('%s=%s\n' % (name, shellquote(value)))
1318 envf.write('%s=%s\n' % (name, shellquote(value)))
1319
1319
1320 for name in testenv:
1320 for name in testenv:
1321 if name in origenviron or name in reqnames:
1321 if name in origenviron or name in reqnames:
1322 continue
1322 continue
1323 envf.write('unset %s\n' % (name,))
1323 envf.write('unset %s\n' % (name,))
1324
1324
1325 def _getenv(self):
1325 def _getenv(self):
1326 """Obtain environment variables to use during test execution."""
1326 """Obtain environment variables to use during test execution."""
1327
1327
1328 def defineport(i):
1328 def defineport(i):
1329 offset = '' if i == 0 else '%s' % i
1329 offset = '' if i == 0 else '%s' % i
1330 env["HGPORT%s" % offset] = '%s' % (self._startport + i)
1330 env["HGPORT%s" % offset] = '%s' % (self._startport + i)
1331
1331
1332 env = os.environ.copy()
1332 env = os.environ.copy()
1333 env['PYTHONUSERBASE'] = sysconfig.get_config_var('userbase') or ''
1333 env['PYTHONUSERBASE'] = sysconfig.get_config_var('userbase') or ''
1334 env['HGEMITWARNINGS'] = '1'
1334 env['HGEMITWARNINGS'] = '1'
1335 env['TESTTMP'] = _bytes2sys(self._testtmp)
1335 env['TESTTMP'] = _bytes2sys(self._testtmp)
1336 env['TESTNAME'] = self.name
1336 env['TESTNAME'] = self.name
1337 env['HOME'] = _bytes2sys(self._testtmp)
1337 env['HOME'] = _bytes2sys(self._testtmp)
1338 formated_timeout = _bytes2sys(b"%d" % default_defaults['timeout'][1])
1338 formated_timeout = _bytes2sys(b"%d" % default_defaults['timeout'][1])
1339 env['HGTEST_TIMEOUT_DEFAULT'] = formated_timeout
1339 env['HGTEST_TIMEOUT_DEFAULT'] = formated_timeout
1340 env['HGTEST_TIMEOUT'] = _bytes2sys(b"%d" % self._timeout)
1340 env['HGTEST_TIMEOUT'] = _bytes2sys(b"%d" % self._timeout)
1341 # This number should match portneeded in _getport
1341 # This number should match portneeded in _getport
1342 for port in xrange(3):
1342 for port in xrange(3):
1343 # This list should be parallel to _portmap in _getreplacements
1343 # This list should be parallel to _portmap in _getreplacements
1344 defineport(port)
1344 defineport(port)
1345 env["HGRCPATH"] = _bytes2sys(os.path.join(self._threadtmp, b'.hgrc'))
1345 env["HGRCPATH"] = _bytes2sys(os.path.join(self._threadtmp, b'.hgrc'))
1346 env["DAEMON_PIDS"] = _bytes2sys(
1346 env["DAEMON_PIDS"] = _bytes2sys(
1347 os.path.join(self._threadtmp, b'daemon.pids')
1347 os.path.join(self._threadtmp, b'daemon.pids')
1348 )
1348 )
1349 env["HGEDITOR"] = (
1349 env["HGEDITOR"] = (
1350 '"' + sysexecutable + '"' + ' -c "import sys; sys.exit(0)"'
1350 '"' + sysexecutable + '"' + ' -c "import sys; sys.exit(0)"'
1351 )
1351 )
1352 env["HGUSER"] = "test"
1352 env["HGUSER"] = "test"
1353 env["HGENCODING"] = "ascii"
1353 env["HGENCODING"] = "ascii"
1354 env["HGENCODINGMODE"] = "strict"
1354 env["HGENCODINGMODE"] = "strict"
1355 env["HGHOSTNAME"] = "test-hostname"
1355 env["HGHOSTNAME"] = "test-hostname"
1356 env['HGIPV6'] = str(int(self._useipv6))
1356 env['HGIPV6'] = str(int(self._useipv6))
1357 # See contrib/catapipe.py for how to use this functionality.
1357 # See contrib/catapipe.py for how to use this functionality.
1358 if 'HGTESTCATAPULTSERVERPIPE' not in env:
1358 if 'HGTESTCATAPULTSERVERPIPE' not in env:
1359 # If we don't have HGTESTCATAPULTSERVERPIPE explicitly set, pull the
1359 # If we don't have HGTESTCATAPULTSERVERPIPE explicitly set, pull the
1360 # non-test one in as a default, otherwise set to devnull
1360 # non-test one in as a default, otherwise set to devnull
1361 env['HGTESTCATAPULTSERVERPIPE'] = env.get(
1361 env['HGTESTCATAPULTSERVERPIPE'] = env.get(
1362 'HGCATAPULTSERVERPIPE', os.devnull
1362 'HGCATAPULTSERVERPIPE', os.devnull
1363 )
1363 )
1364
1364
1365 extraextensions = []
1365 extraextensions = []
1366 for opt in self._extraconfigopts:
1366 for opt in self._extraconfigopts:
1367 section, key = _sys2bytes(opt).split(b'.', 1)
1367 section, key = opt.split('.', 1)
1368 if section != 'extensions':
1368 if section != 'extensions':
1369 continue
1369 continue
1370 name = key.split(b'=', 1)[0]
1370 name = key.split('=', 1)[0]
1371 extraextensions.append(name)
1371 extraextensions.append(name)
1372
1372
1373 if extraextensions:
1373 if extraextensions:
1374 env['HGTESTEXTRAEXTENSIONS'] = b' '.join(extraextensions)
1374 env['HGTESTEXTRAEXTENSIONS'] = ' '.join(extraextensions)
1375
1375
1376 # LOCALIP could be ::1 or 127.0.0.1. Useful for tests that require raw
1376 # LOCALIP could be ::1 or 127.0.0.1. Useful for tests that require raw
1377 # IP addresses.
1377 # IP addresses.
1378 env['LOCALIP'] = _bytes2sys(self._localip())
1378 env['LOCALIP'] = _bytes2sys(self._localip())
1379
1379
1380 # This has the same effect as Py_LegacyWindowsStdioFlag in exewrapper.c,
1380 # This has the same effect as Py_LegacyWindowsStdioFlag in exewrapper.c,
1381 # but this is needed for testing python instances like dummyssh,
1381 # but this is needed for testing python instances like dummyssh,
1382 # dummysmtpd.py, and dumbhttp.py.
1382 # dummysmtpd.py, and dumbhttp.py.
1383 if PYTHON3 and os.name == 'nt':
1383 if PYTHON3 and os.name == 'nt':
1384 env['PYTHONLEGACYWINDOWSSTDIO'] = '1'
1384 env['PYTHONLEGACYWINDOWSSTDIO'] = '1'
1385
1385
1386 # Modified HOME in test environment can confuse Rust tools. So set
1386 # Modified HOME in test environment can confuse Rust tools. So set
1387 # CARGO_HOME and RUSTUP_HOME automatically if a Rust toolchain is
1387 # CARGO_HOME and RUSTUP_HOME automatically if a Rust toolchain is
1388 # present and these variables aren't already defined.
1388 # present and these variables aren't already defined.
1389 cargo_home_path = os.path.expanduser('~/.cargo')
1389 cargo_home_path = os.path.expanduser('~/.cargo')
1390 rustup_home_path = os.path.expanduser('~/.rustup')
1390 rustup_home_path = os.path.expanduser('~/.rustup')
1391
1391
1392 if os.path.exists(cargo_home_path) and b'CARGO_HOME' not in osenvironb:
1392 if os.path.exists(cargo_home_path) and b'CARGO_HOME' not in osenvironb:
1393 env['CARGO_HOME'] = cargo_home_path
1393 env['CARGO_HOME'] = cargo_home_path
1394 if (
1394 if (
1395 os.path.exists(rustup_home_path)
1395 os.path.exists(rustup_home_path)
1396 and b'RUSTUP_HOME' not in osenvironb
1396 and b'RUSTUP_HOME' not in osenvironb
1397 ):
1397 ):
1398 env['RUSTUP_HOME'] = rustup_home_path
1398 env['RUSTUP_HOME'] = rustup_home_path
1399
1399
1400 # Reset some environment variables to well-known values so that
1400 # Reset some environment variables to well-known values so that
1401 # the tests produce repeatable output.
1401 # the tests produce repeatable output.
1402 env['LANG'] = env['LC_ALL'] = env['LANGUAGE'] = 'C'
1402 env['LANG'] = env['LC_ALL'] = env['LANGUAGE'] = 'C'
1403 env['TZ'] = 'GMT'
1403 env['TZ'] = 'GMT'
1404 env["EMAIL"] = "Foo Bar <foo.bar@example.com>"
1404 env["EMAIL"] = "Foo Bar <foo.bar@example.com>"
1405 env['COLUMNS'] = '80'
1405 env['COLUMNS'] = '80'
1406 env['TERM'] = 'xterm'
1406 env['TERM'] = 'xterm'
1407
1407
1408 dropped = [
1408 dropped = [
1409 'CDPATH',
1409 'CDPATH',
1410 'CHGDEBUG',
1410 'CHGDEBUG',
1411 'EDITOR',
1411 'EDITOR',
1412 'GREP_OPTIONS',
1412 'GREP_OPTIONS',
1413 'HG',
1413 'HG',
1414 'HGMERGE',
1414 'HGMERGE',
1415 'HGPLAIN',
1415 'HGPLAIN',
1416 'HGPLAINEXCEPT',
1416 'HGPLAINEXCEPT',
1417 'HGPROF',
1417 'HGPROF',
1418 'http_proxy',
1418 'http_proxy',
1419 'no_proxy',
1419 'no_proxy',
1420 'NO_PROXY',
1420 'NO_PROXY',
1421 'PAGER',
1421 'PAGER',
1422 'VISUAL',
1422 'VISUAL',
1423 ]
1423 ]
1424
1424
1425 for k in dropped:
1425 for k in dropped:
1426 if k in env:
1426 if k in env:
1427 del env[k]
1427 del env[k]
1428
1428
1429 # unset env related to hooks
1429 # unset env related to hooks
1430 for k in list(env):
1430 for k in list(env):
1431 if k.startswith('HG_'):
1431 if k.startswith('HG_'):
1432 del env[k]
1432 del env[k]
1433
1433
1434 if self._usechg:
1434 if self._usechg:
1435 env['CHGSOCKNAME'] = os.path.join(self._chgsockdir, b'server')
1435 env['CHGSOCKNAME'] = os.path.join(self._chgsockdir, b'server')
1436 if self._chgdebug:
1436 if self._chgdebug:
1437 env['CHGDEBUG'] = 'true'
1437 env['CHGDEBUG'] = 'true'
1438
1438
1439 return env
1439 return env
1440
1440
1441 def _createhgrc(self, path):
1441 def _createhgrc(self, path):
1442 """Create an hgrc file for this test."""
1442 """Create an hgrc file for this test."""
1443 with open(path, 'wb') as hgrc:
1443 with open(path, 'wb') as hgrc:
1444 hgrc.write(b'[ui]\n')
1444 hgrc.write(b'[ui]\n')
1445 hgrc.write(b'slash = True\n')
1445 hgrc.write(b'slash = True\n')
1446 hgrc.write(b'interactive = False\n')
1446 hgrc.write(b'interactive = False\n')
1447 hgrc.write(b'detailed-exit-code = True\n')
1447 hgrc.write(b'detailed-exit-code = True\n')
1448 hgrc.write(b'merge = internal:merge\n')
1448 hgrc.write(b'merge = internal:merge\n')
1449 hgrc.write(b'mergemarkers = detailed\n')
1449 hgrc.write(b'mergemarkers = detailed\n')
1450 hgrc.write(b'promptecho = True\n')
1450 hgrc.write(b'promptecho = True\n')
1451 hgrc.write(b'timeout.warn=15\n')
1451 hgrc.write(b'timeout.warn=15\n')
1452 hgrc.write(b'[defaults]\n')
1452 hgrc.write(b'[defaults]\n')
1453 hgrc.write(b'[devel]\n')
1453 hgrc.write(b'[devel]\n')
1454 hgrc.write(b'all-warnings = true\n')
1454 hgrc.write(b'all-warnings = true\n')
1455 hgrc.write(b'default-date = 0 0\n')
1455 hgrc.write(b'default-date = 0 0\n')
1456 hgrc.write(b'[largefiles]\n')
1456 hgrc.write(b'[largefiles]\n')
1457 hgrc.write(
1457 hgrc.write(
1458 b'usercache = %s\n'
1458 b'usercache = %s\n'
1459 % (os.path.join(self._testtmp, b'.cache/largefiles'))
1459 % (os.path.join(self._testtmp, b'.cache/largefiles'))
1460 )
1460 )
1461 hgrc.write(b'[lfs]\n')
1461 hgrc.write(b'[lfs]\n')
1462 hgrc.write(
1462 hgrc.write(
1463 b'usercache = %s\n'
1463 b'usercache = %s\n'
1464 % (os.path.join(self._testtmp, b'.cache/lfs'))
1464 % (os.path.join(self._testtmp, b'.cache/lfs'))
1465 )
1465 )
1466 hgrc.write(b'[web]\n')
1466 hgrc.write(b'[web]\n')
1467 hgrc.write(b'address = localhost\n')
1467 hgrc.write(b'address = localhost\n')
1468 hgrc.write(b'ipv6 = %r\n' % self._useipv6)
1468 hgrc.write(b'ipv6 = %r\n' % self._useipv6)
1469 hgrc.write(b'server-header = testing stub value\n')
1469 hgrc.write(b'server-header = testing stub value\n')
1470
1470
1471 for opt in self._extraconfigopts:
1471 for opt in self._extraconfigopts:
1472 section, key = _sys2bytes(opt).split(b'.', 1)
1472 section, key = _sys2bytes(opt).split(b'.', 1)
1473 assert b'=' in key, (
1473 assert b'=' in key, (
1474 'extra config opt %s must ' 'have an = for assignment' % opt
1474 'extra config opt %s must ' 'have an = for assignment' % opt
1475 )
1475 )
1476 hgrc.write(b'[%s]\n%s\n' % (section, key))
1476 hgrc.write(b'[%s]\n%s\n' % (section, key))
1477
1477
1478 def fail(self, msg):
1478 def fail(self, msg):
1479 # unittest differentiates between errored and failed.
1479 # unittest differentiates between errored and failed.
1480 # Failed is denoted by AssertionError (by default at least).
1480 # Failed is denoted by AssertionError (by default at least).
1481 raise AssertionError(msg)
1481 raise AssertionError(msg)
1482
1482
1483 def _runcommand(self, cmd, env, normalizenewlines=False):
1483 def _runcommand(self, cmd, env, normalizenewlines=False):
1484 """Run command in a sub-process, capturing the output (stdout and
1484 """Run command in a sub-process, capturing the output (stdout and
1485 stderr).
1485 stderr).
1486
1486
1487 Return a tuple (exitcode, output). output is None in debug mode.
1487 Return a tuple (exitcode, output). output is None in debug mode.
1488 """
1488 """
1489 if self._debug:
1489 if self._debug:
1490 proc = subprocess.Popen(
1490 proc = subprocess.Popen(
1491 _bytes2sys(cmd),
1491 _bytes2sys(cmd),
1492 shell=True,
1492 shell=True,
1493 cwd=_bytes2sys(self._testtmp),
1493 cwd=_bytes2sys(self._testtmp),
1494 env=env,
1494 env=env,
1495 )
1495 )
1496 ret = proc.wait()
1496 ret = proc.wait()
1497 return (ret, None)
1497 return (ret, None)
1498
1498
1499 proc = Popen4(cmd, self._testtmp, self._timeout, env)
1499 proc = Popen4(cmd, self._testtmp, self._timeout, env)
1500
1500
1501 def cleanup():
1501 def cleanup():
1502 terminate(proc)
1502 terminate(proc)
1503 ret = proc.wait()
1503 ret = proc.wait()
1504 if ret == 0:
1504 if ret == 0:
1505 ret = signal.SIGTERM << 8
1505 ret = signal.SIGTERM << 8
1506 killdaemons(env['DAEMON_PIDS'])
1506 killdaemons(env['DAEMON_PIDS'])
1507 return ret
1507 return ret
1508
1508
1509 proc.tochild.close()
1509 proc.tochild.close()
1510
1510
1511 try:
1511 try:
1512 output = proc.fromchild.read()
1512 output = proc.fromchild.read()
1513 except KeyboardInterrupt:
1513 except KeyboardInterrupt:
1514 vlog('# Handling keyboard interrupt')
1514 vlog('# Handling keyboard interrupt')
1515 cleanup()
1515 cleanup()
1516 raise
1516 raise
1517
1517
1518 ret = proc.wait()
1518 ret = proc.wait()
1519 if wifexited(ret):
1519 if wifexited(ret):
1520 ret = os.WEXITSTATUS(ret)
1520 ret = os.WEXITSTATUS(ret)
1521
1521
1522 if proc.timeout:
1522 if proc.timeout:
1523 ret = 'timeout'
1523 ret = 'timeout'
1524
1524
1525 if ret:
1525 if ret:
1526 killdaemons(env['DAEMON_PIDS'])
1526 killdaemons(env['DAEMON_PIDS'])
1527
1527
1528 for s, r in self._getreplacements():
1528 for s, r in self._getreplacements():
1529 output = re.sub(s, r, output)
1529 output = re.sub(s, r, output)
1530
1530
1531 if normalizenewlines:
1531 if normalizenewlines:
1532 output = output.replace(b'\r\n', b'\n')
1532 output = output.replace(b'\r\n', b'\n')
1533
1533
1534 return ret, output.splitlines(True)
1534 return ret, output.splitlines(True)
1535
1535
1536
1536
1537 class PythonTest(Test):
1537 class PythonTest(Test):
1538 """A Python-based test."""
1538 """A Python-based test."""
1539
1539
1540 @property
1540 @property
1541 def refpath(self):
1541 def refpath(self):
1542 return os.path.join(self._testdir, b'%s.out' % self.bname)
1542 return os.path.join(self._testdir, b'%s.out' % self.bname)
1543
1543
1544 def _run(self, env):
1544 def _run(self, env):
1545 # Quote the python(3) executable for Windows
1545 # Quote the python(3) executable for Windows
1546 cmd = b'"%s" "%s"' % (PYTHON, self.path)
1546 cmd = b'"%s" "%s"' % (PYTHON, self.path)
1547 vlog("# Running", cmd.decode("utf-8"))
1547 vlog("# Running", cmd.decode("utf-8"))
1548 normalizenewlines = os.name == 'nt'
1548 normalizenewlines = os.name == 'nt'
1549 result = self._runcommand(cmd, env, normalizenewlines=normalizenewlines)
1549 result = self._runcommand(cmd, env, normalizenewlines=normalizenewlines)
1550 if self._aborted:
1550 if self._aborted:
1551 raise KeyboardInterrupt()
1551 raise KeyboardInterrupt()
1552
1552
1553 return result
1553 return result
1554
1554
1555
1555
1556 # Some glob patterns apply only in some circumstances, so the script
1556 # Some glob patterns apply only in some circumstances, so the script
1557 # might want to remove (glob) annotations that otherwise should be
1557 # might want to remove (glob) annotations that otherwise should be
1558 # retained.
1558 # retained.
1559 checkcodeglobpats = [
1559 checkcodeglobpats = [
1560 # On Windows it looks like \ doesn't require a (glob), but we know
1560 # On Windows it looks like \ doesn't require a (glob), but we know
1561 # better.
1561 # better.
1562 re.compile(br'^pushing to \$TESTTMP/.*[^)]$'),
1562 re.compile(br'^pushing to \$TESTTMP/.*[^)]$'),
1563 re.compile(br'^moving \S+/.*[^)]$'),
1563 re.compile(br'^moving \S+/.*[^)]$'),
1564 re.compile(br'^pulling from \$TESTTMP/.*[^)]$'),
1564 re.compile(br'^pulling from \$TESTTMP/.*[^)]$'),
1565 # Not all platforms have 127.0.0.1 as loopback (though most do),
1565 # Not all platforms have 127.0.0.1 as loopback (though most do),
1566 # so we always glob that too.
1566 # so we always glob that too.
1567 re.compile(br'.*\$LOCALIP.*$'),
1567 re.compile(br'.*\$LOCALIP.*$'),
1568 ]
1568 ]
1569
1569
1570 bchr = chr
1570 bchr = chr
1571 if PYTHON3:
1571 if PYTHON3:
1572 bchr = lambda x: bytes([x])
1572 bchr = lambda x: bytes([x])
1573
1573
1574 WARN_UNDEFINED = 1
1574 WARN_UNDEFINED = 1
1575 WARN_YES = 2
1575 WARN_YES = 2
1576 WARN_NO = 3
1576 WARN_NO = 3
1577
1577
1578 MARK_OPTIONAL = b" (?)\n"
1578 MARK_OPTIONAL = b" (?)\n"
1579
1579
1580
1580
1581 def isoptional(line):
1581 def isoptional(line):
1582 return line.endswith(MARK_OPTIONAL)
1582 return line.endswith(MARK_OPTIONAL)
1583
1583
1584
1584
1585 class TTest(Test):
1585 class TTest(Test):
1586 """A "t test" is a test backed by a .t file."""
1586 """A "t test" is a test backed by a .t file."""
1587
1587
1588 SKIPPED_PREFIX = b'skipped: '
1588 SKIPPED_PREFIX = b'skipped: '
1589 FAILED_PREFIX = b'hghave check failed: '
1589 FAILED_PREFIX = b'hghave check failed: '
1590 NEEDESCAPE = re.compile(br'[\x00-\x08\x0b-\x1f\x7f-\xff]').search
1590 NEEDESCAPE = re.compile(br'[\x00-\x08\x0b-\x1f\x7f-\xff]').search
1591
1591
1592 ESCAPESUB = re.compile(br'[\x00-\x08\x0b-\x1f\\\x7f-\xff]').sub
1592 ESCAPESUB = re.compile(br'[\x00-\x08\x0b-\x1f\\\x7f-\xff]').sub
1593 ESCAPEMAP = {bchr(i): br'\x%02x' % i for i in range(256)}
1593 ESCAPEMAP = {bchr(i): br'\x%02x' % i for i in range(256)}
1594 ESCAPEMAP.update({b'\\': b'\\\\', b'\r': br'\r'})
1594 ESCAPEMAP.update({b'\\': b'\\\\', b'\r': br'\r'})
1595
1595
1596 def __init__(self, path, *args, **kwds):
1596 def __init__(self, path, *args, **kwds):
1597 # accept an extra "case" parameter
1597 # accept an extra "case" parameter
1598 case = kwds.pop('case', [])
1598 case = kwds.pop('case', [])
1599 self._case = case
1599 self._case = case
1600 self._allcases = {x for y in parsettestcases(path) for x in y}
1600 self._allcases = {x for y in parsettestcases(path) for x in y}
1601 super(TTest, self).__init__(path, *args, **kwds)
1601 super(TTest, self).__init__(path, *args, **kwds)
1602 if case:
1602 if case:
1603 casepath = b'#'.join(case)
1603 casepath = b'#'.join(case)
1604 self.name = '%s#%s' % (self.name, _bytes2sys(casepath))
1604 self.name = '%s#%s' % (self.name, _bytes2sys(casepath))
1605 self.errpath = b'%s#%s.err' % (self.errpath[:-4], casepath)
1605 self.errpath = b'%s#%s.err' % (self.errpath[:-4], casepath)
1606 self._tmpname += b'-%s' % casepath.replace(b'#', b'-')
1606 self._tmpname += b'-%s' % casepath.replace(b'#', b'-')
1607 self._have = {}
1607 self._have = {}
1608
1608
1609 @property
1609 @property
1610 def refpath(self):
1610 def refpath(self):
1611 return os.path.join(self._testdir, self.bname)
1611 return os.path.join(self._testdir, self.bname)
1612
1612
1613 def _run(self, env):
1613 def _run(self, env):
1614 with open(self.path, 'rb') as f:
1614 with open(self.path, 'rb') as f:
1615 lines = f.readlines()
1615 lines = f.readlines()
1616
1616
1617 # .t file is both reference output and the test input, keep reference
1617 # .t file is both reference output and the test input, keep reference
1618 # output updated with the the test input. This avoids some race
1618 # output updated with the the test input. This avoids some race
1619 # conditions where the reference output does not match the actual test.
1619 # conditions where the reference output does not match the actual test.
1620 if self._refout is not None:
1620 if self._refout is not None:
1621 self._refout = lines
1621 self._refout = lines
1622
1622
1623 salt, script, after, expected = self._parsetest(lines)
1623 salt, script, after, expected = self._parsetest(lines)
1624
1624
1625 # Write out the generated script.
1625 # Write out the generated script.
1626 fname = b'%s.sh' % self._testtmp
1626 fname = b'%s.sh' % self._testtmp
1627 with open(fname, 'wb') as f:
1627 with open(fname, 'wb') as f:
1628 for l in script:
1628 for l in script:
1629 f.write(l)
1629 f.write(l)
1630
1630
1631 cmd = b'%s "%s"' % (self._shell, fname)
1631 cmd = b'%s "%s"' % (self._shell, fname)
1632 vlog("# Running", cmd.decode("utf-8"))
1632 vlog("# Running", cmd.decode("utf-8"))
1633
1633
1634 exitcode, output = self._runcommand(cmd, env)
1634 exitcode, output = self._runcommand(cmd, env)
1635
1635
1636 if self._aborted:
1636 if self._aborted:
1637 raise KeyboardInterrupt()
1637 raise KeyboardInterrupt()
1638
1638
1639 # Do not merge output if skipped. Return hghave message instead.
1639 # Do not merge output if skipped. Return hghave message instead.
1640 # Similarly, with --debug, output is None.
1640 # Similarly, with --debug, output is None.
1641 if exitcode == self.SKIPPED_STATUS or output is None:
1641 if exitcode == self.SKIPPED_STATUS or output is None:
1642 return exitcode, output
1642 return exitcode, output
1643
1643
1644 return self._processoutput(exitcode, output, salt, after, expected)
1644 return self._processoutput(exitcode, output, salt, after, expected)
1645
1645
1646 def _hghave(self, reqs):
1646 def _hghave(self, reqs):
1647 allreqs = b' '.join(reqs)
1647 allreqs = b' '.join(reqs)
1648
1648
1649 self._detectslow(reqs)
1649 self._detectslow(reqs)
1650
1650
1651 if allreqs in self._have:
1651 if allreqs in self._have:
1652 return self._have.get(allreqs)
1652 return self._have.get(allreqs)
1653
1653
1654 # TODO do something smarter when all other uses of hghave are gone.
1654 # TODO do something smarter when all other uses of hghave are gone.
1655 runtestdir = osenvironb[b'RUNTESTDIR']
1655 runtestdir = osenvironb[b'RUNTESTDIR']
1656 tdir = runtestdir.replace(b'\\', b'/')
1656 tdir = runtestdir.replace(b'\\', b'/')
1657 proc = Popen4(
1657 proc = Popen4(
1658 b'%s -c "%s/hghave %s"' % (self._shell, tdir, allreqs),
1658 b'%s -c "%s/hghave %s"' % (self._shell, tdir, allreqs),
1659 self._testtmp,
1659 self._testtmp,
1660 0,
1660 0,
1661 self._getenv(),
1661 self._getenv(),
1662 )
1662 )
1663 stdout, stderr = proc.communicate()
1663 stdout, stderr = proc.communicate()
1664 ret = proc.wait()
1664 ret = proc.wait()
1665 if wifexited(ret):
1665 if wifexited(ret):
1666 ret = os.WEXITSTATUS(ret)
1666 ret = os.WEXITSTATUS(ret)
1667 if ret == 2:
1667 if ret == 2:
1668 print(stdout.decode('utf-8'))
1668 print(stdout.decode('utf-8'))
1669 sys.exit(1)
1669 sys.exit(1)
1670
1670
1671 if ret != 0:
1671 if ret != 0:
1672 self._have[allreqs] = (False, stdout)
1672 self._have[allreqs] = (False, stdout)
1673 return False, stdout
1673 return False, stdout
1674
1674
1675 self._have[allreqs] = (True, None)
1675 self._have[allreqs] = (True, None)
1676 return True, None
1676 return True, None
1677
1677
1678 def _detectslow(self, reqs):
1678 def _detectslow(self, reqs):
1679 """update the timeout of slow test when appropriate"""
1679 """update the timeout of slow test when appropriate"""
1680 if b'slow' in reqs:
1680 if b'slow' in reqs:
1681 self._timeout = self._slowtimeout
1681 self._timeout = self._slowtimeout
1682
1682
1683 def _iftest(self, args):
1683 def _iftest(self, args):
1684 # implements "#if"
1684 # implements "#if"
1685 reqs = []
1685 reqs = []
1686 for arg in args:
1686 for arg in args:
1687 if arg.startswith(b'no-') and arg[3:] in self._allcases:
1687 if arg.startswith(b'no-') and arg[3:] in self._allcases:
1688 if arg[3:] in self._case:
1688 if arg[3:] in self._case:
1689 return False
1689 return False
1690 elif arg in self._allcases:
1690 elif arg in self._allcases:
1691 if arg not in self._case:
1691 if arg not in self._case:
1692 return False
1692 return False
1693 else:
1693 else:
1694 reqs.append(arg)
1694 reqs.append(arg)
1695 self._detectslow(reqs)
1695 self._detectslow(reqs)
1696 return self._hghave(reqs)[0]
1696 return self._hghave(reqs)[0]
1697
1697
1698 def _parsetest(self, lines):
1698 def _parsetest(self, lines):
1699 # We generate a shell script which outputs unique markers to line
1699 # We generate a shell script which outputs unique markers to line
1700 # up script results with our source. These markers include input
1700 # up script results with our source. These markers include input
1701 # line number and the last return code.
1701 # line number and the last return code.
1702 salt = b"SALT%d" % time.time()
1702 salt = b"SALT%d" % time.time()
1703
1703
1704 def addsalt(line, inpython):
1704 def addsalt(line, inpython):
1705 if inpython:
1705 if inpython:
1706 script.append(b'%s %d 0\n' % (salt, line))
1706 script.append(b'%s %d 0\n' % (salt, line))
1707 else:
1707 else:
1708 script.append(b'echo %s %d $?\n' % (salt, line))
1708 script.append(b'echo %s %d $?\n' % (salt, line))
1709
1709
1710 activetrace = []
1710 activetrace = []
1711 session = str(uuid.uuid4())
1711 session = str(uuid.uuid4())
1712 if PYTHON3:
1712 if PYTHON3:
1713 session = session.encode('ascii')
1713 session = session.encode('ascii')
1714 hgcatapult = os.getenv('HGTESTCATAPULTSERVERPIPE') or os.getenv(
1714 hgcatapult = os.getenv('HGTESTCATAPULTSERVERPIPE') or os.getenv(
1715 'HGCATAPULTSERVERPIPE'
1715 'HGCATAPULTSERVERPIPE'
1716 )
1716 )
1717
1717
1718 def toggletrace(cmd=None):
1718 def toggletrace(cmd=None):
1719 if not hgcatapult or hgcatapult == os.devnull:
1719 if not hgcatapult or hgcatapult == os.devnull:
1720 return
1720 return
1721
1721
1722 if activetrace:
1722 if activetrace:
1723 script.append(
1723 script.append(
1724 b'echo END %s %s >> "$HGTESTCATAPULTSERVERPIPE"\n'
1724 b'echo END %s %s >> "$HGTESTCATAPULTSERVERPIPE"\n'
1725 % (session, activetrace[0])
1725 % (session, activetrace[0])
1726 )
1726 )
1727 if cmd is None:
1727 if cmd is None:
1728 return
1728 return
1729
1729
1730 if isinstance(cmd, str):
1730 if isinstance(cmd, str):
1731 quoted = shellquote(cmd.strip())
1731 quoted = shellquote(cmd.strip())
1732 else:
1732 else:
1733 quoted = shellquote(cmd.strip().decode('utf8')).encode('utf8')
1733 quoted = shellquote(cmd.strip().decode('utf8')).encode('utf8')
1734 quoted = quoted.replace(b'\\', b'\\\\')
1734 quoted = quoted.replace(b'\\', b'\\\\')
1735 script.append(
1735 script.append(
1736 b'echo START %s %s >> "$HGTESTCATAPULTSERVERPIPE"\n'
1736 b'echo START %s %s >> "$HGTESTCATAPULTSERVERPIPE"\n'
1737 % (session, quoted)
1737 % (session, quoted)
1738 )
1738 )
1739 activetrace[0:] = [quoted]
1739 activetrace[0:] = [quoted]
1740
1740
1741 script = []
1741 script = []
1742
1742
1743 # After we run the shell script, we re-unify the script output
1743 # After we run the shell script, we re-unify the script output
1744 # with non-active parts of the source, with synchronization by our
1744 # with non-active parts of the source, with synchronization by our
1745 # SALT line number markers. The after table contains the non-active
1745 # SALT line number markers. The after table contains the non-active
1746 # components, ordered by line number.
1746 # components, ordered by line number.
1747 after = {}
1747 after = {}
1748
1748
1749 # Expected shell script output.
1749 # Expected shell script output.
1750 expected = {}
1750 expected = {}
1751
1751
1752 pos = prepos = -1
1752 pos = prepos = -1
1753
1753
1754 # True or False when in a true or false conditional section
1754 # True or False when in a true or false conditional section
1755 skipping = None
1755 skipping = None
1756
1756
1757 # We keep track of whether or not we're in a Python block so we
1757 # We keep track of whether or not we're in a Python block so we
1758 # can generate the surrounding doctest magic.
1758 # can generate the surrounding doctest magic.
1759 inpython = False
1759 inpython = False
1760
1760
1761 if self._debug:
1761 if self._debug:
1762 script.append(b'set -x\n')
1762 script.append(b'set -x\n')
1763 if self._hgcommand != b'hg':
1763 if self._hgcommand != b'hg':
1764 script.append(b'alias hg="%s"\n' % self._hgcommand)
1764 script.append(b'alias hg="%s"\n' % self._hgcommand)
1765 if os.getenv('MSYSTEM'):
1765 if os.getenv('MSYSTEM'):
1766 script.append(b'alias pwd="pwd -W"\n')
1766 script.append(b'alias pwd="pwd -W"\n')
1767
1767
1768 if hgcatapult and hgcatapult != os.devnull:
1768 if hgcatapult and hgcatapult != os.devnull:
1769 if PYTHON3:
1769 if PYTHON3:
1770 hgcatapult = hgcatapult.encode('utf8')
1770 hgcatapult = hgcatapult.encode('utf8')
1771 cataname = self.name.encode('utf8')
1771 cataname = self.name.encode('utf8')
1772 else:
1772 else:
1773 cataname = self.name
1773 cataname = self.name
1774
1774
1775 # Kludge: use a while loop to keep the pipe from getting
1775 # Kludge: use a while loop to keep the pipe from getting
1776 # closed by our echo commands. The still-running file gets
1776 # closed by our echo commands. The still-running file gets
1777 # reaped at the end of the script, which causes the while
1777 # reaped at the end of the script, which causes the while
1778 # loop to exit and closes the pipe. Sigh.
1778 # loop to exit and closes the pipe. Sigh.
1779 script.append(
1779 script.append(
1780 b'rtendtracing() {\n'
1780 b'rtendtracing() {\n'
1781 b' echo END %(session)s %(name)s >> %(catapult)s\n'
1781 b' echo END %(session)s %(name)s >> %(catapult)s\n'
1782 b' rm -f "$TESTTMP/.still-running"\n'
1782 b' rm -f "$TESTTMP/.still-running"\n'
1783 b'}\n'
1783 b'}\n'
1784 b'trap "rtendtracing" 0\n'
1784 b'trap "rtendtracing" 0\n'
1785 b'touch "$TESTTMP/.still-running"\n'
1785 b'touch "$TESTTMP/.still-running"\n'
1786 b'while [ -f "$TESTTMP/.still-running" ]; do sleep 1; done '
1786 b'while [ -f "$TESTTMP/.still-running" ]; do sleep 1; done '
1787 b'> %(catapult)s &\n'
1787 b'> %(catapult)s &\n'
1788 b'HGCATAPULTSESSION=%(session)s ; export HGCATAPULTSESSION\n'
1788 b'HGCATAPULTSESSION=%(session)s ; export HGCATAPULTSESSION\n'
1789 b'echo START %(session)s %(name)s >> %(catapult)s\n'
1789 b'echo START %(session)s %(name)s >> %(catapult)s\n'
1790 % {
1790 % {
1791 b'name': cataname,
1791 b'name': cataname,
1792 b'session': session,
1792 b'session': session,
1793 b'catapult': hgcatapult,
1793 b'catapult': hgcatapult,
1794 }
1794 }
1795 )
1795 )
1796
1796
1797 if self._case:
1797 if self._case:
1798 casestr = b'#'.join(self._case)
1798 casestr = b'#'.join(self._case)
1799 if isinstance(casestr, str):
1799 if isinstance(casestr, str):
1800 quoted = shellquote(casestr)
1800 quoted = shellquote(casestr)
1801 else:
1801 else:
1802 quoted = shellquote(casestr.decode('utf8')).encode('utf8')
1802 quoted = shellquote(casestr.decode('utf8')).encode('utf8')
1803 script.append(b'TESTCASE=%s\n' % quoted)
1803 script.append(b'TESTCASE=%s\n' % quoted)
1804 script.append(b'export TESTCASE\n')
1804 script.append(b'export TESTCASE\n')
1805
1805
1806 n = 0
1806 n = 0
1807 for n, l in enumerate(lines):
1807 for n, l in enumerate(lines):
1808 if not l.endswith(b'\n'):
1808 if not l.endswith(b'\n'):
1809 l += b'\n'
1809 l += b'\n'
1810 if l.startswith(b'#require'):
1810 if l.startswith(b'#require'):
1811 lsplit = l.split()
1811 lsplit = l.split()
1812 if len(lsplit) < 2 or lsplit[0] != b'#require':
1812 if len(lsplit) < 2 or lsplit[0] != b'#require':
1813 after.setdefault(pos, []).append(
1813 after.setdefault(pos, []).append(
1814 b' !!! invalid #require\n'
1814 b' !!! invalid #require\n'
1815 )
1815 )
1816 if not skipping:
1816 if not skipping:
1817 haveresult, message = self._hghave(lsplit[1:])
1817 haveresult, message = self._hghave(lsplit[1:])
1818 if not haveresult:
1818 if not haveresult:
1819 script = [b'echo "%s"\nexit 80\n' % message]
1819 script = [b'echo "%s"\nexit 80\n' % message]
1820 break
1820 break
1821 after.setdefault(pos, []).append(l)
1821 after.setdefault(pos, []).append(l)
1822 elif l.startswith(b'#if'):
1822 elif l.startswith(b'#if'):
1823 lsplit = l.split()
1823 lsplit = l.split()
1824 if len(lsplit) < 2 or lsplit[0] != b'#if':
1824 if len(lsplit) < 2 or lsplit[0] != b'#if':
1825 after.setdefault(pos, []).append(b' !!! invalid #if\n')
1825 after.setdefault(pos, []).append(b' !!! invalid #if\n')
1826 if skipping is not None:
1826 if skipping is not None:
1827 after.setdefault(pos, []).append(b' !!! nested #if\n')
1827 after.setdefault(pos, []).append(b' !!! nested #if\n')
1828 skipping = not self._iftest(lsplit[1:])
1828 skipping = not self._iftest(lsplit[1:])
1829 after.setdefault(pos, []).append(l)
1829 after.setdefault(pos, []).append(l)
1830 elif l.startswith(b'#else'):
1830 elif l.startswith(b'#else'):
1831 if skipping is None:
1831 if skipping is None:
1832 after.setdefault(pos, []).append(b' !!! missing #if\n')
1832 after.setdefault(pos, []).append(b' !!! missing #if\n')
1833 skipping = not skipping
1833 skipping = not skipping
1834 after.setdefault(pos, []).append(l)
1834 after.setdefault(pos, []).append(l)
1835 elif l.startswith(b'#endif'):
1835 elif l.startswith(b'#endif'):
1836 if skipping is None:
1836 if skipping is None:
1837 after.setdefault(pos, []).append(b' !!! missing #if\n')
1837 after.setdefault(pos, []).append(b' !!! missing #if\n')
1838 skipping = None
1838 skipping = None
1839 after.setdefault(pos, []).append(l)
1839 after.setdefault(pos, []).append(l)
1840 elif skipping:
1840 elif skipping:
1841 after.setdefault(pos, []).append(l)
1841 after.setdefault(pos, []).append(l)
1842 elif l.startswith(b' >>> '): # python inlines
1842 elif l.startswith(b' >>> '): # python inlines
1843 after.setdefault(pos, []).append(l)
1843 after.setdefault(pos, []).append(l)
1844 prepos = pos
1844 prepos = pos
1845 pos = n
1845 pos = n
1846 if not inpython:
1846 if not inpython:
1847 # We've just entered a Python block. Add the header.
1847 # We've just entered a Python block. Add the header.
1848 inpython = True
1848 inpython = True
1849 addsalt(prepos, False) # Make sure we report the exit code.
1849 addsalt(prepos, False) # Make sure we report the exit code.
1850 script.append(b'"%s" -m heredoctest <<EOF\n' % PYTHON)
1850 script.append(b'"%s" -m heredoctest <<EOF\n' % PYTHON)
1851 addsalt(n, True)
1851 addsalt(n, True)
1852 script.append(l[2:])
1852 script.append(l[2:])
1853 elif l.startswith(b' ... '): # python inlines
1853 elif l.startswith(b' ... '): # python inlines
1854 after.setdefault(prepos, []).append(l)
1854 after.setdefault(prepos, []).append(l)
1855 script.append(l[2:])
1855 script.append(l[2:])
1856 elif l.startswith(b' $ '): # commands
1856 elif l.startswith(b' $ '): # commands
1857 if inpython:
1857 if inpython:
1858 script.append(b'EOF\n')
1858 script.append(b'EOF\n')
1859 inpython = False
1859 inpython = False
1860 after.setdefault(pos, []).append(l)
1860 after.setdefault(pos, []).append(l)
1861 prepos = pos
1861 prepos = pos
1862 pos = n
1862 pos = n
1863 addsalt(n, False)
1863 addsalt(n, False)
1864 rawcmd = l[4:]
1864 rawcmd = l[4:]
1865 cmd = rawcmd.split()
1865 cmd = rawcmd.split()
1866 toggletrace(rawcmd)
1866 toggletrace(rawcmd)
1867 if len(cmd) == 2 and cmd[0] == b'cd':
1867 if len(cmd) == 2 and cmd[0] == b'cd':
1868 rawcmd = b'cd %s || exit 1\n' % cmd[1]
1868 rawcmd = b'cd %s || exit 1\n' % cmd[1]
1869 script.append(rawcmd)
1869 script.append(rawcmd)
1870 elif l.startswith(b' > '): # continuations
1870 elif l.startswith(b' > '): # continuations
1871 after.setdefault(prepos, []).append(l)
1871 after.setdefault(prepos, []).append(l)
1872 script.append(l[4:])
1872 script.append(l[4:])
1873 elif l.startswith(b' '): # results
1873 elif l.startswith(b' '): # results
1874 # Queue up a list of expected results.
1874 # Queue up a list of expected results.
1875 expected.setdefault(pos, []).append(l[2:])
1875 expected.setdefault(pos, []).append(l[2:])
1876 else:
1876 else:
1877 if inpython:
1877 if inpython:
1878 script.append(b'EOF\n')
1878 script.append(b'EOF\n')
1879 inpython = False
1879 inpython = False
1880 # Non-command/result. Queue up for merged output.
1880 # Non-command/result. Queue up for merged output.
1881 after.setdefault(pos, []).append(l)
1881 after.setdefault(pos, []).append(l)
1882
1882
1883 if inpython:
1883 if inpython:
1884 script.append(b'EOF\n')
1884 script.append(b'EOF\n')
1885 if skipping is not None:
1885 if skipping is not None:
1886 after.setdefault(pos, []).append(b' !!! missing #endif\n')
1886 after.setdefault(pos, []).append(b' !!! missing #endif\n')
1887 addsalt(n + 1, False)
1887 addsalt(n + 1, False)
1888 # Need to end any current per-command trace
1888 # Need to end any current per-command trace
1889 if activetrace:
1889 if activetrace:
1890 toggletrace()
1890 toggletrace()
1891 return salt, script, after, expected
1891 return salt, script, after, expected
1892
1892
1893 def _processoutput(self, exitcode, output, salt, after, expected):
1893 def _processoutput(self, exitcode, output, salt, after, expected):
1894 # Merge the script output back into a unified test.
1894 # Merge the script output back into a unified test.
1895 warnonly = WARN_UNDEFINED # 1: not yet; 2: yes; 3: for sure not
1895 warnonly = WARN_UNDEFINED # 1: not yet; 2: yes; 3: for sure not
1896 if exitcode != 0:
1896 if exitcode != 0:
1897 warnonly = WARN_NO
1897 warnonly = WARN_NO
1898
1898
1899 pos = -1
1899 pos = -1
1900 postout = []
1900 postout = []
1901 for out_rawline in output:
1901 for out_rawline in output:
1902 out_line, cmd_line = out_rawline, None
1902 out_line, cmd_line = out_rawline, None
1903 if salt in out_rawline:
1903 if salt in out_rawline:
1904 out_line, cmd_line = out_rawline.split(salt, 1)
1904 out_line, cmd_line = out_rawline.split(salt, 1)
1905
1905
1906 pos, postout, warnonly = self._process_out_line(
1906 pos, postout, warnonly = self._process_out_line(
1907 out_line, pos, postout, expected, warnonly
1907 out_line, pos, postout, expected, warnonly
1908 )
1908 )
1909 pos, postout = self._process_cmd_line(cmd_line, pos, postout, after)
1909 pos, postout = self._process_cmd_line(cmd_line, pos, postout, after)
1910
1910
1911 if pos in after:
1911 if pos in after:
1912 postout += after.pop(pos)
1912 postout += after.pop(pos)
1913
1913
1914 if warnonly == WARN_YES:
1914 if warnonly == WARN_YES:
1915 exitcode = False # Set exitcode to warned.
1915 exitcode = False # Set exitcode to warned.
1916
1916
1917 return exitcode, postout
1917 return exitcode, postout
1918
1918
1919 def _process_out_line(self, out_line, pos, postout, expected, warnonly):
1919 def _process_out_line(self, out_line, pos, postout, expected, warnonly):
1920 while out_line:
1920 while out_line:
1921 if not out_line.endswith(b'\n'):
1921 if not out_line.endswith(b'\n'):
1922 out_line += b' (no-eol)\n'
1922 out_line += b' (no-eol)\n'
1923
1923
1924 # Find the expected output at the current position.
1924 # Find the expected output at the current position.
1925 els = [None]
1925 els = [None]
1926 if expected.get(pos, None):
1926 if expected.get(pos, None):
1927 els = expected[pos]
1927 els = expected[pos]
1928
1928
1929 optional = []
1929 optional = []
1930 for i, el in enumerate(els):
1930 for i, el in enumerate(els):
1931 r = False
1931 r = False
1932 if el:
1932 if el:
1933 r, exact = self.linematch(el, out_line)
1933 r, exact = self.linematch(el, out_line)
1934 if isinstance(r, str):
1934 if isinstance(r, str):
1935 if r == '-glob':
1935 if r == '-glob':
1936 out_line = ''.join(el.rsplit(' (glob)', 1))
1936 out_line = ''.join(el.rsplit(' (glob)', 1))
1937 r = '' # Warn only this line.
1937 r = '' # Warn only this line.
1938 elif r == "retry":
1938 elif r == "retry":
1939 postout.append(b' ' + el)
1939 postout.append(b' ' + el)
1940 else:
1940 else:
1941 log('\ninfo, unknown linematch result: %r\n' % r)
1941 log('\ninfo, unknown linematch result: %r\n' % r)
1942 r = False
1942 r = False
1943 if r:
1943 if r:
1944 els.pop(i)
1944 els.pop(i)
1945 break
1945 break
1946 if el:
1946 if el:
1947 if isoptional(el):
1947 if isoptional(el):
1948 optional.append(i)
1948 optional.append(i)
1949 else:
1949 else:
1950 m = optline.match(el)
1950 m = optline.match(el)
1951 if m:
1951 if m:
1952 conditions = [c for c in m.group(2).split(b' ')]
1952 conditions = [c for c in m.group(2).split(b' ')]
1953
1953
1954 if not self._iftest(conditions):
1954 if not self._iftest(conditions):
1955 optional.append(i)
1955 optional.append(i)
1956 if exact:
1956 if exact:
1957 # Don't allow line to be matches against a later
1957 # Don't allow line to be matches against a later
1958 # line in the output
1958 # line in the output
1959 els.pop(i)
1959 els.pop(i)
1960 break
1960 break
1961
1961
1962 if r:
1962 if r:
1963 if r == "retry":
1963 if r == "retry":
1964 continue
1964 continue
1965 # clean up any optional leftovers
1965 # clean up any optional leftovers
1966 for i in optional:
1966 for i in optional:
1967 postout.append(b' ' + els[i])
1967 postout.append(b' ' + els[i])
1968 for i in reversed(optional):
1968 for i in reversed(optional):
1969 del els[i]
1969 del els[i]
1970 postout.append(b' ' + el)
1970 postout.append(b' ' + el)
1971 else:
1971 else:
1972 if self.NEEDESCAPE(out_line):
1972 if self.NEEDESCAPE(out_line):
1973 out_line = TTest._stringescape(
1973 out_line = TTest._stringescape(
1974 b'%s (esc)\n' % out_line.rstrip(b'\n')
1974 b'%s (esc)\n' % out_line.rstrip(b'\n')
1975 )
1975 )
1976 postout.append(b' ' + out_line) # Let diff deal with it.
1976 postout.append(b' ' + out_line) # Let diff deal with it.
1977 if r != '': # If line failed.
1977 if r != '': # If line failed.
1978 warnonly = WARN_NO
1978 warnonly = WARN_NO
1979 elif warnonly == WARN_UNDEFINED:
1979 elif warnonly == WARN_UNDEFINED:
1980 warnonly = WARN_YES
1980 warnonly = WARN_YES
1981 break
1981 break
1982 else:
1982 else:
1983 # clean up any optional leftovers
1983 # clean up any optional leftovers
1984 while expected.get(pos, None):
1984 while expected.get(pos, None):
1985 el = expected[pos].pop(0)
1985 el = expected[pos].pop(0)
1986 if el:
1986 if el:
1987 if not isoptional(el):
1987 if not isoptional(el):
1988 m = optline.match(el)
1988 m = optline.match(el)
1989 if m:
1989 if m:
1990 conditions = [c for c in m.group(2).split(b' ')]
1990 conditions = [c for c in m.group(2).split(b' ')]
1991
1991
1992 if self._iftest(conditions):
1992 if self._iftest(conditions):
1993 # Don't append as optional line
1993 # Don't append as optional line
1994 continue
1994 continue
1995 else:
1995 else:
1996 continue
1996 continue
1997 postout.append(b' ' + el)
1997 postout.append(b' ' + el)
1998 return pos, postout, warnonly
1998 return pos, postout, warnonly
1999
1999
2000 def _process_cmd_line(self, cmd_line, pos, postout, after):
2000 def _process_cmd_line(self, cmd_line, pos, postout, after):
2001 """process a "command" part of a line from unified test output"""
2001 """process a "command" part of a line from unified test output"""
2002 if cmd_line:
2002 if cmd_line:
2003 # Add on last return code.
2003 # Add on last return code.
2004 ret = int(cmd_line.split()[1])
2004 ret = int(cmd_line.split()[1])
2005 if ret != 0:
2005 if ret != 0:
2006 postout.append(b' [%d]\n' % ret)
2006 postout.append(b' [%d]\n' % ret)
2007 if pos in after:
2007 if pos in after:
2008 # Merge in non-active test bits.
2008 # Merge in non-active test bits.
2009 postout += after.pop(pos)
2009 postout += after.pop(pos)
2010 pos = int(cmd_line.split()[0])
2010 pos = int(cmd_line.split()[0])
2011 return pos, postout
2011 return pos, postout
2012
2012
2013 @staticmethod
2013 @staticmethod
2014 def rematch(el, l):
2014 def rematch(el, l):
2015 try:
2015 try:
2016 # parse any flags at the beginning of the regex. Only 'i' is
2016 # parse any flags at the beginning of the regex. Only 'i' is
2017 # supported right now, but this should be easy to extend.
2017 # supported right now, but this should be easy to extend.
2018 flags, el = re.match(br'^(\(\?i\))?(.*)', el).groups()[0:2]
2018 flags, el = re.match(br'^(\(\?i\))?(.*)', el).groups()[0:2]
2019 flags = flags or b''
2019 flags = flags or b''
2020 el = flags + b'(?:' + el + b')'
2020 el = flags + b'(?:' + el + b')'
2021 # use \Z to ensure that the regex matches to the end of the string
2021 # use \Z to ensure that the regex matches to the end of the string
2022 if os.name == 'nt':
2022 if os.name == 'nt':
2023 return re.match(el + br'\r?\n\Z', l)
2023 return re.match(el + br'\r?\n\Z', l)
2024 return re.match(el + br'\n\Z', l)
2024 return re.match(el + br'\n\Z', l)
2025 except re.error:
2025 except re.error:
2026 # el is an invalid regex
2026 # el is an invalid regex
2027 return False
2027 return False
2028
2028
2029 @staticmethod
2029 @staticmethod
2030 def globmatch(el, l):
2030 def globmatch(el, l):
2031 # The only supported special characters are * and ? plus / which also
2031 # The only supported special characters are * and ? plus / which also
2032 # matches \ on windows. Escaping of these characters is supported.
2032 # matches \ on windows. Escaping of these characters is supported.
2033 if el + b'\n' == l:
2033 if el + b'\n' == l:
2034 if os.altsep:
2034 if os.altsep:
2035 # matching on "/" is not needed for this line
2035 # matching on "/" is not needed for this line
2036 for pat in checkcodeglobpats:
2036 for pat in checkcodeglobpats:
2037 if pat.match(el):
2037 if pat.match(el):
2038 return True
2038 return True
2039 return b'-glob'
2039 return b'-glob'
2040 return True
2040 return True
2041 el = el.replace(b'$LOCALIP', b'*')
2041 el = el.replace(b'$LOCALIP', b'*')
2042 i, n = 0, len(el)
2042 i, n = 0, len(el)
2043 res = b''
2043 res = b''
2044 while i < n:
2044 while i < n:
2045 c = el[i : i + 1]
2045 c = el[i : i + 1]
2046 i += 1
2046 i += 1
2047 if c == b'\\' and i < n and el[i : i + 1] in b'*?\\/':
2047 if c == b'\\' and i < n and el[i : i + 1] in b'*?\\/':
2048 res += el[i - 1 : i + 1]
2048 res += el[i - 1 : i + 1]
2049 i += 1
2049 i += 1
2050 elif c == b'*':
2050 elif c == b'*':
2051 res += b'.*'
2051 res += b'.*'
2052 elif c == b'?':
2052 elif c == b'?':
2053 res += b'.'
2053 res += b'.'
2054 elif c == b'/' and os.altsep:
2054 elif c == b'/' and os.altsep:
2055 res += b'[/\\\\]'
2055 res += b'[/\\\\]'
2056 else:
2056 else:
2057 res += re.escape(c)
2057 res += re.escape(c)
2058 return TTest.rematch(res, l)
2058 return TTest.rematch(res, l)
2059
2059
2060 def linematch(self, el, l):
2060 def linematch(self, el, l):
2061 if el == l: # perfect match (fast)
2061 if el == l: # perfect match (fast)
2062 return True, True
2062 return True, True
2063 retry = False
2063 retry = False
2064 if isoptional(el):
2064 if isoptional(el):
2065 retry = "retry"
2065 retry = "retry"
2066 el = el[: -len(MARK_OPTIONAL)] + b"\n"
2066 el = el[: -len(MARK_OPTIONAL)] + b"\n"
2067 else:
2067 else:
2068 m = optline.match(el)
2068 m = optline.match(el)
2069 if m:
2069 if m:
2070 conditions = [c for c in m.group(2).split(b' ')]
2070 conditions = [c for c in m.group(2).split(b' ')]
2071
2071
2072 el = m.group(1) + b"\n"
2072 el = m.group(1) + b"\n"
2073 if not self._iftest(conditions):
2073 if not self._iftest(conditions):
2074 # listed feature missing, should not match
2074 # listed feature missing, should not match
2075 return "retry", False
2075 return "retry", False
2076
2076
2077 if el.endswith(b" (esc)\n"):
2077 if el.endswith(b" (esc)\n"):
2078 if PYTHON3:
2078 if PYTHON3:
2079 el = el[:-7].decode('unicode_escape') + '\n'
2079 el = el[:-7].decode('unicode_escape') + '\n'
2080 el = el.encode('latin-1')
2080 el = el.encode('latin-1')
2081 else:
2081 else:
2082 el = el[:-7].decode('string-escape') + '\n'
2082 el = el[:-7].decode('string-escape') + '\n'
2083 if el == l or os.name == 'nt' and el[:-1] + b'\r\n' == l:
2083 if el == l or os.name == 'nt' and el[:-1] + b'\r\n' == l:
2084 return True, True
2084 return True, True
2085 if el.endswith(b" (re)\n"):
2085 if el.endswith(b" (re)\n"):
2086 return (TTest.rematch(el[:-6], l) or retry), False
2086 return (TTest.rematch(el[:-6], l) or retry), False
2087 if el.endswith(b" (glob)\n"):
2087 if el.endswith(b" (glob)\n"):
2088 # ignore '(glob)' added to l by 'replacements'
2088 # ignore '(glob)' added to l by 'replacements'
2089 if l.endswith(b" (glob)\n"):
2089 if l.endswith(b" (glob)\n"):
2090 l = l[:-8] + b"\n"
2090 l = l[:-8] + b"\n"
2091 return (TTest.globmatch(el[:-8], l) or retry), False
2091 return (TTest.globmatch(el[:-8], l) or retry), False
2092 if os.altsep:
2092 if os.altsep:
2093 _l = l.replace(b'\\', b'/')
2093 _l = l.replace(b'\\', b'/')
2094 if el == _l or os.name == 'nt' and el[:-1] + b'\r\n' == _l:
2094 if el == _l or os.name == 'nt' and el[:-1] + b'\r\n' == _l:
2095 return True, True
2095 return True, True
2096 return retry, True
2096 return retry, True
2097
2097
2098 @staticmethod
2098 @staticmethod
2099 def parsehghaveoutput(lines):
2099 def parsehghaveoutput(lines):
2100 """Parse hghave log lines.
2100 """Parse hghave log lines.
2101
2101
2102 Return tuple of lists (missing, failed):
2102 Return tuple of lists (missing, failed):
2103 * the missing/unknown features
2103 * the missing/unknown features
2104 * the features for which existence check failed"""
2104 * the features for which existence check failed"""
2105 missing = []
2105 missing = []
2106 failed = []
2106 failed = []
2107 for line in lines:
2107 for line in lines:
2108 if line.startswith(TTest.SKIPPED_PREFIX):
2108 if line.startswith(TTest.SKIPPED_PREFIX):
2109 line = line.splitlines()[0]
2109 line = line.splitlines()[0]
2110 missing.append(_bytes2sys(line[len(TTest.SKIPPED_PREFIX) :]))
2110 missing.append(_bytes2sys(line[len(TTest.SKIPPED_PREFIX) :]))
2111 elif line.startswith(TTest.FAILED_PREFIX):
2111 elif line.startswith(TTest.FAILED_PREFIX):
2112 line = line.splitlines()[0]
2112 line = line.splitlines()[0]
2113 failed.append(_bytes2sys(line[len(TTest.FAILED_PREFIX) :]))
2113 failed.append(_bytes2sys(line[len(TTest.FAILED_PREFIX) :]))
2114
2114
2115 return missing, failed
2115 return missing, failed
2116
2116
2117 @staticmethod
2117 @staticmethod
2118 def _escapef(m):
2118 def _escapef(m):
2119 return TTest.ESCAPEMAP[m.group(0)]
2119 return TTest.ESCAPEMAP[m.group(0)]
2120
2120
2121 @staticmethod
2121 @staticmethod
2122 def _stringescape(s):
2122 def _stringescape(s):
2123 return TTest.ESCAPESUB(TTest._escapef, s)
2123 return TTest.ESCAPESUB(TTest._escapef, s)
2124
2124
2125
2125
2126 iolock = threading.RLock()
2126 iolock = threading.RLock()
2127 firstlock = threading.RLock()
2127 firstlock = threading.RLock()
2128 firsterror = False
2128 firsterror = False
2129
2129
2130
2130
2131 class TestResult(unittest._TextTestResult):
2131 class TestResult(unittest._TextTestResult):
2132 """Holds results when executing via unittest."""
2132 """Holds results when executing via unittest."""
2133
2133
2134 # Don't worry too much about accessing the non-public _TextTestResult.
2134 # Don't worry too much about accessing the non-public _TextTestResult.
2135 # It is relatively common in Python testing tools.
2135 # It is relatively common in Python testing tools.
2136 def __init__(self, options, *args, **kwargs):
2136 def __init__(self, options, *args, **kwargs):
2137 super(TestResult, self).__init__(*args, **kwargs)
2137 super(TestResult, self).__init__(*args, **kwargs)
2138
2138
2139 self._options = options
2139 self._options = options
2140
2140
2141 # unittest.TestResult didn't have skipped until 2.7. We need to
2141 # unittest.TestResult didn't have skipped until 2.7. We need to
2142 # polyfill it.
2142 # polyfill it.
2143 self.skipped = []
2143 self.skipped = []
2144
2144
2145 # We have a custom "ignored" result that isn't present in any Python
2145 # We have a custom "ignored" result that isn't present in any Python
2146 # unittest implementation. It is very similar to skipped. It may make
2146 # unittest implementation. It is very similar to skipped. It may make
2147 # sense to map it into skip some day.
2147 # sense to map it into skip some day.
2148 self.ignored = []
2148 self.ignored = []
2149
2149
2150 self.times = []
2150 self.times = []
2151 self._firststarttime = None
2151 self._firststarttime = None
2152 # Data stored for the benefit of generating xunit reports.
2152 # Data stored for the benefit of generating xunit reports.
2153 self.successes = []
2153 self.successes = []
2154 self.faildata = {}
2154 self.faildata = {}
2155
2155
2156 if options.color == 'auto':
2156 if options.color == 'auto':
2157 self.color = pygmentspresent and self.stream.isatty()
2157 self.color = pygmentspresent and self.stream.isatty()
2158 elif options.color == 'never':
2158 elif options.color == 'never':
2159 self.color = False
2159 self.color = False
2160 else: # 'always', for testing purposes
2160 else: # 'always', for testing purposes
2161 self.color = pygmentspresent
2161 self.color = pygmentspresent
2162
2162
2163 def onStart(self, test):
2163 def onStart(self, test):
2164 """Can be overriden by custom TestResult"""
2164 """Can be overriden by custom TestResult"""
2165
2165
2166 def onEnd(self):
2166 def onEnd(self):
2167 """Can be overriden by custom TestResult"""
2167 """Can be overriden by custom TestResult"""
2168
2168
2169 def addFailure(self, test, reason):
2169 def addFailure(self, test, reason):
2170 self.failures.append((test, reason))
2170 self.failures.append((test, reason))
2171
2171
2172 if self._options.first:
2172 if self._options.first:
2173 self.stop()
2173 self.stop()
2174 else:
2174 else:
2175 with iolock:
2175 with iolock:
2176 if reason == "timed out":
2176 if reason == "timed out":
2177 self.stream.write('t')
2177 self.stream.write('t')
2178 else:
2178 else:
2179 if not self._options.nodiff:
2179 if not self._options.nodiff:
2180 self.stream.write('\n')
2180 self.stream.write('\n')
2181 # Exclude the '\n' from highlighting to lex correctly
2181 # Exclude the '\n' from highlighting to lex correctly
2182 formatted = 'ERROR: %s output changed\n' % test
2182 formatted = 'ERROR: %s output changed\n' % test
2183 self.stream.write(highlightmsg(formatted, self.color))
2183 self.stream.write(highlightmsg(formatted, self.color))
2184 self.stream.write('!')
2184 self.stream.write('!')
2185
2185
2186 self.stream.flush()
2186 self.stream.flush()
2187
2187
2188 def addSuccess(self, test):
2188 def addSuccess(self, test):
2189 with iolock:
2189 with iolock:
2190 super(TestResult, self).addSuccess(test)
2190 super(TestResult, self).addSuccess(test)
2191 self.successes.append(test)
2191 self.successes.append(test)
2192
2192
2193 def addError(self, test, err):
2193 def addError(self, test, err):
2194 super(TestResult, self).addError(test, err)
2194 super(TestResult, self).addError(test, err)
2195 if self._options.first:
2195 if self._options.first:
2196 self.stop()
2196 self.stop()
2197
2197
2198 # Polyfill.
2198 # Polyfill.
2199 def addSkip(self, test, reason):
2199 def addSkip(self, test, reason):
2200 self.skipped.append((test, reason))
2200 self.skipped.append((test, reason))
2201 with iolock:
2201 with iolock:
2202 if self.showAll:
2202 if self.showAll:
2203 self.stream.writeln('skipped %s' % reason)
2203 self.stream.writeln('skipped %s' % reason)
2204 else:
2204 else:
2205 self.stream.write('s')
2205 self.stream.write('s')
2206 self.stream.flush()
2206 self.stream.flush()
2207
2207
2208 def addIgnore(self, test, reason):
2208 def addIgnore(self, test, reason):
2209 self.ignored.append((test, reason))
2209 self.ignored.append((test, reason))
2210 with iolock:
2210 with iolock:
2211 if self.showAll:
2211 if self.showAll:
2212 self.stream.writeln('ignored %s' % reason)
2212 self.stream.writeln('ignored %s' % reason)
2213 else:
2213 else:
2214 if reason not in ('not retesting', "doesn't match keyword"):
2214 if reason not in ('not retesting', "doesn't match keyword"):
2215 self.stream.write('i')
2215 self.stream.write('i')
2216 else:
2216 else:
2217 self.testsRun += 1
2217 self.testsRun += 1
2218 self.stream.flush()
2218 self.stream.flush()
2219
2219
2220 def addOutputMismatch(self, test, ret, got, expected):
2220 def addOutputMismatch(self, test, ret, got, expected):
2221 """Record a mismatch in test output for a particular test."""
2221 """Record a mismatch in test output for a particular test."""
2222 if self.shouldStop or firsterror:
2222 if self.shouldStop or firsterror:
2223 # don't print, some other test case already failed and
2223 # don't print, some other test case already failed and
2224 # printed, we're just stale and probably failed due to our
2224 # printed, we're just stale and probably failed due to our
2225 # temp dir getting cleaned up.
2225 # temp dir getting cleaned up.
2226 return
2226 return
2227
2227
2228 accepted = False
2228 accepted = False
2229 lines = []
2229 lines = []
2230
2230
2231 with iolock:
2231 with iolock:
2232 if self._options.nodiff:
2232 if self._options.nodiff:
2233 pass
2233 pass
2234 elif self._options.view:
2234 elif self._options.view:
2235 v = self._options.view
2235 v = self._options.view
2236 subprocess.call(
2236 subprocess.call(
2237 r'"%s" "%s" "%s"'
2237 r'"%s" "%s" "%s"'
2238 % (v, _bytes2sys(test.refpath), _bytes2sys(test.errpath)),
2238 % (v, _bytes2sys(test.refpath), _bytes2sys(test.errpath)),
2239 shell=True,
2239 shell=True,
2240 )
2240 )
2241 else:
2241 else:
2242 servefail, lines = getdiff(
2242 servefail, lines = getdiff(
2243 expected, got, test.refpath, test.errpath
2243 expected, got, test.refpath, test.errpath
2244 )
2244 )
2245 self.stream.write('\n')
2245 self.stream.write('\n')
2246 for line in lines:
2246 for line in lines:
2247 line = highlightdiff(line, self.color)
2247 line = highlightdiff(line, self.color)
2248 if PYTHON3:
2248 if PYTHON3:
2249 self.stream.flush()
2249 self.stream.flush()
2250 self.stream.buffer.write(line)
2250 self.stream.buffer.write(line)
2251 self.stream.buffer.flush()
2251 self.stream.buffer.flush()
2252 else:
2252 else:
2253 self.stream.write(line)
2253 self.stream.write(line)
2254 self.stream.flush()
2254 self.stream.flush()
2255
2255
2256 if servefail:
2256 if servefail:
2257 raise test.failureException(
2257 raise test.failureException(
2258 'server failed to start (HGPORT=%s)' % test._startport
2258 'server failed to start (HGPORT=%s)' % test._startport
2259 )
2259 )
2260
2260
2261 # handle interactive prompt without releasing iolock
2261 # handle interactive prompt without releasing iolock
2262 if self._options.interactive:
2262 if self._options.interactive:
2263 if test.readrefout() != expected:
2263 if test.readrefout() != expected:
2264 self.stream.write(
2264 self.stream.write(
2265 'Reference output has changed (run again to prompt '
2265 'Reference output has changed (run again to prompt '
2266 'changes)'
2266 'changes)'
2267 )
2267 )
2268 else:
2268 else:
2269 self.stream.write('Accept this change? [y/N] ')
2269 self.stream.write('Accept this change? [y/N] ')
2270 self.stream.flush()
2270 self.stream.flush()
2271 answer = sys.stdin.readline().strip()
2271 answer = sys.stdin.readline().strip()
2272 if answer.lower() in ('y', 'yes'):
2272 if answer.lower() in ('y', 'yes'):
2273 if test.path.endswith(b'.t'):
2273 if test.path.endswith(b'.t'):
2274 rename(test.errpath, test.path)
2274 rename(test.errpath, test.path)
2275 else:
2275 else:
2276 rename(test.errpath, '%s.out' % test.path)
2276 rename(test.errpath, '%s.out' % test.path)
2277 accepted = True
2277 accepted = True
2278 if not accepted:
2278 if not accepted:
2279 self.faildata[test.name] = b''.join(lines)
2279 self.faildata[test.name] = b''.join(lines)
2280
2280
2281 return accepted
2281 return accepted
2282
2282
2283 def startTest(self, test):
2283 def startTest(self, test):
2284 super(TestResult, self).startTest(test)
2284 super(TestResult, self).startTest(test)
2285
2285
2286 # os.times module computes the user time and system time spent by
2286 # os.times module computes the user time and system time spent by
2287 # child's processes along with real elapsed time taken by a process.
2287 # child's processes along with real elapsed time taken by a process.
2288 # This module has one limitation. It can only work for Linux user
2288 # This module has one limitation. It can only work for Linux user
2289 # and not for Windows. Hence why we fall back to another function
2289 # and not for Windows. Hence why we fall back to another function
2290 # for wall time calculations.
2290 # for wall time calculations.
2291 test.started_times = os.times()
2291 test.started_times = os.times()
2292 # TODO use a monotonic clock once support for Python 2.7 is dropped.
2292 # TODO use a monotonic clock once support for Python 2.7 is dropped.
2293 test.started_time = time.time()
2293 test.started_time = time.time()
2294 if self._firststarttime is None: # thread racy but irrelevant
2294 if self._firststarttime is None: # thread racy but irrelevant
2295 self._firststarttime = test.started_time
2295 self._firststarttime = test.started_time
2296
2296
2297 def stopTest(self, test, interrupted=False):
2297 def stopTest(self, test, interrupted=False):
2298 super(TestResult, self).stopTest(test)
2298 super(TestResult, self).stopTest(test)
2299
2299
2300 test.stopped_times = os.times()
2300 test.stopped_times = os.times()
2301 stopped_time = time.time()
2301 stopped_time = time.time()
2302
2302
2303 starttime = test.started_times
2303 starttime = test.started_times
2304 endtime = test.stopped_times
2304 endtime = test.stopped_times
2305 origin = self._firststarttime
2305 origin = self._firststarttime
2306 self.times.append(
2306 self.times.append(
2307 (
2307 (
2308 test.name,
2308 test.name,
2309 endtime[2] - starttime[2], # user space CPU time
2309 endtime[2] - starttime[2], # user space CPU time
2310 endtime[3] - starttime[3], # sys space CPU time
2310 endtime[3] - starttime[3], # sys space CPU time
2311 stopped_time - test.started_time, # real time
2311 stopped_time - test.started_time, # real time
2312 test.started_time - origin, # start date in run context
2312 test.started_time - origin, # start date in run context
2313 stopped_time - origin, # end date in run context
2313 stopped_time - origin, # end date in run context
2314 )
2314 )
2315 )
2315 )
2316
2316
2317 if interrupted:
2317 if interrupted:
2318 with iolock:
2318 with iolock:
2319 self.stream.writeln(
2319 self.stream.writeln(
2320 'INTERRUPTED: %s (after %d seconds)'
2320 'INTERRUPTED: %s (after %d seconds)'
2321 % (test.name, self.times[-1][3])
2321 % (test.name, self.times[-1][3])
2322 )
2322 )
2323
2323
2324
2324
2325 def getTestResult():
2325 def getTestResult():
2326 """
2326 """
2327 Returns the relevant test result
2327 Returns the relevant test result
2328 """
2328 """
2329 if "CUSTOM_TEST_RESULT" in os.environ:
2329 if "CUSTOM_TEST_RESULT" in os.environ:
2330 testresultmodule = __import__(os.environ["CUSTOM_TEST_RESULT"])
2330 testresultmodule = __import__(os.environ["CUSTOM_TEST_RESULT"])
2331 return testresultmodule.TestResult
2331 return testresultmodule.TestResult
2332 else:
2332 else:
2333 return TestResult
2333 return TestResult
2334
2334
2335
2335
2336 class TestSuite(unittest.TestSuite):
2336 class TestSuite(unittest.TestSuite):
2337 """Custom unittest TestSuite that knows how to execute Mercurial tests."""
2337 """Custom unittest TestSuite that knows how to execute Mercurial tests."""
2338
2338
2339 def __init__(
2339 def __init__(
2340 self,
2340 self,
2341 testdir,
2341 testdir,
2342 jobs=1,
2342 jobs=1,
2343 whitelist=None,
2343 whitelist=None,
2344 blacklist=None,
2344 blacklist=None,
2345 keywords=None,
2345 keywords=None,
2346 loop=False,
2346 loop=False,
2347 runs_per_test=1,
2347 runs_per_test=1,
2348 loadtest=None,
2348 loadtest=None,
2349 showchannels=False,
2349 showchannels=False,
2350 *args,
2350 *args,
2351 **kwargs
2351 **kwargs
2352 ):
2352 ):
2353 """Create a new instance that can run tests with a configuration.
2353 """Create a new instance that can run tests with a configuration.
2354
2354
2355 testdir specifies the directory where tests are executed from. This
2355 testdir specifies the directory where tests are executed from. This
2356 is typically the ``tests`` directory from Mercurial's source
2356 is typically the ``tests`` directory from Mercurial's source
2357 repository.
2357 repository.
2358
2358
2359 jobs specifies the number of jobs to run concurrently. Each test
2359 jobs specifies the number of jobs to run concurrently. Each test
2360 executes on its own thread. Tests actually spawn new processes, so
2360 executes on its own thread. Tests actually spawn new processes, so
2361 state mutation should not be an issue.
2361 state mutation should not be an issue.
2362
2362
2363 If there is only one job, it will use the main thread.
2363 If there is only one job, it will use the main thread.
2364
2364
2365 whitelist and blacklist denote tests that have been whitelisted and
2365 whitelist and blacklist denote tests that have been whitelisted and
2366 blacklisted, respectively. These arguments don't belong in TestSuite.
2366 blacklisted, respectively. These arguments don't belong in TestSuite.
2367 Instead, whitelist and blacklist should be handled by the thing that
2367 Instead, whitelist and blacklist should be handled by the thing that
2368 populates the TestSuite with tests. They are present to preserve
2368 populates the TestSuite with tests. They are present to preserve
2369 backwards compatible behavior which reports skipped tests as part
2369 backwards compatible behavior which reports skipped tests as part
2370 of the results.
2370 of the results.
2371
2371
2372 keywords denotes key words that will be used to filter which tests
2372 keywords denotes key words that will be used to filter which tests
2373 to execute. This arguably belongs outside of TestSuite.
2373 to execute. This arguably belongs outside of TestSuite.
2374
2374
2375 loop denotes whether to loop over tests forever.
2375 loop denotes whether to loop over tests forever.
2376 """
2376 """
2377 super(TestSuite, self).__init__(*args, **kwargs)
2377 super(TestSuite, self).__init__(*args, **kwargs)
2378
2378
2379 self._jobs = jobs
2379 self._jobs = jobs
2380 self._whitelist = whitelist
2380 self._whitelist = whitelist
2381 self._blacklist = blacklist
2381 self._blacklist = blacklist
2382 self._keywords = keywords
2382 self._keywords = keywords
2383 self._loop = loop
2383 self._loop = loop
2384 self._runs_per_test = runs_per_test
2384 self._runs_per_test = runs_per_test
2385 self._loadtest = loadtest
2385 self._loadtest = loadtest
2386 self._showchannels = showchannels
2386 self._showchannels = showchannels
2387
2387
2388 def run(self, result):
2388 def run(self, result):
2389 # We have a number of filters that need to be applied. We do this
2389 # We have a number of filters that need to be applied. We do this
2390 # here instead of inside Test because it makes the running logic for
2390 # here instead of inside Test because it makes the running logic for
2391 # Test simpler.
2391 # Test simpler.
2392 tests = []
2392 tests = []
2393 num_tests = [0]
2393 num_tests = [0]
2394 for test in self._tests:
2394 for test in self._tests:
2395
2395
2396 def get():
2396 def get():
2397 num_tests[0] += 1
2397 num_tests[0] += 1
2398 if getattr(test, 'should_reload', False):
2398 if getattr(test, 'should_reload', False):
2399 return self._loadtest(test, num_tests[0])
2399 return self._loadtest(test, num_tests[0])
2400 return test
2400 return test
2401
2401
2402 if not os.path.exists(test.path):
2402 if not os.path.exists(test.path):
2403 result.addSkip(test, "Doesn't exist")
2403 result.addSkip(test, "Doesn't exist")
2404 continue
2404 continue
2405
2405
2406 is_whitelisted = self._whitelist and (
2406 is_whitelisted = self._whitelist and (
2407 test.relpath in self._whitelist or test.bname in self._whitelist
2407 test.relpath in self._whitelist or test.bname in self._whitelist
2408 )
2408 )
2409 if not is_whitelisted:
2409 if not is_whitelisted:
2410 is_blacklisted = self._blacklist and (
2410 is_blacklisted = self._blacklist and (
2411 test.relpath in self._blacklist
2411 test.relpath in self._blacklist
2412 or test.bname in self._blacklist
2412 or test.bname in self._blacklist
2413 )
2413 )
2414 if is_blacklisted:
2414 if is_blacklisted:
2415 result.addSkip(test, 'blacklisted')
2415 result.addSkip(test, 'blacklisted')
2416 continue
2416 continue
2417 if self._keywords:
2417 if self._keywords:
2418 with open(test.path, 'rb') as f:
2418 with open(test.path, 'rb') as f:
2419 t = f.read().lower() + test.bname.lower()
2419 t = f.read().lower() + test.bname.lower()
2420 ignored = False
2420 ignored = False
2421 for k in self._keywords.lower().split():
2421 for k in self._keywords.lower().split():
2422 if k not in t:
2422 if k not in t:
2423 result.addIgnore(test, "doesn't match keyword")
2423 result.addIgnore(test, "doesn't match keyword")
2424 ignored = True
2424 ignored = True
2425 break
2425 break
2426
2426
2427 if ignored:
2427 if ignored:
2428 continue
2428 continue
2429 for _ in xrange(self._runs_per_test):
2429 for _ in xrange(self._runs_per_test):
2430 tests.append(get())
2430 tests.append(get())
2431
2431
2432 runtests = list(tests)
2432 runtests = list(tests)
2433 done = queue.Queue()
2433 done = queue.Queue()
2434 running = 0
2434 running = 0
2435
2435
2436 channels = [""] * self._jobs
2436 channels = [""] * self._jobs
2437
2437
2438 def job(test, result):
2438 def job(test, result):
2439 for n, v in enumerate(channels):
2439 for n, v in enumerate(channels):
2440 if not v:
2440 if not v:
2441 channel = n
2441 channel = n
2442 break
2442 break
2443 else:
2443 else:
2444 raise ValueError('Could not find output channel')
2444 raise ValueError('Could not find output channel')
2445 channels[channel] = "=" + test.name[5:].split(".")[0]
2445 channels[channel] = "=" + test.name[5:].split(".")[0]
2446 try:
2446 try:
2447 test(result)
2447 test(result)
2448 done.put(None)
2448 done.put(None)
2449 except KeyboardInterrupt:
2449 except KeyboardInterrupt:
2450 pass
2450 pass
2451 except: # re-raises
2451 except: # re-raises
2452 done.put(('!', test, 'run-test raised an error, see traceback'))
2452 done.put(('!', test, 'run-test raised an error, see traceback'))
2453 raise
2453 raise
2454 finally:
2454 finally:
2455 try:
2455 try:
2456 channels[channel] = ''
2456 channels[channel] = ''
2457 except IndexError:
2457 except IndexError:
2458 pass
2458 pass
2459
2459
2460 def stat():
2460 def stat():
2461 count = 0
2461 count = 0
2462 while channels:
2462 while channels:
2463 d = '\n%03s ' % count
2463 d = '\n%03s ' % count
2464 for n, v in enumerate(channels):
2464 for n, v in enumerate(channels):
2465 if v:
2465 if v:
2466 d += v[0]
2466 d += v[0]
2467 channels[n] = v[1:] or '.'
2467 channels[n] = v[1:] or '.'
2468 else:
2468 else:
2469 d += ' '
2469 d += ' '
2470 d += ' '
2470 d += ' '
2471 with iolock:
2471 with iolock:
2472 sys.stdout.write(d + ' ')
2472 sys.stdout.write(d + ' ')
2473 sys.stdout.flush()
2473 sys.stdout.flush()
2474 for x in xrange(10):
2474 for x in xrange(10):
2475 if channels:
2475 if channels:
2476 time.sleep(0.1)
2476 time.sleep(0.1)
2477 count += 1
2477 count += 1
2478
2478
2479 stoppedearly = False
2479 stoppedearly = False
2480
2480
2481 if self._showchannels:
2481 if self._showchannels:
2482 statthread = threading.Thread(target=stat, name="stat")
2482 statthread = threading.Thread(target=stat, name="stat")
2483 statthread.start()
2483 statthread.start()
2484
2484
2485 try:
2485 try:
2486 while tests or running:
2486 while tests or running:
2487 if not done.empty() or running == self._jobs or not tests:
2487 if not done.empty() or running == self._jobs or not tests:
2488 try:
2488 try:
2489 done.get(True, 1)
2489 done.get(True, 1)
2490 running -= 1
2490 running -= 1
2491 if result and result.shouldStop:
2491 if result and result.shouldStop:
2492 stoppedearly = True
2492 stoppedearly = True
2493 break
2493 break
2494 except queue.Empty:
2494 except queue.Empty:
2495 continue
2495 continue
2496 if tests and not running == self._jobs:
2496 if tests and not running == self._jobs:
2497 test = tests.pop(0)
2497 test = tests.pop(0)
2498 if self._loop:
2498 if self._loop:
2499 if getattr(test, 'should_reload', False):
2499 if getattr(test, 'should_reload', False):
2500 num_tests[0] += 1
2500 num_tests[0] += 1
2501 tests.append(self._loadtest(test, num_tests[0]))
2501 tests.append(self._loadtest(test, num_tests[0]))
2502 else:
2502 else:
2503 tests.append(test)
2503 tests.append(test)
2504 if self._jobs == 1:
2504 if self._jobs == 1:
2505 job(test, result)
2505 job(test, result)
2506 else:
2506 else:
2507 t = threading.Thread(
2507 t = threading.Thread(
2508 target=job, name=test.name, args=(test, result)
2508 target=job, name=test.name, args=(test, result)
2509 )
2509 )
2510 t.start()
2510 t.start()
2511 running += 1
2511 running += 1
2512
2512
2513 # If we stop early we still need to wait on started tests to
2513 # If we stop early we still need to wait on started tests to
2514 # finish. Otherwise, there is a race between the test completing
2514 # finish. Otherwise, there is a race between the test completing
2515 # and the test's cleanup code running. This could result in the
2515 # and the test's cleanup code running. This could result in the
2516 # test reporting incorrect.
2516 # test reporting incorrect.
2517 if stoppedearly:
2517 if stoppedearly:
2518 while running:
2518 while running:
2519 try:
2519 try:
2520 done.get(True, 1)
2520 done.get(True, 1)
2521 running -= 1
2521 running -= 1
2522 except queue.Empty:
2522 except queue.Empty:
2523 continue
2523 continue
2524 except KeyboardInterrupt:
2524 except KeyboardInterrupt:
2525 for test in runtests:
2525 for test in runtests:
2526 test.abort()
2526 test.abort()
2527
2527
2528 channels = []
2528 channels = []
2529
2529
2530 return result
2530 return result
2531
2531
2532
2532
2533 # Save the most recent 5 wall-clock runtimes of each test to a
2533 # Save the most recent 5 wall-clock runtimes of each test to a
2534 # human-readable text file named .testtimes. Tests are sorted
2534 # human-readable text file named .testtimes. Tests are sorted
2535 # alphabetically, while times for each test are listed from oldest to
2535 # alphabetically, while times for each test are listed from oldest to
2536 # newest.
2536 # newest.
2537
2537
2538
2538
2539 def loadtimes(outputdir):
2539 def loadtimes(outputdir):
2540 times = []
2540 times = []
2541 try:
2541 try:
2542 with open(os.path.join(outputdir, b'.testtimes')) as fp:
2542 with open(os.path.join(outputdir, b'.testtimes')) as fp:
2543 for line in fp:
2543 for line in fp:
2544 m = re.match('(.*?) ([0-9. ]+)', line)
2544 m = re.match('(.*?) ([0-9. ]+)', line)
2545 times.append(
2545 times.append(
2546 (m.group(1), [float(t) for t in m.group(2).split()])
2546 (m.group(1), [float(t) for t in m.group(2).split()])
2547 )
2547 )
2548 except IOError as err:
2548 except IOError as err:
2549 if err.errno != errno.ENOENT:
2549 if err.errno != errno.ENOENT:
2550 raise
2550 raise
2551 return times
2551 return times
2552
2552
2553
2553
2554 def savetimes(outputdir, result):
2554 def savetimes(outputdir, result):
2555 saved = dict(loadtimes(outputdir))
2555 saved = dict(loadtimes(outputdir))
2556 maxruns = 5
2556 maxruns = 5
2557 skipped = {str(t[0]) for t in result.skipped}
2557 skipped = {str(t[0]) for t in result.skipped}
2558 for tdata in result.times:
2558 for tdata in result.times:
2559 test, real = tdata[0], tdata[3]
2559 test, real = tdata[0], tdata[3]
2560 if test not in skipped:
2560 if test not in skipped:
2561 ts = saved.setdefault(test, [])
2561 ts = saved.setdefault(test, [])
2562 ts.append(real)
2562 ts.append(real)
2563 ts[:] = ts[-maxruns:]
2563 ts[:] = ts[-maxruns:]
2564
2564
2565 fd, tmpname = tempfile.mkstemp(
2565 fd, tmpname = tempfile.mkstemp(
2566 prefix=b'.testtimes', dir=outputdir, text=True
2566 prefix=b'.testtimes', dir=outputdir, text=True
2567 )
2567 )
2568 with os.fdopen(fd, 'w') as fp:
2568 with os.fdopen(fd, 'w') as fp:
2569 for name, ts in sorted(saved.items()):
2569 for name, ts in sorted(saved.items()):
2570 fp.write('%s %s\n' % (name, ' '.join(['%.3f' % (t,) for t in ts])))
2570 fp.write('%s %s\n' % (name, ' '.join(['%.3f' % (t,) for t in ts])))
2571 timepath = os.path.join(outputdir, b'.testtimes')
2571 timepath = os.path.join(outputdir, b'.testtimes')
2572 try:
2572 try:
2573 os.unlink(timepath)
2573 os.unlink(timepath)
2574 except OSError:
2574 except OSError:
2575 pass
2575 pass
2576 try:
2576 try:
2577 os.rename(tmpname, timepath)
2577 os.rename(tmpname, timepath)
2578 except OSError:
2578 except OSError:
2579 pass
2579 pass
2580
2580
2581
2581
2582 class TextTestRunner(unittest.TextTestRunner):
2582 class TextTestRunner(unittest.TextTestRunner):
2583 """Custom unittest test runner that uses appropriate settings."""
2583 """Custom unittest test runner that uses appropriate settings."""
2584
2584
2585 def __init__(self, runner, *args, **kwargs):
2585 def __init__(self, runner, *args, **kwargs):
2586 super(TextTestRunner, self).__init__(*args, **kwargs)
2586 super(TextTestRunner, self).__init__(*args, **kwargs)
2587
2587
2588 self._runner = runner
2588 self._runner = runner
2589
2589
2590 self._result = getTestResult()(
2590 self._result = getTestResult()(
2591 self._runner.options, self.stream, self.descriptions, self.verbosity
2591 self._runner.options, self.stream, self.descriptions, self.verbosity
2592 )
2592 )
2593
2593
2594 def listtests(self, test):
2594 def listtests(self, test):
2595 test = sorted(test, key=lambda t: t.name)
2595 test = sorted(test, key=lambda t: t.name)
2596
2596
2597 self._result.onStart(test)
2597 self._result.onStart(test)
2598
2598
2599 for t in test:
2599 for t in test:
2600 print(t.name)
2600 print(t.name)
2601 self._result.addSuccess(t)
2601 self._result.addSuccess(t)
2602
2602
2603 if self._runner.options.xunit:
2603 if self._runner.options.xunit:
2604 with open(self._runner.options.xunit, "wb") as xuf:
2604 with open(self._runner.options.xunit, "wb") as xuf:
2605 self._writexunit(self._result, xuf)
2605 self._writexunit(self._result, xuf)
2606
2606
2607 if self._runner.options.json:
2607 if self._runner.options.json:
2608 jsonpath = os.path.join(self._runner._outputdir, b'report.json')
2608 jsonpath = os.path.join(self._runner._outputdir, b'report.json')
2609 with open(jsonpath, 'w') as fp:
2609 with open(jsonpath, 'w') as fp:
2610 self._writejson(self._result, fp)
2610 self._writejson(self._result, fp)
2611
2611
2612 return self._result
2612 return self._result
2613
2613
2614 def run(self, test):
2614 def run(self, test):
2615 self._result.onStart(test)
2615 self._result.onStart(test)
2616 test(self._result)
2616 test(self._result)
2617
2617
2618 failed = len(self._result.failures)
2618 failed = len(self._result.failures)
2619 skipped = len(self._result.skipped)
2619 skipped = len(self._result.skipped)
2620 ignored = len(self._result.ignored)
2620 ignored = len(self._result.ignored)
2621
2621
2622 with iolock:
2622 with iolock:
2623 self.stream.writeln('')
2623 self.stream.writeln('')
2624
2624
2625 if not self._runner.options.noskips:
2625 if not self._runner.options.noskips:
2626 for test, msg in sorted(
2626 for test, msg in sorted(
2627 self._result.skipped, key=lambda s: s[0].name
2627 self._result.skipped, key=lambda s: s[0].name
2628 ):
2628 ):
2629 formatted = 'Skipped %s: %s\n' % (test.name, msg)
2629 formatted = 'Skipped %s: %s\n' % (test.name, msg)
2630 msg = highlightmsg(formatted, self._result.color)
2630 msg = highlightmsg(formatted, self._result.color)
2631 self.stream.write(msg)
2631 self.stream.write(msg)
2632 for test, msg in sorted(
2632 for test, msg in sorted(
2633 self._result.failures, key=lambda f: f[0].name
2633 self._result.failures, key=lambda f: f[0].name
2634 ):
2634 ):
2635 formatted = 'Failed %s: %s\n' % (test.name, msg)
2635 formatted = 'Failed %s: %s\n' % (test.name, msg)
2636 self.stream.write(highlightmsg(formatted, self._result.color))
2636 self.stream.write(highlightmsg(formatted, self._result.color))
2637 for test, msg in sorted(
2637 for test, msg in sorted(
2638 self._result.errors, key=lambda e: e[0].name
2638 self._result.errors, key=lambda e: e[0].name
2639 ):
2639 ):
2640 self.stream.writeln('Errored %s: %s' % (test.name, msg))
2640 self.stream.writeln('Errored %s: %s' % (test.name, msg))
2641
2641
2642 if self._runner.options.xunit:
2642 if self._runner.options.xunit:
2643 with open(self._runner.options.xunit, "wb") as xuf:
2643 with open(self._runner.options.xunit, "wb") as xuf:
2644 self._writexunit(self._result, xuf)
2644 self._writexunit(self._result, xuf)
2645
2645
2646 if self._runner.options.json:
2646 if self._runner.options.json:
2647 jsonpath = os.path.join(self._runner._outputdir, b'report.json')
2647 jsonpath = os.path.join(self._runner._outputdir, b'report.json')
2648 with open(jsonpath, 'w') as fp:
2648 with open(jsonpath, 'w') as fp:
2649 self._writejson(self._result, fp)
2649 self._writejson(self._result, fp)
2650
2650
2651 self._runner._checkhglib('Tested')
2651 self._runner._checkhglib('Tested')
2652
2652
2653 savetimes(self._runner._outputdir, self._result)
2653 savetimes(self._runner._outputdir, self._result)
2654
2654
2655 if failed and self._runner.options.known_good_rev:
2655 if failed and self._runner.options.known_good_rev:
2656 self._bisecttests(t for t, m in self._result.failures)
2656 self._bisecttests(t for t, m in self._result.failures)
2657 self.stream.writeln(
2657 self.stream.writeln(
2658 '# Ran %d tests, %d skipped, %d failed.'
2658 '# Ran %d tests, %d skipped, %d failed.'
2659 % (self._result.testsRun, skipped + ignored, failed)
2659 % (self._result.testsRun, skipped + ignored, failed)
2660 )
2660 )
2661 if failed:
2661 if failed:
2662 self.stream.writeln(
2662 self.stream.writeln(
2663 'python hash seed: %s' % os.environ['PYTHONHASHSEED']
2663 'python hash seed: %s' % os.environ['PYTHONHASHSEED']
2664 )
2664 )
2665 if self._runner.options.time:
2665 if self._runner.options.time:
2666 self.printtimes(self._result.times)
2666 self.printtimes(self._result.times)
2667
2667
2668 if self._runner.options.exceptions:
2668 if self._runner.options.exceptions:
2669 exceptions = aggregateexceptions(
2669 exceptions = aggregateexceptions(
2670 os.path.join(self._runner._outputdir, b'exceptions')
2670 os.path.join(self._runner._outputdir, b'exceptions')
2671 )
2671 )
2672
2672
2673 self.stream.writeln('Exceptions Report:')
2673 self.stream.writeln('Exceptions Report:')
2674 self.stream.writeln(
2674 self.stream.writeln(
2675 '%d total from %d frames'
2675 '%d total from %d frames'
2676 % (exceptions['total'], len(exceptions['exceptioncounts']))
2676 % (exceptions['total'], len(exceptions['exceptioncounts']))
2677 )
2677 )
2678 combined = exceptions['combined']
2678 combined = exceptions['combined']
2679 for key in sorted(combined, key=combined.get, reverse=True):
2679 for key in sorted(combined, key=combined.get, reverse=True):
2680 frame, line, exc = key
2680 frame, line, exc = key
2681 totalcount, testcount, leastcount, leasttest = combined[key]
2681 totalcount, testcount, leastcount, leasttest = combined[key]
2682
2682
2683 self.stream.writeln(
2683 self.stream.writeln(
2684 '%d (%d tests)\t%s: %s (%s - %d total)'
2684 '%d (%d tests)\t%s: %s (%s - %d total)'
2685 % (
2685 % (
2686 totalcount,
2686 totalcount,
2687 testcount,
2687 testcount,
2688 frame,
2688 frame,
2689 exc,
2689 exc,
2690 leasttest,
2690 leasttest,
2691 leastcount,
2691 leastcount,
2692 )
2692 )
2693 )
2693 )
2694
2694
2695 self.stream.flush()
2695 self.stream.flush()
2696
2696
2697 return self._result
2697 return self._result
2698
2698
2699 def _bisecttests(self, tests):
2699 def _bisecttests(self, tests):
2700 bisectcmd = ['hg', 'bisect']
2700 bisectcmd = ['hg', 'bisect']
2701 bisectrepo = self._runner.options.bisect_repo
2701 bisectrepo = self._runner.options.bisect_repo
2702 if bisectrepo:
2702 if bisectrepo:
2703 bisectcmd.extend(['-R', os.path.abspath(bisectrepo)])
2703 bisectcmd.extend(['-R', os.path.abspath(bisectrepo)])
2704
2704
2705 def pread(args):
2705 def pread(args):
2706 env = os.environ.copy()
2706 env = os.environ.copy()
2707 env['HGPLAIN'] = '1'
2707 env['HGPLAIN'] = '1'
2708 p = subprocess.Popen(
2708 p = subprocess.Popen(
2709 args, stderr=subprocess.STDOUT, stdout=subprocess.PIPE, env=env
2709 args, stderr=subprocess.STDOUT, stdout=subprocess.PIPE, env=env
2710 )
2710 )
2711 data = p.stdout.read()
2711 data = p.stdout.read()
2712 p.wait()
2712 p.wait()
2713 return data
2713 return data
2714
2714
2715 for test in tests:
2715 for test in tests:
2716 pread(bisectcmd + ['--reset']),
2716 pread(bisectcmd + ['--reset']),
2717 pread(bisectcmd + ['--bad', '.'])
2717 pread(bisectcmd + ['--bad', '.'])
2718 pread(bisectcmd + ['--good', self._runner.options.known_good_rev])
2718 pread(bisectcmd + ['--good', self._runner.options.known_good_rev])
2719 # TODO: we probably need to forward more options
2719 # TODO: we probably need to forward more options
2720 # that alter hg's behavior inside the tests.
2720 # that alter hg's behavior inside the tests.
2721 opts = ''
2721 opts = ''
2722 withhg = self._runner.options.with_hg
2722 withhg = self._runner.options.with_hg
2723 if withhg:
2723 if withhg:
2724 opts += ' --with-hg=%s ' % shellquote(_bytes2sys(withhg))
2724 opts += ' --with-hg=%s ' % shellquote(_bytes2sys(withhg))
2725 rtc = '%s %s %s %s' % (sysexecutable, sys.argv[0], opts, test)
2725 rtc = '%s %s %s %s' % (sysexecutable, sys.argv[0], opts, test)
2726 data = pread(bisectcmd + ['--command', rtc])
2726 data = pread(bisectcmd + ['--command', rtc])
2727 m = re.search(
2727 m = re.search(
2728 (
2728 (
2729 br'\nThe first (?P<goodbad>bad|good) revision '
2729 br'\nThe first (?P<goodbad>bad|good) revision '
2730 br'is:\nchangeset: +\d+:(?P<node>[a-f0-9]+)\n.*\n'
2730 br'is:\nchangeset: +\d+:(?P<node>[a-f0-9]+)\n.*\n'
2731 br'summary: +(?P<summary>[^\n]+)\n'
2731 br'summary: +(?P<summary>[^\n]+)\n'
2732 ),
2732 ),
2733 data,
2733 data,
2734 (re.MULTILINE | re.DOTALL),
2734 (re.MULTILINE | re.DOTALL),
2735 )
2735 )
2736 if m is None:
2736 if m is None:
2737 self.stream.writeln(
2737 self.stream.writeln(
2738 'Failed to identify failure point for %s' % test
2738 'Failed to identify failure point for %s' % test
2739 )
2739 )
2740 continue
2740 continue
2741 dat = m.groupdict()
2741 dat = m.groupdict()
2742 verb = 'broken' if dat['goodbad'] == b'bad' else 'fixed'
2742 verb = 'broken' if dat['goodbad'] == b'bad' else 'fixed'
2743 self.stream.writeln(
2743 self.stream.writeln(
2744 '%s %s by %s (%s)'
2744 '%s %s by %s (%s)'
2745 % (
2745 % (
2746 test,
2746 test,
2747 verb,
2747 verb,
2748 dat['node'].decode('ascii'),
2748 dat['node'].decode('ascii'),
2749 dat['summary'].decode('utf8', 'ignore'),
2749 dat['summary'].decode('utf8', 'ignore'),
2750 )
2750 )
2751 )
2751 )
2752
2752
2753 def printtimes(self, times):
2753 def printtimes(self, times):
2754 # iolock held by run
2754 # iolock held by run
2755 self.stream.writeln('# Producing time report')
2755 self.stream.writeln('# Producing time report')
2756 times.sort(key=lambda t: (t[3]))
2756 times.sort(key=lambda t: (t[3]))
2757 cols = '%7.3f %7.3f %7.3f %7.3f %7.3f %s'
2757 cols = '%7.3f %7.3f %7.3f %7.3f %7.3f %s'
2758 self.stream.writeln(
2758 self.stream.writeln(
2759 '%-7s %-7s %-7s %-7s %-7s %s'
2759 '%-7s %-7s %-7s %-7s %-7s %s'
2760 % ('start', 'end', 'cuser', 'csys', 'real', 'Test')
2760 % ('start', 'end', 'cuser', 'csys', 'real', 'Test')
2761 )
2761 )
2762 for tdata in times:
2762 for tdata in times:
2763 test = tdata[0]
2763 test = tdata[0]
2764 cuser, csys, real, start, end = tdata[1:6]
2764 cuser, csys, real, start, end = tdata[1:6]
2765 self.stream.writeln(cols % (start, end, cuser, csys, real, test))
2765 self.stream.writeln(cols % (start, end, cuser, csys, real, test))
2766
2766
2767 @staticmethod
2767 @staticmethod
2768 def _writexunit(result, outf):
2768 def _writexunit(result, outf):
2769 # See http://llg.cubic.org/docs/junit/ for a reference.
2769 # See http://llg.cubic.org/docs/junit/ for a reference.
2770 timesd = {t[0]: t[3] for t in result.times}
2770 timesd = {t[0]: t[3] for t in result.times}
2771 doc = minidom.Document()
2771 doc = minidom.Document()
2772 s = doc.createElement('testsuite')
2772 s = doc.createElement('testsuite')
2773 s.setAttribute('errors', "0") # TODO
2773 s.setAttribute('errors', "0") # TODO
2774 s.setAttribute('failures', str(len(result.failures)))
2774 s.setAttribute('failures', str(len(result.failures)))
2775 s.setAttribute('name', 'run-tests')
2775 s.setAttribute('name', 'run-tests')
2776 s.setAttribute(
2776 s.setAttribute(
2777 'skipped', str(len(result.skipped) + len(result.ignored))
2777 'skipped', str(len(result.skipped) + len(result.ignored))
2778 )
2778 )
2779 s.setAttribute('tests', str(result.testsRun))
2779 s.setAttribute('tests', str(result.testsRun))
2780 doc.appendChild(s)
2780 doc.appendChild(s)
2781 for tc in result.successes:
2781 for tc in result.successes:
2782 t = doc.createElement('testcase')
2782 t = doc.createElement('testcase')
2783 t.setAttribute('name', tc.name)
2783 t.setAttribute('name', tc.name)
2784 tctime = timesd.get(tc.name)
2784 tctime = timesd.get(tc.name)
2785 if tctime is not None:
2785 if tctime is not None:
2786 t.setAttribute('time', '%.3f' % tctime)
2786 t.setAttribute('time', '%.3f' % tctime)
2787 s.appendChild(t)
2787 s.appendChild(t)
2788 for tc, err in sorted(result.faildata.items()):
2788 for tc, err in sorted(result.faildata.items()):
2789 t = doc.createElement('testcase')
2789 t = doc.createElement('testcase')
2790 t.setAttribute('name', tc)
2790 t.setAttribute('name', tc)
2791 tctime = timesd.get(tc)
2791 tctime = timesd.get(tc)
2792 if tctime is not None:
2792 if tctime is not None:
2793 t.setAttribute('time', '%.3f' % tctime)
2793 t.setAttribute('time', '%.3f' % tctime)
2794 # createCDATASection expects a unicode or it will
2794 # createCDATASection expects a unicode or it will
2795 # convert using default conversion rules, which will
2795 # convert using default conversion rules, which will
2796 # fail if string isn't ASCII.
2796 # fail if string isn't ASCII.
2797 err = cdatasafe(err).decode('utf-8', 'replace')
2797 err = cdatasafe(err).decode('utf-8', 'replace')
2798 cd = doc.createCDATASection(err)
2798 cd = doc.createCDATASection(err)
2799 # Use 'failure' here instead of 'error' to match errors = 0,
2799 # Use 'failure' here instead of 'error' to match errors = 0,
2800 # failures = len(result.failures) in the testsuite element.
2800 # failures = len(result.failures) in the testsuite element.
2801 failelem = doc.createElement('failure')
2801 failelem = doc.createElement('failure')
2802 failelem.setAttribute('message', 'output changed')
2802 failelem.setAttribute('message', 'output changed')
2803 failelem.setAttribute('type', 'output-mismatch')
2803 failelem.setAttribute('type', 'output-mismatch')
2804 failelem.appendChild(cd)
2804 failelem.appendChild(cd)
2805 t.appendChild(failelem)
2805 t.appendChild(failelem)
2806 s.appendChild(t)
2806 s.appendChild(t)
2807 for tc, message in result.skipped:
2807 for tc, message in result.skipped:
2808 # According to the schema, 'skipped' has no attributes. So store
2808 # According to the schema, 'skipped' has no attributes. So store
2809 # the skip message as a text node instead.
2809 # the skip message as a text node instead.
2810 t = doc.createElement('testcase')
2810 t = doc.createElement('testcase')
2811 t.setAttribute('name', tc.name)
2811 t.setAttribute('name', tc.name)
2812 binmessage = message.encode('utf-8')
2812 binmessage = message.encode('utf-8')
2813 message = cdatasafe(binmessage).decode('utf-8', 'replace')
2813 message = cdatasafe(binmessage).decode('utf-8', 'replace')
2814 cd = doc.createCDATASection(message)
2814 cd = doc.createCDATASection(message)
2815 skipelem = doc.createElement('skipped')
2815 skipelem = doc.createElement('skipped')
2816 skipelem.appendChild(cd)
2816 skipelem.appendChild(cd)
2817 t.appendChild(skipelem)
2817 t.appendChild(skipelem)
2818 s.appendChild(t)
2818 s.appendChild(t)
2819 outf.write(doc.toprettyxml(indent=' ', encoding='utf-8'))
2819 outf.write(doc.toprettyxml(indent=' ', encoding='utf-8'))
2820
2820
2821 @staticmethod
2821 @staticmethod
2822 def _writejson(result, outf):
2822 def _writejson(result, outf):
2823 timesd = {}
2823 timesd = {}
2824 for tdata in result.times:
2824 for tdata in result.times:
2825 test = tdata[0]
2825 test = tdata[0]
2826 timesd[test] = tdata[1:]
2826 timesd[test] = tdata[1:]
2827
2827
2828 outcome = {}
2828 outcome = {}
2829 groups = [
2829 groups = [
2830 ('success', ((tc, None) for tc in result.successes)),
2830 ('success', ((tc, None) for tc in result.successes)),
2831 ('failure', result.failures),
2831 ('failure', result.failures),
2832 ('skip', result.skipped),
2832 ('skip', result.skipped),
2833 ]
2833 ]
2834 for res, testcases in groups:
2834 for res, testcases in groups:
2835 for tc, __ in testcases:
2835 for tc, __ in testcases:
2836 if tc.name in timesd:
2836 if tc.name in timesd:
2837 diff = result.faildata.get(tc.name, b'')
2837 diff = result.faildata.get(tc.name, b'')
2838 try:
2838 try:
2839 diff = diff.decode('unicode_escape')
2839 diff = diff.decode('unicode_escape')
2840 except UnicodeDecodeError as e:
2840 except UnicodeDecodeError as e:
2841 diff = '%r decoding diff, sorry' % e
2841 diff = '%r decoding diff, sorry' % e
2842 tres = {
2842 tres = {
2843 'result': res,
2843 'result': res,
2844 'time': ('%0.3f' % timesd[tc.name][2]),
2844 'time': ('%0.3f' % timesd[tc.name][2]),
2845 'cuser': ('%0.3f' % timesd[tc.name][0]),
2845 'cuser': ('%0.3f' % timesd[tc.name][0]),
2846 'csys': ('%0.3f' % timesd[tc.name][1]),
2846 'csys': ('%0.3f' % timesd[tc.name][1]),
2847 'start': ('%0.3f' % timesd[tc.name][3]),
2847 'start': ('%0.3f' % timesd[tc.name][3]),
2848 'end': ('%0.3f' % timesd[tc.name][4]),
2848 'end': ('%0.3f' % timesd[tc.name][4]),
2849 'diff': diff,
2849 'diff': diff,
2850 }
2850 }
2851 else:
2851 else:
2852 # blacklisted test
2852 # blacklisted test
2853 tres = {'result': res}
2853 tres = {'result': res}
2854
2854
2855 outcome[tc.name] = tres
2855 outcome[tc.name] = tres
2856 jsonout = json.dumps(
2856 jsonout = json.dumps(
2857 outcome, sort_keys=True, indent=4, separators=(',', ': ')
2857 outcome, sort_keys=True, indent=4, separators=(',', ': ')
2858 )
2858 )
2859 outf.writelines(("testreport =", jsonout))
2859 outf.writelines(("testreport =", jsonout))
2860
2860
2861
2861
2862 def sorttests(testdescs, previoustimes, shuffle=False):
2862 def sorttests(testdescs, previoustimes, shuffle=False):
2863 """Do an in-place sort of tests."""
2863 """Do an in-place sort of tests."""
2864 if shuffle:
2864 if shuffle:
2865 random.shuffle(testdescs)
2865 random.shuffle(testdescs)
2866 return
2866 return
2867
2867
2868 if previoustimes:
2868 if previoustimes:
2869
2869
2870 def sortkey(f):
2870 def sortkey(f):
2871 f = f['path']
2871 f = f['path']
2872 if f in previoustimes:
2872 if f in previoustimes:
2873 # Use most recent time as estimate
2873 # Use most recent time as estimate
2874 return -(previoustimes[f][-1])
2874 return -(previoustimes[f][-1])
2875 else:
2875 else:
2876 # Default to a rather arbitrary value of 1 second for new tests
2876 # Default to a rather arbitrary value of 1 second for new tests
2877 return -1.0
2877 return -1.0
2878
2878
2879 else:
2879 else:
2880 # keywords for slow tests
2880 # keywords for slow tests
2881 slow = {
2881 slow = {
2882 b'svn': 10,
2882 b'svn': 10,
2883 b'cvs': 10,
2883 b'cvs': 10,
2884 b'hghave': 10,
2884 b'hghave': 10,
2885 b'largefiles-update': 10,
2885 b'largefiles-update': 10,
2886 b'run-tests': 10,
2886 b'run-tests': 10,
2887 b'corruption': 10,
2887 b'corruption': 10,
2888 b'race': 10,
2888 b'race': 10,
2889 b'i18n': 10,
2889 b'i18n': 10,
2890 b'check': 100,
2890 b'check': 100,
2891 b'gendoc': 100,
2891 b'gendoc': 100,
2892 b'contrib-perf': 200,
2892 b'contrib-perf': 200,
2893 b'merge-combination': 100,
2893 b'merge-combination': 100,
2894 }
2894 }
2895 perf = {}
2895 perf = {}
2896
2896
2897 def sortkey(f):
2897 def sortkey(f):
2898 # run largest tests first, as they tend to take the longest
2898 # run largest tests first, as they tend to take the longest
2899 f = f['path']
2899 f = f['path']
2900 try:
2900 try:
2901 return perf[f]
2901 return perf[f]
2902 except KeyError:
2902 except KeyError:
2903 try:
2903 try:
2904 val = -os.stat(f).st_size
2904 val = -os.stat(f).st_size
2905 except OSError as e:
2905 except OSError as e:
2906 if e.errno != errno.ENOENT:
2906 if e.errno != errno.ENOENT:
2907 raise
2907 raise
2908 perf[f] = -1e9 # file does not exist, tell early
2908 perf[f] = -1e9 # file does not exist, tell early
2909 return -1e9
2909 return -1e9
2910 for kw, mul in slow.items():
2910 for kw, mul in slow.items():
2911 if kw in f:
2911 if kw in f:
2912 val *= mul
2912 val *= mul
2913 if f.endswith(b'.py'):
2913 if f.endswith(b'.py'):
2914 val /= 10.0
2914 val /= 10.0
2915 perf[f] = val / 1000.0
2915 perf[f] = val / 1000.0
2916 return perf[f]
2916 return perf[f]
2917
2917
2918 testdescs.sort(key=sortkey)
2918 testdescs.sort(key=sortkey)
2919
2919
2920
2920
2921 class TestRunner(object):
2921 class TestRunner(object):
2922 """Holds context for executing tests.
2922 """Holds context for executing tests.
2923
2923
2924 Tests rely on a lot of state. This object holds it for them.
2924 Tests rely on a lot of state. This object holds it for them.
2925 """
2925 """
2926
2926
2927 # Programs required to run tests.
2927 # Programs required to run tests.
2928 REQUIREDTOOLS = [
2928 REQUIREDTOOLS = [
2929 b'diff',
2929 b'diff',
2930 b'grep',
2930 b'grep',
2931 b'unzip',
2931 b'unzip',
2932 b'gunzip',
2932 b'gunzip',
2933 b'bunzip2',
2933 b'bunzip2',
2934 b'sed',
2934 b'sed',
2935 ]
2935 ]
2936
2936
2937 # Maps file extensions to test class.
2937 # Maps file extensions to test class.
2938 TESTTYPES = [
2938 TESTTYPES = [
2939 (b'.py', PythonTest),
2939 (b'.py', PythonTest),
2940 (b'.t', TTest),
2940 (b'.t', TTest),
2941 ]
2941 ]
2942
2942
2943 def __init__(self):
2943 def __init__(self):
2944 self.options = None
2944 self.options = None
2945 self._hgroot = None
2945 self._hgroot = None
2946 self._testdir = None
2946 self._testdir = None
2947 self._outputdir = None
2947 self._outputdir = None
2948 self._hgtmp = None
2948 self._hgtmp = None
2949 self._installdir = None
2949 self._installdir = None
2950 self._bindir = None
2950 self._bindir = None
2951 self._tmpbindir = None
2951 self._tmpbindir = None
2952 self._pythondir = None
2952 self._pythondir = None
2953 self._coveragefile = None
2953 self._coveragefile = None
2954 self._createdfiles = []
2954 self._createdfiles = []
2955 self._hgcommand = None
2955 self._hgcommand = None
2956 self._hgpath = None
2956 self._hgpath = None
2957 self._portoffset = 0
2957 self._portoffset = 0
2958 self._ports = {}
2958 self._ports = {}
2959
2959
2960 def run(self, args, parser=None):
2960 def run(self, args, parser=None):
2961 """Run the test suite."""
2961 """Run the test suite."""
2962 oldmask = os.umask(0o22)
2962 oldmask = os.umask(0o22)
2963 try:
2963 try:
2964 parser = parser or getparser()
2964 parser = parser or getparser()
2965 options = parseargs(args, parser)
2965 options = parseargs(args, parser)
2966 tests = [_sys2bytes(a) for a in options.tests]
2966 tests = [_sys2bytes(a) for a in options.tests]
2967 if options.test_list is not None:
2967 if options.test_list is not None:
2968 for listfile in options.test_list:
2968 for listfile in options.test_list:
2969 with open(listfile, 'rb') as f:
2969 with open(listfile, 'rb') as f:
2970 tests.extend(t for t in f.read().splitlines() if t)
2970 tests.extend(t for t in f.read().splitlines() if t)
2971 self.options = options
2971 self.options = options
2972
2972
2973 self._checktools()
2973 self._checktools()
2974 testdescs = self.findtests(tests)
2974 testdescs = self.findtests(tests)
2975 if options.profile_runner:
2975 if options.profile_runner:
2976 import statprof
2976 import statprof
2977
2977
2978 statprof.start()
2978 statprof.start()
2979 result = self._run(testdescs)
2979 result = self._run(testdescs)
2980 if options.profile_runner:
2980 if options.profile_runner:
2981 statprof.stop()
2981 statprof.stop()
2982 statprof.display()
2982 statprof.display()
2983 return result
2983 return result
2984
2984
2985 finally:
2985 finally:
2986 os.umask(oldmask)
2986 os.umask(oldmask)
2987
2987
2988 def _run(self, testdescs):
2988 def _run(self, testdescs):
2989 testdir = getcwdb()
2989 testdir = getcwdb()
2990 self._testdir = osenvironb[b'TESTDIR'] = getcwdb()
2990 self._testdir = osenvironb[b'TESTDIR'] = getcwdb()
2991 # assume all tests in same folder for now
2991 # assume all tests in same folder for now
2992 if testdescs:
2992 if testdescs:
2993 pathname = os.path.dirname(testdescs[0]['path'])
2993 pathname = os.path.dirname(testdescs[0]['path'])
2994 if pathname:
2994 if pathname:
2995 testdir = os.path.join(testdir, pathname)
2995 testdir = os.path.join(testdir, pathname)
2996 self._testdir = osenvironb[b'TESTDIR'] = testdir
2996 self._testdir = osenvironb[b'TESTDIR'] = testdir
2997 if self.options.outputdir:
2997 if self.options.outputdir:
2998 self._outputdir = canonpath(_sys2bytes(self.options.outputdir))
2998 self._outputdir = canonpath(_sys2bytes(self.options.outputdir))
2999 else:
2999 else:
3000 self._outputdir = getcwdb()
3000 self._outputdir = getcwdb()
3001 if testdescs and pathname:
3001 if testdescs and pathname:
3002 self._outputdir = os.path.join(self._outputdir, pathname)
3002 self._outputdir = os.path.join(self._outputdir, pathname)
3003 previoustimes = {}
3003 previoustimes = {}
3004 if self.options.order_by_runtime:
3004 if self.options.order_by_runtime:
3005 previoustimes = dict(loadtimes(self._outputdir))
3005 previoustimes = dict(loadtimes(self._outputdir))
3006 sorttests(testdescs, previoustimes, shuffle=self.options.random)
3006 sorttests(testdescs, previoustimes, shuffle=self.options.random)
3007
3007
3008 if 'PYTHONHASHSEED' not in os.environ:
3008 if 'PYTHONHASHSEED' not in os.environ:
3009 # use a random python hash seed all the time
3009 # use a random python hash seed all the time
3010 # we do the randomness ourself to know what seed is used
3010 # we do the randomness ourself to know what seed is used
3011 os.environ['PYTHONHASHSEED'] = str(random.getrandbits(32))
3011 os.environ['PYTHONHASHSEED'] = str(random.getrandbits(32))
3012
3012
3013 # Rayon (Rust crate for multi-threading) will use all logical CPU cores
3013 # Rayon (Rust crate for multi-threading) will use all logical CPU cores
3014 # by default, causing thrashing on high-cpu-count systems.
3014 # by default, causing thrashing on high-cpu-count systems.
3015 # Setting its limit to 3 during tests should still let us uncover
3015 # Setting its limit to 3 during tests should still let us uncover
3016 # multi-threading bugs while keeping the thrashing reasonable.
3016 # multi-threading bugs while keeping the thrashing reasonable.
3017 os.environ.setdefault("RAYON_NUM_THREADS", "3")
3017 os.environ.setdefault("RAYON_NUM_THREADS", "3")
3018
3018
3019 if self.options.tmpdir:
3019 if self.options.tmpdir:
3020 self.options.keep_tmpdir = True
3020 self.options.keep_tmpdir = True
3021 tmpdir = _sys2bytes(self.options.tmpdir)
3021 tmpdir = _sys2bytes(self.options.tmpdir)
3022 if os.path.exists(tmpdir):
3022 if os.path.exists(tmpdir):
3023 # Meaning of tmpdir has changed since 1.3: we used to create
3023 # Meaning of tmpdir has changed since 1.3: we used to create
3024 # HGTMP inside tmpdir; now HGTMP is tmpdir. So fail if
3024 # HGTMP inside tmpdir; now HGTMP is tmpdir. So fail if
3025 # tmpdir already exists.
3025 # tmpdir already exists.
3026 print("error: temp dir %r already exists" % tmpdir)
3026 print("error: temp dir %r already exists" % tmpdir)
3027 return 1
3027 return 1
3028
3028
3029 os.makedirs(tmpdir)
3029 os.makedirs(tmpdir)
3030 else:
3030 else:
3031 d = None
3031 d = None
3032 if os.name == 'nt':
3032 if os.name == 'nt':
3033 # without this, we get the default temp dir location, but
3033 # without this, we get the default temp dir location, but
3034 # in all lowercase, which causes troubles with paths (issue3490)
3034 # in all lowercase, which causes troubles with paths (issue3490)
3035 d = osenvironb.get(b'TMP', None)
3035 d = osenvironb.get(b'TMP', None)
3036 tmpdir = tempfile.mkdtemp(b'', b'hgtests.', d)
3036 tmpdir = tempfile.mkdtemp(b'', b'hgtests.', d)
3037
3037
3038 self._hgtmp = osenvironb[b'HGTMP'] = os.path.realpath(tmpdir)
3038 self._hgtmp = osenvironb[b'HGTMP'] = os.path.realpath(tmpdir)
3039
3039
3040 if self.options.with_hg:
3040 if self.options.with_hg:
3041 self._installdir = None
3041 self._installdir = None
3042 whg = self.options.with_hg
3042 whg = self.options.with_hg
3043 self._bindir = os.path.dirname(os.path.realpath(whg))
3043 self._bindir = os.path.dirname(os.path.realpath(whg))
3044 assert isinstance(self._bindir, bytes)
3044 assert isinstance(self._bindir, bytes)
3045 self._hgcommand = os.path.basename(whg)
3045 self._hgcommand = os.path.basename(whg)
3046 self._tmpbindir = os.path.join(self._hgtmp, b'install', b'bin')
3046 self._tmpbindir = os.path.join(self._hgtmp, b'install', b'bin')
3047 os.makedirs(self._tmpbindir)
3047 os.makedirs(self._tmpbindir)
3048
3048
3049 normbin = os.path.normpath(os.path.abspath(whg))
3049 normbin = os.path.normpath(os.path.abspath(whg))
3050 normbin = normbin.replace(_sys2bytes(os.sep), b'/')
3050 normbin = normbin.replace(_sys2bytes(os.sep), b'/')
3051
3051
3052 # Other Python scripts in the test harness need to
3052 # Other Python scripts in the test harness need to
3053 # `import mercurial`. If `hg` is a Python script, we assume
3053 # `import mercurial`. If `hg` is a Python script, we assume
3054 # the Mercurial modules are relative to its path and tell the tests
3054 # the Mercurial modules are relative to its path and tell the tests
3055 # to load Python modules from its directory.
3055 # to load Python modules from its directory.
3056 with open(whg, 'rb') as fh:
3056 with open(whg, 'rb') as fh:
3057 initial = fh.read(1024)
3057 initial = fh.read(1024)
3058
3058
3059 if re.match(b'#!.*python', initial):
3059 if re.match(b'#!.*python', initial):
3060 self._pythondir = self._bindir
3060 self._pythondir = self._bindir
3061 # If it looks like our in-repo Rust binary, use the source root.
3061 # If it looks like our in-repo Rust binary, use the source root.
3062 # This is a bit hacky. But rhg is still not supported outside the
3062 # This is a bit hacky. But rhg is still not supported outside the
3063 # source directory. So until it is, do the simple thing.
3063 # source directory. So until it is, do the simple thing.
3064 elif re.search(b'/rust/target/[^/]+/hg', normbin):
3064 elif re.search(b'/rust/target/[^/]+/hg', normbin):
3065 self._pythondir = os.path.dirname(self._testdir)
3065 self._pythondir = os.path.dirname(self._testdir)
3066 # Fall back to the legacy behavior.
3066 # Fall back to the legacy behavior.
3067 else:
3067 else:
3068 self._pythondir = self._bindir
3068 self._pythondir = self._bindir
3069
3069
3070 else:
3070 else:
3071 self._installdir = os.path.join(self._hgtmp, b"install")
3071 self._installdir = os.path.join(self._hgtmp, b"install")
3072 self._bindir = os.path.join(self._installdir, b"bin")
3072 self._bindir = os.path.join(self._installdir, b"bin")
3073 self._hgcommand = b'hg'
3073 self._hgcommand = b'hg'
3074 self._tmpbindir = self._bindir
3074 self._tmpbindir = self._bindir
3075 self._pythondir = os.path.join(self._installdir, b"lib", b"python")
3075 self._pythondir = os.path.join(self._installdir, b"lib", b"python")
3076
3076
3077 # Force the use of hg.exe instead of relying on MSYS to recognize hg is
3077 # Force the use of hg.exe instead of relying on MSYS to recognize hg is
3078 # a python script and feed it to python.exe. Legacy stdio is force
3078 # a python script and feed it to python.exe. Legacy stdio is force
3079 # enabled by hg.exe, and this is a more realistic way to launch hg
3079 # enabled by hg.exe, and this is a more realistic way to launch hg
3080 # anyway.
3080 # anyway.
3081 if os.name == 'nt' and not self._hgcommand.endswith(b'.exe'):
3081 if os.name == 'nt' and not self._hgcommand.endswith(b'.exe'):
3082 self._hgcommand += b'.exe'
3082 self._hgcommand += b'.exe'
3083
3083
3084 # set CHGHG, then replace "hg" command by "chg"
3084 # set CHGHG, then replace "hg" command by "chg"
3085 chgbindir = self._bindir
3085 chgbindir = self._bindir
3086 if self.options.chg or self.options.with_chg:
3086 if self.options.chg or self.options.with_chg:
3087 osenvironb[b'CHGHG'] = os.path.join(self._bindir, self._hgcommand)
3087 osenvironb[b'CHGHG'] = os.path.join(self._bindir, self._hgcommand)
3088 else:
3088 else:
3089 osenvironb.pop(b'CHGHG', None) # drop flag for hghave
3089 osenvironb.pop(b'CHGHG', None) # drop flag for hghave
3090 if self.options.chg:
3090 if self.options.chg:
3091 self._hgcommand = b'chg'
3091 self._hgcommand = b'chg'
3092 elif self.options.with_chg:
3092 elif self.options.with_chg:
3093 chgbindir = os.path.dirname(os.path.realpath(self.options.with_chg))
3093 chgbindir = os.path.dirname(os.path.realpath(self.options.with_chg))
3094 self._hgcommand = os.path.basename(self.options.with_chg)
3094 self._hgcommand = os.path.basename(self.options.with_chg)
3095
3095
3096 osenvironb[b"BINDIR"] = self._bindir
3096 osenvironb[b"BINDIR"] = self._bindir
3097 osenvironb[b"PYTHON"] = PYTHON
3097 osenvironb[b"PYTHON"] = PYTHON
3098
3098
3099 fileb = _sys2bytes(__file__)
3099 fileb = _sys2bytes(__file__)
3100 runtestdir = os.path.abspath(os.path.dirname(fileb))
3100 runtestdir = os.path.abspath(os.path.dirname(fileb))
3101 osenvironb[b'RUNTESTDIR'] = runtestdir
3101 osenvironb[b'RUNTESTDIR'] = runtestdir
3102 if PYTHON3:
3102 if PYTHON3:
3103 sepb = _sys2bytes(os.pathsep)
3103 sepb = _sys2bytes(os.pathsep)
3104 else:
3104 else:
3105 sepb = os.pathsep
3105 sepb = os.pathsep
3106 path = [self._bindir, runtestdir] + osenvironb[b"PATH"].split(sepb)
3106 path = [self._bindir, runtestdir] + osenvironb[b"PATH"].split(sepb)
3107 if os.path.islink(__file__):
3107 if os.path.islink(__file__):
3108 # test helper will likely be at the end of the symlink
3108 # test helper will likely be at the end of the symlink
3109 realfile = os.path.realpath(fileb)
3109 realfile = os.path.realpath(fileb)
3110 realdir = os.path.abspath(os.path.dirname(realfile))
3110 realdir = os.path.abspath(os.path.dirname(realfile))
3111 path.insert(2, realdir)
3111 path.insert(2, realdir)
3112 if chgbindir != self._bindir:
3112 if chgbindir != self._bindir:
3113 path.insert(1, chgbindir)
3113 path.insert(1, chgbindir)
3114 if self._testdir != runtestdir:
3114 if self._testdir != runtestdir:
3115 path = [self._testdir] + path
3115 path = [self._testdir] + path
3116 if self._tmpbindir != self._bindir:
3116 if self._tmpbindir != self._bindir:
3117 path = [self._tmpbindir] + path
3117 path = [self._tmpbindir] + path
3118 osenvironb[b"PATH"] = sepb.join(path)
3118 osenvironb[b"PATH"] = sepb.join(path)
3119
3119
3120 # Include TESTDIR in PYTHONPATH so that out-of-tree extensions
3120 # Include TESTDIR in PYTHONPATH so that out-of-tree extensions
3121 # can run .../tests/run-tests.py test-foo where test-foo
3121 # can run .../tests/run-tests.py test-foo where test-foo
3122 # adds an extension to HGRC. Also include run-test.py directory to
3122 # adds an extension to HGRC. Also include run-test.py directory to
3123 # import modules like heredoctest.
3123 # import modules like heredoctest.
3124 pypath = [self._pythondir, self._testdir, runtestdir]
3124 pypath = [self._pythondir, self._testdir, runtestdir]
3125 # We have to augment PYTHONPATH, rather than simply replacing
3125 # We have to augment PYTHONPATH, rather than simply replacing
3126 # it, in case external libraries are only available via current
3126 # it, in case external libraries are only available via current
3127 # PYTHONPATH. (In particular, the Subversion bindings on OS X
3127 # PYTHONPATH. (In particular, the Subversion bindings on OS X
3128 # are in /opt/subversion.)
3128 # are in /opt/subversion.)
3129 oldpypath = osenvironb.get(IMPL_PATH)
3129 oldpypath = osenvironb.get(IMPL_PATH)
3130 if oldpypath:
3130 if oldpypath:
3131 pypath.append(oldpypath)
3131 pypath.append(oldpypath)
3132 osenvironb[IMPL_PATH] = sepb.join(pypath)
3132 osenvironb[IMPL_PATH] = sepb.join(pypath)
3133
3133
3134 if self.options.pure:
3134 if self.options.pure:
3135 os.environ["HGTEST_RUN_TESTS_PURE"] = "--pure"
3135 os.environ["HGTEST_RUN_TESTS_PURE"] = "--pure"
3136 os.environ["HGMODULEPOLICY"] = "py"
3136 os.environ["HGMODULEPOLICY"] = "py"
3137 if self.options.rust:
3137 if self.options.rust:
3138 os.environ["HGMODULEPOLICY"] = "rust+c"
3138 os.environ["HGMODULEPOLICY"] = "rust+c"
3139 if self.options.no_rust:
3139 if self.options.no_rust:
3140 current_policy = os.environ.get("HGMODULEPOLICY", "")
3140 current_policy = os.environ.get("HGMODULEPOLICY", "")
3141 if current_policy.startswith("rust+"):
3141 if current_policy.startswith("rust+"):
3142 os.environ["HGMODULEPOLICY"] = current_policy[len("rust+") :]
3142 os.environ["HGMODULEPOLICY"] = current_policy[len("rust+") :]
3143 os.environ.pop("HGWITHRUSTEXT", None)
3143 os.environ.pop("HGWITHRUSTEXT", None)
3144
3144
3145 if self.options.allow_slow_tests:
3145 if self.options.allow_slow_tests:
3146 os.environ["HGTEST_SLOW"] = "slow"
3146 os.environ["HGTEST_SLOW"] = "slow"
3147 elif 'HGTEST_SLOW' in os.environ:
3147 elif 'HGTEST_SLOW' in os.environ:
3148 del os.environ['HGTEST_SLOW']
3148 del os.environ['HGTEST_SLOW']
3149
3149
3150 self._coveragefile = os.path.join(self._testdir, b'.coverage')
3150 self._coveragefile = os.path.join(self._testdir, b'.coverage')
3151
3151
3152 if self.options.exceptions:
3152 if self.options.exceptions:
3153 exceptionsdir = os.path.join(self._outputdir, b'exceptions')
3153 exceptionsdir = os.path.join(self._outputdir, b'exceptions')
3154 try:
3154 try:
3155 os.makedirs(exceptionsdir)
3155 os.makedirs(exceptionsdir)
3156 except OSError as e:
3156 except OSError as e:
3157 if e.errno != errno.EEXIST:
3157 if e.errno != errno.EEXIST:
3158 raise
3158 raise
3159
3159
3160 # Remove all existing exception reports.
3160 # Remove all existing exception reports.
3161 for f in os.listdir(exceptionsdir):
3161 for f in os.listdir(exceptionsdir):
3162 os.unlink(os.path.join(exceptionsdir, f))
3162 os.unlink(os.path.join(exceptionsdir, f))
3163
3163
3164 osenvironb[b'HGEXCEPTIONSDIR'] = exceptionsdir
3164 osenvironb[b'HGEXCEPTIONSDIR'] = exceptionsdir
3165 logexceptions = os.path.join(self._testdir, b'logexceptions.py')
3165 logexceptions = os.path.join(self._testdir, b'logexceptions.py')
3166 self.options.extra_config_opt.append(
3166 self.options.extra_config_opt.append(
3167 'extensions.logexceptions=%s' % logexceptions.decode('utf-8')
3167 'extensions.logexceptions=%s' % logexceptions.decode('utf-8')
3168 )
3168 )
3169
3169
3170 vlog("# Using TESTDIR", _bytes2sys(self._testdir))
3170 vlog("# Using TESTDIR", _bytes2sys(self._testdir))
3171 vlog("# Using RUNTESTDIR", _bytes2sys(osenvironb[b'RUNTESTDIR']))
3171 vlog("# Using RUNTESTDIR", _bytes2sys(osenvironb[b'RUNTESTDIR']))
3172 vlog("# Using HGTMP", _bytes2sys(self._hgtmp))
3172 vlog("# Using HGTMP", _bytes2sys(self._hgtmp))
3173 vlog("# Using PATH", os.environ["PATH"])
3173 vlog("# Using PATH", os.environ["PATH"])
3174 vlog(
3174 vlog(
3175 "# Using",
3175 "# Using",
3176 _bytes2sys(IMPL_PATH),
3176 _bytes2sys(IMPL_PATH),
3177 _bytes2sys(osenvironb[IMPL_PATH]),
3177 _bytes2sys(osenvironb[IMPL_PATH]),
3178 )
3178 )
3179 vlog("# Writing to directory", _bytes2sys(self._outputdir))
3179 vlog("# Writing to directory", _bytes2sys(self._outputdir))
3180
3180
3181 try:
3181 try:
3182 return self._runtests(testdescs) or 0
3182 return self._runtests(testdescs) or 0
3183 finally:
3183 finally:
3184 time.sleep(0.1)
3184 time.sleep(0.1)
3185 self._cleanup()
3185 self._cleanup()
3186
3186
3187 def findtests(self, args):
3187 def findtests(self, args):
3188 """Finds possible test files from arguments.
3188 """Finds possible test files from arguments.
3189
3189
3190 If you wish to inject custom tests into the test harness, this would
3190 If you wish to inject custom tests into the test harness, this would
3191 be a good function to monkeypatch or override in a derived class.
3191 be a good function to monkeypatch or override in a derived class.
3192 """
3192 """
3193 if not args:
3193 if not args:
3194 if self.options.changed:
3194 if self.options.changed:
3195 proc = Popen4(
3195 proc = Popen4(
3196 b'hg st --rev "%s" -man0 .'
3196 b'hg st --rev "%s" -man0 .'
3197 % _sys2bytes(self.options.changed),
3197 % _sys2bytes(self.options.changed),
3198 None,
3198 None,
3199 0,
3199 0,
3200 )
3200 )
3201 stdout, stderr = proc.communicate()
3201 stdout, stderr = proc.communicate()
3202 args = stdout.strip(b'\0').split(b'\0')
3202 args = stdout.strip(b'\0').split(b'\0')
3203 else:
3203 else:
3204 args = os.listdir(b'.')
3204 args = os.listdir(b'.')
3205
3205
3206 expanded_args = []
3206 expanded_args = []
3207 for arg in args:
3207 for arg in args:
3208 if os.path.isdir(arg):
3208 if os.path.isdir(arg):
3209 if not arg.endswith(b'/'):
3209 if not arg.endswith(b'/'):
3210 arg += b'/'
3210 arg += b'/'
3211 expanded_args.extend([arg + a for a in os.listdir(arg)])
3211 expanded_args.extend([arg + a for a in os.listdir(arg)])
3212 else:
3212 else:
3213 expanded_args.append(arg)
3213 expanded_args.append(arg)
3214 args = expanded_args
3214 args = expanded_args
3215
3215
3216 testcasepattern = re.compile(br'([\w-]+\.t|py)(?:#([a-zA-Z0-9_\-.#]+))')
3216 testcasepattern = re.compile(br'([\w-]+\.t|py)(?:#([a-zA-Z0-9_\-.#]+))')
3217 tests = []
3217 tests = []
3218 for t in args:
3218 for t in args:
3219 case = []
3219 case = []
3220
3220
3221 if not (
3221 if not (
3222 os.path.basename(t).startswith(b'test-')
3222 os.path.basename(t).startswith(b'test-')
3223 and (t.endswith(b'.py') or t.endswith(b'.t'))
3223 and (t.endswith(b'.py') or t.endswith(b'.t'))
3224 ):
3224 ):
3225
3225
3226 m = testcasepattern.match(os.path.basename(t))
3226 m = testcasepattern.match(os.path.basename(t))
3227 if m is not None:
3227 if m is not None:
3228 t_basename, casestr = m.groups()
3228 t_basename, casestr = m.groups()
3229 t = os.path.join(os.path.dirname(t), t_basename)
3229 t = os.path.join(os.path.dirname(t), t_basename)
3230 if casestr:
3230 if casestr:
3231 case = casestr.split(b'#')
3231 case = casestr.split(b'#')
3232 else:
3232 else:
3233 continue
3233 continue
3234
3234
3235 if t.endswith(b'.t'):
3235 if t.endswith(b'.t'):
3236 # .t file may contain multiple test cases
3236 # .t file may contain multiple test cases
3237 casedimensions = parsettestcases(t)
3237 casedimensions = parsettestcases(t)
3238 if casedimensions:
3238 if casedimensions:
3239 cases = []
3239 cases = []
3240
3240
3241 def addcases(case, casedimensions):
3241 def addcases(case, casedimensions):
3242 if not casedimensions:
3242 if not casedimensions:
3243 cases.append(case)
3243 cases.append(case)
3244 else:
3244 else:
3245 for c in casedimensions[0]:
3245 for c in casedimensions[0]:
3246 addcases(case + [c], casedimensions[1:])
3246 addcases(case + [c], casedimensions[1:])
3247
3247
3248 addcases([], casedimensions)
3248 addcases([], casedimensions)
3249 if case and case in cases:
3249 if case and case in cases:
3250 cases = [case]
3250 cases = [case]
3251 elif case:
3251 elif case:
3252 # Ignore invalid cases
3252 # Ignore invalid cases
3253 cases = []
3253 cases = []
3254 else:
3254 else:
3255 pass
3255 pass
3256 tests += [{'path': t, 'case': c} for c in sorted(cases)]
3256 tests += [{'path': t, 'case': c} for c in sorted(cases)]
3257 else:
3257 else:
3258 tests.append({'path': t})
3258 tests.append({'path': t})
3259 else:
3259 else:
3260 tests.append({'path': t})
3260 tests.append({'path': t})
3261
3261
3262 if self.options.retest:
3262 if self.options.retest:
3263 retest_args = []
3263 retest_args = []
3264 for test in tests:
3264 for test in tests:
3265 errpath = self._geterrpath(test)
3265 errpath = self._geterrpath(test)
3266 if os.path.exists(errpath):
3266 if os.path.exists(errpath):
3267 retest_args.append(test)
3267 retest_args.append(test)
3268 tests = retest_args
3268 tests = retest_args
3269 return tests
3269 return tests
3270
3270
3271 def _runtests(self, testdescs):
3271 def _runtests(self, testdescs):
3272 def _reloadtest(test, i):
3272 def _reloadtest(test, i):
3273 # convert a test back to its description dict
3273 # convert a test back to its description dict
3274 desc = {'path': test.path}
3274 desc = {'path': test.path}
3275 case = getattr(test, '_case', [])
3275 case = getattr(test, '_case', [])
3276 if case:
3276 if case:
3277 desc['case'] = case
3277 desc['case'] = case
3278 return self._gettest(desc, i)
3278 return self._gettest(desc, i)
3279
3279
3280 try:
3280 try:
3281 if self.options.restart:
3281 if self.options.restart:
3282 orig = list(testdescs)
3282 orig = list(testdescs)
3283 while testdescs:
3283 while testdescs:
3284 desc = testdescs[0]
3284 desc = testdescs[0]
3285 errpath = self._geterrpath(desc)
3285 errpath = self._geterrpath(desc)
3286 if os.path.exists(errpath):
3286 if os.path.exists(errpath):
3287 break
3287 break
3288 testdescs.pop(0)
3288 testdescs.pop(0)
3289 if not testdescs:
3289 if not testdescs:
3290 print("running all tests")
3290 print("running all tests")
3291 testdescs = orig
3291 testdescs = orig
3292
3292
3293 tests = [self._gettest(d, i) for i, d in enumerate(testdescs)]
3293 tests = [self._gettest(d, i) for i, d in enumerate(testdescs)]
3294 num_tests = len(tests) * self.options.runs_per_test
3294 num_tests = len(tests) * self.options.runs_per_test
3295
3295
3296 jobs = min(num_tests, self.options.jobs)
3296 jobs = min(num_tests, self.options.jobs)
3297
3297
3298 failed = False
3298 failed = False
3299 kws = self.options.keywords
3299 kws = self.options.keywords
3300 if kws is not None and PYTHON3:
3300 if kws is not None and PYTHON3:
3301 kws = kws.encode('utf-8')
3301 kws = kws.encode('utf-8')
3302
3302
3303 suite = TestSuite(
3303 suite = TestSuite(
3304 self._testdir,
3304 self._testdir,
3305 jobs=jobs,
3305 jobs=jobs,
3306 whitelist=self.options.whitelisted,
3306 whitelist=self.options.whitelisted,
3307 blacklist=self.options.blacklist,
3307 blacklist=self.options.blacklist,
3308 keywords=kws,
3308 keywords=kws,
3309 loop=self.options.loop,
3309 loop=self.options.loop,
3310 runs_per_test=self.options.runs_per_test,
3310 runs_per_test=self.options.runs_per_test,
3311 showchannels=self.options.showchannels,
3311 showchannels=self.options.showchannels,
3312 tests=tests,
3312 tests=tests,
3313 loadtest=_reloadtest,
3313 loadtest=_reloadtest,
3314 )
3314 )
3315 verbosity = 1
3315 verbosity = 1
3316 if self.options.list_tests:
3316 if self.options.list_tests:
3317 verbosity = 0
3317 verbosity = 0
3318 elif self.options.verbose:
3318 elif self.options.verbose:
3319 verbosity = 2
3319 verbosity = 2
3320 runner = TextTestRunner(self, verbosity=verbosity)
3320 runner = TextTestRunner(self, verbosity=verbosity)
3321
3321
3322 if self.options.list_tests:
3322 if self.options.list_tests:
3323 result = runner.listtests(suite)
3323 result = runner.listtests(suite)
3324 else:
3324 else:
3325 if self._installdir:
3325 if self._installdir:
3326 self._installhg()
3326 self._installhg()
3327 self._checkhglib("Testing")
3327 self._checkhglib("Testing")
3328 else:
3328 else:
3329 self._usecorrectpython()
3329 self._usecorrectpython()
3330 if self.options.chg:
3330 if self.options.chg:
3331 assert self._installdir
3331 assert self._installdir
3332 self._installchg()
3332 self._installchg()
3333
3333
3334 log(
3334 log(
3335 'running %d tests using %d parallel processes'
3335 'running %d tests using %d parallel processes'
3336 % (num_tests, jobs)
3336 % (num_tests, jobs)
3337 )
3337 )
3338
3338
3339 result = runner.run(suite)
3339 result = runner.run(suite)
3340
3340
3341 if result.failures or result.errors:
3341 if result.failures or result.errors:
3342 failed = True
3342 failed = True
3343
3343
3344 result.onEnd()
3344 result.onEnd()
3345
3345
3346 if self.options.anycoverage:
3346 if self.options.anycoverage:
3347 self._outputcoverage()
3347 self._outputcoverage()
3348 except KeyboardInterrupt:
3348 except KeyboardInterrupt:
3349 failed = True
3349 failed = True
3350 print("\ninterrupted!")
3350 print("\ninterrupted!")
3351
3351
3352 if failed:
3352 if failed:
3353 return 1
3353 return 1
3354
3354
3355 def _geterrpath(self, test):
3355 def _geterrpath(self, test):
3356 # test['path'] is a relative path
3356 # test['path'] is a relative path
3357 if 'case' in test:
3357 if 'case' in test:
3358 # for multiple dimensions test cases
3358 # for multiple dimensions test cases
3359 casestr = b'#'.join(test['case'])
3359 casestr = b'#'.join(test['case'])
3360 errpath = b'%s#%s.err' % (test['path'], casestr)
3360 errpath = b'%s#%s.err' % (test['path'], casestr)
3361 else:
3361 else:
3362 errpath = b'%s.err' % test['path']
3362 errpath = b'%s.err' % test['path']
3363 if self.options.outputdir:
3363 if self.options.outputdir:
3364 self._outputdir = canonpath(_sys2bytes(self.options.outputdir))
3364 self._outputdir = canonpath(_sys2bytes(self.options.outputdir))
3365 errpath = os.path.join(self._outputdir, errpath)
3365 errpath = os.path.join(self._outputdir, errpath)
3366 return errpath
3366 return errpath
3367
3367
3368 def _getport(self, count):
3368 def _getport(self, count):
3369 port = self._ports.get(count) # do we have a cached entry?
3369 port = self._ports.get(count) # do we have a cached entry?
3370 if port is None:
3370 if port is None:
3371 portneeded = 3
3371 portneeded = 3
3372 # above 100 tries we just give up and let test reports failure
3372 # above 100 tries we just give up and let test reports failure
3373 for tries in xrange(100):
3373 for tries in xrange(100):
3374 allfree = True
3374 allfree = True
3375 port = self.options.port + self._portoffset
3375 port = self.options.port + self._portoffset
3376 for idx in xrange(portneeded):
3376 for idx in xrange(portneeded):
3377 if not checkportisavailable(port + idx):
3377 if not checkportisavailable(port + idx):
3378 allfree = False
3378 allfree = False
3379 break
3379 break
3380 self._portoffset += portneeded
3380 self._portoffset += portneeded
3381 if allfree:
3381 if allfree:
3382 break
3382 break
3383 self._ports[count] = port
3383 self._ports[count] = port
3384 return port
3384 return port
3385
3385
3386 def _gettest(self, testdesc, count):
3386 def _gettest(self, testdesc, count):
3387 """Obtain a Test by looking at its filename.
3387 """Obtain a Test by looking at its filename.
3388
3388
3389 Returns a Test instance. The Test may not be runnable if it doesn't
3389 Returns a Test instance. The Test may not be runnable if it doesn't
3390 map to a known type.
3390 map to a known type.
3391 """
3391 """
3392 path = testdesc['path']
3392 path = testdesc['path']
3393 lctest = path.lower()
3393 lctest = path.lower()
3394 testcls = Test
3394 testcls = Test
3395
3395
3396 for ext, cls in self.TESTTYPES:
3396 for ext, cls in self.TESTTYPES:
3397 if lctest.endswith(ext):
3397 if lctest.endswith(ext):
3398 testcls = cls
3398 testcls = cls
3399 break
3399 break
3400
3400
3401 refpath = os.path.join(getcwdb(), path)
3401 refpath = os.path.join(getcwdb(), path)
3402 tmpdir = os.path.join(self._hgtmp, b'child%d' % count)
3402 tmpdir = os.path.join(self._hgtmp, b'child%d' % count)
3403
3403
3404 # extra keyword parameters. 'case' is used by .t tests
3404 # extra keyword parameters. 'case' is used by .t tests
3405 kwds = {k: testdesc[k] for k in ['case'] if k in testdesc}
3405 kwds = {k: testdesc[k] for k in ['case'] if k in testdesc}
3406
3406
3407 t = testcls(
3407 t = testcls(
3408 refpath,
3408 refpath,
3409 self._outputdir,
3409 self._outputdir,
3410 tmpdir,
3410 tmpdir,
3411 keeptmpdir=self.options.keep_tmpdir,
3411 keeptmpdir=self.options.keep_tmpdir,
3412 debug=self.options.debug,
3412 debug=self.options.debug,
3413 first=self.options.first,
3413 first=self.options.first,
3414 timeout=self.options.timeout,
3414 timeout=self.options.timeout,
3415 startport=self._getport(count),
3415 startport=self._getport(count),
3416 extraconfigopts=self.options.extra_config_opt,
3416 extraconfigopts=self.options.extra_config_opt,
3417 shell=self.options.shell,
3417 shell=self.options.shell,
3418 hgcommand=self._hgcommand,
3418 hgcommand=self._hgcommand,
3419 usechg=bool(self.options.with_chg or self.options.chg),
3419 usechg=bool(self.options.with_chg or self.options.chg),
3420 chgdebug=self.options.chg_debug,
3420 chgdebug=self.options.chg_debug,
3421 useipv6=useipv6,
3421 useipv6=useipv6,
3422 **kwds
3422 **kwds
3423 )
3423 )
3424 t.should_reload = True
3424 t.should_reload = True
3425 return t
3425 return t
3426
3426
3427 def _cleanup(self):
3427 def _cleanup(self):
3428 """Clean up state from this test invocation."""
3428 """Clean up state from this test invocation."""
3429 if self.options.keep_tmpdir:
3429 if self.options.keep_tmpdir:
3430 return
3430 return
3431
3431
3432 vlog("# Cleaning up HGTMP", _bytes2sys(self._hgtmp))
3432 vlog("# Cleaning up HGTMP", _bytes2sys(self._hgtmp))
3433 shutil.rmtree(self._hgtmp, True)
3433 shutil.rmtree(self._hgtmp, True)
3434 for f in self._createdfiles:
3434 for f in self._createdfiles:
3435 try:
3435 try:
3436 os.remove(f)
3436 os.remove(f)
3437 except OSError:
3437 except OSError:
3438 pass
3438 pass
3439
3439
3440 def _usecorrectpython(self):
3440 def _usecorrectpython(self):
3441 """Configure the environment to use the appropriate Python in tests."""
3441 """Configure the environment to use the appropriate Python in tests."""
3442 # Tests must use the same interpreter as us or bad things will happen.
3442 # Tests must use the same interpreter as us or bad things will happen.
3443 pyexename = sys.platform == 'win32' and b'python.exe' or b'python3'
3443 pyexename = sys.platform == 'win32' and b'python.exe' or b'python3'
3444
3444
3445 # os.symlink() is a thing with py3 on Windows, but it requires
3445 # os.symlink() is a thing with py3 on Windows, but it requires
3446 # Administrator rights.
3446 # Administrator rights.
3447 if getattr(os, 'symlink', None) and os.name != 'nt':
3447 if getattr(os, 'symlink', None) and os.name != 'nt':
3448 vlog(
3448 vlog(
3449 "# Making python executable in test path a symlink to '%s'"
3449 "# Making python executable in test path a symlink to '%s'"
3450 % sysexecutable
3450 % sysexecutable
3451 )
3451 )
3452 mypython = os.path.join(self._tmpbindir, pyexename)
3452 mypython = os.path.join(self._tmpbindir, pyexename)
3453 try:
3453 try:
3454 if os.readlink(mypython) == sysexecutable:
3454 if os.readlink(mypython) == sysexecutable:
3455 return
3455 return
3456 os.unlink(mypython)
3456 os.unlink(mypython)
3457 except OSError as err:
3457 except OSError as err:
3458 if err.errno != errno.ENOENT:
3458 if err.errno != errno.ENOENT:
3459 raise
3459 raise
3460 if self._findprogram(pyexename) != sysexecutable:
3460 if self._findprogram(pyexename) != sysexecutable:
3461 try:
3461 try:
3462 os.symlink(sysexecutable, mypython)
3462 os.symlink(sysexecutable, mypython)
3463 self._createdfiles.append(mypython)
3463 self._createdfiles.append(mypython)
3464 except OSError as err:
3464 except OSError as err:
3465 # child processes may race, which is harmless
3465 # child processes may race, which is harmless
3466 if err.errno != errno.EEXIST:
3466 if err.errno != errno.EEXIST:
3467 raise
3467 raise
3468 else:
3468 else:
3469 # Windows doesn't have `python3.exe`, and MSYS cannot understand the
3469 # Windows doesn't have `python3.exe`, and MSYS cannot understand the
3470 # reparse point with that name provided by Microsoft. Copy the
3470 # reparse point with that name provided by Microsoft. Copy the
3471 # current interpreter to PATH with that name so the shebang lines
3471 # current interpreter to PATH with that name so the shebang lines
3472 # work.
3472 # work.
3473 if os.getenv('MSYSTEM'):
3473 if os.getenv('MSYSTEM'):
3474 shutil.copy(
3474 shutil.copy(
3475 sys.executable,
3475 sys.executable,
3476 _bytes2sys(self._tmpbindir + b'/python3.exe'),
3476 _bytes2sys(self._tmpbindir + b'/python3.exe'),
3477 )
3477 )
3478
3478
3479 exedir, exename = os.path.split(sysexecutable)
3479 exedir, exename = os.path.split(sysexecutable)
3480 vlog(
3480 vlog(
3481 "# Modifying search path to find %s as %s in '%s'"
3481 "# Modifying search path to find %s as %s in '%s'"
3482 % (exename, pyexename, exedir)
3482 % (exename, pyexename, exedir)
3483 )
3483 )
3484 path = os.environ['PATH'].split(os.pathsep)
3484 path = os.environ['PATH'].split(os.pathsep)
3485 while exedir in path:
3485 while exedir in path:
3486 path.remove(exedir)
3486 path.remove(exedir)
3487
3487
3488 # Binaries installed by pip into the user area like pylint.exe may
3488 # Binaries installed by pip into the user area like pylint.exe may
3489 # not be in PATH by default.
3489 # not be in PATH by default.
3490 extra_paths = [exedir]
3490 extra_paths = [exedir]
3491 vi = sys.version_info
3491 vi = sys.version_info
3492 if 'APPDATA' in os.environ:
3492 if 'APPDATA' in os.environ:
3493 scripts_dir = os.path.join(
3493 scripts_dir = os.path.join(
3494 os.environ['APPDATA'],
3494 os.environ['APPDATA'],
3495 'Python',
3495 'Python',
3496 'Python%d%d' % (vi[0], vi[1]),
3496 'Python%d%d' % (vi[0], vi[1]),
3497 'Scripts',
3497 'Scripts',
3498 )
3498 )
3499
3499
3500 if vi.major == 2:
3500 if vi.major == 2:
3501 scripts_dir = os.path.join(
3501 scripts_dir = os.path.join(
3502 os.environ['APPDATA'],
3502 os.environ['APPDATA'],
3503 'Python',
3503 'Python',
3504 'Scripts',
3504 'Scripts',
3505 )
3505 )
3506
3506
3507 extra_paths.append(scripts_dir)
3507 extra_paths.append(scripts_dir)
3508
3508
3509 os.environ['PATH'] = os.pathsep.join(extra_paths + path)
3509 os.environ['PATH'] = os.pathsep.join(extra_paths + path)
3510 if not self._findprogram(pyexename):
3510 if not self._findprogram(pyexename):
3511 print("WARNING: Cannot find %s in search path" % pyexename)
3511 print("WARNING: Cannot find %s in search path" % pyexename)
3512
3512
3513 def _installhg(self):
3513 def _installhg(self):
3514 """Install hg into the test environment.
3514 """Install hg into the test environment.
3515
3515
3516 This will also configure hg with the appropriate testing settings.
3516 This will also configure hg with the appropriate testing settings.
3517 """
3517 """
3518 vlog("# Performing temporary installation of HG")
3518 vlog("# Performing temporary installation of HG")
3519 installerrs = os.path.join(self._hgtmp, b"install.err")
3519 installerrs = os.path.join(self._hgtmp, b"install.err")
3520 compiler = ''
3520 compiler = ''
3521 if self.options.compiler:
3521 if self.options.compiler:
3522 compiler = '--compiler ' + self.options.compiler
3522 compiler = '--compiler ' + self.options.compiler
3523 setup_opts = b""
3523 setup_opts = b""
3524 if self.options.pure:
3524 if self.options.pure:
3525 setup_opts = b"--pure"
3525 setup_opts = b"--pure"
3526 elif self.options.rust:
3526 elif self.options.rust:
3527 setup_opts = b"--rust"
3527 setup_opts = b"--rust"
3528 elif self.options.no_rust:
3528 elif self.options.no_rust:
3529 setup_opts = b"--no-rust"
3529 setup_opts = b"--no-rust"
3530
3530
3531 # Run installer in hg root
3531 # Run installer in hg root
3532 script = os.path.realpath(sys.argv[0])
3532 script = os.path.realpath(sys.argv[0])
3533 exe = sysexecutable
3533 exe = sysexecutable
3534 if PYTHON3:
3534 if PYTHON3:
3535 compiler = _sys2bytes(compiler)
3535 compiler = _sys2bytes(compiler)
3536 script = _sys2bytes(script)
3536 script = _sys2bytes(script)
3537 exe = _sys2bytes(exe)
3537 exe = _sys2bytes(exe)
3538 hgroot = os.path.dirname(os.path.dirname(script))
3538 hgroot = os.path.dirname(os.path.dirname(script))
3539 self._hgroot = hgroot
3539 self._hgroot = hgroot
3540 os.chdir(hgroot)
3540 os.chdir(hgroot)
3541 nohome = b'--home=""'
3541 nohome = b'--home=""'
3542 if os.name == 'nt':
3542 if os.name == 'nt':
3543 # The --home="" trick works only on OS where os.sep == '/'
3543 # The --home="" trick works only on OS where os.sep == '/'
3544 # because of a distutils convert_path() fast-path. Avoid it at
3544 # because of a distutils convert_path() fast-path. Avoid it at
3545 # least on Windows for now, deal with .pydistutils.cfg bugs
3545 # least on Windows for now, deal with .pydistutils.cfg bugs
3546 # when they happen.
3546 # when they happen.
3547 nohome = b''
3547 nohome = b''
3548 cmd = (
3548 cmd = (
3549 b'"%(exe)s" setup.py %(setup_opts)s clean --all'
3549 b'"%(exe)s" setup.py %(setup_opts)s clean --all'
3550 b' build %(compiler)s --build-base="%(base)s"'
3550 b' build %(compiler)s --build-base="%(base)s"'
3551 b' install --force --prefix="%(prefix)s"'
3551 b' install --force --prefix="%(prefix)s"'
3552 b' --install-lib="%(libdir)s"'
3552 b' --install-lib="%(libdir)s"'
3553 b' --install-scripts="%(bindir)s" %(nohome)s >%(logfile)s 2>&1'
3553 b' --install-scripts="%(bindir)s" %(nohome)s >%(logfile)s 2>&1'
3554 % {
3554 % {
3555 b'exe': exe,
3555 b'exe': exe,
3556 b'setup_opts': setup_opts,
3556 b'setup_opts': setup_opts,
3557 b'compiler': compiler,
3557 b'compiler': compiler,
3558 b'base': os.path.join(self._hgtmp, b"build"),
3558 b'base': os.path.join(self._hgtmp, b"build"),
3559 b'prefix': self._installdir,
3559 b'prefix': self._installdir,
3560 b'libdir': self._pythondir,
3560 b'libdir': self._pythondir,
3561 b'bindir': self._bindir,
3561 b'bindir': self._bindir,
3562 b'nohome': nohome,
3562 b'nohome': nohome,
3563 b'logfile': installerrs,
3563 b'logfile': installerrs,
3564 }
3564 }
3565 )
3565 )
3566
3566
3567 # setuptools requires install directories to exist.
3567 # setuptools requires install directories to exist.
3568 def makedirs(p):
3568 def makedirs(p):
3569 try:
3569 try:
3570 os.makedirs(p)
3570 os.makedirs(p)
3571 except OSError as e:
3571 except OSError as e:
3572 if e.errno != errno.EEXIST:
3572 if e.errno != errno.EEXIST:
3573 raise
3573 raise
3574
3574
3575 makedirs(self._pythondir)
3575 makedirs(self._pythondir)
3576 makedirs(self._bindir)
3576 makedirs(self._bindir)
3577
3577
3578 vlog("# Running", cmd.decode("utf-8"))
3578 vlog("# Running", cmd.decode("utf-8"))
3579 if subprocess.call(_bytes2sys(cmd), shell=True) == 0:
3579 if subprocess.call(_bytes2sys(cmd), shell=True) == 0:
3580 if not self.options.verbose:
3580 if not self.options.verbose:
3581 try:
3581 try:
3582 os.remove(installerrs)
3582 os.remove(installerrs)
3583 except OSError as e:
3583 except OSError as e:
3584 if e.errno != errno.ENOENT:
3584 if e.errno != errno.ENOENT:
3585 raise
3585 raise
3586 else:
3586 else:
3587 with open(installerrs, 'rb') as f:
3587 with open(installerrs, 'rb') as f:
3588 for line in f:
3588 for line in f:
3589 if PYTHON3:
3589 if PYTHON3:
3590 sys.stdout.buffer.write(line)
3590 sys.stdout.buffer.write(line)
3591 else:
3591 else:
3592 sys.stdout.write(line)
3592 sys.stdout.write(line)
3593 sys.exit(1)
3593 sys.exit(1)
3594 os.chdir(self._testdir)
3594 os.chdir(self._testdir)
3595
3595
3596 self._usecorrectpython()
3596 self._usecorrectpython()
3597
3597
3598 hgbat = os.path.join(self._bindir, b'hg.bat')
3598 hgbat = os.path.join(self._bindir, b'hg.bat')
3599 if os.path.isfile(hgbat):
3599 if os.path.isfile(hgbat):
3600 # hg.bat expects to be put in bin/scripts while run-tests.py
3600 # hg.bat expects to be put in bin/scripts while run-tests.py
3601 # installation layout put it in bin/ directly. Fix it
3601 # installation layout put it in bin/ directly. Fix it
3602 with open(hgbat, 'rb') as f:
3602 with open(hgbat, 'rb') as f:
3603 data = f.read()
3603 data = f.read()
3604 if br'"%~dp0..\python" "%~dp0hg" %*' in data:
3604 if br'"%~dp0..\python" "%~dp0hg" %*' in data:
3605 data = data.replace(
3605 data = data.replace(
3606 br'"%~dp0..\python" "%~dp0hg" %*',
3606 br'"%~dp0..\python" "%~dp0hg" %*',
3607 b'"%~dp0python" "%~dp0hg" %*',
3607 b'"%~dp0python" "%~dp0hg" %*',
3608 )
3608 )
3609 with open(hgbat, 'wb') as f:
3609 with open(hgbat, 'wb') as f:
3610 f.write(data)
3610 f.write(data)
3611 else:
3611 else:
3612 print('WARNING: cannot fix hg.bat reference to python.exe')
3612 print('WARNING: cannot fix hg.bat reference to python.exe')
3613
3613
3614 if self.options.anycoverage:
3614 if self.options.anycoverage:
3615 custom = os.path.join(
3615 custom = os.path.join(
3616 osenvironb[b'RUNTESTDIR'], b'sitecustomize.py'
3616 osenvironb[b'RUNTESTDIR'], b'sitecustomize.py'
3617 )
3617 )
3618 target = os.path.join(self._pythondir, b'sitecustomize.py')
3618 target = os.path.join(self._pythondir, b'sitecustomize.py')
3619 vlog('# Installing coverage trigger to %s' % target)
3619 vlog('# Installing coverage trigger to %s' % target)
3620 shutil.copyfile(custom, target)
3620 shutil.copyfile(custom, target)
3621 rc = os.path.join(self._testdir, b'.coveragerc')
3621 rc = os.path.join(self._testdir, b'.coveragerc')
3622 vlog('# Installing coverage rc to %s' % rc)
3622 vlog('# Installing coverage rc to %s' % rc)
3623 osenvironb[b'COVERAGE_PROCESS_START'] = rc
3623 osenvironb[b'COVERAGE_PROCESS_START'] = rc
3624 covdir = os.path.join(self._installdir, b'..', b'coverage')
3624 covdir = os.path.join(self._installdir, b'..', b'coverage')
3625 try:
3625 try:
3626 os.mkdir(covdir)
3626 os.mkdir(covdir)
3627 except OSError as e:
3627 except OSError as e:
3628 if e.errno != errno.EEXIST:
3628 if e.errno != errno.EEXIST:
3629 raise
3629 raise
3630
3630
3631 osenvironb[b'COVERAGE_DIR'] = covdir
3631 osenvironb[b'COVERAGE_DIR'] = covdir
3632
3632
3633 def _checkhglib(self, verb):
3633 def _checkhglib(self, verb):
3634 """Ensure that the 'mercurial' package imported by python is
3634 """Ensure that the 'mercurial' package imported by python is
3635 the one we expect it to be. If not, print a warning to stderr."""
3635 the one we expect it to be. If not, print a warning to stderr."""
3636 if (self._bindir == self._pythondir) and (
3636 if (self._bindir == self._pythondir) and (
3637 self._bindir != self._tmpbindir
3637 self._bindir != self._tmpbindir
3638 ):
3638 ):
3639 # The pythondir has been inferred from --with-hg flag.
3639 # The pythondir has been inferred from --with-hg flag.
3640 # We cannot expect anything sensible here.
3640 # We cannot expect anything sensible here.
3641 return
3641 return
3642 expecthg = os.path.join(self._pythondir, b'mercurial')
3642 expecthg = os.path.join(self._pythondir, b'mercurial')
3643 actualhg = self._gethgpath()
3643 actualhg = self._gethgpath()
3644 if os.path.abspath(actualhg) != os.path.abspath(expecthg):
3644 if os.path.abspath(actualhg) != os.path.abspath(expecthg):
3645 sys.stderr.write(
3645 sys.stderr.write(
3646 'warning: %s with unexpected mercurial lib: %s\n'
3646 'warning: %s with unexpected mercurial lib: %s\n'
3647 ' (expected %s)\n' % (verb, actualhg, expecthg)
3647 ' (expected %s)\n' % (verb, actualhg, expecthg)
3648 )
3648 )
3649
3649
3650 def _gethgpath(self):
3650 def _gethgpath(self):
3651 """Return the path to the mercurial package that is actually found by
3651 """Return the path to the mercurial package that is actually found by
3652 the current Python interpreter."""
3652 the current Python interpreter."""
3653 if self._hgpath is not None:
3653 if self._hgpath is not None:
3654 return self._hgpath
3654 return self._hgpath
3655
3655
3656 cmd = b'"%s" -c "import mercurial; print (mercurial.__path__[0])"'
3656 cmd = b'"%s" -c "import mercurial; print (mercurial.__path__[0])"'
3657 cmd = cmd % PYTHON
3657 cmd = cmd % PYTHON
3658 if PYTHON3:
3658 if PYTHON3:
3659 cmd = _bytes2sys(cmd)
3659 cmd = _bytes2sys(cmd)
3660
3660
3661 p = subprocess.Popen(cmd, stdout=subprocess.PIPE, shell=True)
3661 p = subprocess.Popen(cmd, stdout=subprocess.PIPE, shell=True)
3662 out, err = p.communicate()
3662 out, err = p.communicate()
3663
3663
3664 self._hgpath = out.strip()
3664 self._hgpath = out.strip()
3665
3665
3666 return self._hgpath
3666 return self._hgpath
3667
3667
3668 def _installchg(self):
3668 def _installchg(self):
3669 """Install chg into the test environment"""
3669 """Install chg into the test environment"""
3670 vlog('# Performing temporary installation of CHG')
3670 vlog('# Performing temporary installation of CHG')
3671 assert os.path.dirname(self._bindir) == self._installdir
3671 assert os.path.dirname(self._bindir) == self._installdir
3672 assert self._hgroot, 'must be called after _installhg()'
3672 assert self._hgroot, 'must be called after _installhg()'
3673 cmd = b'"%(make)s" clean install PREFIX="%(prefix)s"' % {
3673 cmd = b'"%(make)s" clean install PREFIX="%(prefix)s"' % {
3674 b'make': b'make', # TODO: switch by option or environment?
3674 b'make': b'make', # TODO: switch by option or environment?
3675 b'prefix': self._installdir,
3675 b'prefix': self._installdir,
3676 }
3676 }
3677 cwd = os.path.join(self._hgroot, b'contrib', b'chg')
3677 cwd = os.path.join(self._hgroot, b'contrib', b'chg')
3678 vlog("# Running", cmd)
3678 vlog("# Running", cmd)
3679 proc = subprocess.Popen(
3679 proc = subprocess.Popen(
3680 cmd,
3680 cmd,
3681 shell=True,
3681 shell=True,
3682 cwd=cwd,
3682 cwd=cwd,
3683 stdin=subprocess.PIPE,
3683 stdin=subprocess.PIPE,
3684 stdout=subprocess.PIPE,
3684 stdout=subprocess.PIPE,
3685 stderr=subprocess.STDOUT,
3685 stderr=subprocess.STDOUT,
3686 )
3686 )
3687 out, _err = proc.communicate()
3687 out, _err = proc.communicate()
3688 if proc.returncode != 0:
3688 if proc.returncode != 0:
3689 if PYTHON3:
3689 if PYTHON3:
3690 sys.stdout.buffer.write(out)
3690 sys.stdout.buffer.write(out)
3691 else:
3691 else:
3692 sys.stdout.write(out)
3692 sys.stdout.write(out)
3693 sys.exit(1)
3693 sys.exit(1)
3694
3694
3695 def _outputcoverage(self):
3695 def _outputcoverage(self):
3696 """Produce code coverage output."""
3696 """Produce code coverage output."""
3697 import coverage
3697 import coverage
3698
3698
3699 coverage = coverage.coverage
3699 coverage = coverage.coverage
3700
3700
3701 vlog('# Producing coverage report')
3701 vlog('# Producing coverage report')
3702 # chdir is the easiest way to get short, relative paths in the
3702 # chdir is the easiest way to get short, relative paths in the
3703 # output.
3703 # output.
3704 os.chdir(self._hgroot)
3704 os.chdir(self._hgroot)
3705 covdir = os.path.join(_bytes2sys(self._installdir), '..', 'coverage')
3705 covdir = os.path.join(_bytes2sys(self._installdir), '..', 'coverage')
3706 cov = coverage(data_file=os.path.join(covdir, 'cov'))
3706 cov = coverage(data_file=os.path.join(covdir, 'cov'))
3707
3707
3708 # Map install directory paths back to source directory.
3708 # Map install directory paths back to source directory.
3709 cov.config.paths['srcdir'] = ['.', _bytes2sys(self._pythondir)]
3709 cov.config.paths['srcdir'] = ['.', _bytes2sys(self._pythondir)]
3710
3710
3711 cov.combine()
3711 cov.combine()
3712
3712
3713 omit = [
3713 omit = [
3714 _bytes2sys(os.path.join(x, b'*'))
3714 _bytes2sys(os.path.join(x, b'*'))
3715 for x in [self._bindir, self._testdir]
3715 for x in [self._bindir, self._testdir]
3716 ]
3716 ]
3717 cov.report(ignore_errors=True, omit=omit)
3717 cov.report(ignore_errors=True, omit=omit)
3718
3718
3719 if self.options.htmlcov:
3719 if self.options.htmlcov:
3720 htmldir = os.path.join(_bytes2sys(self._outputdir), 'htmlcov')
3720 htmldir = os.path.join(_bytes2sys(self._outputdir), 'htmlcov')
3721 cov.html_report(directory=htmldir, omit=omit)
3721 cov.html_report(directory=htmldir, omit=omit)
3722 if self.options.annotate:
3722 if self.options.annotate:
3723 adir = os.path.join(_bytes2sys(self._outputdir), 'annotated')
3723 adir = os.path.join(_bytes2sys(self._outputdir), 'annotated')
3724 if not os.path.isdir(adir):
3724 if not os.path.isdir(adir):
3725 os.mkdir(adir)
3725 os.mkdir(adir)
3726 cov.annotate(directory=adir, omit=omit)
3726 cov.annotate(directory=adir, omit=omit)
3727
3727
3728 def _findprogram(self, program):
3728 def _findprogram(self, program):
3729 """Search PATH for a executable program"""
3729 """Search PATH for a executable program"""
3730 dpb = _sys2bytes(os.defpath)
3730 dpb = _sys2bytes(os.defpath)
3731 sepb = _sys2bytes(os.pathsep)
3731 sepb = _sys2bytes(os.pathsep)
3732 for p in osenvironb.get(b'PATH', dpb).split(sepb):
3732 for p in osenvironb.get(b'PATH', dpb).split(sepb):
3733 name = os.path.join(p, program)
3733 name = os.path.join(p, program)
3734 if os.name == 'nt' or os.access(name, os.X_OK):
3734 if os.name == 'nt' or os.access(name, os.X_OK):
3735 return _bytes2sys(name)
3735 return _bytes2sys(name)
3736 return None
3736 return None
3737
3737
3738 def _checktools(self):
3738 def _checktools(self):
3739 """Ensure tools required to run tests are present."""
3739 """Ensure tools required to run tests are present."""
3740 for p in self.REQUIREDTOOLS:
3740 for p in self.REQUIREDTOOLS:
3741 if os.name == 'nt' and not p.endswith(b'.exe'):
3741 if os.name == 'nt' and not p.endswith(b'.exe'):
3742 p += b'.exe'
3742 p += b'.exe'
3743 found = self._findprogram(p)
3743 found = self._findprogram(p)
3744 p = p.decode("utf-8")
3744 p = p.decode("utf-8")
3745 if found:
3745 if found:
3746 vlog("# Found prerequisite", p, "at", found)
3746 vlog("# Found prerequisite", p, "at", found)
3747 else:
3747 else:
3748 print("WARNING: Did not find prerequisite tool: %s " % p)
3748 print("WARNING: Did not find prerequisite tool: %s " % p)
3749
3749
3750
3750
3751 def aggregateexceptions(path):
3751 def aggregateexceptions(path):
3752 exceptioncounts = collections.Counter()
3752 exceptioncounts = collections.Counter()
3753 testsbyfailure = collections.defaultdict(set)
3753 testsbyfailure = collections.defaultdict(set)
3754 failuresbytest = collections.defaultdict(set)
3754 failuresbytest = collections.defaultdict(set)
3755
3755
3756 for f in os.listdir(path):
3756 for f in os.listdir(path):
3757 with open(os.path.join(path, f), 'rb') as fh:
3757 with open(os.path.join(path, f), 'rb') as fh:
3758 data = fh.read().split(b'\0')
3758 data = fh.read().split(b'\0')
3759 if len(data) != 5:
3759 if len(data) != 5:
3760 continue
3760 continue
3761
3761
3762 exc, mainframe, hgframe, hgline, testname = data
3762 exc, mainframe, hgframe, hgline, testname = data
3763 exc = exc.decode('utf-8')
3763 exc = exc.decode('utf-8')
3764 mainframe = mainframe.decode('utf-8')
3764 mainframe = mainframe.decode('utf-8')
3765 hgframe = hgframe.decode('utf-8')
3765 hgframe = hgframe.decode('utf-8')
3766 hgline = hgline.decode('utf-8')
3766 hgline = hgline.decode('utf-8')
3767 testname = testname.decode('utf-8')
3767 testname = testname.decode('utf-8')
3768
3768
3769 key = (hgframe, hgline, exc)
3769 key = (hgframe, hgline, exc)
3770 exceptioncounts[key] += 1
3770 exceptioncounts[key] += 1
3771 testsbyfailure[key].add(testname)
3771 testsbyfailure[key].add(testname)
3772 failuresbytest[testname].add(key)
3772 failuresbytest[testname].add(key)
3773
3773
3774 # Find test having fewest failures for each failure.
3774 # Find test having fewest failures for each failure.
3775 leastfailing = {}
3775 leastfailing = {}
3776 for key, tests in testsbyfailure.items():
3776 for key, tests in testsbyfailure.items():
3777 fewesttest = None
3777 fewesttest = None
3778 fewestcount = 99999999
3778 fewestcount = 99999999
3779 for test in sorted(tests):
3779 for test in sorted(tests):
3780 if len(failuresbytest[test]) < fewestcount:
3780 if len(failuresbytest[test]) < fewestcount:
3781 fewesttest = test
3781 fewesttest = test
3782 fewestcount = len(failuresbytest[test])
3782 fewestcount = len(failuresbytest[test])
3783
3783
3784 leastfailing[key] = (fewestcount, fewesttest)
3784 leastfailing[key] = (fewestcount, fewesttest)
3785
3785
3786 # Create a combined counter so we can sort by total occurrences and
3786 # Create a combined counter so we can sort by total occurrences and
3787 # impacted tests.
3787 # impacted tests.
3788 combined = {}
3788 combined = {}
3789 for key in exceptioncounts:
3789 for key in exceptioncounts:
3790 combined[key] = (
3790 combined[key] = (
3791 exceptioncounts[key],
3791 exceptioncounts[key],
3792 len(testsbyfailure[key]),
3792 len(testsbyfailure[key]),
3793 leastfailing[key][0],
3793 leastfailing[key][0],
3794 leastfailing[key][1],
3794 leastfailing[key][1],
3795 )
3795 )
3796
3796
3797 return {
3797 return {
3798 'exceptioncounts': exceptioncounts,
3798 'exceptioncounts': exceptioncounts,
3799 'total': sum(exceptioncounts.values()),
3799 'total': sum(exceptioncounts.values()),
3800 'combined': combined,
3800 'combined': combined,
3801 'leastfailing': leastfailing,
3801 'leastfailing': leastfailing,
3802 'byfailure': testsbyfailure,
3802 'byfailure': testsbyfailure,
3803 'bytest': failuresbytest,
3803 'bytest': failuresbytest,
3804 }
3804 }
3805
3805
3806
3806
3807 if __name__ == '__main__':
3807 if __name__ == '__main__':
3808 runner = TestRunner()
3808 runner = TestRunner()
3809
3809
3810 try:
3810 try:
3811 import msvcrt
3811 import msvcrt
3812
3812
3813 msvcrt.setmode(sys.stdin.fileno(), os.O_BINARY)
3813 msvcrt.setmode(sys.stdin.fileno(), os.O_BINARY)
3814 msvcrt.setmode(sys.stdout.fileno(), os.O_BINARY)
3814 msvcrt.setmode(sys.stdout.fileno(), os.O_BINARY)
3815 msvcrt.setmode(sys.stderr.fileno(), os.O_BINARY)
3815 msvcrt.setmode(sys.stderr.fileno(), os.O_BINARY)
3816 except ImportError:
3816 except ImportError:
3817 pass
3817 pass
3818
3818
3819 sys.exit(runner.run(sys.argv[1:]))
3819 sys.exit(runner.run(sys.argv[1:]))
@@ -1,2031 +1,2034 b''
1 This file tests the behavior of run-tests.py itself.
1 This file tests the behavior of run-tests.py itself.
2
2
3 Avoid interference from actual test env:
3 Avoid interference from actual test env:
4
4
5 $ . "$TESTDIR/helper-runtests.sh"
5 $ . "$TESTDIR/helper-runtests.sh"
6
6
7 Smoke test with install
7 Smoke test with install
8 ============
8 ============
9 $ "$PYTHON" $TESTDIR/run-tests.py $HGTEST_RUN_TESTS_PURE -l
9 $ "$PYTHON" $TESTDIR/run-tests.py $HGTEST_RUN_TESTS_PURE -l
10 running 0 tests using 0 parallel processes
10 running 0 tests using 0 parallel processes
11
11
12 # Ran 0 tests, 0 skipped, 0 failed.
12 # Ran 0 tests, 0 skipped, 0 failed.
13
13
14 Define a helper to avoid the install step
14 Define a helper to avoid the install step
15 =============
15 =============
16 $ rt()
16 $ rt()
17 > {
17 > {
18 > "$PYTHON" $TESTDIR/run-tests.py --with-hg=`which hg` -j1 "$@"
18 > "$PYTHON" $TESTDIR/run-tests.py --with-hg=`which hg` -j1 "$@"
19 > }
19 > }
20
20
21 error paths
21 error paths
22
22
23 #if symlink
23 #if symlink
24 $ ln -s `which true` hg
24 $ ln -s `which true` hg
25 $ "$PYTHON" $TESTDIR/run-tests.py --with-hg=./hg
25 $ "$PYTHON" $TESTDIR/run-tests.py --with-hg=./hg
26 warning: --with-hg should specify an hg script
26 warning: --with-hg should specify an hg script
27 running 0 tests using 0 parallel processes
27 running 0 tests using 0 parallel processes
28
28
29 # Ran 0 tests, 0 skipped, 0 failed.
29 # Ran 0 tests, 0 skipped, 0 failed.
30 $ rm hg
30 $ rm hg
31 #endif
31 #endif
32
32
33 #if execbit
33 #if execbit
34 $ touch hg
34 $ touch hg
35 $ "$PYTHON" $TESTDIR/run-tests.py --with-hg=./hg
35 $ "$PYTHON" $TESTDIR/run-tests.py --with-hg=./hg
36 usage: run-tests.py [options] [tests]
36 usage: run-tests.py [options] [tests]
37 run-tests.py: error: --with-hg must specify an executable hg script
37 run-tests.py: error: --with-hg must specify an executable hg script
38 [2]
38 [2]
39 $ rm hg
39 $ rm hg
40 #endif
40 #endif
41
41
42 Features for testing optional lines
42 Features for testing optional lines
43 ===================================
43 ===================================
44
44
45 $ cat > hghaveaddon.py <<EOF
45 $ cat > hghaveaddon.py <<EOF
46 > import hghave
46 > import hghave
47 > @hghave.check("custom", "custom hghave feature")
47 > @hghave.check("custom", "custom hghave feature")
48 > def has_custom():
48 > def has_custom():
49 > return True
49 > return True
50 > @hghave.check("missing", "missing hghave feature")
50 > @hghave.check("missing", "missing hghave feature")
51 > def has_missing():
51 > def has_missing():
52 > return False
52 > return False
53 > EOF
53 > EOF
54
54
55 an empty test
55 an empty test
56 =======================
56 =======================
57
57
58 $ touch test-empty.t
58 $ touch test-empty.t
59 $ rt
59 $ rt
60 running 1 tests using 1 parallel processes
60 running 1 tests using 1 parallel processes
61 .
61 .
62 # Ran 1 tests, 0 skipped, 0 failed.
62 # Ran 1 tests, 0 skipped, 0 failed.
63 $ rm test-empty.t
63 $ rm test-empty.t
64
64
65 a succesful test
65 a succesful test
66 =======================
66 =======================
67
67
68 $ cat > test-success.t << EOF
68 $ cat > test-success.t << EOF
69 > $ echo babar
69 > $ echo babar
70 > babar
70 > babar
71 > $ echo xyzzy
71 > $ echo xyzzy
72 > dont_print (?)
72 > dont_print (?)
73 > nothing[42]line (re) (?)
73 > nothing[42]line (re) (?)
74 > never*happens (glob) (?)
74 > never*happens (glob) (?)
75 > more_nothing (?)
75 > more_nothing (?)
76 > xyzzy
76 > xyzzy
77 > nor this (?)
77 > nor this (?)
78 > $ printf 'abc\ndef\nxyz\n'
78 > $ printf 'abc\ndef\nxyz\n'
79 > 123 (?)
79 > 123 (?)
80 > abc
80 > abc
81 > def (?)
81 > def (?)
82 > 456 (?)
82 > 456 (?)
83 > xyz
83 > xyz
84 > $ printf 'zyx\nwvu\ntsr\n'
84 > $ printf 'zyx\nwvu\ntsr\n'
85 > abc (?)
85 > abc (?)
86 > zyx (custom !)
86 > zyx (custom !)
87 > wvu
87 > wvu
88 > no_print (no-custom !)
88 > no_print (no-custom !)
89 > tsr (no-missing !)
89 > tsr (no-missing !)
90 > missing (missing !)
90 > missing (missing !)
91 > EOF
91 > EOF
92
92
93 $ rt
93 $ rt
94 running 1 tests using 1 parallel processes
94 running 1 tests using 1 parallel processes
95 .
95 .
96 # Ran 1 tests, 0 skipped, 0 failed.
96 # Ran 1 tests, 0 skipped, 0 failed.
97
97
98 failing test
98 failing test
99 ==================
99 ==================
100
100
101 test churn with globs
101 test churn with globs
102 $ cat > test-failure.t <<EOF
102 $ cat > test-failure.t <<EOF
103 > $ echo "bar-baz"; echo "bar-bad"; echo foo
103 > $ echo "bar-baz"; echo "bar-bad"; echo foo
104 > bar*bad (glob)
104 > bar*bad (glob)
105 > bar*baz (glob)
105 > bar*baz (glob)
106 > | fo (re)
106 > | fo (re)
107 > EOF
107 > EOF
108 $ rt test-failure.t
108 $ rt test-failure.t
109 running 1 tests using 1 parallel processes
109 running 1 tests using 1 parallel processes
110
110
111 --- $TESTTMP/test-failure.t
111 --- $TESTTMP/test-failure.t
112 +++ $TESTTMP/test-failure.t.err
112 +++ $TESTTMP/test-failure.t.err
113 @@ -1,4 +1,4 @@
113 @@ -1,4 +1,4 @@
114 $ echo "bar-baz"; echo "bar-bad"; echo foo
114 $ echo "bar-baz"; echo "bar-bad"; echo foo
115 + bar*baz (glob)
115 + bar*baz (glob)
116 bar*bad (glob)
116 bar*bad (glob)
117 - bar*baz (glob)
117 - bar*baz (glob)
118 - | fo (re)
118 - | fo (re)
119 + foo
119 + foo
120
120
121 ERROR: test-failure.t output changed
121 ERROR: test-failure.t output changed
122 !
122 !
123 Failed test-failure.t: output changed
123 Failed test-failure.t: output changed
124 # Ran 1 tests, 0 skipped, 1 failed.
124 # Ran 1 tests, 0 skipped, 1 failed.
125 python hash seed: * (glob)
125 python hash seed: * (glob)
126 [1]
126 [1]
127
127
128 test how multiple globs gets matched with lines in output
128 test how multiple globs gets matched with lines in output
129 $ cat > test-failure-globs.t <<EOF
129 $ cat > test-failure-globs.t <<EOF
130 > $ echo "context"; echo "context"; \
130 > $ echo "context"; echo "context"; \
131 > echo "key: 1"; echo "value: not a"; \
131 > echo "key: 1"; echo "value: not a"; \
132 > echo "key: 2"; echo "value: not b"; \
132 > echo "key: 2"; echo "value: not b"; \
133 > echo "key: 3"; echo "value: c"; \
133 > echo "key: 3"; echo "value: c"; \
134 > echo "key: 4"; echo "value: d"
134 > echo "key: 4"; echo "value: d"
135 > context
135 > context
136 > context
136 > context
137 > key: 1
137 > key: 1
138 > value: a
138 > value: a
139 > key: 2
139 > key: 2
140 > value: b
140 > value: b
141 > key: 3
141 > key: 3
142 > value: * (glob)
142 > value: * (glob)
143 > key: 4
143 > key: 4
144 > value: * (glob)
144 > value: * (glob)
145 > EOF
145 > EOF
146 $ rt test-failure-globs.t
146 $ rt test-failure-globs.t
147 running 1 tests using 1 parallel processes
147 running 1 tests using 1 parallel processes
148
148
149 --- $TESTTMP/test-failure-globs.t
149 --- $TESTTMP/test-failure-globs.t
150 +++ $TESTTMP/test-failure-globs.t.err
150 +++ $TESTTMP/test-failure-globs.t.err
151 @@ -2,9 +2,9 @@
151 @@ -2,9 +2,9 @@
152 context
152 context
153 context
153 context
154 key: 1
154 key: 1
155 - value: a
155 - value: a
156 + value: not a
156 + value: not a
157 key: 2
157 key: 2
158 - value: b
158 - value: b
159 + value: not b
159 + value: not b
160 key: 3
160 key: 3
161 value: * (glob)
161 value: * (glob)
162 key: 4
162 key: 4
163
163
164 ERROR: test-failure-globs.t output changed
164 ERROR: test-failure-globs.t output changed
165 !
165 !
166 Failed test-failure-globs.t: output changed
166 Failed test-failure-globs.t: output changed
167 # Ran 1 tests, 0 skipped, 1 failed.
167 # Ran 1 tests, 0 skipped, 1 failed.
168 python hash seed: * (glob)
168 python hash seed: * (glob)
169 [1]
169 [1]
170 $ rm test-failure-globs.t
170 $ rm test-failure-globs.t
171
171
172 test diff colorisation
172 test diff colorisation
173
173
174 #if no-windows pygments
174 #if no-windows pygments
175 $ rt test-failure.t --color always
175 $ rt test-failure.t --color always
176 running 1 tests using 1 parallel processes
176 running 1 tests using 1 parallel processes
177
177
178 \x1b[38;5;124m--- $TESTTMP/test-failure.t\x1b[39m (esc)
178 \x1b[38;5;124m--- $TESTTMP/test-failure.t\x1b[39m (esc)
179 \x1b[38;5;34m+++ $TESTTMP/test-failure.t.err\x1b[39m (esc)
179 \x1b[38;5;34m+++ $TESTTMP/test-failure.t.err\x1b[39m (esc)
180 \x1b[38;5;90;01m@@ -1,4 +1,4 @@\x1b[39;00m (esc)
180 \x1b[38;5;90;01m@@ -1,4 +1,4 @@\x1b[39;00m (esc)
181 $ echo "bar-baz"; echo "bar-bad"; echo foo
181 $ echo "bar-baz"; echo "bar-bad"; echo foo
182 \x1b[38;5;34m+ bar*baz (glob)\x1b[39m (esc)
182 \x1b[38;5;34m+ bar*baz (glob)\x1b[39m (esc)
183 bar*bad (glob)
183 bar*bad (glob)
184 \x1b[38;5;124m- bar*baz (glob)\x1b[39m (esc)
184 \x1b[38;5;124m- bar*baz (glob)\x1b[39m (esc)
185 \x1b[38;5;124m- | fo (re)\x1b[39m (esc)
185 \x1b[38;5;124m- | fo (re)\x1b[39m (esc)
186 \x1b[38;5;34m+ foo\x1b[39m (esc)
186 \x1b[38;5;34m+ foo\x1b[39m (esc)
187
187
188 \x1b[38;5;88mERROR: \x1b[39m\x1b[38;5;9mtest-failure.t\x1b[39m\x1b[38;5;88m output changed\x1b[39m (esc)
188 \x1b[38;5;88mERROR: \x1b[39m\x1b[38;5;9mtest-failure.t\x1b[39m\x1b[38;5;88m output changed\x1b[39m (esc)
189 !
189 !
190 \x1b[38;5;88mFailed \x1b[39m\x1b[38;5;9mtest-failure.t\x1b[39m\x1b[38;5;88m: output changed\x1b[39m (esc)
190 \x1b[38;5;88mFailed \x1b[39m\x1b[38;5;9mtest-failure.t\x1b[39m\x1b[38;5;88m: output changed\x1b[39m (esc)
191 # Ran 1 tests, 0 skipped, 1 failed.
191 # Ran 1 tests, 0 skipped, 1 failed.
192 python hash seed: * (glob)
192 python hash seed: * (glob)
193 [1]
193 [1]
194
194
195 $ rt test-failure.t 2> tmp.log
195 $ rt test-failure.t 2> tmp.log
196 running 1 tests using 1 parallel processes
196 running 1 tests using 1 parallel processes
197 [1]
197 [1]
198 $ cat tmp.log
198 $ cat tmp.log
199
199
200 --- $TESTTMP/test-failure.t
200 --- $TESTTMP/test-failure.t
201 +++ $TESTTMP/test-failure.t.err
201 +++ $TESTTMP/test-failure.t.err
202 @@ -1,4 +1,4 @@
202 @@ -1,4 +1,4 @@
203 $ echo "bar-baz"; echo "bar-bad"; echo foo
203 $ echo "bar-baz"; echo "bar-bad"; echo foo
204 + bar*baz (glob)
204 + bar*baz (glob)
205 bar*bad (glob)
205 bar*bad (glob)
206 - bar*baz (glob)
206 - bar*baz (glob)
207 - | fo (re)
207 - | fo (re)
208 + foo
208 + foo
209
209
210 ERROR: test-failure.t output changed
210 ERROR: test-failure.t output changed
211 !
211 !
212 Failed test-failure.t: output changed
212 Failed test-failure.t: output changed
213 # Ran 1 tests, 0 skipped, 1 failed.
213 # Ran 1 tests, 0 skipped, 1 failed.
214 python hash seed: * (glob)
214 python hash seed: * (glob)
215 #endif
215 #endif
216
216
217 $ cat > test-failure.t << EOF
217 $ cat > test-failure.t << EOF
218 > $ true
218 > $ true
219 > should go away (true !)
219 > should go away (true !)
220 > $ true
220 > $ true
221 > should stay (false !)
221 > should stay (false !)
222 >
222 >
223 > Should remove first line, not second or third
223 > Should remove first line, not second or third
224 > $ echo 'testing'
224 > $ echo 'testing'
225 > baz*foo (glob) (true !)
225 > baz*foo (glob) (true !)
226 > foobar*foo (glob) (false !)
226 > foobar*foo (glob) (false !)
227 > te*ting (glob) (true !)
227 > te*ting (glob) (true !)
228 >
228 >
229 > Should keep first two lines, remove third and last
229 > Should keep first two lines, remove third and last
230 > $ echo 'testing'
230 > $ echo 'testing'
231 > test.ng (re) (true !)
231 > test.ng (re) (true !)
232 > foo.ar (re) (false !)
232 > foo.ar (re) (false !)
233 > b.r (re) (true !)
233 > b.r (re) (true !)
234 > missing (?)
234 > missing (?)
235 > awol (true !)
235 > awol (true !)
236 >
236 >
237 > The "missing" line should stay, even though awol is dropped
237 > The "missing" line should stay, even though awol is dropped
238 > $ echo 'testing'
238 > $ echo 'testing'
239 > test.ng (re) (true !)
239 > test.ng (re) (true !)
240 > foo.ar (?)
240 > foo.ar (?)
241 > awol
241 > awol
242 > missing (?)
242 > missing (?)
243 > EOF
243 > EOF
244 $ rt test-failure.t
244 $ rt test-failure.t
245 running 1 tests using 1 parallel processes
245 running 1 tests using 1 parallel processes
246
246
247 --- $TESTTMP/test-failure.t
247 --- $TESTTMP/test-failure.t
248 +++ $TESTTMP/test-failure.t.err
248 +++ $TESTTMP/test-failure.t.err
249 @@ -1,11 +1,9 @@
249 @@ -1,11 +1,9 @@
250 $ true
250 $ true
251 - should go away (true !)
251 - should go away (true !)
252 $ true
252 $ true
253 should stay (false !)
253 should stay (false !)
254
254
255 Should remove first line, not second or third
255 Should remove first line, not second or third
256 $ echo 'testing'
256 $ echo 'testing'
257 - baz*foo (glob) (true !)
257 - baz*foo (glob) (true !)
258 foobar*foo (glob) (false !)
258 foobar*foo (glob) (false !)
259 te*ting (glob) (true !)
259 te*ting (glob) (true !)
260
260
261 foo.ar (re) (false !)
261 foo.ar (re) (false !)
262 missing (?)
262 missing (?)
263 @@ -13,13 +11,10 @@
263 @@ -13,13 +11,10 @@
264 $ echo 'testing'
264 $ echo 'testing'
265 test.ng (re) (true !)
265 test.ng (re) (true !)
266 foo.ar (re) (false !)
266 foo.ar (re) (false !)
267 - b.r (re) (true !)
267 - b.r (re) (true !)
268 missing (?)
268 missing (?)
269 - awol (true !)
269 - awol (true !)
270
270
271 The "missing" line should stay, even though awol is dropped
271 The "missing" line should stay, even though awol is dropped
272 $ echo 'testing'
272 $ echo 'testing'
273 test.ng (re) (true !)
273 test.ng (re) (true !)
274 foo.ar (?)
274 foo.ar (?)
275 - awol
275 - awol
276 missing (?)
276 missing (?)
277
277
278 ERROR: test-failure.t output changed
278 ERROR: test-failure.t output changed
279 !
279 !
280 Failed test-failure.t: output changed
280 Failed test-failure.t: output changed
281 # Ran 1 tests, 0 skipped, 1 failed.
281 # Ran 1 tests, 0 skipped, 1 failed.
282 python hash seed: * (glob)
282 python hash seed: * (glob)
283 [1]
283 [1]
284
284
285 basic failing test
285 basic failing test
286 $ cat > test-failure.t << EOF
286 $ cat > test-failure.t << EOF
287 > $ echo babar
287 > $ echo babar
288 > rataxes
288 > rataxes
289 > This is a noop statement so that
289 > This is a noop statement so that
290 > this test is still more bytes than success.
290 > this test is still more bytes than success.
291 > pad pad pad pad............................................................
291 > pad pad pad pad............................................................
292 > pad pad pad pad............................................................
292 > pad pad pad pad............................................................
293 > pad pad pad pad............................................................
293 > pad pad pad pad............................................................
294 > pad pad pad pad............................................................
294 > pad pad pad pad............................................................
295 > pad pad pad pad............................................................
295 > pad pad pad pad............................................................
296 > pad pad pad pad............................................................
296 > pad pad pad pad............................................................
297 > EOF
297 > EOF
298
298
299 >>> fh = open('test-failure-unicode.t', 'wb')
299 >>> fh = open('test-failure-unicode.t', 'wb')
300 >>> fh.write(u' $ echo babar\u03b1\n'.encode('utf-8')) and None
300 >>> fh.write(u' $ echo babar\u03b1\n'.encode('utf-8')) and None
301 >>> fh.write(u' l\u03b5\u03b5t\n'.encode('utf-8')) and None
301 >>> fh.write(u' l\u03b5\u03b5t\n'.encode('utf-8')) and None
302
302
303 $ rt
303 $ rt
304 running 3 tests using 1 parallel processes
304 running 3 tests using 1 parallel processes
305
305
306 --- $TESTTMP/test-failure.t
306 --- $TESTTMP/test-failure.t
307 +++ $TESTTMP/test-failure.t.err
307 +++ $TESTTMP/test-failure.t.err
308 @@ -1,5 +1,5 @@
308 @@ -1,5 +1,5 @@
309 $ echo babar
309 $ echo babar
310 - rataxes
310 - rataxes
311 + babar
311 + babar
312 This is a noop statement so that
312 This is a noop statement so that
313 this test is still more bytes than success.
313 this test is still more bytes than success.
314 pad pad pad pad............................................................
314 pad pad pad pad............................................................
315
315
316 ERROR: test-failure.t output changed
316 ERROR: test-failure.t output changed
317 !.
317 !.
318 --- $TESTTMP/test-failure-unicode.t
318 --- $TESTTMP/test-failure-unicode.t
319 +++ $TESTTMP/test-failure-unicode.t.err
319 +++ $TESTTMP/test-failure-unicode.t.err
320 @@ -1,2 +1,2 @@
320 @@ -1,2 +1,2 @@
321 $ echo babar\xce\xb1 (esc)
321 $ echo babar\xce\xb1 (esc)
322 - l\xce\xb5\xce\xb5t (esc)
322 - l\xce\xb5\xce\xb5t (esc)
323 + babar\xce\xb1 (esc)
323 + babar\xce\xb1 (esc)
324
324
325 ERROR: test-failure-unicode.t output changed
325 ERROR: test-failure-unicode.t output changed
326 !
326 !
327 Failed test-failure-unicode.t: output changed
327 Failed test-failure-unicode.t: output changed
328 Failed test-failure.t: output changed
328 Failed test-failure.t: output changed
329 # Ran 3 tests, 0 skipped, 2 failed.
329 # Ran 3 tests, 0 skipped, 2 failed.
330 python hash seed: * (glob)
330 python hash seed: * (glob)
331 [1]
331 [1]
332
332
333 test --outputdir
333 test --outputdir
334 $ mkdir output
334 $ mkdir output
335 $ rt --outputdir output
335 $ rt --outputdir output
336 running 3 tests using 1 parallel processes
336 running 3 tests using 1 parallel processes
337
337
338 --- $TESTTMP/test-failure.t
338 --- $TESTTMP/test-failure.t
339 +++ $TESTTMP/output/test-failure.t.err
339 +++ $TESTTMP/output/test-failure.t.err
340 @@ -1,5 +1,5 @@
340 @@ -1,5 +1,5 @@
341 $ echo babar
341 $ echo babar
342 - rataxes
342 - rataxes
343 + babar
343 + babar
344 This is a noop statement so that
344 This is a noop statement so that
345 this test is still more bytes than success.
345 this test is still more bytes than success.
346 pad pad pad pad............................................................
346 pad pad pad pad............................................................
347
347
348 ERROR: test-failure.t output changed
348 ERROR: test-failure.t output changed
349 !.
349 !.
350 --- $TESTTMP/test-failure-unicode.t
350 --- $TESTTMP/test-failure-unicode.t
351 +++ $TESTTMP/output/test-failure-unicode.t.err
351 +++ $TESTTMP/output/test-failure-unicode.t.err
352 @@ -1,2 +1,2 @@
352 @@ -1,2 +1,2 @@
353 $ echo babar\xce\xb1 (esc)
353 $ echo babar\xce\xb1 (esc)
354 - l\xce\xb5\xce\xb5t (esc)
354 - l\xce\xb5\xce\xb5t (esc)
355 + babar\xce\xb1 (esc)
355 + babar\xce\xb1 (esc)
356
356
357 ERROR: test-failure-unicode.t output changed
357 ERROR: test-failure-unicode.t output changed
358 !
358 !
359 Failed test-failure-unicode.t: output changed
359 Failed test-failure-unicode.t: output changed
360 Failed test-failure.t: output changed
360 Failed test-failure.t: output changed
361 # Ran 3 tests, 0 skipped, 2 failed.
361 # Ran 3 tests, 0 skipped, 2 failed.
362 python hash seed: * (glob)
362 python hash seed: * (glob)
363 [1]
363 [1]
364 $ ls -a output
364 $ ls -a output
365 .
365 .
366 ..
366 ..
367 .testtimes
367 .testtimes
368 test-failure-unicode.t.err
368 test-failure-unicode.t.err
369 test-failure.t.err
369 test-failure.t.err
370
370
371 test --xunit support
371 test --xunit support
372 $ rt --xunit=xunit.xml
372 $ rt --xunit=xunit.xml
373 running 3 tests using 1 parallel processes
373 running 3 tests using 1 parallel processes
374
374
375 --- $TESTTMP/test-failure.t
375 --- $TESTTMP/test-failure.t
376 +++ $TESTTMP/test-failure.t.err
376 +++ $TESTTMP/test-failure.t.err
377 @@ -1,5 +1,5 @@
377 @@ -1,5 +1,5 @@
378 $ echo babar
378 $ echo babar
379 - rataxes
379 - rataxes
380 + babar
380 + babar
381 This is a noop statement so that
381 This is a noop statement so that
382 this test is still more bytes than success.
382 this test is still more bytes than success.
383 pad pad pad pad............................................................
383 pad pad pad pad............................................................
384
384
385 ERROR: test-failure.t output changed
385 ERROR: test-failure.t output changed
386 !.
386 !.
387 --- $TESTTMP/test-failure-unicode.t
387 --- $TESTTMP/test-failure-unicode.t
388 +++ $TESTTMP/test-failure-unicode.t.err
388 +++ $TESTTMP/test-failure-unicode.t.err
389 @@ -1,2 +1,2 @@
389 @@ -1,2 +1,2 @@
390 $ echo babar\xce\xb1 (esc)
390 $ echo babar\xce\xb1 (esc)
391 - l\xce\xb5\xce\xb5t (esc)
391 - l\xce\xb5\xce\xb5t (esc)
392 + babar\xce\xb1 (esc)
392 + babar\xce\xb1 (esc)
393
393
394 ERROR: test-failure-unicode.t output changed
394 ERROR: test-failure-unicode.t output changed
395 !
395 !
396 Failed test-failure-unicode.t: output changed
396 Failed test-failure-unicode.t: output changed
397 Failed test-failure.t: output changed
397 Failed test-failure.t: output changed
398 # Ran 3 tests, 0 skipped, 2 failed.
398 # Ran 3 tests, 0 skipped, 2 failed.
399 python hash seed: * (glob)
399 python hash seed: * (glob)
400 [1]
400 [1]
401 $ cat xunit.xml
401 $ cat xunit.xml
402 <?xml version="1.0" encoding="utf-8"?>
402 <?xml version="1.0" encoding="utf-8"?>
403 <testsuite errors="0" failures="2" name="run-tests" skipped="0" tests="3">
403 <testsuite errors="0" failures="2" name="run-tests" skipped="0" tests="3">
404 <testcase name="test-success.t" time="*"/> (glob)
404 <testcase name="test-success.t" time="*"/> (glob)
405 <testcase name="test-failure-unicode.t" time="*"> (glob)
405 <testcase name="test-failure-unicode.t" time="*"> (glob)
406 <failure message="output changed" type="output-mismatch"><![CDATA[--- $TESTTMP/test-failure-unicode.t (py38 !)
406 <failure message="output changed" type="output-mismatch"><![CDATA[--- $TESTTMP/test-failure-unicode.t (py38 !)
407 <failure message="output changed" type="output-mismatch"> (no-py38 !)
407 <failure message="output changed" type="output-mismatch"> (no-py38 !)
408 <![CDATA[--- $TESTTMP/test-failure-unicode.t (no-py38 !)
408 <![CDATA[--- $TESTTMP/test-failure-unicode.t (no-py38 !)
409 +++ $TESTTMP/test-failure-unicode.t.err
409 +++ $TESTTMP/test-failure-unicode.t.err
410 @@ -1,2 +1,2 @@
410 @@ -1,2 +1,2 @@
411 $ echo babar\xce\xb1 (esc)
411 $ echo babar\xce\xb1 (esc)
412 - l\xce\xb5\xce\xb5t (esc)
412 - l\xce\xb5\xce\xb5t (esc)
413 + babar\xce\xb1 (esc)
413 + babar\xce\xb1 (esc)
414 ]]></failure> (py38 !)
414 ]]></failure> (py38 !)
415 ]]> </failure> (no-py38 !)
415 ]]> </failure> (no-py38 !)
416 </testcase>
416 </testcase>
417 <testcase name="test-failure.t" time="*"> (glob)
417 <testcase name="test-failure.t" time="*"> (glob)
418 <failure message="output changed" type="output-mismatch"><![CDATA[--- $TESTTMP/test-failure.t (py38 !)
418 <failure message="output changed" type="output-mismatch"><![CDATA[--- $TESTTMP/test-failure.t (py38 !)
419 <failure message="output changed" type="output-mismatch"> (no-py38 !)
419 <failure message="output changed" type="output-mismatch"> (no-py38 !)
420 <![CDATA[--- $TESTTMP/test-failure.t (no-py38 !)
420 <![CDATA[--- $TESTTMP/test-failure.t (no-py38 !)
421 +++ $TESTTMP/test-failure.t.err
421 +++ $TESTTMP/test-failure.t.err
422 @@ -1,5 +1,5 @@
422 @@ -1,5 +1,5 @@
423 $ echo babar
423 $ echo babar
424 - rataxes
424 - rataxes
425 + babar
425 + babar
426 This is a noop statement so that
426 This is a noop statement so that
427 this test is still more bytes than success.
427 this test is still more bytes than success.
428 pad pad pad pad............................................................
428 pad pad pad pad............................................................
429 ]]></failure> (py38 !)
429 ]]></failure> (py38 !)
430 ]]> </failure> (no-py38 !)
430 ]]> </failure> (no-py38 !)
431 </testcase>
431 </testcase>
432 </testsuite>
432 </testsuite>
433
433
434 $ cat .testtimes
434 $ cat .testtimes
435 test-empty.t * (glob)
435 test-empty.t * (glob)
436 test-failure-globs.t * (glob)
436 test-failure-globs.t * (glob)
437 test-failure-unicode.t * (glob)
437 test-failure-unicode.t * (glob)
438 test-failure.t * (glob)
438 test-failure.t * (glob)
439 test-success.t * (glob)
439 test-success.t * (glob)
440
440
441 $ rt --list-tests
441 $ rt --list-tests
442 test-failure-unicode.t
442 test-failure-unicode.t
443 test-failure.t
443 test-failure.t
444 test-success.t
444 test-success.t
445
445
446 $ rt --list-tests --json
446 $ rt --list-tests --json
447 test-failure-unicode.t
447 test-failure-unicode.t
448 test-failure.t
448 test-failure.t
449 test-success.t
449 test-success.t
450 $ cat report.json
450 $ cat report.json
451 testreport ={
451 testreport ={
452 "test-failure-unicode.t": {
452 "test-failure-unicode.t": {
453 "result": "success"
453 "result": "success"
454 },
454 },
455 "test-failure.t": {
455 "test-failure.t": {
456 "result": "success"
456 "result": "success"
457 },
457 },
458 "test-success.t": {
458 "test-success.t": {
459 "result": "success"
459 "result": "success"
460 }
460 }
461 } (no-eol)
461 } (no-eol)
462
462
463 $ rt --list-tests --xunit=xunit.xml
463 $ rt --list-tests --xunit=xunit.xml
464 test-failure-unicode.t
464 test-failure-unicode.t
465 test-failure.t
465 test-failure.t
466 test-success.t
466 test-success.t
467 $ cat xunit.xml
467 $ cat xunit.xml
468 <?xml version="1.0" encoding="utf-8"?>
468 <?xml version="1.0" encoding="utf-8"?>
469 <testsuite errors="0" failures="0" name="run-tests" skipped="0" tests="0">
469 <testsuite errors="0" failures="0" name="run-tests" skipped="0" tests="0">
470 <testcase name="test-failure-unicode.t"/>
470 <testcase name="test-failure-unicode.t"/>
471 <testcase name="test-failure.t"/>
471 <testcase name="test-failure.t"/>
472 <testcase name="test-success.t"/>
472 <testcase name="test-success.t"/>
473 </testsuite>
473 </testsuite>
474
474
475 $ rt --list-tests test-failure* --json --xunit=xunit.xml --outputdir output
475 $ rt --list-tests test-failure* --json --xunit=xunit.xml --outputdir output
476 test-failure-unicode.t
476 test-failure-unicode.t
477 test-failure.t
477 test-failure.t
478 $ cat output/report.json
478 $ cat output/report.json
479 testreport ={
479 testreport ={
480 "test-failure-unicode.t": {
480 "test-failure-unicode.t": {
481 "result": "success"
481 "result": "success"
482 },
482 },
483 "test-failure.t": {
483 "test-failure.t": {
484 "result": "success"
484 "result": "success"
485 }
485 }
486 } (no-eol)
486 } (no-eol)
487 $ cat xunit.xml
487 $ cat xunit.xml
488 <?xml version="1.0" encoding="utf-8"?>
488 <?xml version="1.0" encoding="utf-8"?>
489 <testsuite errors="0" failures="0" name="run-tests" skipped="0" tests="0">
489 <testsuite errors="0" failures="0" name="run-tests" skipped="0" tests="0">
490 <testcase name="test-failure-unicode.t"/>
490 <testcase name="test-failure-unicode.t"/>
491 <testcase name="test-failure.t"/>
491 <testcase name="test-failure.t"/>
492 </testsuite>
492 </testsuite>
493
493
494 $ rm test-failure-unicode.t
494 $ rm test-failure-unicode.t
495
495
496 test for --retest
496 test for --retest
497 ====================
497 ====================
498
498
499 $ rt --retest
499 $ rt --retest
500 running 1 tests using 1 parallel processes
500 running 1 tests using 1 parallel processes
501
501
502 --- $TESTTMP/test-failure.t
502 --- $TESTTMP/test-failure.t
503 +++ $TESTTMP/test-failure.t.err
503 +++ $TESTTMP/test-failure.t.err
504 @@ -1,5 +1,5 @@
504 @@ -1,5 +1,5 @@
505 $ echo babar
505 $ echo babar
506 - rataxes
506 - rataxes
507 + babar
507 + babar
508 This is a noop statement so that
508 This is a noop statement so that
509 this test is still more bytes than success.
509 this test is still more bytes than success.
510 pad pad pad pad............................................................
510 pad pad pad pad............................................................
511
511
512 ERROR: test-failure.t output changed
512 ERROR: test-failure.t output changed
513 !
513 !
514 Failed test-failure.t: output changed
514 Failed test-failure.t: output changed
515 # Ran 1 tests, 0 skipped, 1 failed.
515 # Ran 1 tests, 0 skipped, 1 failed.
516 python hash seed: * (glob)
516 python hash seed: * (glob)
517 [1]
517 [1]
518
518
519 --retest works with --outputdir
519 --retest works with --outputdir
520 $ rm -r output
520 $ rm -r output
521 $ mkdir output
521 $ mkdir output
522 $ mv test-failure.t.err output
522 $ mv test-failure.t.err output
523 $ rt --retest --outputdir output
523 $ rt --retest --outputdir output
524 running 1 tests using 1 parallel processes
524 running 1 tests using 1 parallel processes
525
525
526 --- $TESTTMP/test-failure.t
526 --- $TESTTMP/test-failure.t
527 +++ $TESTTMP/output/test-failure.t.err
527 +++ $TESTTMP/output/test-failure.t.err
528 @@ -1,5 +1,5 @@
528 @@ -1,5 +1,5 @@
529 $ echo babar
529 $ echo babar
530 - rataxes
530 - rataxes
531 + babar
531 + babar
532 This is a noop statement so that
532 This is a noop statement so that
533 this test is still more bytes than success.
533 this test is still more bytes than success.
534 pad pad pad pad............................................................
534 pad pad pad pad............................................................
535
535
536 ERROR: test-failure.t output changed
536 ERROR: test-failure.t output changed
537 !
537 !
538 Failed test-failure.t: output changed
538 Failed test-failure.t: output changed
539 # Ran 1 tests, 0 skipped, 1 failed.
539 # Ran 1 tests, 0 skipped, 1 failed.
540 python hash seed: * (glob)
540 python hash seed: * (glob)
541 [1]
541 [1]
542
542
543 Selecting Tests To Run
543 Selecting Tests To Run
544 ======================
544 ======================
545
545
546 successful
546 successful
547
547
548 $ rt test-success.t
548 $ rt test-success.t
549 running 1 tests using 1 parallel processes
549 running 1 tests using 1 parallel processes
550 .
550 .
551 # Ran 1 tests, 0 skipped, 0 failed.
551 # Ran 1 tests, 0 skipped, 0 failed.
552
552
553 success w/ keyword
553 success w/ keyword
554 $ rt -k xyzzy
554 $ rt -k xyzzy
555 running 2 tests using 1 parallel processes
555 running 2 tests using 1 parallel processes
556 .
556 .
557 # Ran 2 tests, 1 skipped, 0 failed.
557 # Ran 2 tests, 1 skipped, 0 failed.
558
558
559 failed
559 failed
560
560
561 $ rt test-failure.t
561 $ rt test-failure.t
562 running 1 tests using 1 parallel processes
562 running 1 tests using 1 parallel processes
563
563
564 --- $TESTTMP/test-failure.t
564 --- $TESTTMP/test-failure.t
565 +++ $TESTTMP/test-failure.t.err
565 +++ $TESTTMP/test-failure.t.err
566 @@ -1,5 +1,5 @@
566 @@ -1,5 +1,5 @@
567 $ echo babar
567 $ echo babar
568 - rataxes
568 - rataxes
569 + babar
569 + babar
570 This is a noop statement so that
570 This is a noop statement so that
571 this test is still more bytes than success.
571 this test is still more bytes than success.
572 pad pad pad pad............................................................
572 pad pad pad pad............................................................
573
573
574 ERROR: test-failure.t output changed
574 ERROR: test-failure.t output changed
575 !
575 !
576 Failed test-failure.t: output changed
576 Failed test-failure.t: output changed
577 # Ran 1 tests, 0 skipped, 1 failed.
577 # Ran 1 tests, 0 skipped, 1 failed.
578 python hash seed: * (glob)
578 python hash seed: * (glob)
579 [1]
579 [1]
580
580
581 failure w/ keyword
581 failure w/ keyword
582 $ rt -k rataxes
582 $ rt -k rataxes
583 running 2 tests using 1 parallel processes
583 running 2 tests using 1 parallel processes
584
584
585 --- $TESTTMP/test-failure.t
585 --- $TESTTMP/test-failure.t
586 +++ $TESTTMP/test-failure.t.err
586 +++ $TESTTMP/test-failure.t.err
587 @@ -1,5 +1,5 @@
587 @@ -1,5 +1,5 @@
588 $ echo babar
588 $ echo babar
589 - rataxes
589 - rataxes
590 + babar
590 + babar
591 This is a noop statement so that
591 This is a noop statement so that
592 this test is still more bytes than success.
592 this test is still more bytes than success.
593 pad pad pad pad............................................................
593 pad pad pad pad............................................................
594
594
595 ERROR: test-failure.t output changed
595 ERROR: test-failure.t output changed
596 !
596 !
597 Failed test-failure.t: output changed
597 Failed test-failure.t: output changed
598 # Ran 2 tests, 1 skipped, 1 failed.
598 # Ran 2 tests, 1 skipped, 1 failed.
599 python hash seed: * (glob)
599 python hash seed: * (glob)
600 [1]
600 [1]
601
601
602 Verify that when a process fails to start we show a useful message
602 Verify that when a process fails to start we show a useful message
603 ==================================================================
603 ==================================================================
604
604
605 $ cat > test-serve-fail.t <<EOF
605 $ cat > test-serve-fail.t <<EOF
606 > $ echo 'abort: child process failed to start blah'
606 > $ echo 'abort: child process failed to start blah'
607 > EOF
607 > EOF
608 $ rt test-serve-fail.t
608 $ rt test-serve-fail.t
609 running 1 tests using 1 parallel processes
609 running 1 tests using 1 parallel processes
610
610
611 --- $TESTTMP/test-serve-fail.t
611 --- $TESTTMP/test-serve-fail.t
612 +++ $TESTTMP/test-serve-fail.t.err
612 +++ $TESTTMP/test-serve-fail.t.err
613 @@ -1* +1,2 @@ (glob)
613 @@ -1* +1,2 @@ (glob)
614 $ echo 'abort: child process failed to start blah'
614 $ echo 'abort: child process failed to start blah'
615 + abort: child process failed to start blah
615 + abort: child process failed to start blah
616
616
617 ERROR: test-serve-fail.t output changed
617 ERROR: test-serve-fail.t output changed
618 !
618 !
619 Failed test-serve-fail.t: server failed to start (HGPORT=*) (glob)
619 Failed test-serve-fail.t: server failed to start (HGPORT=*) (glob)
620 # Ran 1 tests, 0 skipped, 1 failed.
620 # Ran 1 tests, 0 skipped, 1 failed.
621 python hash seed: * (glob)
621 python hash seed: * (glob)
622 [1]
622 [1]
623 $ rm test-serve-fail.t
623 $ rm test-serve-fail.t
624
624
625 Verify that we can try other ports
625 Verify that we can try other ports
626 ===================================
626 ===================================
627
627
628 Extensions aren't inherited by the invoked run-tests.py. An extension
628 Extensions aren't inherited by the invoked run-tests.py. An extension
629 introducing a repository requirement could cause this to fail. So we force
629 introducing a repository requirement could cause this to fail. So we force
630 HGRCPATH to get a clean environment.
630 HGRCPATH to get a clean environment.
631
631
632 $ HGRCPATH= hg init inuse
632 $ HGRCPATH= hg init inuse
633 $ hg serve -R inuse -p $HGPORT -d --pid-file=blocks.pid
633 $ hg serve -R inuse -p $HGPORT -d --pid-file=blocks.pid
634 $ cat blocks.pid >> $DAEMON_PIDS
634 $ cat blocks.pid >> $DAEMON_PIDS
635 $ cat > test-serve-inuse.t <<EOF
635 $ cat > test-serve-inuse.t <<EOF
636 > $ hg serve -R `pwd`/inuse -p \$HGPORT -d --pid-file=hg.pid
636 > $ hg serve -R `pwd`/inuse -p \$HGPORT -d --pid-file=hg.pid
637 > $ cat hg.pid >> \$DAEMON_PIDS
637 > $ cat hg.pid >> \$DAEMON_PIDS
638 > EOF
638 > EOF
639 $ rt test-serve-inuse.t
639 $ rt test-serve-inuse.t
640 running 1 tests using 1 parallel processes
640 running 1 tests using 1 parallel processes
641 .
641 .
642 # Ran 1 tests, 0 skipped, 0 failed.
642 # Ran 1 tests, 0 skipped, 0 failed.
643 $ rm test-serve-inuse.t
643 $ rm test-serve-inuse.t
644 $ killdaemons.py $DAEMON_PIDS
644 $ killdaemons.py $DAEMON_PIDS
645
645
646 Running In Debug Mode
646 Running In Debug Mode
647 ======================
647 ======================
648
648
649 $ rt --debug 2>&1 | grep -v pwd
649 $ rt --debug 2>&1 | grep -v pwd
650 running 2 tests using 1 parallel processes
650 running 2 tests using 1 parallel processes
651 + alias hg=hg.exe (windows !)
651 + alias hg=hg.exe (windows !)
652 + echo *SALT* 0 0 (glob)
652 + echo *SALT* 0 0 (glob)
653 *SALT* 0 0 (glob)
653 *SALT* 0 0 (glob)
654 + echo babar
654 + echo babar
655 babar
655 babar
656 + echo *SALT* 10 0 (glob)
656 + echo *SALT* 10 0 (glob)
657 *SALT* 10 0 (glob)
657 *SALT* 10 0 (glob)
658 .+ alias hg=hg.exe (windows !)
658 .+ alias hg=hg.exe (windows !)
659 *+ echo *SALT* 0 0 (glob)
659 *+ echo *SALT* 0 0 (glob)
660 *SALT* 0 0 (glob)
660 *SALT* 0 0 (glob)
661 + echo babar
661 + echo babar
662 babar
662 babar
663 + echo *SALT* 2 0 (glob)
663 + echo *SALT* 2 0 (glob)
664 *SALT* 2 0 (glob)
664 *SALT* 2 0 (glob)
665 + echo xyzzy
665 + echo xyzzy
666 xyzzy
666 xyzzy
667 + echo *SALT* 9 0 (glob)
667 + echo *SALT* 9 0 (glob)
668 *SALT* 9 0 (glob)
668 *SALT* 9 0 (glob)
669 + printf *abc\ndef\nxyz\n* (glob)
669 + printf *abc\ndef\nxyz\n* (glob)
670 abc
670 abc
671 def
671 def
672 xyz
672 xyz
673 + echo *SALT* 15 0 (glob)
673 + echo *SALT* 15 0 (glob)
674 *SALT* 15 0 (glob)
674 *SALT* 15 0 (glob)
675 + printf *zyx\nwvu\ntsr\n* (glob)
675 + printf *zyx\nwvu\ntsr\n* (glob)
676 zyx
676 zyx
677 wvu
677 wvu
678 tsr
678 tsr
679 + echo *SALT* 22 0 (glob)
679 + echo *SALT* 22 0 (glob)
680 *SALT* 22 0 (glob)
680 *SALT* 22 0 (glob)
681 .
681 .
682 # Ran 2 tests, 0 skipped, 0 failed.
682 # Ran 2 tests, 0 skipped, 0 failed.
683
683
684 Parallel runs
684 Parallel runs
685 ==============
685 ==============
686
686
687 (duplicate the failing test to get predictable output)
687 (duplicate the failing test to get predictable output)
688 $ cp test-failure.t test-failure-copy.t
688 $ cp test-failure.t test-failure-copy.t
689
689
690 $ rt --jobs 2 test-failure*.t -n
690 $ rt --jobs 2 test-failure*.t -n
691 running 2 tests using 2 parallel processes
691 running 2 tests using 2 parallel processes
692 !!
692 !!
693 Failed test-failure*.t: output changed (glob)
693 Failed test-failure*.t: output changed (glob)
694 Failed test-failure*.t: output changed (glob)
694 Failed test-failure*.t: output changed (glob)
695 # Ran 2 tests, 0 skipped, 2 failed.
695 # Ran 2 tests, 0 skipped, 2 failed.
696 python hash seed: * (glob)
696 python hash seed: * (glob)
697 [1]
697 [1]
698
698
699 failures in parallel with --first should only print one failure
699 failures in parallel with --first should only print one failure
700 $ rt --jobs 2 --first test-failure*.t
700 $ rt --jobs 2 --first test-failure*.t
701 running 2 tests using 2 parallel processes
701 running 2 tests using 2 parallel processes
702
702
703 --- $TESTTMP/test-failure*.t (glob)
703 --- $TESTTMP/test-failure*.t (glob)
704 +++ $TESTTMP/test-failure*.t.err (glob)
704 +++ $TESTTMP/test-failure*.t.err (glob)
705 @@ -1,5 +1,5 @@
705 @@ -1,5 +1,5 @@
706 $ echo babar
706 $ echo babar
707 - rataxes
707 - rataxes
708 + babar
708 + babar
709 This is a noop statement so that
709 This is a noop statement so that
710 this test is still more bytes than success.
710 this test is still more bytes than success.
711 pad pad pad pad............................................................
711 pad pad pad pad............................................................
712
712
713 Failed test-failure*.t: output changed (glob)
713 Failed test-failure*.t: output changed (glob)
714 Failed test-failure*.t: output changed (glob)
714 Failed test-failure*.t: output changed (glob)
715 # Ran 2 tests, 0 skipped, 2 failed.
715 # Ran 2 tests, 0 skipped, 2 failed.
716 python hash seed: * (glob)
716 python hash seed: * (glob)
717 [1]
717 [1]
718
718
719
719
720 (delete the duplicated test file)
720 (delete the duplicated test file)
721 $ rm test-failure-copy.t
721 $ rm test-failure-copy.t
722
722
723 multiple runs per test should be parallelized
723 multiple runs per test should be parallelized
724
724
725 $ rt --jobs 2 --runs-per-test 2 test-success.t
725 $ rt --jobs 2 --runs-per-test 2 test-success.t
726 running 2 tests using 2 parallel processes
726 running 2 tests using 2 parallel processes
727 ..
727 ..
728 # Ran 2 tests, 0 skipped, 0 failed.
728 # Ran 2 tests, 0 skipped, 0 failed.
729
729
730 Interactive run
730 Interactive run
731 ===============
731 ===============
732
732
733 (backup the failing test)
733 (backup the failing test)
734 $ cp test-failure.t backup
734 $ cp test-failure.t backup
735
735
736 Refuse the fix
736 Refuse the fix
737
737
738 $ echo 'n' | rt -i
738 $ echo 'n' | rt -i
739 running 2 tests using 1 parallel processes
739 running 2 tests using 1 parallel processes
740
740
741 --- $TESTTMP/test-failure.t
741 --- $TESTTMP/test-failure.t
742 +++ $TESTTMP/test-failure.t.err
742 +++ $TESTTMP/test-failure.t.err
743 @@ -1,5 +1,5 @@
743 @@ -1,5 +1,5 @@
744 $ echo babar
744 $ echo babar
745 - rataxes
745 - rataxes
746 + babar
746 + babar
747 This is a noop statement so that
747 This is a noop statement so that
748 this test is still more bytes than success.
748 this test is still more bytes than success.
749 pad pad pad pad............................................................
749 pad pad pad pad............................................................
750 Accept this change? [y/N]
750 Accept this change? [y/N]
751 ERROR: test-failure.t output changed
751 ERROR: test-failure.t output changed
752 !.
752 !.
753 Failed test-failure.t: output changed
753 Failed test-failure.t: output changed
754 # Ran 2 tests, 0 skipped, 1 failed.
754 # Ran 2 tests, 0 skipped, 1 failed.
755 python hash seed: * (glob)
755 python hash seed: * (glob)
756 [1]
756 [1]
757
757
758 $ cat test-failure.t
758 $ cat test-failure.t
759 $ echo babar
759 $ echo babar
760 rataxes
760 rataxes
761 This is a noop statement so that
761 This is a noop statement so that
762 this test is still more bytes than success.
762 this test is still more bytes than success.
763 pad pad pad pad............................................................
763 pad pad pad pad............................................................
764 pad pad pad pad............................................................
764 pad pad pad pad............................................................
765 pad pad pad pad............................................................
765 pad pad pad pad............................................................
766 pad pad pad pad............................................................
766 pad pad pad pad............................................................
767 pad pad pad pad............................................................
767 pad pad pad pad............................................................
768 pad pad pad pad............................................................
768 pad pad pad pad............................................................
769
769
770 Interactive with custom view
770 Interactive with custom view
771
771
772 $ echo 'n' | rt -i --view echo
772 $ echo 'n' | rt -i --view echo
773 running 2 tests using 1 parallel processes
773 running 2 tests using 1 parallel processes
774 $TESTTMP/test-failure.t $TESTTMP/test-failure.t.err
774 $TESTTMP/test-failure.t $TESTTMP/test-failure.t.err
775 Accept this change? [y/N]* (glob)
775 Accept this change? [y/N]* (glob)
776 ERROR: test-failure.t output changed
776 ERROR: test-failure.t output changed
777 !.
777 !.
778 Failed test-failure.t: output changed
778 Failed test-failure.t: output changed
779 # Ran 2 tests, 0 skipped, 1 failed.
779 # Ran 2 tests, 0 skipped, 1 failed.
780 python hash seed: * (glob)
780 python hash seed: * (glob)
781 [1]
781 [1]
782
782
783 View the fix
783 View the fix
784
784
785 $ echo 'y' | rt --view echo
785 $ echo 'y' | rt --view echo
786 running 2 tests using 1 parallel processes
786 running 2 tests using 1 parallel processes
787 $TESTTMP/test-failure.t $TESTTMP/test-failure.t.err
787 $TESTTMP/test-failure.t $TESTTMP/test-failure.t.err
788
788
789 ERROR: test-failure.t output changed
789 ERROR: test-failure.t output changed
790 !.
790 !.
791 Failed test-failure.t: output changed
791 Failed test-failure.t: output changed
792 # Ran 2 tests, 0 skipped, 1 failed.
792 # Ran 2 tests, 0 skipped, 1 failed.
793 python hash seed: * (glob)
793 python hash seed: * (glob)
794 [1]
794 [1]
795
795
796 Accept the fix
796 Accept the fix
797
797
798 $ cat >> test-failure.t <<EOF
798 $ cat >> test-failure.t <<EOF
799 > $ echo 'saved backup bundle to \$TESTTMP/foo.hg'
799 > $ echo 'saved backup bundle to \$TESTTMP/foo.hg'
800 > saved backup bundle to \$TESTTMP/foo.hg
800 > saved backup bundle to \$TESTTMP/foo.hg
801 > $ echo 'saved backup bundle to \$TESTTMP/foo.hg'
801 > $ echo 'saved backup bundle to \$TESTTMP/foo.hg'
802 > saved backup bundle to $TESTTMP\\foo.hg
802 > saved backup bundle to $TESTTMP\\foo.hg
803 > $ echo 'saved backup bundle to \$TESTTMP/foo.hg'
803 > $ echo 'saved backup bundle to \$TESTTMP/foo.hg'
804 > saved backup bundle to \$TESTTMP/*.hg (glob)
804 > saved backup bundle to \$TESTTMP/*.hg (glob)
805 > EOF
805 > EOF
806 $ echo 'y' | rt -i 2>&1
806 $ echo 'y' | rt -i 2>&1
807 running 2 tests using 1 parallel processes
807 running 2 tests using 1 parallel processes
808
808
809 --- $TESTTMP/test-failure.t
809 --- $TESTTMP/test-failure.t
810 +++ $TESTTMP/test-failure.t.err
810 +++ $TESTTMP/test-failure.t.err
811 @@ -1,5 +1,5 @@
811 @@ -1,5 +1,5 @@
812 $ echo babar
812 $ echo babar
813 - rataxes
813 - rataxes
814 + babar
814 + babar
815 This is a noop statement so that
815 This is a noop statement so that
816 this test is still more bytes than success.
816 this test is still more bytes than success.
817 pad pad pad pad............................................................
817 pad pad pad pad............................................................
818 @@ -11,6 +11,6 @@
818 @@ -11,6 +11,6 @@
819 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
819 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
820 saved backup bundle to $TESTTMP/foo.hg
820 saved backup bundle to $TESTTMP/foo.hg
821 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
821 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
822 - saved backup bundle to $TESTTMP\foo.hg
822 - saved backup bundle to $TESTTMP\foo.hg
823 + saved backup bundle to $TESTTMP/foo.hg
823 + saved backup bundle to $TESTTMP/foo.hg
824 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
824 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
825 saved backup bundle to $TESTTMP/*.hg (glob)
825 saved backup bundle to $TESTTMP/*.hg (glob)
826 Accept this change? [y/N] ..
826 Accept this change? [y/N] ..
827 # Ran 2 tests, 0 skipped, 0 failed.
827 # Ran 2 tests, 0 skipped, 0 failed.
828
828
829 $ sed -e 's,(glob)$,&<,g' test-failure.t
829 $ sed -e 's,(glob)$,&<,g' test-failure.t
830 $ echo babar
830 $ echo babar
831 babar
831 babar
832 This is a noop statement so that
832 This is a noop statement so that
833 this test is still more bytes than success.
833 this test is still more bytes than success.
834 pad pad pad pad............................................................
834 pad pad pad pad............................................................
835 pad pad pad pad............................................................
835 pad pad pad pad............................................................
836 pad pad pad pad............................................................
836 pad pad pad pad............................................................
837 pad pad pad pad............................................................
837 pad pad pad pad............................................................
838 pad pad pad pad............................................................
838 pad pad pad pad............................................................
839 pad pad pad pad............................................................
839 pad pad pad pad............................................................
840 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
840 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
841 saved backup bundle to $TESTTMP/foo.hg
841 saved backup bundle to $TESTTMP/foo.hg
842 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
842 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
843 saved backup bundle to $TESTTMP/foo.hg
843 saved backup bundle to $TESTTMP/foo.hg
844 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
844 $ echo 'saved backup bundle to $TESTTMP/foo.hg'
845 saved backup bundle to $TESTTMP/*.hg (glob)<
845 saved backup bundle to $TESTTMP/*.hg (glob)<
846
846
847 $ rm test-failure.t
847 $ rm test-failure.t
848
848
849 Race condition - test file was modified when test is running
849 Race condition - test file was modified when test is running
850
850
851 $ TESTRACEDIR=`pwd`
851 $ TESTRACEDIR=`pwd`
852 $ export TESTRACEDIR
852 $ export TESTRACEDIR
853 $ cat > test-race.t <<EOF
853 $ cat > test-race.t <<EOF
854 > $ echo 1
854 > $ echo 1
855 > $ echo "# a new line" >> $TESTRACEDIR/test-race.t
855 > $ echo "# a new line" >> $TESTRACEDIR/test-race.t
856 > EOF
856 > EOF
857
857
858 $ rt -i test-race.t
858 $ rt -i test-race.t
859 running 1 tests using 1 parallel processes
859 running 1 tests using 1 parallel processes
860
860
861 --- $TESTTMP/test-race.t
861 --- $TESTTMP/test-race.t
862 +++ $TESTTMP/test-race.t.err
862 +++ $TESTTMP/test-race.t.err
863 @@ -1,2 +1,3 @@
863 @@ -1,2 +1,3 @@
864 $ echo 1
864 $ echo 1
865 + 1
865 + 1
866 $ echo "# a new line" >> $TESTTMP/test-race.t
866 $ echo "# a new line" >> $TESTTMP/test-race.t
867 Reference output has changed (run again to prompt changes)
867 Reference output has changed (run again to prompt changes)
868 ERROR: test-race.t output changed
868 ERROR: test-race.t output changed
869 !
869 !
870 Failed test-race.t: output changed
870 Failed test-race.t: output changed
871 # Ran 1 tests, 0 skipped, 1 failed.
871 # Ran 1 tests, 0 skipped, 1 failed.
872 python hash seed: * (glob)
872 python hash seed: * (glob)
873 [1]
873 [1]
874
874
875 $ rm test-race.t
875 $ rm test-race.t
876
876
877 When "#testcases" is used in .t files
877 When "#testcases" is used in .t files
878
878
879 $ cat >> test-cases.t <<EOF
879 $ cat >> test-cases.t <<EOF
880 > #testcases a b
880 > #testcases a b
881 > #if a
881 > #if a
882 > $ echo 1
882 > $ echo 1
883 > #endif
883 > #endif
884 > #if b
884 > #if b
885 > $ echo 2
885 > $ echo 2
886 > #endif
886 > #endif
887 > EOF
887 > EOF
888
888
889 $ cat <<EOF | rt -i test-cases.t 2>&1
889 $ cat <<EOF | rt -i test-cases.t 2>&1
890 > y
890 > y
891 > y
891 > y
892 > EOF
892 > EOF
893 running 2 tests using 1 parallel processes
893 running 2 tests using 1 parallel processes
894
894
895 --- $TESTTMP/test-cases.t
895 --- $TESTTMP/test-cases.t
896 +++ $TESTTMP/test-cases.t#a.err
896 +++ $TESTTMP/test-cases.t#a.err
897 @@ -1,6 +1,7 @@
897 @@ -1,6 +1,7 @@
898 #testcases a b
898 #testcases a b
899 #if a
899 #if a
900 $ echo 1
900 $ echo 1
901 + 1
901 + 1
902 #endif
902 #endif
903 #if b
903 #if b
904 $ echo 2
904 $ echo 2
905 Accept this change? [y/N] .
905 Accept this change? [y/N] .
906 --- $TESTTMP/test-cases.t
906 --- $TESTTMP/test-cases.t
907 +++ $TESTTMP/test-cases.t#b.err
907 +++ $TESTTMP/test-cases.t#b.err
908 @@ -5,4 +5,5 @@
908 @@ -5,4 +5,5 @@
909 #endif
909 #endif
910 #if b
910 #if b
911 $ echo 2
911 $ echo 2
912 + 2
912 + 2
913 #endif
913 #endif
914 Accept this change? [y/N] .
914 Accept this change? [y/N] .
915 # Ran 2 tests, 0 skipped, 0 failed.
915 # Ran 2 tests, 0 skipped, 0 failed.
916
916
917 $ cat test-cases.t
917 $ cat test-cases.t
918 #testcases a b
918 #testcases a b
919 #if a
919 #if a
920 $ echo 1
920 $ echo 1
921 1
921 1
922 #endif
922 #endif
923 #if b
923 #if b
924 $ echo 2
924 $ echo 2
925 2
925 2
926 #endif
926 #endif
927
927
928 $ cat >> test-cases.t <<'EOF'
928 $ cat >> test-cases.t <<'EOF'
929 > #if a
929 > #if a
930 > $ NAME=A
930 > $ NAME=A
931 > #else
931 > #else
932 > $ NAME=B
932 > $ NAME=B
933 > #endif
933 > #endif
934 > $ echo $NAME
934 > $ echo $NAME
935 > A (a !)
935 > A (a !)
936 > B (b !)
936 > B (b !)
937 > EOF
937 > EOF
938 $ rt test-cases.t
938 $ rt test-cases.t
939 running 2 tests using 1 parallel processes
939 running 2 tests using 1 parallel processes
940 ..
940 ..
941 # Ran 2 tests, 0 skipped, 0 failed.
941 # Ran 2 tests, 0 skipped, 0 failed.
942
942
943 When using multiple dimensions of "#testcases" in .t files
943 When using multiple dimensions of "#testcases" in .t files
944
944
945 $ cat > test-cases.t <<'EOF'
945 $ cat > test-cases.t <<'EOF'
946 > #testcases a b
946 > #testcases a b
947 > #testcases c d
947 > #testcases c d
948 > #if a d
948 > #if a d
949 > $ echo $TESTCASE
949 > $ echo $TESTCASE
950 > a#d
950 > a#d
951 > #endif
951 > #endif
952 > #if b c
952 > #if b c
953 > $ echo yes
953 > $ echo yes
954 > no
954 > no
955 > #endif
955 > #endif
956 > EOF
956 > EOF
957 $ rt test-cases.t
957 $ rt test-cases.t
958 running 4 tests using 1 parallel processes
958 running 4 tests using 1 parallel processes
959 ..
959 ..
960 --- $TESTTMP/test-cases.t
960 --- $TESTTMP/test-cases.t
961 +++ $TESTTMP/test-cases.t#b#c.err
961 +++ $TESTTMP/test-cases.t#b#c.err
962 @@ -6,5 +6,5 @@
962 @@ -6,5 +6,5 @@
963 #endif
963 #endif
964 #if b c
964 #if b c
965 $ echo yes
965 $ echo yes
966 - no
966 - no
967 + yes
967 + yes
968 #endif
968 #endif
969
969
970 ERROR: test-cases.t#b#c output changed
970 ERROR: test-cases.t#b#c output changed
971 !.
971 !.
972 Failed test-cases.t#b#c: output changed
972 Failed test-cases.t#b#c: output changed
973 # Ran 4 tests, 0 skipped, 1 failed.
973 # Ran 4 tests, 0 skipped, 1 failed.
974 python hash seed: * (glob)
974 python hash seed: * (glob)
975 [1]
975 [1]
976
976
977 $ rt --retest
977 $ rt --retest
978 running 1 tests using 1 parallel processes
978 running 1 tests using 1 parallel processes
979
979
980 --- $TESTTMP/test-cases.t
980 --- $TESTTMP/test-cases.t
981 +++ $TESTTMP/test-cases.t#b#c.err
981 +++ $TESTTMP/test-cases.t#b#c.err
982 @@ -6,5 +6,5 @@
982 @@ -6,5 +6,5 @@
983 #endif
983 #endif
984 #if b c
984 #if b c
985 $ echo yes
985 $ echo yes
986 - no
986 - no
987 + yes
987 + yes
988 #endif
988 #endif
989
989
990 ERROR: test-cases.t#b#c output changed
990 ERROR: test-cases.t#b#c output changed
991 !
991 !
992 Failed test-cases.t#b#c: output changed
992 Failed test-cases.t#b#c: output changed
993 # Ran 1 tests, 0 skipped, 1 failed.
993 # Ran 1 tests, 0 skipped, 1 failed.
994 python hash seed: * (glob)
994 python hash seed: * (glob)
995 [1]
995 [1]
996 $ rm test-cases.t#b#c.err
996 $ rm test-cases.t#b#c.err
997 $ rm test-cases.t
997 $ rm test-cases.t
998
998
999 (reinstall)
999 (reinstall)
1000 $ mv backup test-failure.t
1000 $ mv backup test-failure.t
1001
1001
1002 No Diff
1002 No Diff
1003 ===============
1003 ===============
1004
1004
1005 $ rt --nodiff
1005 $ rt --nodiff
1006 running 2 tests using 1 parallel processes
1006 running 2 tests using 1 parallel processes
1007 !.
1007 !.
1008 Failed test-failure.t: output changed
1008 Failed test-failure.t: output changed
1009 # Ran 2 tests, 0 skipped, 1 failed.
1009 # Ran 2 tests, 0 skipped, 1 failed.
1010 python hash seed: * (glob)
1010 python hash seed: * (glob)
1011 [1]
1011 [1]
1012
1012
1013 test --tmpdir support
1013 test --tmpdir support
1014 $ rt --tmpdir=$TESTTMP/keep test-success.t
1014 $ rt --tmpdir=$TESTTMP/keep test-success.t
1015 running 1 tests using 1 parallel processes
1015 running 1 tests using 1 parallel processes
1016
1016
1017 Keeping testtmp dir: $TESTTMP/keep/child1/test-success.t
1017 Keeping testtmp dir: $TESTTMP/keep/child1/test-success.t
1018 Keeping threadtmp dir: $TESTTMP/keep/child1
1018 Keeping threadtmp dir: $TESTTMP/keep/child1
1019 .
1019 .
1020 # Ran 1 tests, 0 skipped, 0 failed.
1020 # Ran 1 tests, 0 skipped, 0 failed.
1021
1021
1022 timeouts
1022 timeouts
1023 ========
1023 ========
1024 $ cat > test-timeout.t <<EOF
1024 $ cat > test-timeout.t <<EOF
1025 > $ sleep 2
1025 > $ sleep 2
1026 > $ echo pass
1026 > $ echo pass
1027 > pass
1027 > pass
1028 > EOF
1028 > EOF
1029 > echo '#require slow' > test-slow-timeout.t
1029 > echo '#require slow' > test-slow-timeout.t
1030 > cat test-timeout.t >> test-slow-timeout.t
1030 > cat test-timeout.t >> test-slow-timeout.t
1031 $ rt --timeout=1 --slowtimeout=3 test-timeout.t test-slow-timeout.t
1031 $ rt --timeout=1 --slowtimeout=3 test-timeout.t test-slow-timeout.t
1032 running 2 tests using 1 parallel processes
1032 running 2 tests using 1 parallel processes
1033 st
1033 st
1034 Skipped test-slow-timeout.t: missing feature: allow slow tests (use --allow-slow-tests)
1034 Skipped test-slow-timeout.t: missing feature: allow slow tests (use --allow-slow-tests)
1035 Failed test-timeout.t: timed out
1035 Failed test-timeout.t: timed out
1036 # Ran 1 tests, 1 skipped, 1 failed.
1036 # Ran 1 tests, 1 skipped, 1 failed.
1037 python hash seed: * (glob)
1037 python hash seed: * (glob)
1038 [1]
1038 [1]
1039 $ rt --timeout=1 --slowtimeout=3 \
1039 $ rt --timeout=1 --slowtimeout=3 \
1040 > test-timeout.t test-slow-timeout.t --allow-slow-tests
1040 > test-timeout.t test-slow-timeout.t --allow-slow-tests
1041 running 2 tests using 1 parallel processes
1041 running 2 tests using 1 parallel processes
1042 .t
1042 .t
1043 Failed test-timeout.t: timed out
1043 Failed test-timeout.t: timed out
1044 # Ran 2 tests, 0 skipped, 1 failed.
1044 # Ran 2 tests, 0 skipped, 1 failed.
1045 python hash seed: * (glob)
1045 python hash seed: * (glob)
1046 [1]
1046 [1]
1047 $ rm test-timeout.t test-slow-timeout.t
1047 $ rm test-timeout.t test-slow-timeout.t
1048
1048
1049 test for --time
1049 test for --time
1050 ==================
1050 ==================
1051
1051
1052 $ rt test-success.t --time
1052 $ rt test-success.t --time
1053 running 1 tests using 1 parallel processes
1053 running 1 tests using 1 parallel processes
1054 .
1054 .
1055 # Ran 1 tests, 0 skipped, 0 failed.
1055 # Ran 1 tests, 0 skipped, 0 failed.
1056 # Producing time report
1056 # Producing time report
1057 start end cuser csys real Test
1057 start end cuser csys real Test
1058 \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} test-success.t (re)
1058 \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} test-success.t (re)
1059
1059
1060 test for --time with --job enabled
1060 test for --time with --job enabled
1061 ====================================
1061 ====================================
1062
1062
1063 $ rt test-success.t --time --jobs 2
1063 $ rt test-success.t --time --jobs 2
1064 running 1 tests using 1 parallel processes
1064 running 1 tests using 1 parallel processes
1065 .
1065 .
1066 # Ran 1 tests, 0 skipped, 0 failed.
1066 # Ran 1 tests, 0 skipped, 0 failed.
1067 # Producing time report
1067 # Producing time report
1068 start end cuser csys real Test
1068 start end cuser csys real Test
1069 \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} test-success.t (re)
1069 \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} \s*[\d\.]{5,8} test-success.t (re)
1070
1070
1071 Skips
1071 Skips
1072 ================
1072 ================
1073 $ cat > test-skip.t <<EOF
1073 $ cat > test-skip.t <<EOF
1074 > $ echo xyzzy
1074 > $ echo xyzzy
1075 > #if true
1075 > #if true
1076 > #require false
1076 > #require false
1077 > #end
1077 > #end
1078 > EOF
1078 > EOF
1079 $ cat > test-noskip.t <<EOF
1079 $ cat > test-noskip.t <<EOF
1080 > #if false
1080 > #if false
1081 > #require false
1081 > #require false
1082 > #endif
1082 > #endif
1083 > EOF
1083 > EOF
1084 $ rt --nodiff
1084 $ rt --nodiff
1085 running 4 tests using 1 parallel processes
1085 running 4 tests using 1 parallel processes
1086 !.s.
1086 !.s.
1087 Skipped test-skip.t: missing feature: nail clipper
1087 Skipped test-skip.t: missing feature: nail clipper
1088 Failed test-failure.t: output changed
1088 Failed test-failure.t: output changed
1089 # Ran 3 tests, 1 skipped, 1 failed.
1089 # Ran 3 tests, 1 skipped, 1 failed.
1090 python hash seed: * (glob)
1090 python hash seed: * (glob)
1091 [1]
1091 [1]
1092
1092
1093 $ rm test-noskip.t
1093 $ rm test-noskip.t
1094 $ rt --keyword xyzzy
1094 $ rt --keyword xyzzy
1095 running 3 tests using 1 parallel processes
1095 running 3 tests using 1 parallel processes
1096 .s
1096 .s
1097 Skipped test-skip.t: missing feature: nail clipper
1097 Skipped test-skip.t: missing feature: nail clipper
1098 # Ran 2 tests, 2 skipped, 0 failed.
1098 # Ran 2 tests, 2 skipped, 0 failed.
1099
1099
1100 Skips with xml
1100 Skips with xml
1101 $ rt --keyword xyzzy \
1101 $ rt --keyword xyzzy \
1102 > --xunit=xunit.xml
1102 > --xunit=xunit.xml
1103 running 3 tests using 1 parallel processes
1103 running 3 tests using 1 parallel processes
1104 .s
1104 .s
1105 Skipped test-skip.t: missing feature: nail clipper
1105 Skipped test-skip.t: missing feature: nail clipper
1106 # Ran 2 tests, 2 skipped, 0 failed.
1106 # Ran 2 tests, 2 skipped, 0 failed.
1107 $ cat xunit.xml
1107 $ cat xunit.xml
1108 <?xml version="1.0" encoding="utf-8"?>
1108 <?xml version="1.0" encoding="utf-8"?>
1109 <testsuite errors="0" failures="0" name="run-tests" skipped="2" tests="2">
1109 <testsuite errors="0" failures="0" name="run-tests" skipped="2" tests="2">
1110 <testcase name="test-success.t" time="*"/> (glob)
1110 <testcase name="test-success.t" time="*"/> (glob)
1111 <testcase name="test-skip.t">
1111 <testcase name="test-skip.t">
1112 <skipped><![CDATA[missing feature: nail clipper]]></skipped> (py38 !)
1112 <skipped><![CDATA[missing feature: nail clipper]]></skipped> (py38 !)
1113 <skipped> (no-py38 !)
1113 <skipped> (no-py38 !)
1114 <![CDATA[missing feature: nail clipper]]> </skipped> (no-py38 !)
1114 <![CDATA[missing feature: nail clipper]]> </skipped> (no-py38 !)
1115 </testcase>
1115 </testcase>
1116 </testsuite>
1116 </testsuite>
1117
1117
1118 Missing skips or blacklisted skips don't count as executed:
1118 Missing skips or blacklisted skips don't count as executed:
1119 $ echo test-failure.t > blacklist
1119 $ echo test-failure.t > blacklist
1120 $ rt --blacklist=blacklist --json\
1120 $ rt --blacklist=blacklist --json\
1121 > test-failure.t test-bogus.t
1121 > test-failure.t test-bogus.t
1122 running 2 tests using 1 parallel processes
1122 running 2 tests using 1 parallel processes
1123 ss
1123 ss
1124 Skipped test-bogus.t: Doesn't exist
1124 Skipped test-bogus.t: Doesn't exist
1125 Skipped test-failure.t: blacklisted
1125 Skipped test-failure.t: blacklisted
1126 # Ran 0 tests, 2 skipped, 0 failed.
1126 # Ran 0 tests, 2 skipped, 0 failed.
1127 $ cat report.json
1127 $ cat report.json
1128 testreport ={
1128 testreport ={
1129 "test-bogus.t": {
1129 "test-bogus.t": {
1130 "result": "skip"
1130 "result": "skip"
1131 },
1131 },
1132 "test-failure.t": {
1132 "test-failure.t": {
1133 "result": "skip"
1133 "result": "skip"
1134 }
1134 }
1135 } (no-eol)
1135 } (no-eol)
1136
1136
1137 Whitelist trumps blacklist
1137 Whitelist trumps blacklist
1138 $ echo test-failure.t > whitelist
1138 $ echo test-failure.t > whitelist
1139 $ rt --blacklist=blacklist --whitelist=whitelist --json\
1139 $ rt --blacklist=blacklist --whitelist=whitelist --json\
1140 > test-failure.t test-bogus.t
1140 > test-failure.t test-bogus.t
1141 running 2 tests using 1 parallel processes
1141 running 2 tests using 1 parallel processes
1142 s
1142 s
1143 --- $TESTTMP/test-failure.t
1143 --- $TESTTMP/test-failure.t
1144 +++ $TESTTMP/test-failure.t.err
1144 +++ $TESTTMP/test-failure.t.err
1145 @@ -1,5 +1,5 @@
1145 @@ -1,5 +1,5 @@
1146 $ echo babar
1146 $ echo babar
1147 - rataxes
1147 - rataxes
1148 + babar
1148 + babar
1149 This is a noop statement so that
1149 This is a noop statement so that
1150 this test is still more bytes than success.
1150 this test is still more bytes than success.
1151 pad pad pad pad............................................................
1151 pad pad pad pad............................................................
1152
1152
1153 ERROR: test-failure.t output changed
1153 ERROR: test-failure.t output changed
1154 !
1154 !
1155 Skipped test-bogus.t: Doesn't exist
1155 Skipped test-bogus.t: Doesn't exist
1156 Failed test-failure.t: output changed
1156 Failed test-failure.t: output changed
1157 # Ran 1 tests, 1 skipped, 1 failed.
1157 # Ran 1 tests, 1 skipped, 1 failed.
1158 python hash seed: * (glob)
1158 python hash seed: * (glob)
1159 [1]
1159 [1]
1160
1160
1161 Ensure that --test-list causes only the tests listed in that file to
1161 Ensure that --test-list causes only the tests listed in that file to
1162 be executed.
1162 be executed.
1163 $ echo test-success.t >> onlytest
1163 $ echo test-success.t >> onlytest
1164 $ rt --test-list=onlytest
1164 $ rt --test-list=onlytest
1165 running 1 tests using 1 parallel processes
1165 running 1 tests using 1 parallel processes
1166 .
1166 .
1167 # Ran 1 tests, 0 skipped, 0 failed.
1167 # Ran 1 tests, 0 skipped, 0 failed.
1168 $ echo test-bogus.t >> anothertest
1168 $ echo test-bogus.t >> anothertest
1169 $ rt --test-list=onlytest --test-list=anothertest
1169 $ rt --test-list=onlytest --test-list=anothertest
1170 running 2 tests using 1 parallel processes
1170 running 2 tests using 1 parallel processes
1171 s.
1171 s.
1172 Skipped test-bogus.t: Doesn't exist
1172 Skipped test-bogus.t: Doesn't exist
1173 # Ran 1 tests, 1 skipped, 0 failed.
1173 # Ran 1 tests, 1 skipped, 0 failed.
1174 $ rm onlytest anothertest
1174 $ rm onlytest anothertest
1175
1175
1176 test for --json
1176 test for --json
1177 ==================
1177 ==================
1178
1178
1179 $ rt --json
1179 $ rt --json
1180 running 3 tests using 1 parallel processes
1180 running 3 tests using 1 parallel processes
1181
1181
1182 --- $TESTTMP/test-failure.t
1182 --- $TESTTMP/test-failure.t
1183 +++ $TESTTMP/test-failure.t.err
1183 +++ $TESTTMP/test-failure.t.err
1184 @@ -1,5 +1,5 @@
1184 @@ -1,5 +1,5 @@
1185 $ echo babar
1185 $ echo babar
1186 - rataxes
1186 - rataxes
1187 + babar
1187 + babar
1188 This is a noop statement so that
1188 This is a noop statement so that
1189 this test is still more bytes than success.
1189 this test is still more bytes than success.
1190 pad pad pad pad............................................................
1190 pad pad pad pad............................................................
1191
1191
1192 ERROR: test-failure.t output changed
1192 ERROR: test-failure.t output changed
1193 !.s
1193 !.s
1194 Skipped test-skip.t: missing feature: nail clipper
1194 Skipped test-skip.t: missing feature: nail clipper
1195 Failed test-failure.t: output changed
1195 Failed test-failure.t: output changed
1196 # Ran 2 tests, 1 skipped, 1 failed.
1196 # Ran 2 tests, 1 skipped, 1 failed.
1197 python hash seed: * (glob)
1197 python hash seed: * (glob)
1198 [1]
1198 [1]
1199
1199
1200 $ cat report.json
1200 $ cat report.json
1201 testreport ={
1201 testreport ={
1202 "test-failure.t": [\{] (re)
1202 "test-failure.t": [\{] (re)
1203 "csys": "\s*\d+\.\d{3,4}", ? (re)
1203 "csys": "\s*\d+\.\d{3,4}", ? (re)
1204 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1204 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1205 "diff": "---.+\+\+\+.+", ? (re)
1205 "diff": "---.+\+\+\+.+", ? (re)
1206 "end": "\s*\d+\.\d{3,4}", ? (re)
1206 "end": "\s*\d+\.\d{3,4}", ? (re)
1207 "result": "failure", ? (re)
1207 "result": "failure", ? (re)
1208 "start": "\s*\d+\.\d{3,4}", ? (re)
1208 "start": "\s*\d+\.\d{3,4}", ? (re)
1209 "time": "\s*\d+\.\d{3,4}" (re)
1209 "time": "\s*\d+\.\d{3,4}" (re)
1210 }, ? (re)
1210 }, ? (re)
1211 "test-skip.t": {
1211 "test-skip.t": {
1212 "csys": "\s*\d+\.\d{3,4}", ? (re)
1212 "csys": "\s*\d+\.\d{3,4}", ? (re)
1213 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1213 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1214 "diff": "", ? (re)
1214 "diff": "", ? (re)
1215 "end": "\s*\d+\.\d{3,4}", ? (re)
1215 "end": "\s*\d+\.\d{3,4}", ? (re)
1216 "result": "skip", ? (re)
1216 "result": "skip", ? (re)
1217 "start": "\s*\d+\.\d{3,4}", ? (re)
1217 "start": "\s*\d+\.\d{3,4}", ? (re)
1218 "time": "\s*\d+\.\d{3,4}" (re)
1218 "time": "\s*\d+\.\d{3,4}" (re)
1219 }, ? (re)
1219 }, ? (re)
1220 "test-success.t": [\{] (re)
1220 "test-success.t": [\{] (re)
1221 "csys": "\s*\d+\.\d{3,4}", ? (re)
1221 "csys": "\s*\d+\.\d{3,4}", ? (re)
1222 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1222 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1223 "diff": "", ? (re)
1223 "diff": "", ? (re)
1224 "end": "\s*\d+\.\d{3,4}", ? (re)
1224 "end": "\s*\d+\.\d{3,4}", ? (re)
1225 "result": "success", ? (re)
1225 "result": "success", ? (re)
1226 "start": "\s*\d+\.\d{3,4}", ? (re)
1226 "start": "\s*\d+\.\d{3,4}", ? (re)
1227 "time": "\s*\d+\.\d{3,4}" (re)
1227 "time": "\s*\d+\.\d{3,4}" (re)
1228 }
1228 }
1229 } (no-eol)
1229 } (no-eol)
1230 --json with --outputdir
1230 --json with --outputdir
1231
1231
1232 $ rm report.json
1232 $ rm report.json
1233 $ rm -r output
1233 $ rm -r output
1234 $ mkdir output
1234 $ mkdir output
1235 $ rt --json --outputdir output
1235 $ rt --json --outputdir output
1236 running 3 tests using 1 parallel processes
1236 running 3 tests using 1 parallel processes
1237
1237
1238 --- $TESTTMP/test-failure.t
1238 --- $TESTTMP/test-failure.t
1239 +++ $TESTTMP/output/test-failure.t.err
1239 +++ $TESTTMP/output/test-failure.t.err
1240 @@ -1,5 +1,5 @@
1240 @@ -1,5 +1,5 @@
1241 $ echo babar
1241 $ echo babar
1242 - rataxes
1242 - rataxes
1243 + babar
1243 + babar
1244 This is a noop statement so that
1244 This is a noop statement so that
1245 this test is still more bytes than success.
1245 this test is still more bytes than success.
1246 pad pad pad pad............................................................
1246 pad pad pad pad............................................................
1247
1247
1248 ERROR: test-failure.t output changed
1248 ERROR: test-failure.t output changed
1249 !.s
1249 !.s
1250 Skipped test-skip.t: missing feature: nail clipper
1250 Skipped test-skip.t: missing feature: nail clipper
1251 Failed test-failure.t: output changed
1251 Failed test-failure.t: output changed
1252 # Ran 2 tests, 1 skipped, 1 failed.
1252 # Ran 2 tests, 1 skipped, 1 failed.
1253 python hash seed: * (glob)
1253 python hash seed: * (glob)
1254 [1]
1254 [1]
1255 $ f report.json
1255 $ f report.json
1256 report.json: file not found
1256 report.json: file not found
1257 $ cat output/report.json
1257 $ cat output/report.json
1258 testreport ={
1258 testreport ={
1259 "test-failure.t": [\{] (re)
1259 "test-failure.t": [\{] (re)
1260 "csys": "\s*\d+\.\d{3,4}", ? (re)
1260 "csys": "\s*\d+\.\d{3,4}", ? (re)
1261 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1261 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1262 "diff": "---.+\+\+\+.+", ? (re)
1262 "diff": "---.+\+\+\+.+", ? (re)
1263 "end": "\s*\d+\.\d{3,4}", ? (re)
1263 "end": "\s*\d+\.\d{3,4}", ? (re)
1264 "result": "failure", ? (re)
1264 "result": "failure", ? (re)
1265 "start": "\s*\d+\.\d{3,4}", ? (re)
1265 "start": "\s*\d+\.\d{3,4}", ? (re)
1266 "time": "\s*\d+\.\d{3,4}" (re)
1266 "time": "\s*\d+\.\d{3,4}" (re)
1267 }, ? (re)
1267 }, ? (re)
1268 "test-skip.t": {
1268 "test-skip.t": {
1269 "csys": "\s*\d+\.\d{3,4}", ? (re)
1269 "csys": "\s*\d+\.\d{3,4}", ? (re)
1270 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1270 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1271 "diff": "", ? (re)
1271 "diff": "", ? (re)
1272 "end": "\s*\d+\.\d{3,4}", ? (re)
1272 "end": "\s*\d+\.\d{3,4}", ? (re)
1273 "result": "skip", ? (re)
1273 "result": "skip", ? (re)
1274 "start": "\s*\d+\.\d{3,4}", ? (re)
1274 "start": "\s*\d+\.\d{3,4}", ? (re)
1275 "time": "\s*\d+\.\d{3,4}" (re)
1275 "time": "\s*\d+\.\d{3,4}" (re)
1276 }, ? (re)
1276 }, ? (re)
1277 "test-success.t": [\{] (re)
1277 "test-success.t": [\{] (re)
1278 "csys": "\s*\d+\.\d{3,4}", ? (re)
1278 "csys": "\s*\d+\.\d{3,4}", ? (re)
1279 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1279 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1280 "diff": "", ? (re)
1280 "diff": "", ? (re)
1281 "end": "\s*\d+\.\d{3,4}", ? (re)
1281 "end": "\s*\d+\.\d{3,4}", ? (re)
1282 "result": "success", ? (re)
1282 "result": "success", ? (re)
1283 "start": "\s*\d+\.\d{3,4}", ? (re)
1283 "start": "\s*\d+\.\d{3,4}", ? (re)
1284 "time": "\s*\d+\.\d{3,4}" (re)
1284 "time": "\s*\d+\.\d{3,4}" (re)
1285 }
1285 }
1286 } (no-eol)
1286 } (no-eol)
1287 $ ls -a output
1287 $ ls -a output
1288 .
1288 .
1289 ..
1289 ..
1290 .testtimes
1290 .testtimes
1291 report.json
1291 report.json
1292 test-failure.t.err
1292 test-failure.t.err
1293
1293
1294 Test that failed test accepted through interactive are properly reported:
1294 Test that failed test accepted through interactive are properly reported:
1295
1295
1296 $ cp test-failure.t backup
1296 $ cp test-failure.t backup
1297 $ echo y | rt --json -i
1297 $ echo y | rt --json -i
1298 running 3 tests using 1 parallel processes
1298 running 3 tests using 1 parallel processes
1299
1299
1300 --- $TESTTMP/test-failure.t
1300 --- $TESTTMP/test-failure.t
1301 +++ $TESTTMP/test-failure.t.err
1301 +++ $TESTTMP/test-failure.t.err
1302 @@ -1,5 +1,5 @@
1302 @@ -1,5 +1,5 @@
1303 $ echo babar
1303 $ echo babar
1304 - rataxes
1304 - rataxes
1305 + babar
1305 + babar
1306 This is a noop statement so that
1306 This is a noop statement so that
1307 this test is still more bytes than success.
1307 this test is still more bytes than success.
1308 pad pad pad pad............................................................
1308 pad pad pad pad............................................................
1309 Accept this change? [y/N] ..s
1309 Accept this change? [y/N] ..s
1310 Skipped test-skip.t: missing feature: nail clipper
1310 Skipped test-skip.t: missing feature: nail clipper
1311 # Ran 2 tests, 1 skipped, 0 failed.
1311 # Ran 2 tests, 1 skipped, 0 failed.
1312
1312
1313 $ cat report.json
1313 $ cat report.json
1314 testreport ={
1314 testreport ={
1315 "test-failure.t": [\{] (re)
1315 "test-failure.t": [\{] (re)
1316 "csys": "\s*\d+\.\d{3,4}", ? (re)
1316 "csys": "\s*\d+\.\d{3,4}", ? (re)
1317 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1317 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1318 "diff": "", ? (re)
1318 "diff": "", ? (re)
1319 "end": "\s*\d+\.\d{3,4}", ? (re)
1319 "end": "\s*\d+\.\d{3,4}", ? (re)
1320 "result": "success", ? (re)
1320 "result": "success", ? (re)
1321 "start": "\s*\d+\.\d{3,4}", ? (re)
1321 "start": "\s*\d+\.\d{3,4}", ? (re)
1322 "time": "\s*\d+\.\d{3,4}" (re)
1322 "time": "\s*\d+\.\d{3,4}" (re)
1323 }, ? (re)
1323 }, ? (re)
1324 "test-skip.t": {
1324 "test-skip.t": {
1325 "csys": "\s*\d+\.\d{3,4}", ? (re)
1325 "csys": "\s*\d+\.\d{3,4}", ? (re)
1326 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1326 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1327 "diff": "", ? (re)
1327 "diff": "", ? (re)
1328 "end": "\s*\d+\.\d{3,4}", ? (re)
1328 "end": "\s*\d+\.\d{3,4}", ? (re)
1329 "result": "skip", ? (re)
1329 "result": "skip", ? (re)
1330 "start": "\s*\d+\.\d{3,4}", ? (re)
1330 "start": "\s*\d+\.\d{3,4}", ? (re)
1331 "time": "\s*\d+\.\d{3,4}" (re)
1331 "time": "\s*\d+\.\d{3,4}" (re)
1332 }, ? (re)
1332 }, ? (re)
1333 "test-success.t": [\{] (re)
1333 "test-success.t": [\{] (re)
1334 "csys": "\s*\d+\.\d{3,4}", ? (re)
1334 "csys": "\s*\d+\.\d{3,4}", ? (re)
1335 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1335 "cuser": "\s*\d+\.\d{3,4}", ? (re)
1336 "diff": "", ? (re)
1336 "diff": "", ? (re)
1337 "end": "\s*\d+\.\d{3,4}", ? (re)
1337 "end": "\s*\d+\.\d{3,4}", ? (re)
1338 "result": "success", ? (re)
1338 "result": "success", ? (re)
1339 "start": "\s*\d+\.\d{3,4}", ? (re)
1339 "start": "\s*\d+\.\d{3,4}", ? (re)
1340 "time": "\s*\d+\.\d{3,4}" (re)
1340 "time": "\s*\d+\.\d{3,4}" (re)
1341 }
1341 }
1342 } (no-eol)
1342 } (no-eol)
1343 $ mv backup test-failure.t
1343 $ mv backup test-failure.t
1344
1344
1345 backslash on end of line with glob matching is handled properly
1345 backslash on end of line with glob matching is handled properly
1346
1346
1347 $ cat > test-glob-backslash.t << EOF
1347 $ cat > test-glob-backslash.t << EOF
1348 > $ echo 'foo bar \\'
1348 > $ echo 'foo bar \\'
1349 > foo * \ (glob)
1349 > foo * \ (glob)
1350 > EOF
1350 > EOF
1351
1351
1352 $ rt test-glob-backslash.t
1352 $ rt test-glob-backslash.t
1353 running 1 tests using 1 parallel processes
1353 running 1 tests using 1 parallel processes
1354 .
1354 .
1355 # Ran 1 tests, 0 skipped, 0 failed.
1355 # Ran 1 tests, 0 skipped, 0 failed.
1356
1356
1357 $ rm -f test-glob-backslash.t
1357 $ rm -f test-glob-backslash.t
1358
1358
1359 Test globbing of local IP addresses
1359 Test globbing of local IP addresses
1360 $ echo 172.16.18.1
1360 $ echo 172.16.18.1
1361 $LOCALIP (glob)
1361 $LOCALIP (glob)
1362 $ echo dead:beef::1
1362 $ echo dead:beef::1
1363 $LOCALIP (glob)
1363 $LOCALIP (glob)
1364
1364
1365 Add support for external test formatter
1365 Add support for external test formatter
1366 =======================================
1366 =======================================
1367
1367
1368 $ CUSTOM_TEST_RESULT=basic_test_result "$PYTHON" $TESTDIR/run-tests.py --with-hg=`which hg` -j1 "$@" test-success.t test-failure.t
1368 $ CUSTOM_TEST_RESULT=basic_test_result "$PYTHON" $TESTDIR/run-tests.py --with-hg=`which hg` -j1 "$@" test-success.t test-failure.t
1369 running 2 tests using 1 parallel processes
1369 running 2 tests using 1 parallel processes
1370
1370
1371 # Ran 2 tests, 0 skipped, 0 failed.
1371 # Ran 2 tests, 0 skipped, 0 failed.
1372 ON_START! <__main__.TestSuite tests=[<__main__.TTest testMethod=test-failure.t>, <__main__.TTest testMethod=test-success.t>]>
1372 ON_START! <__main__.TestSuite tests=[<__main__.TTest testMethod=test-failure.t>, <__main__.TTest testMethod=test-success.t>]>
1373 FAILURE! test-failure.t output changed
1373 FAILURE! test-failure.t output changed
1374 SUCCESS! test-success.t
1374 SUCCESS! test-success.t
1375 ON_END!
1375 ON_END!
1376
1376
1377 Test reusability for third party tools
1377 Test reusability for third party tools
1378 ======================================
1378 ======================================
1379
1379
1380 $ mkdir "$TESTTMP"/anothertests
1380 $ mkdir "$TESTTMP"/anothertests
1381 $ cd "$TESTTMP"/anothertests
1381 $ cd "$TESTTMP"/anothertests
1382
1382
1383 test that `run-tests.py` can execute hghave, even if it runs not in
1383 test that `run-tests.py` can execute hghave, even if it runs not in
1384 Mercurial source tree.
1384 Mercurial source tree.
1385
1385
1386 $ cat > test-hghave.t <<EOF
1386 $ cat > test-hghave.t <<EOF
1387 > #require true
1387 > #require true
1388 > $ echo foo
1388 > $ echo foo
1389 > foo
1389 > foo
1390 > EOF
1390 > EOF
1391 $ rt test-hghave.t
1391 $ rt test-hghave.t
1392 running 1 tests using 1 parallel processes
1392 running 1 tests using 1 parallel processes
1393 .
1393 .
1394 # Ran 1 tests, 0 skipped, 0 failed.
1394 # Ran 1 tests, 0 skipped, 0 failed.
1395
1395
1396 test that RUNTESTDIR refers the directory, in which `run-tests.py` now
1396 test that RUNTESTDIR refers the directory, in which `run-tests.py` now
1397 running is placed.
1397 running is placed.
1398
1398
1399 $ cat > test-runtestdir.t <<EOF
1399 $ cat > test-runtestdir.t <<EOF
1400 > - $TESTDIR, in which test-run-tests.t is placed
1400 > - $TESTDIR, in which test-run-tests.t is placed
1401 > - \$TESTDIR, in which test-runtestdir.t is placed (expanded at runtime)
1401 > - \$TESTDIR, in which test-runtestdir.t is placed (expanded at runtime)
1402 > - \$RUNTESTDIR, in which run-tests.py is placed (expanded at runtime)
1402 > - \$RUNTESTDIR, in which run-tests.py is placed (expanded at runtime)
1403 >
1403 >
1404 > #if windows
1404 > #if windows
1405 > $ test "\$TESTDIR" = "$TESTTMP\anothertests"
1405 > $ test "\$TESTDIR" = "$TESTTMP\anothertests"
1406 > #else
1406 > #else
1407 > $ test "\$TESTDIR" = "$TESTTMP"/anothertests
1407 > $ test "\$TESTDIR" = "$TESTTMP"/anothertests
1408 > #endif
1408 > #endif
1409 > If this prints a path, that means RUNTESTDIR didn't equal
1409 > If this prints a path, that means RUNTESTDIR didn't equal
1410 > TESTDIR as it should have.
1410 > TESTDIR as it should have.
1411 > $ test "\$RUNTESTDIR" = "$TESTDIR" || echo "\$RUNTESTDIR"
1411 > $ test "\$RUNTESTDIR" = "$TESTDIR" || echo "\$RUNTESTDIR"
1412 > This should print the start of check-code. If this passes but the
1412 > This should print the start of check-code. If this passes but the
1413 > previous check failed, that means we found a copy of check-code at whatever
1413 > previous check failed, that means we found a copy of check-code at whatever
1414 > RUNTESTSDIR ended up containing, even though it doesn't match TESTDIR.
1414 > RUNTESTSDIR ended up containing, even though it doesn't match TESTDIR.
1415 > $ head -n 3 "\$RUNTESTDIR"/../contrib/check-code.py | sed 's@.!.*python3@#!USRBINENVPY@'
1415 > $ head -n 3 "\$RUNTESTDIR"/../contrib/check-code.py | sed 's@.!.*python3@#!USRBINENVPY@'
1416 > #!USRBINENVPY
1416 > #!USRBINENVPY
1417 > #
1417 > #
1418 > # check-code - a style and portability checker for Mercurial
1418 > # check-code - a style and portability checker for Mercurial
1419 > EOF
1419 > EOF
1420 $ rt test-runtestdir.t
1420 $ rt test-runtestdir.t
1421 running 1 tests using 1 parallel processes
1421 running 1 tests using 1 parallel processes
1422 .
1422 .
1423 # Ran 1 tests, 0 skipped, 0 failed.
1423 # Ran 1 tests, 0 skipped, 0 failed.
1424
1424
1425 #if execbit
1425 #if execbit
1426
1426
1427 test that TESTDIR is referred in PATH
1427 test that TESTDIR is referred in PATH
1428
1428
1429 $ cat > custom-command.sh <<EOF
1429 $ cat > custom-command.sh <<EOF
1430 > #!/bin/sh
1430 > #!/bin/sh
1431 > echo "hello world"
1431 > echo "hello world"
1432 > EOF
1432 > EOF
1433 $ chmod +x custom-command.sh
1433 $ chmod +x custom-command.sh
1434 $ cat > test-testdir-path.t <<EOF
1434 $ cat > test-testdir-path.t <<EOF
1435 > $ custom-command.sh
1435 > $ custom-command.sh
1436 > hello world
1436 > hello world
1437 > EOF
1437 > EOF
1438 $ rt test-testdir-path.t
1438 $ rt test-testdir-path.t
1439 running 1 tests using 1 parallel processes
1439 running 1 tests using 1 parallel processes
1440 .
1440 .
1441 # Ran 1 tests, 0 skipped, 0 failed.
1441 # Ran 1 tests, 0 skipped, 0 failed.
1442
1442
1443 #endif
1443 #endif
1444
1444
1445 test support for --allow-slow-tests
1445 test support for --allow-slow-tests
1446 $ cat > test-very-slow-test.t <<EOF
1446 $ cat > test-very-slow-test.t <<EOF
1447 > #require slow
1447 > #require slow
1448 > $ echo pass
1448 > $ echo pass
1449 > pass
1449 > pass
1450 > EOF
1450 > EOF
1451 $ rt test-very-slow-test.t
1451 $ rt test-very-slow-test.t
1452 running 1 tests using 1 parallel processes
1452 running 1 tests using 1 parallel processes
1453 s
1453 s
1454 Skipped test-very-slow-test.t: missing feature: allow slow tests (use --allow-slow-tests)
1454 Skipped test-very-slow-test.t: missing feature: allow slow tests (use --allow-slow-tests)
1455 # Ran 0 tests, 1 skipped, 0 failed.
1455 # Ran 0 tests, 1 skipped, 0 failed.
1456 $ rt $HGTEST_RUN_TESTS_PURE --allow-slow-tests test-very-slow-test.t
1456 $ rt $HGTEST_RUN_TESTS_PURE --allow-slow-tests test-very-slow-test.t
1457 running 1 tests using 1 parallel processes
1457 running 1 tests using 1 parallel processes
1458 .
1458 .
1459 # Ran 1 tests, 0 skipped, 0 failed.
1459 # Ran 1 tests, 0 skipped, 0 failed.
1460
1460
1461 support for running a test outside the current directory
1461 support for running a test outside the current directory
1462 $ mkdir nonlocal
1462 $ mkdir nonlocal
1463 $ cat > nonlocal/test-is-not-here.t << EOF
1463 $ cat > nonlocal/test-is-not-here.t << EOF
1464 > $ echo pass
1464 > $ echo pass
1465 > pass
1465 > pass
1466 > EOF
1466 > EOF
1467 $ rt nonlocal/test-is-not-here.t
1467 $ rt nonlocal/test-is-not-here.t
1468 running 1 tests using 1 parallel processes
1468 running 1 tests using 1 parallel processes
1469 .
1469 .
1470 # Ran 1 tests, 0 skipped, 0 failed.
1470 # Ran 1 tests, 0 skipped, 0 failed.
1471
1471
1472 support for automatically discovering test if arg is a folder
1472 support for automatically discovering test if arg is a folder
1473 $ mkdir tmp && cd tmp
1473 $ mkdir tmp && cd tmp
1474
1474
1475 $ cat > test-uno.t << EOF
1475 $ cat > test-uno.t << EOF
1476 > $ echo line
1476 > $ echo line
1477 > line
1477 > line
1478 > EOF
1478 > EOF
1479
1479
1480 $ cp test-uno.t test-dos.t
1480 $ cp test-uno.t test-dos.t
1481 $ cd ..
1481 $ cd ..
1482 $ cp -R tmp tmpp
1482 $ cp -R tmp tmpp
1483 $ cp tmp/test-uno.t test-solo.t
1483 $ cp tmp/test-uno.t test-solo.t
1484
1484
1485 $ rt tmp/ test-solo.t tmpp
1485 $ rt tmp/ test-solo.t tmpp
1486 running 5 tests using 1 parallel processes
1486 running 5 tests using 1 parallel processes
1487 .....
1487 .....
1488 # Ran 5 tests, 0 skipped, 0 failed.
1488 # Ran 5 tests, 0 skipped, 0 failed.
1489 $ rm -rf tmp tmpp
1489 $ rm -rf tmp tmpp
1490
1490
1491 support for running run-tests.py from another directory
1491 support for running run-tests.py from another directory
1492 $ mkdir tmp && cd tmp
1492 $ mkdir tmp && cd tmp
1493
1493
1494 $ cat > useful-file.sh << EOF
1494 $ cat > useful-file.sh << EOF
1495 > important command
1495 > important command
1496 > EOF
1496 > EOF
1497
1497
1498 $ cat > test-folder.t << EOF
1498 $ cat > test-folder.t << EOF
1499 > $ cat \$TESTDIR/useful-file.sh
1499 > $ cat \$TESTDIR/useful-file.sh
1500 > important command
1500 > important command
1501 > EOF
1501 > EOF
1502
1502
1503 $ cat > test-folder-fail.t << EOF
1503 $ cat > test-folder-fail.t << EOF
1504 > $ cat \$TESTDIR/useful-file.sh
1504 > $ cat \$TESTDIR/useful-file.sh
1505 > important commando
1505 > important commando
1506 > EOF
1506 > EOF
1507
1507
1508 $ cd ..
1508 $ cd ..
1509 $ rt tmp/test-*.t
1509 $ rt tmp/test-*.t
1510 running 2 tests using 1 parallel processes
1510 running 2 tests using 1 parallel processes
1511
1511
1512 --- $TESTTMP/anothertests/tmp/test-folder-fail.t
1512 --- $TESTTMP/anothertests/tmp/test-folder-fail.t
1513 +++ $TESTTMP/anothertests/tmp/test-folder-fail.t.err
1513 +++ $TESTTMP/anothertests/tmp/test-folder-fail.t.err
1514 @@ -1,2 +1,2 @@
1514 @@ -1,2 +1,2 @@
1515 $ cat $TESTDIR/useful-file.sh
1515 $ cat $TESTDIR/useful-file.sh
1516 - important commando
1516 - important commando
1517 + important command
1517 + important command
1518
1518
1519 ERROR: test-folder-fail.t output changed
1519 ERROR: test-folder-fail.t output changed
1520 !.
1520 !.
1521 Failed test-folder-fail.t: output changed
1521 Failed test-folder-fail.t: output changed
1522 # Ran 2 tests, 0 skipped, 1 failed.
1522 # Ran 2 tests, 0 skipped, 1 failed.
1523 python hash seed: * (glob)
1523 python hash seed: * (glob)
1524 [1]
1524 [1]
1525
1525
1526 support for bisecting failed tests automatically
1526 support for bisecting failed tests automatically
1527 $ hg init bisect
1527 $ hg init bisect
1528 $ cd bisect
1528 $ cd bisect
1529 $ cat >> test-bisect.t <<EOF
1529 $ cat >> test-bisect.t <<EOF
1530 > $ echo pass
1530 > $ echo pass
1531 > pass
1531 > pass
1532 > EOF
1532 > EOF
1533 $ hg add test-bisect.t
1533 $ hg add test-bisect.t
1534 $ hg ci -m 'good'
1534 $ hg ci -m 'good'
1535 $ cat >> test-bisect.t <<EOF
1535 $ cat >> test-bisect.t <<EOF
1536 > $ echo pass
1536 > $ echo pass
1537 > fail
1537 > fail
1538 > EOF
1538 > EOF
1539 $ hg ci -m 'bad'
1539 $ hg ci -m 'bad'
1540 $ rt --known-good-rev=0 test-bisect.t
1540 $ rt --known-good-rev=0 test-bisect.t
1541 running 1 tests using 1 parallel processes
1541 running 1 tests using 1 parallel processes
1542
1542
1543 --- $TESTTMP/anothertests/bisect/test-bisect.t
1543 --- $TESTTMP/anothertests/bisect/test-bisect.t
1544 +++ $TESTTMP/anothertests/bisect/test-bisect.t.err
1544 +++ $TESTTMP/anothertests/bisect/test-bisect.t.err
1545 @@ -1,4 +1,4 @@
1545 @@ -1,4 +1,4 @@
1546 $ echo pass
1546 $ echo pass
1547 pass
1547 pass
1548 $ echo pass
1548 $ echo pass
1549 - fail
1549 - fail
1550 + pass
1550 + pass
1551
1551
1552 ERROR: test-bisect.t output changed
1552 ERROR: test-bisect.t output changed
1553 !
1553 !
1554 Failed test-bisect.t: output changed
1554 Failed test-bisect.t: output changed
1555 test-bisect.t broken by 72cbf122d116 (bad)
1555 test-bisect.t broken by 72cbf122d116 (bad)
1556 # Ran 1 tests, 0 skipped, 1 failed.
1556 # Ran 1 tests, 0 skipped, 1 failed.
1557 python hash seed: * (glob)
1557 python hash seed: * (glob)
1558 [1]
1558 [1]
1559
1559
1560 $ cd ..
1560 $ cd ..
1561
1561
1562 support bisecting a separate repo
1562 support bisecting a separate repo
1563
1563
1564 $ hg init bisect-dependent
1564 $ hg init bisect-dependent
1565 $ cd bisect-dependent
1565 $ cd bisect-dependent
1566 $ cat > test-bisect-dependent.t <<EOF
1566 $ cat > test-bisect-dependent.t <<EOF
1567 > $ tail -1 \$TESTDIR/../bisect/test-bisect.t
1567 > $ tail -1 \$TESTDIR/../bisect/test-bisect.t
1568 > pass
1568 > pass
1569 > EOF
1569 > EOF
1570 $ hg commit -Am dependent test-bisect-dependent.t
1570 $ hg commit -Am dependent test-bisect-dependent.t
1571
1571
1572 $ rt --known-good-rev=0 test-bisect-dependent.t
1572 $ rt --known-good-rev=0 test-bisect-dependent.t
1573 running 1 tests using 1 parallel processes
1573 running 1 tests using 1 parallel processes
1574
1574
1575 --- $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t
1575 --- $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t
1576 +++ $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t.err
1576 +++ $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t.err
1577 @@ -1,2 +1,2 @@
1577 @@ -1,2 +1,2 @@
1578 $ tail -1 $TESTDIR/../bisect/test-bisect.t
1578 $ tail -1 $TESTDIR/../bisect/test-bisect.t
1579 - pass
1579 - pass
1580 + fail
1580 + fail
1581
1581
1582 ERROR: test-bisect-dependent.t output changed
1582 ERROR: test-bisect-dependent.t output changed
1583 !
1583 !
1584 Failed test-bisect-dependent.t: output changed
1584 Failed test-bisect-dependent.t: output changed
1585 Failed to identify failure point for test-bisect-dependent.t
1585 Failed to identify failure point for test-bisect-dependent.t
1586 # Ran 1 tests, 0 skipped, 1 failed.
1586 # Ran 1 tests, 0 skipped, 1 failed.
1587 python hash seed: * (glob)
1587 python hash seed: * (glob)
1588 [1]
1588 [1]
1589
1589
1590 $ rt --bisect-repo=../test-bisect test-bisect-dependent.t
1590 $ rt --bisect-repo=../test-bisect test-bisect-dependent.t
1591 usage: run-tests.py [options] [tests]
1591 usage: run-tests.py [options] [tests]
1592 run-tests.py: error: --bisect-repo cannot be used without --known-good-rev
1592 run-tests.py: error: --bisect-repo cannot be used without --known-good-rev
1593 [2]
1593 [2]
1594
1594
1595 $ rt --known-good-rev=0 --bisect-repo=../bisect test-bisect-dependent.t
1595 $ rt --known-good-rev=0 --bisect-repo=../bisect test-bisect-dependent.t
1596 running 1 tests using 1 parallel processes
1596 running 1 tests using 1 parallel processes
1597
1597
1598 --- $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t
1598 --- $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t
1599 +++ $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t.err
1599 +++ $TESTTMP/anothertests/bisect-dependent/test-bisect-dependent.t.err
1600 @@ -1,2 +1,2 @@
1600 @@ -1,2 +1,2 @@
1601 $ tail -1 $TESTDIR/../bisect/test-bisect.t
1601 $ tail -1 $TESTDIR/../bisect/test-bisect.t
1602 - pass
1602 - pass
1603 + fail
1603 + fail
1604
1604
1605 ERROR: test-bisect-dependent.t output changed
1605 ERROR: test-bisect-dependent.t output changed
1606 !
1606 !
1607 Failed test-bisect-dependent.t: output changed
1607 Failed test-bisect-dependent.t: output changed
1608 test-bisect-dependent.t broken by 72cbf122d116 (bad)
1608 test-bisect-dependent.t broken by 72cbf122d116 (bad)
1609 # Ran 1 tests, 0 skipped, 1 failed.
1609 # Ran 1 tests, 0 skipped, 1 failed.
1610 python hash seed: * (glob)
1610 python hash seed: * (glob)
1611 [1]
1611 [1]
1612
1612
1613 $ cd ..
1613 $ cd ..
1614
1614
1615 Test a broken #if statement doesn't break run-tests threading.
1615 Test a broken #if statement doesn't break run-tests threading.
1616 ==============================================================
1616 ==============================================================
1617 $ mkdir broken
1617 $ mkdir broken
1618 $ cd broken
1618 $ cd broken
1619 $ cat > test-broken.t <<EOF
1619 $ cat > test-broken.t <<EOF
1620 > true
1620 > true
1621 > #if notarealhghavefeature
1621 > #if notarealhghavefeature
1622 > $ false
1622 > $ false
1623 > #endif
1623 > #endif
1624 > EOF
1624 > EOF
1625 $ for f in 1 2 3 4 ; do
1625 $ for f in 1 2 3 4 ; do
1626 > cat > test-works-$f.t <<EOF
1626 > cat > test-works-$f.t <<EOF
1627 > This is test case $f
1627 > This is test case $f
1628 > $ sleep 1
1628 > $ sleep 1
1629 > EOF
1629 > EOF
1630 > done
1630 > done
1631 $ rt -j 2
1631 $ rt -j 2
1632 running 5 tests using 2 parallel processes
1632 running 5 tests using 2 parallel processes
1633 ....
1633 ....
1634 # Ran 5 tests, 0 skipped, 0 failed.
1634 # Ran 5 tests, 0 skipped, 0 failed.
1635 skipped: unknown feature: notarealhghavefeature
1635 skipped: unknown feature: notarealhghavefeature
1636
1636
1637 $ cd ..
1637 $ cd ..
1638 $ rm -rf broken
1638 $ rm -rf broken
1639
1639
1640 Test cases in .t files
1640 Test cases in .t files
1641 ======================
1641 ======================
1642 $ mkdir cases
1642 $ mkdir cases
1643 $ cd cases
1643 $ cd cases
1644 $ cat > test-cases-abc.t <<'EOF'
1644 $ cat > test-cases-abc.t <<'EOF'
1645 > #testcases A B C
1645 > #testcases A B C
1646 > $ V=B
1646 > $ V=B
1647 > #if A
1647 > #if A
1648 > $ V=A
1648 > $ V=A
1649 > #endif
1649 > #endif
1650 > #if C
1650 > #if C
1651 > $ V=C
1651 > $ V=C
1652 > #endif
1652 > #endif
1653 > $ echo $V | sed 's/A/C/'
1653 > $ echo $V | sed 's/A/C/'
1654 > C
1654 > C
1655 > #if C
1655 > #if C
1656 > $ [ $V = C ]
1656 > $ [ $V = C ]
1657 > #endif
1657 > #endif
1658 > #if A
1658 > #if A
1659 > $ [ $V = C ]
1659 > $ [ $V = C ]
1660 > [1]
1660 > [1]
1661 > #endif
1661 > #endif
1662 > #if no-C
1662 > #if no-C
1663 > $ [ $V = C ]
1663 > $ [ $V = C ]
1664 > [1]
1664 > [1]
1665 > #endif
1665 > #endif
1666 > $ [ $V = D ]
1666 > $ [ $V = D ]
1667 > [1]
1667 > [1]
1668 > EOF
1668 > EOF
1669 $ rt
1669 $ rt
1670 running 3 tests using 1 parallel processes
1670 running 3 tests using 1 parallel processes
1671 .
1671 .
1672 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1672 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1673 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1673 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1674 @@ -7,7 +7,7 @@
1674 @@ -7,7 +7,7 @@
1675 $ V=C
1675 $ V=C
1676 #endif
1676 #endif
1677 $ echo $V | sed 's/A/C/'
1677 $ echo $V | sed 's/A/C/'
1678 - C
1678 - C
1679 + B
1679 + B
1680 #if C
1680 #if C
1681 $ [ $V = C ]
1681 $ [ $V = C ]
1682 #endif
1682 #endif
1683
1683
1684 ERROR: test-cases-abc.t#B output changed
1684 ERROR: test-cases-abc.t#B output changed
1685 !.
1685 !.
1686 Failed test-cases-abc.t#B: output changed
1686 Failed test-cases-abc.t#B: output changed
1687 # Ran 3 tests, 0 skipped, 1 failed.
1687 # Ran 3 tests, 0 skipped, 1 failed.
1688 python hash seed: * (glob)
1688 python hash seed: * (glob)
1689 [1]
1689 [1]
1690
1690
1691 --restart works
1691 --restart works
1692
1692
1693 $ rt --restart
1693 $ rt --restart
1694 running 2 tests using 1 parallel processes
1694 running 2 tests using 1 parallel processes
1695
1695
1696 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1696 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1697 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1697 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1698 @@ -7,7 +7,7 @@
1698 @@ -7,7 +7,7 @@
1699 $ V=C
1699 $ V=C
1700 #endif
1700 #endif
1701 $ echo $V | sed 's/A/C/'
1701 $ echo $V | sed 's/A/C/'
1702 - C
1702 - C
1703 + B
1703 + B
1704 #if C
1704 #if C
1705 $ [ $V = C ]
1705 $ [ $V = C ]
1706 #endif
1706 #endif
1707
1707
1708 ERROR: test-cases-abc.t#B output changed
1708 ERROR: test-cases-abc.t#B output changed
1709 !.
1709 !.
1710 Failed test-cases-abc.t#B: output changed
1710 Failed test-cases-abc.t#B: output changed
1711 # Ran 2 tests, 0 skipped, 1 failed.
1711 # Ran 2 tests, 0 skipped, 1 failed.
1712 python hash seed: * (glob)
1712 python hash seed: * (glob)
1713 [1]
1713 [1]
1714
1714
1715 --restart works with outputdir
1715 --restart works with outputdir
1716
1716
1717 $ mkdir output
1717 $ mkdir output
1718 $ mv test-cases-abc.t#B.err output
1718 $ mv test-cases-abc.t#B.err output
1719 $ rt --restart --outputdir output
1719 $ rt --restart --outputdir output
1720 running 2 tests using 1 parallel processes
1720 running 2 tests using 1 parallel processes
1721
1721
1722 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1722 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1723 +++ $TESTTMP/anothertests/cases/output/test-cases-abc.t#B.err
1723 +++ $TESTTMP/anothertests/cases/output/test-cases-abc.t#B.err
1724 @@ -7,7 +7,7 @@
1724 @@ -7,7 +7,7 @@
1725 $ V=C
1725 $ V=C
1726 #endif
1726 #endif
1727 $ echo $V | sed 's/A/C/'
1727 $ echo $V | sed 's/A/C/'
1728 - C
1728 - C
1729 + B
1729 + B
1730 #if C
1730 #if C
1731 $ [ $V = C ]
1731 $ [ $V = C ]
1732 #endif
1732 #endif
1733
1733
1734 ERROR: test-cases-abc.t#B output changed
1734 ERROR: test-cases-abc.t#B output changed
1735 !.
1735 !.
1736 Failed test-cases-abc.t#B: output changed
1736 Failed test-cases-abc.t#B: output changed
1737 # Ran 2 tests, 0 skipped, 1 failed.
1737 # Ran 2 tests, 0 skipped, 1 failed.
1738 python hash seed: * (glob)
1738 python hash seed: * (glob)
1739 [1]
1739 [1]
1740
1740
1741 Test TESTCASE variable
1741 Test TESTCASE variable
1742
1742
1743 $ cat > test-cases-ab.t <<'EOF'
1743 $ cat > test-cases-ab.t <<'EOF'
1744 > $ dostuff() {
1744 > $ dostuff() {
1745 > > echo "In case $TESTCASE"
1745 > > echo "In case $TESTCASE"
1746 > > }
1746 > > }
1747 > #testcases A B
1747 > #testcases A B
1748 > #if A
1748 > #if A
1749 > $ dostuff
1749 > $ dostuff
1750 > In case A
1750 > In case A
1751 > #endif
1751 > #endif
1752 > #if B
1752 > #if B
1753 > $ dostuff
1753 > $ dostuff
1754 > In case B
1754 > In case B
1755 > #endif
1755 > #endif
1756 > EOF
1756 > EOF
1757 $ rt test-cases-ab.t
1757 $ rt test-cases-ab.t
1758 running 2 tests using 1 parallel processes
1758 running 2 tests using 1 parallel processes
1759 ..
1759 ..
1760 # Ran 2 tests, 0 skipped, 0 failed.
1760 # Ran 2 tests, 0 skipped, 0 failed.
1761
1761
1762 Support running a specific test case
1762 Support running a specific test case
1763
1763
1764 $ rt "test-cases-abc.t#B"
1764 $ rt "test-cases-abc.t#B"
1765 running 1 tests using 1 parallel processes
1765 running 1 tests using 1 parallel processes
1766
1766
1767 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1767 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1768 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1768 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1769 @@ -7,7 +7,7 @@
1769 @@ -7,7 +7,7 @@
1770 $ V=C
1770 $ V=C
1771 #endif
1771 #endif
1772 $ echo $V | sed 's/A/C/'
1772 $ echo $V | sed 's/A/C/'
1773 - C
1773 - C
1774 + B
1774 + B
1775 #if C
1775 #if C
1776 $ [ $V = C ]
1776 $ [ $V = C ]
1777 #endif
1777 #endif
1778
1778
1779 ERROR: test-cases-abc.t#B output changed
1779 ERROR: test-cases-abc.t#B output changed
1780 !
1780 !
1781 Failed test-cases-abc.t#B: output changed
1781 Failed test-cases-abc.t#B: output changed
1782 # Ran 1 tests, 0 skipped, 1 failed.
1782 # Ran 1 tests, 0 skipped, 1 failed.
1783 python hash seed: * (glob)
1783 python hash seed: * (glob)
1784 [1]
1784 [1]
1785
1785
1786 Support running multiple test cases in the same file
1786 Support running multiple test cases in the same file
1787
1787
1788 $ rt test-cases-abc.t#B test-cases-abc.t#C
1788 $ rt test-cases-abc.t#B test-cases-abc.t#C
1789 running 2 tests using 1 parallel processes
1789 running 2 tests using 1 parallel processes
1790
1790
1791 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1791 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1792 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1792 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1793 @@ -7,7 +7,7 @@
1793 @@ -7,7 +7,7 @@
1794 $ V=C
1794 $ V=C
1795 #endif
1795 #endif
1796 $ echo $V | sed 's/A/C/'
1796 $ echo $V | sed 's/A/C/'
1797 - C
1797 - C
1798 + B
1798 + B
1799 #if C
1799 #if C
1800 $ [ $V = C ]
1800 $ [ $V = C ]
1801 #endif
1801 #endif
1802
1802
1803 ERROR: test-cases-abc.t#B output changed
1803 ERROR: test-cases-abc.t#B output changed
1804 !.
1804 !.
1805 Failed test-cases-abc.t#B: output changed
1805 Failed test-cases-abc.t#B: output changed
1806 # Ran 2 tests, 0 skipped, 1 failed.
1806 # Ran 2 tests, 0 skipped, 1 failed.
1807 python hash seed: * (glob)
1807 python hash seed: * (glob)
1808 [1]
1808 [1]
1809
1809
1810 Support ignoring invalid test cases
1810 Support ignoring invalid test cases
1811
1811
1812 $ rt test-cases-abc.t#B test-cases-abc.t#D
1812 $ rt test-cases-abc.t#B test-cases-abc.t#D
1813 running 1 tests using 1 parallel processes
1813 running 1 tests using 1 parallel processes
1814
1814
1815 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1815 --- $TESTTMP/anothertests/cases/test-cases-abc.t
1816 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1816 +++ $TESTTMP/anothertests/cases/test-cases-abc.t#B.err
1817 @@ -7,7 +7,7 @@
1817 @@ -7,7 +7,7 @@
1818 $ V=C
1818 $ V=C
1819 #endif
1819 #endif
1820 $ echo $V | sed 's/A/C/'
1820 $ echo $V | sed 's/A/C/'
1821 - C
1821 - C
1822 + B
1822 + B
1823 #if C
1823 #if C
1824 $ [ $V = C ]
1824 $ [ $V = C ]
1825 #endif
1825 #endif
1826
1826
1827 ERROR: test-cases-abc.t#B output changed
1827 ERROR: test-cases-abc.t#B output changed
1828 !
1828 !
1829 Failed test-cases-abc.t#B: output changed
1829 Failed test-cases-abc.t#B: output changed
1830 # Ran 1 tests, 0 skipped, 1 failed.
1830 # Ran 1 tests, 0 skipped, 1 failed.
1831 python hash seed: * (glob)
1831 python hash seed: * (glob)
1832 [1]
1832 [1]
1833
1833
1834 Support running complex test cases names
1834 Support running complex test cases names
1835
1835
1836 $ cat > test-cases-advanced-cases.t <<'EOF'
1836 $ cat > test-cases-advanced-cases.t <<'EOF'
1837 > #testcases simple case-with-dashes casewith_-.chars
1837 > #testcases simple case-with-dashes casewith_-.chars
1838 > $ echo $TESTCASE
1838 > $ echo $TESTCASE
1839 > simple
1839 > simple
1840 > EOF
1840 > EOF
1841
1841
1842 $ cat test-cases-advanced-cases.t
1842 $ cat test-cases-advanced-cases.t
1843 #testcases simple case-with-dashes casewith_-.chars
1843 #testcases simple case-with-dashes casewith_-.chars
1844 $ echo $TESTCASE
1844 $ echo $TESTCASE
1845 simple
1845 simple
1846
1846
1847 $ rt test-cases-advanced-cases.t
1847 $ rt test-cases-advanced-cases.t
1848 running 3 tests using 1 parallel processes
1848 running 3 tests using 1 parallel processes
1849
1849
1850 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1850 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1851 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#case-with-dashes.err
1851 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#case-with-dashes.err
1852 @@ -1,3 +1,3 @@
1852 @@ -1,3 +1,3 @@
1853 #testcases simple case-with-dashes casewith_-.chars
1853 #testcases simple case-with-dashes casewith_-.chars
1854 $ echo $TESTCASE
1854 $ echo $TESTCASE
1855 - simple
1855 - simple
1856 + case-with-dashes
1856 + case-with-dashes
1857
1857
1858 ERROR: test-cases-advanced-cases.t#case-with-dashes output changed
1858 ERROR: test-cases-advanced-cases.t#case-with-dashes output changed
1859 !
1859 !
1860 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1860 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1861 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#casewith_-.chars.err
1861 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#casewith_-.chars.err
1862 @@ -1,3 +1,3 @@
1862 @@ -1,3 +1,3 @@
1863 #testcases simple case-with-dashes casewith_-.chars
1863 #testcases simple case-with-dashes casewith_-.chars
1864 $ echo $TESTCASE
1864 $ echo $TESTCASE
1865 - simple
1865 - simple
1866 + casewith_-.chars
1866 + casewith_-.chars
1867
1867
1868 ERROR: test-cases-advanced-cases.t#casewith_-.chars output changed
1868 ERROR: test-cases-advanced-cases.t#casewith_-.chars output changed
1869 !.
1869 !.
1870 Failed test-cases-advanced-cases.t#case-with-dashes: output changed
1870 Failed test-cases-advanced-cases.t#case-with-dashes: output changed
1871 Failed test-cases-advanced-cases.t#casewith_-.chars: output changed
1871 Failed test-cases-advanced-cases.t#casewith_-.chars: output changed
1872 # Ran 3 tests, 0 skipped, 2 failed.
1872 # Ran 3 tests, 0 skipped, 2 failed.
1873 python hash seed: * (glob)
1873 python hash seed: * (glob)
1874 [1]
1874 [1]
1875
1875
1876 $ rt "test-cases-advanced-cases.t#case-with-dashes"
1876 $ rt "test-cases-advanced-cases.t#case-with-dashes"
1877 running 1 tests using 1 parallel processes
1877 running 1 tests using 1 parallel processes
1878
1878
1879 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1879 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1880 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#case-with-dashes.err
1880 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#case-with-dashes.err
1881 @@ -1,3 +1,3 @@
1881 @@ -1,3 +1,3 @@
1882 #testcases simple case-with-dashes casewith_-.chars
1882 #testcases simple case-with-dashes casewith_-.chars
1883 $ echo $TESTCASE
1883 $ echo $TESTCASE
1884 - simple
1884 - simple
1885 + case-with-dashes
1885 + case-with-dashes
1886
1886
1887 ERROR: test-cases-advanced-cases.t#case-with-dashes output changed
1887 ERROR: test-cases-advanced-cases.t#case-with-dashes output changed
1888 !
1888 !
1889 Failed test-cases-advanced-cases.t#case-with-dashes: output changed
1889 Failed test-cases-advanced-cases.t#case-with-dashes: output changed
1890 # Ran 1 tests, 0 skipped, 1 failed.
1890 # Ran 1 tests, 0 skipped, 1 failed.
1891 python hash seed: * (glob)
1891 python hash seed: * (glob)
1892 [1]
1892 [1]
1893
1893
1894 $ rt "test-cases-advanced-cases.t#casewith_-.chars"
1894 $ rt "test-cases-advanced-cases.t#casewith_-.chars"
1895 running 1 tests using 1 parallel processes
1895 running 1 tests using 1 parallel processes
1896
1896
1897 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1897 --- $TESTTMP/anothertests/cases/test-cases-advanced-cases.t
1898 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#casewith_-.chars.err
1898 +++ $TESTTMP/anothertests/cases/test-cases-advanced-cases.t#casewith_-.chars.err
1899 @@ -1,3 +1,3 @@
1899 @@ -1,3 +1,3 @@
1900 #testcases simple case-with-dashes casewith_-.chars
1900 #testcases simple case-with-dashes casewith_-.chars
1901 $ echo $TESTCASE
1901 $ echo $TESTCASE
1902 - simple
1902 - simple
1903 + casewith_-.chars
1903 + casewith_-.chars
1904
1904
1905 ERROR: test-cases-advanced-cases.t#casewith_-.chars output changed
1905 ERROR: test-cases-advanced-cases.t#casewith_-.chars output changed
1906 !
1906 !
1907 Failed test-cases-advanced-cases.t#casewith_-.chars: output changed
1907 Failed test-cases-advanced-cases.t#casewith_-.chars: output changed
1908 # Ran 1 tests, 0 skipped, 1 failed.
1908 # Ran 1 tests, 0 skipped, 1 failed.
1909 python hash seed: * (glob)
1909 python hash seed: * (glob)
1910 [1]
1910 [1]
1911
1911
1912 Test automatic pattern replacement
1912 Test automatic pattern replacement
1913 ==================================
1913 ==================================
1914
1914
1915 $ cat << EOF >> common-pattern.py
1915 $ cat << EOF >> common-pattern.py
1916 > substitutions = [
1916 > substitutions = [
1917 > (br'foo-(.*)\\b',
1917 > (br'foo-(.*)\\b',
1918 > br'\$XXX=\\1\$'),
1918 > br'\$XXX=\\1\$'),
1919 > (br'bar\\n',
1919 > (br'bar\\n',
1920 > br'\$YYY$\\n'),
1920 > br'\$YYY$\\n'),
1921 > ]
1921 > ]
1922 > EOF
1922 > EOF
1923
1923
1924 $ cat << EOF >> test-substitution.t
1924 $ cat << EOF >> test-substitution.t
1925 > $ echo foo-12
1925 > $ echo foo-12
1926 > \$XXX=12$
1926 > \$XXX=12$
1927 > $ echo foo-42
1927 > $ echo foo-42
1928 > \$XXX=42$
1928 > \$XXX=42$
1929 > $ echo bar prior
1929 > $ echo bar prior
1930 > bar prior
1930 > bar prior
1931 > $ echo lastbar
1931 > $ echo lastbar
1932 > last\$YYY$
1932 > last\$YYY$
1933 > $ echo foo-bar foo-baz
1933 > $ echo foo-bar foo-baz
1934 > EOF
1934 > EOF
1935
1935
1936 $ rt test-substitution.t
1936 $ rt test-substitution.t
1937 running 1 tests using 1 parallel processes
1937 running 1 tests using 1 parallel processes
1938
1938
1939 --- $TESTTMP/anothertests/cases/test-substitution.t
1939 --- $TESTTMP/anothertests/cases/test-substitution.t
1940 +++ $TESTTMP/anothertests/cases/test-substitution.t.err
1940 +++ $TESTTMP/anothertests/cases/test-substitution.t.err
1941 @@ -7,3 +7,4 @@
1941 @@ -7,3 +7,4 @@
1942 $ echo lastbar
1942 $ echo lastbar
1943 last$YYY$
1943 last$YYY$
1944 $ echo foo-bar foo-baz
1944 $ echo foo-bar foo-baz
1945 + $XXX=bar foo-baz$
1945 + $XXX=bar foo-baz$
1946
1946
1947 ERROR: test-substitution.t output changed
1947 ERROR: test-substitution.t output changed
1948 !
1948 !
1949 Failed test-substitution.t: output changed
1949 Failed test-substitution.t: output changed
1950 # Ran 1 tests, 0 skipped, 1 failed.
1950 # Ran 1 tests, 0 skipped, 1 failed.
1951 python hash seed: * (glob)
1951 python hash seed: * (glob)
1952 [1]
1952 [1]
1953
1953
1954 --extra-config-opt works
1954 --extra-config-opt works
1955
1955
1956 $ cat << EOF >> test-config-opt.t
1956 $ cat << EOF >> test-config-opt.t
1957 > $ hg init test-config-opt
1957 > $ hg init test-config-opt
1958 > $ hg -R test-config-opt purge
1958 > $ hg -R test-config-opt purge
1959 > $ echo "HGTESTEXTRAEXTENSIONS: \$HGTESTEXTRAEXTENSIONS"
1960 > HGTESTEXTRAEXTENSIONS: purge
1959 > EOF
1961 > EOF
1960
1962
1961 $ rt --extra-config-opt extensions.purge= test-config-opt.t
1963 $ rt --extra-config-opt extensions.purge= \
1964 > --extra-config-opt not.an.extension=True test-config-opt.t
1962 running 1 tests using 1 parallel processes
1965 running 1 tests using 1 parallel processes
1963 .
1966 .
1964 # Ran 1 tests, 0 skipped, 0 failed.
1967 # Ran 1 tests, 0 skipped, 0 failed.
1965
1968
1966 Test conditional output matching
1969 Test conditional output matching
1967 ================================
1970 ================================
1968
1971
1969 $ cat << EOF >> test-conditional-matching.t
1972 $ cat << EOF >> test-conditional-matching.t
1970 > #testcases foo bar
1973 > #testcases foo bar
1971 > $ echo richtig
1974 > $ echo richtig
1972 > richtig (true !)
1975 > richtig (true !)
1973 > $ echo falsch
1976 > $ echo falsch
1974 > falsch (false !)
1977 > falsch (false !)
1975 > #if foo
1978 > #if foo
1976 > $ echo arthur
1979 > $ echo arthur
1977 > arthur (bar !)
1980 > arthur (bar !)
1978 > #endif
1981 > #endif
1979 > $ echo celeste
1982 > $ echo celeste
1980 > celeste (foo !)
1983 > celeste (foo !)
1981 > $ echo zephir
1984 > $ echo zephir
1982 > zephir (bar !)
1985 > zephir (bar !)
1983 > EOF
1986 > EOF
1984
1987
1985 $ rt test-conditional-matching.t
1988 $ rt test-conditional-matching.t
1986 running 2 tests using 1 parallel processes
1989 running 2 tests using 1 parallel processes
1987
1990
1988 --- $TESTTMP/anothertests/cases/test-conditional-matching.t
1991 --- $TESTTMP/anothertests/cases/test-conditional-matching.t
1989 +++ $TESTTMP/anothertests/cases/test-conditional-matching.t#bar.err
1992 +++ $TESTTMP/anothertests/cases/test-conditional-matching.t#bar.err
1990 @@ -3,11 +3,13 @@
1993 @@ -3,11 +3,13 @@
1991 richtig (true !)
1994 richtig (true !)
1992 $ echo falsch
1995 $ echo falsch
1993 falsch (false !)
1996 falsch (false !)
1994 + falsch
1997 + falsch
1995 #if foo
1998 #if foo
1996 $ echo arthur
1999 $ echo arthur
1997 arthur \(bar !\) (re)
2000 arthur \(bar !\) (re)
1998 #endif
2001 #endif
1999 $ echo celeste
2002 $ echo celeste
2000 celeste \(foo !\) (re)
2003 celeste \(foo !\) (re)
2001 + celeste
2004 + celeste
2002 $ echo zephir
2005 $ echo zephir
2003 zephir \(bar !\) (re)
2006 zephir \(bar !\) (re)
2004
2007
2005 ERROR: test-conditional-matching.t#bar output changed
2008 ERROR: test-conditional-matching.t#bar output changed
2006 !
2009 !
2007 --- $TESTTMP/anothertests/cases/test-conditional-matching.t
2010 --- $TESTTMP/anothertests/cases/test-conditional-matching.t
2008 +++ $TESTTMP/anothertests/cases/test-conditional-matching.t#foo.err
2011 +++ $TESTTMP/anothertests/cases/test-conditional-matching.t#foo.err
2009 @@ -3,11 +3,14 @@
2012 @@ -3,11 +3,14 @@
2010 richtig (true !)
2013 richtig (true !)
2011 $ echo falsch
2014 $ echo falsch
2012 falsch (false !)
2015 falsch (false !)
2013 + falsch
2016 + falsch
2014 #if foo
2017 #if foo
2015 $ echo arthur
2018 $ echo arthur
2016 arthur \(bar !\) (re)
2019 arthur \(bar !\) (re)
2017 + arthur
2020 + arthur
2018 #endif
2021 #endif
2019 $ echo celeste
2022 $ echo celeste
2020 celeste \(foo !\) (re)
2023 celeste \(foo !\) (re)
2021 $ echo zephir
2024 $ echo zephir
2022 zephir \(bar !\) (re)
2025 zephir \(bar !\) (re)
2023 + zephir
2026 + zephir
2024
2027
2025 ERROR: test-conditional-matching.t#foo output changed
2028 ERROR: test-conditional-matching.t#foo output changed
2026 !
2029 !
2027 Failed test-conditional-matching.t#bar: output changed
2030 Failed test-conditional-matching.t#bar: output changed
2028 Failed test-conditional-matching.t#foo: output changed
2031 Failed test-conditional-matching.t#foo: output changed
2029 # Ran 2 tests, 0 skipped, 2 failed.
2032 # Ran 2 tests, 0 skipped, 2 failed.
2030 python hash seed: * (glob)
2033 python hash seed: * (glob)
2031 [1]
2034 [1]
General Comments 0
You need to be logged in to leave comments. Login now