31
27
from cStringIO import StringIO
38
from pprint import pformat
43
from subprocess import Popen, PIPE, STDOUT
50
# nb: check this before importing anything else from within it
51
_testtools_version = getattr(testtools, '__version__', ())
52
if _testtools_version < (0, 9, 5):
53
raise ImportError("need at least testtools 0.9.5: %s is %r"
54
% (testtools.__file__, _testtools_version))
55
from testtools import content
52
58
from bzrlib import (
62
commands as _mod_commands,
72
plugin as _mod_plugin,
79
transport as _mod_transport,
68
import bzrlib.commands
69
import bzrlib.timestamp
71
import bzrlib.inventory
72
import bzrlib.iterablefile
75
83
import bzrlib.lsprof
76
84
except ImportError:
77
85
# lsprof not available
79
from bzrlib.merge import merge_inner
82
from bzrlib.smart import client, request, server
84
from bzrlib import symbol_versioning
87
from bzrlib.smart import client, request
88
from bzrlib.transport import (
85
92
from bzrlib.symbol_versioning import (
87
93
deprecated_function,
92
from bzrlib.transport import get_transport
93
import bzrlib.transport
94
from bzrlib.transport.local import LocalURLServer
95
from bzrlib.transport.memory import MemoryServer
96
from bzrlib.transport.readonly import ReadonlyServer
97
from bzrlib.trace import mutter, note
98
from bzrlib.tests import TestUtil
99
from bzrlib.tests.http_server import HttpServer
100
from bzrlib.tests.TestUtil import (
104
from bzrlib.tests.treeshape import build_tree_contents
96
from bzrlib.tests import (
101
from bzrlib.ui import NullProgressView
105
102
from bzrlib.ui.text import TextUIFactory
106
import bzrlib.version_info_formats.format_custom
107
from bzrlib.workingtree import WorkingTree, WorkingTreeFormat2
109
104
# Mark this python module as being part of the implementation
110
105
# of unittest: this gives us better tracebacks where the last
111
106
# shown frame is the test code, not our assertXYZ.
114
default_transport = LocalURLServer
117
class ExtendedTestResult(unittest._TextTestResult):
109
default_transport = test_server.LocalURLServer
112
_unitialized_attr = object()
113
"""A sentinel needed to act as a default value in a method signature."""
116
# Subunit result codes, defined here to prevent a hard dependency on subunit.
120
# These are intentionally brought into this namespace. That way plugins, etc
121
# can just "from bzrlib.tests import TestCase, TestLoader, etc"
122
TestSuite = TestUtil.TestSuite
123
TestLoader = TestUtil.TestLoader
125
# Tests should run in a clean and clearly defined environment. The goal is to
126
# keep them isolated from the running environment as mush as possible. The test
127
# framework ensures the variables defined below are set (or deleted if the
128
# value is None) before a test is run and reset to their original value after
129
# the test is run. Generally if some code depends on an environment variable,
130
# the tests should start without this variable in the environment. There are a
131
# few exceptions but you shouldn't violate this rule lightly.
135
# bzr now uses the Win32 API and doesn't rely on APPDATA, but the
136
# tests do check our impls match APPDATA
137
'BZR_EDITOR': None, # test_msgeditor manipulates this variable
141
'BZREMAIL': None, # may still be present in the environment
142
'EMAIL': 'jrandom@example.com', # set EMAIL as bzr does not guess
143
'BZR_PROGRESS_BAR': None,
144
# This should trap leaks to ~/.bzr.log. This occurs when tests use TestCase
145
# as a base class instead of TestCaseInTempDir. Tests inheriting from
146
# TestCase should not use disk resources, BZR_LOG is one.
147
'BZR_LOG': '/you-should-use-TestCaseInTempDir-if-you-need-a-log-file',
148
'BZR_PLUGIN_PATH': None,
149
'BZR_DISABLE_PLUGINS': None,
150
'BZR_PLUGINS_AT': None,
151
'BZR_CONCURRENCY': None,
152
# Make sure that any text ui tests are consistent regardless of
153
# the environment the test case is run in; you may want tests that
154
# test other combinations. 'dumb' is a reasonable guess for tests
155
# going to a pipe or a StringIO.
161
'SSH_AUTH_SOCK': None,
171
# Nobody cares about ftp_proxy, FTP_PROXY AFAIK. So far at
172
# least. If you do (care), please update this comment
176
'BZR_REMOTE_PATH': None,
177
# Generally speaking, we don't want apport reporting on crashes in
178
# the test envirnoment unless we're specifically testing apport,
179
# so that it doesn't leak into the real system environment. We
180
# use an env var so it propagates to subprocesses.
181
'APPORT_DISABLE': '1',
185
def override_os_environ(test, env=None):
186
"""Modify os.environ keeping a copy.
188
:param test: A test instance
190
:param env: A dict containing variable definitions to be installed
193
env = isolated_environ
194
test._original_os_environ = dict([(var, value)
195
for var, value in os.environ.iteritems()])
196
for var, value in env.iteritems():
197
osutils.set_or_unset_env(var, value)
198
if var not in test._original_os_environ:
199
# The var is new, add it with a value of None, so
200
# restore_os_environ will delete it
201
test._original_os_environ[var] = None
204
def restore_os_environ(test):
205
"""Restore os.environ to its original state.
207
:param test: A test instance previously passed to override_os_environ.
209
for var, value in test._original_os_environ.iteritems():
210
# Restore the original value (or delete it if the value has been set to
211
# None in override_os_environ).
212
osutils.set_or_unset_env(var, value)
215
def _clear__type_equality_funcs(test):
216
"""Cleanup bound methods stored on TestCase instances
218
Clear the dict breaking a few (mostly) harmless cycles in the affected
219
unittests released with Python 2.6 and initial Python 2.7 versions.
221
For a few revisions between Python 2.7.1 and Python 2.7.2 that annoyingly
222
shipped in Oneiric, an object with no clear method was used, hence the
223
extra complications, see bug 809048 for details.
225
type_equality_funcs = getattr(test, "_type_equality_funcs", None)
226
if type_equality_funcs is not None:
227
tef_clear = getattr(type_equality_funcs, "clear", None)
228
if tef_clear is None:
229
tef_instance_dict = getattr(type_equality_funcs, "__dict__", None)
230
if tef_instance_dict is not None:
231
tef_clear = tef_instance_dict.clear
232
if tef_clear is not None:
236
class ExtendedTestResult(testtools.TextTestResult):
118
237
"""Accepts, reports and accumulates the results of running tests.
120
239
Compared to the unittest version this class adds support for
170
288
self._overall_start_time = time.time()
171
289
self._strict = strict
290
self._first_thread_leaker_id = None
291
self._tests_leaking_threads_count = 0
292
self._traceback_from_test = None
294
def stopTestRun(self):
297
stopTime = time.time()
298
timeTaken = stopTime - self.startTime
299
# GZ 2010-07-19: Seems testtools has no printErrors method, and though
300
# the parent class method is similar have to duplicate
301
self._show_list('ERROR', self.errors)
302
self._show_list('FAIL', self.failures)
303
self.stream.write(self.sep2)
304
self.stream.write("%s %d test%s in %.3fs\n\n" % (actionTaken,
305
run, run != 1 and "s" or "", timeTaken))
306
if not self.wasSuccessful():
307
self.stream.write("FAILED (")
308
failed, errored = map(len, (self.failures, self.errors))
310
self.stream.write("failures=%d" % failed)
312
if failed: self.stream.write(", ")
313
self.stream.write("errors=%d" % errored)
314
if self.known_failure_count:
315
if failed or errored: self.stream.write(", ")
316
self.stream.write("known_failure_count=%d" %
317
self.known_failure_count)
318
self.stream.write(")\n")
320
if self.known_failure_count:
321
self.stream.write("OK (known_failures=%d)\n" %
322
self.known_failure_count)
324
self.stream.write("OK\n")
325
if self.skip_count > 0:
326
skipped = self.skip_count
327
self.stream.write('%d test%s skipped\n' %
328
(skipped, skipped != 1 and "s" or ""))
330
for feature, count in sorted(self.unsupported.items()):
331
self.stream.write("Missing feature '%s' skipped %d tests.\n" %
175
334
ok = self.wasStrictlySuccessful()
177
336
ok = self.wasSuccessful()
179
self.stream.write('tests passed\n')
181
self.stream.write('tests failed\n')
182
if TestCase._first_thread_leaker_id:
337
if self._first_thread_leaker_id:
183
338
self.stream.write(
184
339
'%s is leaking threads among %d leaking tests.\n' % (
185
TestCase._first_thread_leaker_id,
186
TestCase._leaking_threads_tests))
188
def _extractBenchmarkTime(self, testCase):
340
self._first_thread_leaker_id,
341
self._tests_leaking_threads_count))
342
# We don't report the main thread as an active one.
344
'%d non-main threads were left active in the end.\n'
345
% (len(self._active_threads) - 1))
347
def getDescription(self, test):
350
def _extractBenchmarkTime(self, testCase, details=None):
189
351
"""Add a benchmark time for the current test case."""
352
if details and 'benchtime' in details:
353
return float(''.join(details['benchtime'].iter_bytes()))
190
354
return getattr(testCase, "_benchtime", None)
192
356
def _elapsedTestTimeString(self):
193
357
"""Return a time string for the overall time the current test has taken."""
194
return self._formatTime(time.time() - self._start_time)
358
return self._formatTime(self._delta_to_float(
359
self._now() - self._start_datetime))
196
361
def _testTimeString(self, testCase):
197
362
benchmark_time = self._extractBenchmarkTime(testCase)
209
374
def _shortened_test_description(self, test):
211
what = re.sub(r'^bzrlib\.(tests|benchmarks)\.', '', what)
376
what = re.sub(r'^bzrlib\.tests\.', '', what)
379
# GZ 2010-10-04: Cloned tests may end up harmlessly calling this method
380
# multiple times in a row, because the handler is added for
381
# each test but the container list is shared between cases.
382
# See lp:498869 lp:625574 and lp:637725 for background.
383
def _record_traceback_from_test(self, exc_info):
384
"""Store the traceback from passed exc_info tuple till"""
385
self._traceback_from_test = exc_info[2]
214
387
def startTest(self, test):
215
unittest.TestResult.startTest(self, test)
388
super(ExtendedTestResult, self).startTest(test)
216
389
if self.count == 0:
217
390
self.startTests()
218
392
self.report_test_start(test)
219
393
test.number = self.count
220
394
self._recordTestStartTime()
395
# Make testtools cases give us the real traceback on failure
396
addOnException = getattr(test, "addOnException", None)
397
if addOnException is not None:
398
addOnException(self._record_traceback_from_test)
399
# Only check for thread leaks on bzrlib derived test cases
400
if isinstance(test, TestCase):
401
test.addCleanup(self._check_leaked_threads, test)
403
def stopTest(self, test):
404
super(ExtendedTestResult, self).stopTest(test)
405
# Manually break cycles, means touching various private things but hey
406
getDetails = getattr(test, "getDetails", None)
407
if getDetails is not None:
409
_clear__type_equality_funcs(test)
410
self._traceback_from_test = None
222
412
def startTests(self):
224
'testing: %s\n' % (osutils.realpath(sys.argv[0]),))
226
' %s (%s python%s)\n' % (
228
bzrlib.version_string,
229
bzrlib._format_version_tuple(sys.version_info),
231
self.stream.write('\n')
413
self.report_tests_starting()
414
self._active_threads = threading.enumerate()
416
def _check_leaked_threads(self, test):
417
"""See if any threads have leaked since last call
419
A sample of live threads is stored in the _active_threads attribute,
420
when this method runs it compares the current live threads and any not
421
in the previous sample are treated as having leaked.
423
now_active_threads = set(threading.enumerate())
424
threads_leaked = now_active_threads.difference(self._active_threads)
426
self._report_thread_leak(test, threads_leaked, now_active_threads)
427
self._tests_leaking_threads_count += 1
428
if self._first_thread_leaker_id is None:
429
self._first_thread_leaker_id = test.id()
430
self._active_threads = now_active_threads
233
432
def _recordTestStartTime(self):
234
433
"""Record that a test has started."""
235
self._start_time = time.time()
237
def _cleanupLogFile(self, test):
238
# We can only do this if we have one of our TestCases, not if
240
setKeepLogfile = getattr(test, 'setKeepLogfile', None)
241
if setKeepLogfile is not None:
434
self._start_datetime = self._now()
244
436
def addError(self, test, err):
245
437
"""Tell result that test finished with an error.
266
452
Called from the TestCase run() method when the test
267
453
fails because e.g. an assert() method failed.
269
self._testConcluded(test)
270
if isinstance(err[1], KnownFailure):
271
return self._addKnownFailure(test, err)
273
unittest.TestResult.addFailure(self, test, err)
274
self.failure_count += 1
275
self.report_failure(test, err)
278
self._cleanupLogFile(test)
455
self._post_mortem(self._traceback_from_test)
456
super(ExtendedTestResult, self).addFailure(test, err)
457
self.failure_count += 1
458
self.report_failure(test, err)
280
def addSuccess(self, test):
462
def addSuccess(self, test, details=None):
281
463
"""Tell result that test completed successfully.
283
465
Called from the TestCase run()
285
self._testConcluded(test)
286
467
if self._bench_history is not None:
287
benchmark_time = self._extractBenchmarkTime(test)
468
benchmark_time = self._extractBenchmarkTime(test, details)
288
469
if benchmark_time is not None:
289
470
self._bench_history.write("%s %s\n" % (
290
471
self._formatTime(benchmark_time),
292
473
self.report_success(test)
293
self._cleanupLogFile(test)
294
unittest.TestResult.addSuccess(self, test)
474
super(ExtendedTestResult, self).addSuccess(test)
295
475
test._log_contents = ''
297
def _testConcluded(self, test):
298
"""Common code when a test has finished.
300
Called regardless of whether it succeded, failed, etc.
304
def _addKnownFailure(self, test, err):
477
def addExpectedFailure(self, test, err):
305
478
self.known_failure_count += 1
306
479
self.report_known_failure(test, err)
481
def addUnexpectedSuccess(self, test, details=None):
482
"""Tell result the test unexpectedly passed, counting as a failure
484
When the minimum version of testtools required becomes 0.9.8 this
485
can be updated to use the new handling there.
487
super(ExtendedTestResult, self).addFailure(test, details=details)
488
self.failure_count += 1
489
self.report_unexpected_success(test,
490
"".join(details["reason"].iter_text()))
308
494
def addNotSupported(self, test, feature):
309
495
"""The test will not be run because of a missing feature.
311
497
# this can be called in two different ways: it may be that the
312
# test started running, and then raised (through addError)
498
# test started running, and then raised (through requireFeature)
313
499
# UnavailableFeature. Alternatively this method can be called
314
# while probing for features before running the tests; in that
315
# case we will see startTest and stopTest, but the test will never
500
# while probing for features before running the test code proper; in
501
# that case we will see startTest and stopTest, but the test will
502
# never actually run.
317
503
self.unsupported.setdefault(str(feature), 0)
318
504
self.unsupported[str(feature)] += 1
319
505
self.report_unsupported(test, feature)
323
509
self.skip_count += 1
324
510
self.report_skip(test, reason)
326
def _addNotApplicable(self, test, skip_excinfo):
327
if isinstance(skip_excinfo[1], TestNotApplicable):
328
self.not_applicable_count += 1
329
self.report_not_applicable(test, skip_excinfo)
332
except KeyboardInterrupt:
335
self.addError(test, test.exc_info())
337
# seems best to treat this as success from point-of-view of unittest
338
# -- it actually does nothing so it barely matters :)
339
unittest.TestResult.addSuccess(self, test)
340
test._log_contents = ''
342
def printErrorList(self, flavour, errors):
343
for test, err in errors:
344
self.stream.writeln(self.separator1)
345
self.stream.write("%s: " % flavour)
346
self.stream.writeln(self.getDescription(test))
347
if getattr(test, '_get_log', None) is not None:
348
log_contents = test._get_log()
350
self.stream.write('\n')
352
('vvvv[log from %s]' % test.id()).ljust(78,'-'))
353
self.stream.write('\n')
354
self.stream.write(log_contents)
355
self.stream.write('\n')
357
('^^^^[log from %s]' % test.id()).ljust(78,'-'))
358
self.stream.write('\n')
359
self.stream.writeln(self.separator2)
360
self.stream.writeln("%s" % err)
365
def report_cleaning_up(self):
512
def addNotApplicable(self, test, reason):
513
self.not_applicable_count += 1
514
self.report_not_applicable(test, reason)
516
def _count_stored_tests(self):
517
"""Count of tests instances kept alive due to not succeeding"""
518
return self.error_count + self.failure_count + self.known_failure_count
520
def _post_mortem(self, tb=None):
521
"""Start a PDB post mortem session."""
522
if os.environ.get('BZR_TEST_PDB', None):
526
def progress(self, offset, whence):
527
"""The test is adjusting the count of tests to run."""
528
if whence == SUBUNIT_SEEK_SET:
529
self.num_tests = offset
530
elif whence == SUBUNIT_SEEK_CUR:
531
self.num_tests += offset
533
raise errors.BzrError("Unknown whence %r" % whence)
535
def report_tests_starting(self):
536
"""Display information before the test run begins"""
537
if getattr(sys, 'frozen', None) is None:
538
bzr_path = osutils.realpath(sys.argv[0])
540
bzr_path = sys.executable
542
'bzr selftest: %s\n' % (bzr_path,))
545
bzrlib.__path__[0],))
547
' bzr-%s python-%s %s\n' % (
548
bzrlib.version_string,
549
bzrlib._format_version_tuple(sys.version_info),
550
platform.platform(aliased=1),
552
self.stream.write('\n')
554
def report_test_start(self, test):
555
"""Display information on the test just about to be run"""
557
def _report_thread_leak(self, test, leaked_threads, active_threads):
558
"""Display information on a test that leaked one or more threads"""
559
# GZ 2010-09-09: A leak summary reported separately from the general
560
# thread debugging would be nice. Tests under subunit
561
# need something not using stream, perhaps adding a
562
# testtools details object would be fitting.
563
if 'threads' in selftest_debug_flags:
564
self.stream.write('%s is leaking, active is now %d\n' %
565
(test.id(), len(active_threads)))
567
def startTestRun(self):
568
self.startTime = time.time()
368
570
def report_success(self, test):
499
709
return '%s%s' % (indent, err[1])
501
711
def report_error(self, test, err):
502
self.stream.writeln('ERROR %s\n%s'
712
self.stream.write('ERROR %s\n%s\n'
503
713
% (self._testTimeString(test),
504
714
self._error_summary(err)))
506
716
def report_failure(self, test, err):
507
self.stream.writeln(' FAIL %s\n%s'
717
self.stream.write(' FAIL %s\n%s\n'
508
718
% (self._testTimeString(test),
509
719
self._error_summary(err)))
511
721
def report_known_failure(self, test, err):
512
self.stream.writeln('XFAIL %s\n%s'
722
self.stream.write('XFAIL %s\n%s\n'
513
723
% (self._testTimeString(test),
514
724
self._error_summary(err)))
726
def report_unexpected_success(self, test, reason):
727
self.stream.write(' FAIL %s\n%s: %s\n'
728
% (self._testTimeString(test),
729
"Unexpected success. Should have failed",
516
732
def report_success(self, test):
517
self.stream.writeln(' OK %s' % self._testTimeString(test))
733
self.stream.write(' OK %s\n' % self._testTimeString(test))
518
734
for bench_called, stats in getattr(test, '_benchcalls', []):
519
self.stream.writeln('LSProf output for %s(%s, %s)' % bench_called)
735
self.stream.write('LSProf output for %s(%s, %s)\n' % bench_called)
520
736
stats.pprint(file=self.stream)
521
737
# flush the stream so that we get smooth output. This verbose mode is
522
738
# used to show the output in PQM.
523
739
self.stream.flush()
525
741
def report_skip(self, test, reason):
526
self.stream.writeln(' SKIP %s\n%s'
742
self.stream.write(' SKIP %s\n%s\n'
527
743
% (self._testTimeString(test), reason))
529
def report_not_applicable(self, test, skip_excinfo):
530
self.stream.writeln(' N/A %s\n%s'
531
% (self._testTimeString(test),
532
self._error_summary(skip_excinfo)))
745
def report_not_applicable(self, test, reason):
746
self.stream.write(' N/A %s\n %s\n'
747
% (self._testTimeString(test), reason))
534
749
def report_unsupported(self, test, feature):
535
750
"""test cannot be run because feature is missing."""
536
self.stream.writeln("NODEP %s\n The feature '%s' is not available."
751
self.stream.write("NODEP %s\n The feature '%s' is not available.\n"
537
752
%(self._testTimeString(test), feature))
547
762
bench_history=None,
764
result_decorators=None,
551
self.stream = unittest._WritelnDecorator(stream)
766
"""Create a TextTestRunner.
768
:param result_decorators: An optional list of decorators to apply
769
to the result object being used by the runner. Decorators are
770
applied left to right - the first element in the list is the
773
# stream may know claim to know to write unicode strings, but in older
774
# pythons this goes sufficiently wrong that it is a bad idea. (
775
# specifically a built in file with encoding 'UTF-8' will still try
776
# to encode using ascii.
777
new_encoding = osutils.get_terminal_encoding()
778
codec = codecs.lookup(new_encoding)
779
if type(codec) is tuple:
783
encode = codec.encode
784
# GZ 2010-09-08: Really we don't want to be writing arbitrary bytes,
785
# so should swap to the plain codecs.StreamWriter
786
stream = osutils.UnicodeOrBytesToBytesWriter(encode, stream,
788
stream.encoding = new_encoding
552
790
self.descriptions = descriptions
553
791
self.verbosity = verbosity
554
792
self._bench_history = bench_history
555
self.list_only = list_only
556
793
self._strict = strict
794
self._result_decorators = result_decorators or []
558
796
def run(self, test):
559
797
"Run the given test case or test suite."
560
startTime = time.time()
561
798
if self.verbosity == 1:
562
799
result_class = TextTestResult
563
800
elif self.verbosity >= 2:
564
801
result_class = VerboseTestResult
565
result = result_class(self.stream,
802
original_result = result_class(self.stream,
566
803
self.descriptions,
568
805
bench_history=self._bench_history,
569
num_tests=test.countTestCases(),
570
806
strict=self._strict,
572
result.stop_early = self.stop_on_failure
573
result.report_starting()
575
if self.verbosity >= 2:
576
self.stream.writeln("Listing tests only ...\n")
578
for t in iter_suite_tests(test):
579
self.stream.writeln("%s" % (t.id()))
588
if isinstance(test, testtools.ConcurrentTestSuite):
589
# We need to catch bzr specific behaviors
590
test.run(BZRTransformingResult(result))
593
run = result.testsRun
595
stopTime = time.time()
596
timeTaken = stopTime - startTime
598
self.stream.writeln(result.separator2)
599
self.stream.writeln("%s %d test%s in %.3fs" % (actionTaken,
600
run, run != 1 and "s" or "", timeTaken))
601
self.stream.writeln()
602
if not result.wasSuccessful():
603
self.stream.write("FAILED (")
604
failed, errored = map(len, (result.failures, result.errors))
606
self.stream.write("failures=%d" % failed)
608
if failed: self.stream.write(", ")
609
self.stream.write("errors=%d" % errored)
610
if result.known_failure_count:
611
if failed or errored: self.stream.write(", ")
612
self.stream.write("known_failure_count=%d" %
613
result.known_failure_count)
614
self.stream.writeln(")")
616
if result.known_failure_count:
617
self.stream.writeln("OK (known_failures=%d)" %
618
result.known_failure_count)
620
self.stream.writeln("OK")
621
if result.skip_count > 0:
622
skipped = result.skip_count
623
self.stream.writeln('%d test%s skipped' %
624
(skipped, skipped != 1 and "s" or ""))
625
if result.unsupported:
626
for feature, count in sorted(result.unsupported.items()):
627
self.stream.writeln("Missing feature '%s' skipped %d tests." %
808
# Signal to result objects that look at stop early policy to stop,
809
original_result.stop_early = self.stop_on_failure
810
result = original_result
811
for decorator in self._result_decorators:
812
result = decorator(result)
813
result.stop_early = self.stop_on_failure
814
result.startTestRun()
819
# higher level code uses our extended protocol to determine
820
# what exit code to give.
821
return original_result
633
824
def iter_suite_tests(suite):
770
975
routine, and to build and check bzr trees.
772
977
In addition to the usual method of overriding tearDown(), this class also
773
allows subclasses to register functions into the _cleanups list, which is
978
allows subclasses to register cleanup functions via addCleanup, which are
774
979
run in order as the object is torn down. It's less likely this will be
775
980
accidentally overlooked.
778
_active_threads = None
779
_leaking_threads_tests = 0
780
_first_thread_leaker_id = None
781
_log_file_name = None
783
_keep_log_file = False
784
984
# record lsprof data when performing benchmark calls.
785
985
_gather_lsprof_in_benchmarks = False
786
attrs_to_keep = ('id', '_testMethodName', '_testMethodDoc',
787
'_log_contents', '_log_file_name', '_benchtime',
788
'_TestCase__testMethodName', '_TestCase__testMethodDoc',)
790
987
def __init__(self, methodName='testMethod'):
791
988
super(TestCase, self).__init__(methodName)
793
self._bzr_test_setUp_run = False
794
self._bzr_test_tearDown_run = False
989
self._directory_isolation = True
990
self.exception_handlers.insert(0,
991
(UnavailableFeature, self._do_unsupported_or_skip))
992
self.exception_handlers.insert(0,
993
(TestNotApplicable, self._do_not_applicable))
797
unittest.TestCase.setUp(self)
798
self._bzr_test_setUp_run = True
996
super(TestCase, self).setUp()
997
for feature in getattr(self, '_test_needs_features', []):
998
self.requireFeature(feature)
799
999
self._cleanEnvironment()
1000
if bzrlib.global_state is not None:
1001
self.overrideAttr(bzrlib.global_state, 'cmdline_overrides',
1002
config.CommandLineSection())
800
1003
self._silenceUI()
801
1004
self._startLogFile()
802
1005
self._benchcalls = []
803
1006
self._benchtime = None
804
1007
self._clear_hooks()
805
# Track locks - needs to be called before _clear_debug_flags.
1008
self._track_transports()
806
1009
self._track_locks()
807
1010
self._clear_debug_flags()
808
TestCase._active_threads = threading.activeCount()
809
self.addCleanup(self._check_leaked_threads)
1011
# Isolate global verbosity level, to make sure it's reproducible
1012
# between tests. We should get rid of this altogether: bug 656694. --
1014
self.overrideAttr(bzrlib.trace, '_verbosity_level', 0)
1015
# Isolate config option expansion until its default value for bzrlib is
1016
# settled on or a the FIXME associated with _get_expand_default_value
1017
# is addressed -- vila 20110219
1018
self.overrideAttr(config, '_expand_default_value', None)
1019
self._log_files = set()
1020
# Each key in the ``_counters`` dict holds a value for a different
1021
# counter. When the test ends, addDetail() should be used to output the
1022
# counter values. This happens in install_counter_hook().
1024
if 'config_stats' in selftest_debug_flags:
1025
self._install_config_stats_hooks()
1026
# Do not use i18n for tests (unless the test reverses this)
811
1029
def debug(self):
812
1030
# debug a frame up.
814
pdb.Pdb().set_trace(sys._getframe().f_back)
816
def _check_leaked_threads(self):
817
active = threading.activeCount()
818
leaked_threads = active - TestCase._active_threads
819
TestCase._active_threads = active
821
TestCase._leaking_threads_tests += 1
822
if TestCase._first_thread_leaker_id is None:
823
TestCase._first_thread_leaker_id = self.id()
1032
# The sys preserved stdin/stdout should allow blackbox tests debugging
1033
pdb.Pdb(stdin=sys.__stdin__, stdout=sys.__stdout__
1034
).set_trace(sys._getframe().f_back)
1036
def discardDetail(self, name):
1037
"""Extend the addDetail, getDetails api so we can remove a detail.
1039
eg. bzr always adds the 'log' detail at startup, but we don't want to
1040
include it for skipped, xfail, etc tests.
1042
It is safe to call this for a detail that doesn't exist, in case this
1043
gets called multiple times.
1045
# We cheat. details is stored in __details which means we shouldn't
1046
# touch it. but getDetails() returns the dict directly, so we can
1048
details = self.getDetails()
1052
def install_counter_hook(self, hooks, name, counter_name=None):
1053
"""Install a counting hook.
1055
Any hook can be counted as long as it doesn't need to return a value.
1057
:param hooks: Where the hook should be installed.
1059
:param name: The hook name that will be counted.
1061
:param counter_name: The counter identifier in ``_counters``, defaults
1064
_counters = self._counters # Avoid closing over self
1065
if counter_name is None:
1067
if _counters.has_key(counter_name):
1068
raise AssertionError('%s is already used as a counter name'
1070
_counters[counter_name] = 0
1071
self.addDetail(counter_name, content.Content(content.UTF8_TEXT,
1072
lambda: ['%d' % (_counters[counter_name],)]))
1073
def increment_counter(*args, **kwargs):
1074
_counters[counter_name] += 1
1075
label = 'count %s calls' % (counter_name,)
1076
hooks.install_named_hook(name, increment_counter, label)
1077
self.addCleanup(hooks.uninstall_named_hook, name, label)
1079
def _install_config_stats_hooks(self):
1080
"""Install config hooks to count hook calls.
1083
for hook_name in ('get', 'set', 'remove', 'load', 'save'):
1084
self.install_counter_hook(config.ConfigHooks, hook_name,
1085
'config.%s' % (hook_name,))
1087
# The OldConfigHooks are private and need special handling to protect
1088
# against recursive tests (tests that run other tests), so we just do
1089
# manually what registering them into _builtin_known_hooks will provide
1091
self.overrideAttr(config, 'OldConfigHooks', config._OldConfigHooks())
1092
for hook_name in ('get', 'set', 'remove', 'load', 'save'):
1093
self.install_counter_hook(config.OldConfigHooks, hook_name,
1094
'old_config.%s' % (hook_name,))
825
1096
def _clear_debug_flags(self):
826
1097
"""Prevent externally set debug flags affecting tests.
828
1099
Tests that want to use debug flags can just set them in the
829
1100
debug_flags set during setup/teardown.
831
self._preserved_debug_flags = set(debug.debug_flags)
1102
# Start with a copy of the current debug flags we can safely modify.
1103
self.overrideAttr(debug, 'debug_flags', set(debug.debug_flags))
832
1104
if 'allow_debug' not in selftest_debug_flags:
833
1105
debug.debug_flags.clear()
834
self.addCleanup(self._restore_debug_flags)
1106
if 'disable_lock_checks' not in selftest_debug_flags:
1107
debug.debug_flags.add('strict_locks')
836
1109
def _clear_hooks(self):
837
1110
# prevent hooks affecting tests
1111
known_hooks = hooks.known_hooks
838
1112
self._preserved_hooks = {}
839
for key, factory in hooks.known_hooks.items():
840
parent, name = hooks.known_hooks_key_to_parent_and_attribute(key)
841
current_hooks = hooks.known_hooks_key_to_object(key)
1113
for key, (parent, name) in known_hooks.iter_parent_objects():
1114
current_hooks = getattr(parent, name)
842
1115
self._preserved_hooks[parent] = (name, current_hooks)
1116
self._preserved_lazy_hooks = hooks._lazy_hooks
1117
hooks._lazy_hooks = {}
843
1118
self.addCleanup(self._restoreHooks)
844
for key, factory in hooks.known_hooks.items():
845
parent, name = hooks.known_hooks_key_to_parent_and_attribute(key)
1119
for key, (parent, name) in known_hooks.iter_parent_objects():
1120
factory = known_hooks.get(key)
846
1121
setattr(parent, name, factory())
847
1122
# this hook should always be installed
848
1123
request._install_hook()
1125
def disable_directory_isolation(self):
1126
"""Turn off directory isolation checks."""
1127
self._directory_isolation = False
1129
def enable_directory_isolation(self):
1130
"""Enable directory isolation checks."""
1131
self._directory_isolation = True
850
1133
def _silenceUI(self):
851
1134
"""Turn off UI for duration of test"""
852
1135
# by default the UI is off; tests can turn it on if they want it.
853
saved = ui.ui_factory
855
ui.ui_factory = saved
856
ui.ui_factory = ui.SilentUIFactory()
857
self.addCleanup(_restore)
1136
self.overrideAttr(ui, 'ui_factory', ui.SilentUIFactory())
859
1138
def _check_locks(self):
860
1139
"""Check that all lock take/release actions have been paired."""
861
# once we have fixed all the current lock problems, we can change the
862
# following code to always check for mismatched locks, but only do
863
# traceback showing with -Dlock (self._lock_check_thorough is True).
864
# For now, because the test suite will fail, we only assert that lock
865
# matching has occured with -Dlock.
1140
# We always check for mismatched locks. If a mismatch is found, we
1141
# fail unless -Edisable_lock_checks is supplied to selftest, in which
1142
# case we just print a warning.
867
1144
acquired_locks = [lock for action, lock in self._lock_actions
868
1145
if action == 'acquired']
905
1190
def _lock_broken(self, result):
906
1191
self._lock_actions.append(('broken', result))
1193
def permit_dir(self, name):
1194
"""Permit a directory to be used by this test. See permit_url."""
1195
name_transport = _mod_transport.get_transport_from_path(name)
1196
self.permit_url(name)
1197
self.permit_url(name_transport.base)
1199
def permit_url(self, url):
1200
"""Declare that url is an ok url to use in this test.
1202
Do this for memory transports, temporary test directory etc.
1204
Do not do this for the current working directory, /tmp, or any other
1205
preexisting non isolated url.
1207
if not url.endswith('/'):
1209
self._bzr_selftest_roots.append(url)
1211
def permit_source_tree_branch_repo(self):
1212
"""Permit the source tree bzr is running from to be opened.
1214
Some code such as bzrlib.version attempts to read from the bzr branch
1215
that bzr is executing from (if any). This method permits that directory
1216
to be used in the test suite.
1218
path = self.get_source_path()
1219
self.record_directory_isolation()
1222
workingtree.WorkingTree.open(path)
1223
except (errors.NotBranchError, errors.NoWorkingTree):
1224
raise TestSkipped('Needs a working tree of bzr sources')
1226
self.enable_directory_isolation()
1228
def _preopen_isolate_transport(self, transport):
1229
"""Check that all transport openings are done in the test work area."""
1230
while isinstance(transport, pathfilter.PathFilteringTransport):
1231
# Unwrap pathfiltered transports
1232
transport = transport.server.backing_transport.clone(
1233
transport._filter('.'))
1234
url = transport.base
1235
# ReadonlySmartTCPServer_for_testing decorates the backing transport
1236
# urls it is given by prepending readonly+. This is appropriate as the
1237
# client shouldn't know that the server is readonly (or not readonly).
1238
# We could register all servers twice, with readonly+ prepending, but
1239
# that makes for a long list; this is about the same but easier to
1241
if url.startswith('readonly+'):
1242
url = url[len('readonly+'):]
1243
self._preopen_isolate_url(url)
1245
def _preopen_isolate_url(self, url):
1246
if not self._directory_isolation:
1248
if self._directory_isolation == 'record':
1249
self._bzr_selftest_roots.append(url)
1251
# This prevents all transports, including e.g. sftp ones backed on disk
1252
# from working unless they are explicitly granted permission. We then
1253
# depend on the code that sets up test transports to check that they are
1254
# appropriately isolated and enable their use by calling
1255
# self.permit_transport()
1256
if not osutils.is_inside_any(self._bzr_selftest_roots, url):
1257
raise errors.BzrError("Attempt to escape test isolation: %r %r"
1258
% (url, self._bzr_selftest_roots))
1260
def record_directory_isolation(self):
1261
"""Gather accessed directories to permit later access.
1263
This is used for tests that access the branch bzr is running from.
1265
self._directory_isolation = "record"
1267
def start_server(self, transport_server, backing_server=None):
1268
"""Start transport_server for this test.
1270
This starts the server, registers a cleanup for it and permits the
1271
server's urls to be used.
1273
if backing_server is None:
1274
transport_server.start_server()
1276
transport_server.start_server(backing_server)
1277
self.addCleanup(transport_server.stop_server)
1278
# Obtain a real transport because if the server supplies a password, it
1279
# will be hidden from the base on the client side.
1280
t = _mod_transport.get_transport_from_url(transport_server.get_url())
1281
# Some transport servers effectively chroot the backing transport;
1282
# others like SFTPServer don't - users of the transport can walk up the
1283
# transport to read the entire backing transport. This wouldn't matter
1284
# except that the workdir tests are given - and that they expect the
1285
# server's url to point at - is one directory under the safety net. So
1286
# Branch operations into the transport will attempt to walk up one
1287
# directory. Chrooting all servers would avoid this but also mean that
1288
# we wouldn't be testing directly against non-root urls. Alternatively
1289
# getting the test framework to start the server with a backing server
1290
# at the actual safety net directory would work too, but this then
1291
# means that the self.get_url/self.get_transport methods would need
1292
# to transform all their results. On balance its cleaner to handle it
1293
# here, and permit a higher url when we have one of these transports.
1294
if t.base.endswith('/work/'):
1295
# we have safety net/test root/work
1296
t = t.clone('../..')
1297
elif isinstance(transport_server,
1298
test_server.SmartTCPServer_for_testing):
1299
# The smart server adds a path similar to work, which is traversed
1300
# up from by the client. But the server is chrooted - the actual
1301
# backing transport is not escaped from, and VFS requests to the
1302
# root will error (because they try to escape the chroot).
1304
while t2.base != t.base:
1307
self.permit_url(t.base)
1309
def _track_transports(self):
1310
"""Install checks for transport usage."""
1311
# TestCase has no safe place it can write to.
1312
self._bzr_selftest_roots = []
1313
# Currently the easiest way to be sure that nothing is going on is to
1314
# hook into bzr dir opening. This leaves a small window of error for
1315
# transport tests, but they are well known, and we can improve on this
1317
bzrdir.BzrDir.hooks.install_named_hook("pre_open",
1318
self._preopen_isolate_transport, "Check bzr directories are safe.")
908
1320
def _ndiff_strings(self, a, b):
909
1321
"""Return ndiff between two strings containing lines.
1286
1723
The file is removed as the test is torn down.
1288
fileno, name = tempfile.mkstemp(suffix='.log', prefix='testbzr')
1289
self._log_file = os.fdopen(fileno, 'w+')
1290
self._log_memento = bzrlib.trace.push_log_file(self._log_file)
1291
self._log_file_name = name
1725
pseudo_log_file = StringIO()
1726
def _get_log_contents_for_weird_testtools_api():
1727
return [pseudo_log_file.getvalue().decode(
1728
"utf-8", "replace").encode("utf-8")]
1729
self.addDetail("log", content.Content(content.ContentType("text",
1730
"plain", {"charset": "utf8"}),
1731
_get_log_contents_for_weird_testtools_api))
1732
self._log_file = pseudo_log_file
1733
self._log_memento = trace.push_log_file(self._log_file)
1292
1734
self.addCleanup(self._finishLogFile)
1294
1736
def _finishLogFile(self):
1295
1737
"""Finished with the log file.
1297
Close the file and delete it, unless setKeepLogfile was called.
1299
if self._log_file is None:
1301
bzrlib.trace.pop_log_file(self._log_memento)
1302
self._log_file.close()
1303
self._log_file = None
1304
if not self._keep_log_file:
1305
os.remove(self._log_file_name)
1306
self._log_file_name = None
1308
def setKeepLogfile(self):
1309
"""Make the logfile not be deleted when _finishLogFile is called."""
1310
self._keep_log_file = True
1312
def addCleanup(self, callable, *args, **kwargs):
1313
"""Arrange to run a callable when this case is torn down.
1315
Callables are run in the reverse of the order they are registered,
1316
ie last-in first-out.
1318
self._cleanups.append((callable, args, kwargs))
1739
Close the file and delete it.
1741
if trace._trace_file:
1742
# flush the log file, to get all content
1743
trace._trace_file.flush()
1744
trace.pop_log_file(self._log_memento)
1746
def thisFailsStrictLockCheck(self):
1747
"""It is known that this test would fail with -Dstrict_locks.
1749
By default, all tests are run with strict lock checking unless
1750
-Edisable_lock_checks is supplied. However there are some tests which
1751
we know fail strict locks at this point that have not been fixed.
1752
They should call this function to disable the strict checking.
1754
This should be used sparingly, it is much better to fix the locking
1755
issues rather than papering over the problem by calling this function.
1757
debug.debug_flags.discard('strict_locks')
1759
def overrideAttr(self, obj, attr_name, new=_unitialized_attr):
1760
"""Overrides an object attribute restoring it after the test.
1762
:note: This should be used with discretion; you should think about
1763
whether it's better to make the code testable without monkey-patching.
1765
:param obj: The object that will be mutated.
1767
:param attr_name: The attribute name we want to preserve/override in
1770
:param new: The optional value we want to set the attribute to.
1772
:returns: The actual attr value.
1774
value = getattr(obj, attr_name)
1775
# The actual value is captured by the call below
1776
self.addCleanup(setattr, obj, attr_name, value)
1777
if new is not _unitialized_attr:
1778
setattr(obj, attr_name, new)
1781
def overrideEnv(self, name, new):
1782
"""Set an environment variable, and reset it after the test.
1784
:param name: The environment variable name.
1786
:param new: The value to set the variable to. If None, the
1787
variable is deleted from the environment.
1789
:returns: The actual variable value.
1791
value = osutils.set_or_unset_env(name, new)
1792
self.addCleanup(osutils.set_or_unset_env, name, value)
1795
def recordCalls(self, obj, attr_name):
1796
"""Monkeypatch in a wrapper that will record calls.
1798
The monkeypatch is automatically removed when the test concludes.
1800
:param obj: The namespace holding the reference to be replaced;
1801
typically a module, class, or object.
1802
:param attr_name: A string for the name of the attribute to
1804
:returns: A list that will be extended with one item every time the
1805
function is called, with a tuple of (args, kwargs).
1809
def decorator(*args, **kwargs):
1810
calls.append((args, kwargs))
1811
return orig(*args, **kwargs)
1812
orig = self.overrideAttr(obj, attr_name, decorator)
1320
1815
def _cleanEnvironment(self):
1322
'BZR_HOME': None, # Don't inherit BZR_HOME to all the tests.
1323
'HOME': os.getcwd(),
1324
# bzr now uses the Win32 API and doesn't rely on APPDATA, but the
1325
# tests do check our impls match APPDATA
1326
'BZR_EDITOR': None, # test_msgeditor manipulates this variable
1330
'BZREMAIL': None, # may still be present in the environment
1332
'BZR_PROGRESS_BAR': None,
1334
'BZR_PLUGIN_PATH': None,
1335
# Make sure that any text ui tests are consistent regardless of
1336
# the environment the test case is run in; you may want tests that
1337
# test other combinations. 'dumb' is a reasonable guess for tests
1338
# going to a pipe or a StringIO.
1343
'SSH_AUTH_SOCK': None,
1347
'https_proxy': None,
1348
'HTTPS_PROXY': None,
1353
# Nobody cares about ftp_proxy, FTP_PROXY AFAIK. So far at
1354
# least. If you do (care), please update this comment
1358
'BZR_REMOTE_PATH': None,
1361
self.addCleanup(self._restoreEnvironment)
1362
for name, value in new_env.iteritems():
1363
self._captureVar(name, value)
1365
def _captureVar(self, name, newvalue):
1366
"""Set an environment variable, and reset it when finished."""
1367
self.__old_env[name] = osutils.set_or_unset_env(name, newvalue)
1369
def _restore_debug_flags(self):
1370
debug.debug_flags.clear()
1371
debug.debug_flags.update(self._preserved_debug_flags)
1373
def _restoreEnvironment(self):
1374
for name, value in self.__old_env.iteritems():
1375
osutils.set_or_unset_env(name, value)
1816
for name, value in isolated_environ.iteritems():
1817
self.overrideEnv(name, value)
1377
1819
def _restoreHooks(self):
1378
1820
for klass, (name, hooks) in self._preserved_hooks.items():
1379
1821
setattr(klass, name, hooks)
1822
self._preserved_hooks.clear()
1823
bzrlib.hooks._lazy_hooks = self._preserved_lazy_hooks
1824
self._preserved_lazy_hooks.clear()
1381
1826
def knownFailure(self, reason):
1382
"""This test has failed for some known reason."""
1383
raise KnownFailure(reason)
1827
"""Declare that this test fails for a known reason
1829
Tests that are known to fail should generally be using expectedFailure
1830
with an appropriate reverse assertion if a change could cause the test
1831
to start passing. Conversely if the test has no immediate prospect of
1832
succeeding then using skip is more suitable.
1834
When this method is called while an exception is being handled, that
1835
traceback will be used, otherwise a new exception will be thrown to
1836
provide one but won't be reported.
1838
self._add_reason(reason)
1840
exc_info = sys.exc_info()
1841
if exc_info != (None, None, None):
1842
self._report_traceback(exc_info)
1845
raise self.failureException(reason)
1846
except self.failureException:
1847
exc_info = sys.exc_info()
1848
# GZ 02-08-2011: Maybe cleanup this err.exc_info attribute too?
1849
raise testtools.testcase._ExpectedFailure(exc_info)
1853
def _suppress_log(self):
1854
"""Remove the log info from details."""
1855
self.discardDetail('log')
1385
1857
def _do_skip(self, result, reason):
1858
self._suppress_log()
1386
1859
addSkip = getattr(result, 'addSkip', None)
1387
1860
if not callable(addSkip):
1388
result.addError(self, sys.exc_info())
1861
result.addSuccess(result)
1390
1863
addSkip(self, reason)
1392
def run(self, result=None):
1393
if result is None: result = self.defaultTestResult()
1394
for feature in getattr(self, '_test_needs_features', []):
1395
if not feature.available():
1396
result.startTest(self)
1397
if getattr(result, 'addNotSupported', None):
1398
result.addNotSupported(self, feature)
1400
result.addSuccess(self)
1401
result.stopTest(self)
1405
result.startTest(self)
1406
absent_attr = object()
1408
method_name = getattr(self, '_testMethodName', absent_attr)
1409
if method_name is absent_attr:
1411
method_name = getattr(self, '_TestCase__testMethodName')
1412
testMethod = getattr(self, method_name)
1416
if not self._bzr_test_setUp_run:
1418
"test setUp did not invoke "
1419
"bzrlib.tests.TestCase's setUp")
1420
except KeyboardInterrupt:
1423
except TestSkipped, e:
1424
self._do_skip(result, e.args[0])
1428
result.addError(self, sys.exc_info())
1436
except self.failureException:
1437
result.addFailure(self, sys.exc_info())
1438
except TestSkipped, e:
1440
reason = "No reason given."
1443
self._do_skip(result, reason)
1444
except KeyboardInterrupt:
1448
result.addError(self, sys.exc_info())
1452
if not self._bzr_test_tearDown_run:
1454
"test tearDown did not invoke "
1455
"bzrlib.tests.TestCase's tearDown")
1456
except KeyboardInterrupt:
1460
result.addError(self, sys.exc_info())
1463
if ok: result.addSuccess(self)
1465
result.stopTest(self)
1467
except TestNotApplicable:
1468
# Not moved from the result [yet].
1471
except KeyboardInterrupt:
1476
for attr_name in self.attrs_to_keep:
1477
if attr_name in self.__dict__:
1478
saved_attrs[attr_name] = self.__dict__[attr_name]
1479
self.__dict__ = saved_attrs
1483
self._log_contents = ''
1484
self._bzr_test_tearDown_run = True
1485
unittest.TestCase.tearDown(self)
1866
def _do_known_failure(self, result, e):
1867
self._suppress_log()
1868
err = sys.exc_info()
1869
addExpectedFailure = getattr(result, 'addExpectedFailure', None)
1870
if addExpectedFailure is not None:
1871
addExpectedFailure(self, err)
1873
result.addSuccess(self)
1876
def _do_not_applicable(self, result, e):
1878
reason = 'No reason given'
1881
self._suppress_log ()
1882
addNotApplicable = getattr(result, 'addNotApplicable', None)
1883
if addNotApplicable is not None:
1884
result.addNotApplicable(self, reason)
1886
self._do_skip(result, reason)
1889
def _report_skip(self, result, err):
1890
"""Override the default _report_skip.
1892
We want to strip the 'log' detail. If we waint until _do_skip, it has
1893
already been formatted into the 'reason' string, and we can't pull it
1896
self._suppress_log()
1897
super(TestCase, self)._report_skip(self, result, err)
1900
def _report_expected_failure(self, result, err):
1903
See _report_skip for motivation.
1905
self._suppress_log()
1906
super(TestCase, self)._report_expected_failure(self, result, err)
1909
def _do_unsupported_or_skip(self, result, e):
1911
self._suppress_log()
1912
addNotSupported = getattr(result, 'addNotSupported', None)
1913
if addNotSupported is not None:
1914
result.addNotSupported(self, reason)
1916
self._do_skip(result, reason)
1487
1918
def time(self, callable, *args, **kwargs):
1488
1919
"""Run callable and accrue the time it takes to the benchmark time.
2832
class TestDecorator(TestSuite):
3373
class TestDecorator(TestUtil.TestSuite):
2833
3374
"""A decorator for TestCase/TestSuite objects.
2835
Usually, subclasses should override __iter__(used when flattening test
2836
suites), which we do to filter, reorder, parallelise and so on, run() and
3376
Contains rather than flattening suite passed on construction
2840
def __init__(self, suite):
2841
TestSuite.__init__(self)
2844
def countTestCases(self):
2847
cases += test.countTestCases()
3379
def __init__(self, suite=None):
3380
super(TestDecorator, self).__init__()
3381
if suite is not None:
3384
# Don't need subclass run method with suite emptying
3385
run = unittest.TestSuite.run
3388
class CountingDecorator(TestDecorator):
3389
"""A decorator which calls result.progress(self.countTestCases)."""
2854
3391
def run(self, result):
2855
# Use iteration on self, not self._tests, to allow subclasses to hook
2858
if result.shouldStop:
3392
progress_method = getattr(result, 'progress', None)
3393
if callable(progress_method):
3394
progress_method(self.countTestCases(), SUBUNIT_SEEK_SET)
3395
return super(CountingDecorator, self).run(result)
2864
3398
class ExcludeDecorator(TestDecorator):
2865
3399
"""A decorator which excludes test matching an exclude pattern."""
2867
3401
def __init__(self, suite, exclude_pattern):
2868
TestDecorator.__init__(self, suite)
2869
self.exclude_pattern = exclude_pattern
2870
self.excluded = False
2874
return iter(self._tests)
2875
self.excluded = True
2876
suite = exclude_tests_by_re(self, self.exclude_pattern)
2878
self.addTests(suite)
2879
return iter(self._tests)
3402
super(ExcludeDecorator, self).__init__(
3403
exclude_tests_by_re(suite, exclude_pattern))
2882
3406
class FilterTestsDecorator(TestDecorator):
2883
3407
"""A decorator which filters tests to those matching a pattern."""
2885
3409
def __init__(self, suite, pattern):
2886
TestDecorator.__init__(self, suite)
2887
self.pattern = pattern
2888
self.filtered = False
2892
return iter(self._tests)
2893
self.filtered = True
2894
suite = filter_suite_by_re(self, self.pattern)
2896
self.addTests(suite)
2897
return iter(self._tests)
3410
super(FilterTestsDecorator, self).__init__(
3411
filter_suite_by_re(suite, pattern))
2900
3414
class RandomDecorator(TestDecorator):
2901
3415
"""A decorator which randomises the order of its tests."""
2903
3417
def __init__(self, suite, random_seed, stream):
2904
TestDecorator.__init__(self, suite)
2905
self.random_seed = random_seed
2906
self.randomised = False
2907
self.stream = stream
2911
return iter(self._tests)
2912
self.randomised = True
2913
self.stream.writeln("Randomizing test order using seed %s\n" %
2914
(self.actual_seed()))
3418
random_seed = self.actual_seed(random_seed)
3419
stream.write("Randomizing test order using seed %s\n\n" %
2915
3421
# Initialise the random number generator.
2916
random.seed(self.actual_seed())
2917
suite = randomize_suite(self)
2919
self.addTests(suite)
2920
return iter(self._tests)
3422
random.seed(random_seed)
3423
super(RandomDecorator, self).__init__(randomize_suite(suite))
2922
def actual_seed(self):
2923
if self.random_seed == "now":
3426
def actual_seed(seed):
2924
3428
# We convert the seed to a long to make it reuseable across
2925
3429
# invocations (because the user can reenter it).
2926
self.random_seed = long(time.time())
3430
return long(time.time())
2928
3432
# Convert the seed to a long if we can
2930
self.random_seed = long(self.random_seed)
3435
except (TypeError, ValueError):
2933
return self.random_seed
2936
3440
class TestFirstDecorator(TestDecorator):
2937
3441
"""A decorator which moves named tests to the front."""
2939
3443
def __init__(self, suite, pattern):
2940
TestDecorator.__init__(self, suite)
2941
self.pattern = pattern
2942
self.filtered = False
2946
return iter(self._tests)
2947
self.filtered = True
2948
suites = split_suite_by_re(self, self.pattern)
2950
self.addTests(suites)
2951
return iter(self._tests)
3444
super(TestFirstDecorator, self).__init__()
3445
self.addTests(split_suite_by_re(suite, pattern))
2954
3448
def partition_tests(suite, count):
2955
3449
"""Partition suite into count lists of tests."""
2957
tests = list(iter_suite_tests(suite))
2958
tests_per_process = int(math.ceil(float(len(tests)) / count))
2959
for block in range(count):
2960
low_test = block * tests_per_process
2961
high_test = low_test + tests_per_process
2962
process_tests = tests[low_test:high_test]
2963
result.append(process_tests)
3450
# This just assigns tests in a round-robin fashion. On one hand this
3451
# splits up blocks of related tests that might run faster if they shared
3452
# resources, but on the other it avoids assigning blocks of slow tests to
3453
# just one partition. So the slowest partition shouldn't be much slower
3455
partitions = [list() for i in range(count)]
3456
tests = iter_suite_tests(suite)
3457
for partition, test in itertools.izip(itertools.cycle(partitions), tests):
3458
partition.append(test)
3462
def workaround_zealous_crypto_random():
3463
"""Crypto.Random want to help us being secure, but we don't care here.
3465
This workaround some test failure related to the sftp server. Once paramiko
3466
stop using the controversial API in Crypto.Random, we may get rid of it.
3469
from Crypto.Random import atfork
2967
3475
def fork_for_tests(suite):
3073
class BZRTransformingResult(unittest.TestResult):
3591
class ProfileResult(testtools.ExtendedToOriginalDecorator):
3592
"""Generate profiling data for all activity between start and success.
3594
The profile data is appended to the test's _benchcalls attribute and can
3595
be accessed by the forwarded-to TestResult.
3075
def __init__(self, target):
3076
unittest.TestResult.__init__(self)
3077
self.result = target
3597
While it might be cleaner do accumulate this in stopTest, addSuccess is
3598
where our existing output support for lsprof is, and this class aims to
3599
fit in with that: while it could be moved it's not necessary to accomplish
3600
test profiling, nor would it be dramatically cleaner.
3079
3603
def startTest(self, test):
3080
self.result.startTest(test)
3604
self.profiler = bzrlib.lsprof.BzrProfiler()
3605
# Prevent deadlocks in tests that use lsprof: those tests will
3607
bzrlib.lsprof.BzrProfiler.profiler_block = 0
3608
self.profiler.start()
3609
testtools.ExtendedToOriginalDecorator.startTest(self, test)
3611
def addSuccess(self, test):
3612
stats = self.profiler.stop()
3614
calls = test._benchcalls
3615
except AttributeError:
3616
test._benchcalls = []
3617
calls = test._benchcalls
3618
calls.append(((test.id(), "", ""), stats))
3619
testtools.ExtendedToOriginalDecorator.addSuccess(self, test)
3082
3621
def stopTest(self, test):
3083
self.result.stopTest(test)
3085
def addError(self, test, err):
3086
feature = self._error_looks_like('UnavailableFeature: ', err)
3087
if feature is not None:
3088
self.result.addNotSupported(test, feature)
3090
self.result.addError(test, err)
3092
def addFailure(self, test, err):
3093
known = self._error_looks_like('KnownFailure: ', err)
3094
if known is not None:
3095
self.result._addKnownFailure(test, [KnownFailure,
3096
KnownFailure(known), None])
3098
self.result.addFailure(test, err)
3100
def addSkip(self, test, reason):
3101
self.result.addSkip(test, reason)
3103
def addSuccess(self, test):
3104
self.result.addSuccess(test)
3106
def _error_looks_like(self, prefix, err):
3107
"""Deserialize exception and returns the stringify value."""
3111
if isinstance(exc, subunit.RemoteException):
3112
# stringify the exception gives access to the remote traceback
3113
# We search the last line for 'prefix'
3114
lines = str(exc).split('\n')
3115
while lines and not lines[-1]:
3118
if lines[-1].startswith(prefix):
3119
value = lines[-1][len(prefix):]
3622
testtools.ExtendedToOriginalDecorator.stopTest(self, test)
3623
self.profiler = None
3123
3626
# Controlled by "bzr selftest -E=..." option
3627
# Currently supported:
3628
# -Eallow_debug Will no longer clear debug.debug_flags() so it
3629
# preserves any flags supplied at the command line.
3630
# -Edisable_lock_checks Turns errors in mismatched locks into simple prints
3631
# rather than failing tests. And no longer raise
3632
# LockContention when fctnl locks are not being used
3633
# with proper exclusion rules.
3634
# -Ethreads Will display thread ident at creation/join time to
3635
# help track thread leaks
3636
# -Euncollected_cases Display the identity of any test cases that weren't
3637
# deallocated after being completed.
3638
# -Econfig_stats Will collect statistics using addDetail
3124
3639
selftest_debug_flags = set()
3338
3869
test_prefix_alias_registry.register('bp', 'bzrlib.plugins')
3872
def _test_suite_testmod_names():
3873
"""Return the standard list of test module names to test."""
3876
'bzrlib.tests.blackbox',
3877
'bzrlib.tests.commands',
3878
'bzrlib.tests.doc_generate',
3879
'bzrlib.tests.per_branch',
3880
'bzrlib.tests.per_bzrdir',
3881
'bzrlib.tests.per_controldir',
3882
'bzrlib.tests.per_controldir_colo',
3883
'bzrlib.tests.per_foreign_vcs',
3884
'bzrlib.tests.per_interrepository',
3885
'bzrlib.tests.per_intertree',
3886
'bzrlib.tests.per_inventory',
3887
'bzrlib.tests.per_interbranch',
3888
'bzrlib.tests.per_lock',
3889
'bzrlib.tests.per_merger',
3890
'bzrlib.tests.per_transport',
3891
'bzrlib.tests.per_tree',
3892
'bzrlib.tests.per_pack_repository',
3893
'bzrlib.tests.per_repository',
3894
'bzrlib.tests.per_repository_chk',
3895
'bzrlib.tests.per_repository_reference',
3896
'bzrlib.tests.per_repository_vf',
3897
'bzrlib.tests.per_uifactory',
3898
'bzrlib.tests.per_versionedfile',
3899
'bzrlib.tests.per_workingtree',
3900
'bzrlib.tests.test__annotator',
3901
'bzrlib.tests.test__bencode',
3902
'bzrlib.tests.test__btree_serializer',
3903
'bzrlib.tests.test__chk_map',
3904
'bzrlib.tests.test__dirstate_helpers',
3905
'bzrlib.tests.test__groupcompress',
3906
'bzrlib.tests.test__known_graph',
3907
'bzrlib.tests.test__rio',
3908
'bzrlib.tests.test__simple_set',
3909
'bzrlib.tests.test__static_tuple',
3910
'bzrlib.tests.test__walkdirs_win32',
3911
'bzrlib.tests.test_ancestry',
3912
'bzrlib.tests.test_annotate',
3913
'bzrlib.tests.test_api',
3914
'bzrlib.tests.test_atomicfile',
3915
'bzrlib.tests.test_bad_files',
3916
'bzrlib.tests.test_bisect_multi',
3917
'bzrlib.tests.test_branch',
3918
'bzrlib.tests.test_branchbuilder',
3919
'bzrlib.tests.test_btree_index',
3920
'bzrlib.tests.test_bugtracker',
3921
'bzrlib.tests.test_bundle',
3922
'bzrlib.tests.test_bzrdir',
3923
'bzrlib.tests.test__chunks_to_lines',
3924
'bzrlib.tests.test_cache_utf8',
3925
'bzrlib.tests.test_chk_map',
3926
'bzrlib.tests.test_chk_serializer',
3927
'bzrlib.tests.test_chunk_writer',
3928
'bzrlib.tests.test_clean_tree',
3929
'bzrlib.tests.test_cleanup',
3930
'bzrlib.tests.test_cmdline',
3931
'bzrlib.tests.test_commands',
3932
'bzrlib.tests.test_commit',
3933
'bzrlib.tests.test_commit_merge',
3934
'bzrlib.tests.test_config',
3935
'bzrlib.tests.test_conflicts',
3936
'bzrlib.tests.test_controldir',
3937
'bzrlib.tests.test_counted_lock',
3938
'bzrlib.tests.test_crash',
3939
'bzrlib.tests.test_decorators',
3940
'bzrlib.tests.test_delta',
3941
'bzrlib.tests.test_debug',
3942
'bzrlib.tests.test_diff',
3943
'bzrlib.tests.test_directory_service',
3944
'bzrlib.tests.test_dirstate',
3945
'bzrlib.tests.test_email_message',
3946
'bzrlib.tests.test_eol_filters',
3947
'bzrlib.tests.test_errors',
3948
'bzrlib.tests.test_estimate_compressed_size',
3949
'bzrlib.tests.test_export',
3950
'bzrlib.tests.test_export_pot',
3951
'bzrlib.tests.test_extract',
3952
'bzrlib.tests.test_features',
3953
'bzrlib.tests.test_fetch',
3954
'bzrlib.tests.test_fixtures',
3955
'bzrlib.tests.test_fifo_cache',
3956
'bzrlib.tests.test_filters',
3957
'bzrlib.tests.test_filter_tree',
3958
'bzrlib.tests.test_ftp_transport',
3959
'bzrlib.tests.test_foreign',
3960
'bzrlib.tests.test_generate_docs',
3961
'bzrlib.tests.test_generate_ids',
3962
'bzrlib.tests.test_globbing',
3963
'bzrlib.tests.test_gpg',
3964
'bzrlib.tests.test_graph',
3965
'bzrlib.tests.test_groupcompress',
3966
'bzrlib.tests.test_hashcache',
3967
'bzrlib.tests.test_help',
3968
'bzrlib.tests.test_hooks',
3969
'bzrlib.tests.test_http',
3970
'bzrlib.tests.test_http_response',
3971
'bzrlib.tests.test_https_ca_bundle',
3972
'bzrlib.tests.test_i18n',
3973
'bzrlib.tests.test_identitymap',
3974
'bzrlib.tests.test_ignores',
3975
'bzrlib.tests.test_index',
3976
'bzrlib.tests.test_import_tariff',
3977
'bzrlib.tests.test_info',
3978
'bzrlib.tests.test_inv',
3979
'bzrlib.tests.test_inventory_delta',
3980
'bzrlib.tests.test_knit',
3981
'bzrlib.tests.test_lazy_import',
3982
'bzrlib.tests.test_lazy_regex',
3983
'bzrlib.tests.test_library_state',
3984
'bzrlib.tests.test_lock',
3985
'bzrlib.tests.test_lockable_files',
3986
'bzrlib.tests.test_lockdir',
3987
'bzrlib.tests.test_log',
3988
'bzrlib.tests.test_lru_cache',
3989
'bzrlib.tests.test_lsprof',
3990
'bzrlib.tests.test_mail_client',
3991
'bzrlib.tests.test_matchers',
3992
'bzrlib.tests.test_memorytree',
3993
'bzrlib.tests.test_merge',
3994
'bzrlib.tests.test_merge3',
3995
'bzrlib.tests.test_merge_core',
3996
'bzrlib.tests.test_merge_directive',
3997
'bzrlib.tests.test_mergetools',
3998
'bzrlib.tests.test_missing',
3999
'bzrlib.tests.test_msgeditor',
4000
'bzrlib.tests.test_multiparent',
4001
'bzrlib.tests.test_mutabletree',
4002
'bzrlib.tests.test_nonascii',
4003
'bzrlib.tests.test_options',
4004
'bzrlib.tests.test_osutils',
4005
'bzrlib.tests.test_osutils_encodings',
4006
'bzrlib.tests.test_pack',
4007
'bzrlib.tests.test_patch',
4008
'bzrlib.tests.test_patches',
4009
'bzrlib.tests.test_permissions',
4010
'bzrlib.tests.test_plugins',
4011
'bzrlib.tests.test_progress',
4012
'bzrlib.tests.test_pyutils',
4013
'bzrlib.tests.test_read_bundle',
4014
'bzrlib.tests.test_reconcile',
4015
'bzrlib.tests.test_reconfigure',
4016
'bzrlib.tests.test_registry',
4017
'bzrlib.tests.test_remote',
4018
'bzrlib.tests.test_rename_map',
4019
'bzrlib.tests.test_repository',
4020
'bzrlib.tests.test_revert',
4021
'bzrlib.tests.test_revision',
4022
'bzrlib.tests.test_revisionspec',
4023
'bzrlib.tests.test_revisiontree',
4024
'bzrlib.tests.test_rio',
4025
'bzrlib.tests.test_rules',
4026
'bzrlib.tests.test_sampler',
4027
'bzrlib.tests.test_scenarios',
4028
'bzrlib.tests.test_script',
4029
'bzrlib.tests.test_selftest',
4030
'bzrlib.tests.test_serializer',
4031
'bzrlib.tests.test_setup',
4032
'bzrlib.tests.test_sftp_transport',
4033
'bzrlib.tests.test_shelf',
4034
'bzrlib.tests.test_shelf_ui',
4035
'bzrlib.tests.test_smart',
4036
'bzrlib.tests.test_smart_add',
4037
'bzrlib.tests.test_smart_request',
4038
'bzrlib.tests.test_smart_signals',
4039
'bzrlib.tests.test_smart_transport',
4040
'bzrlib.tests.test_smtp_connection',
4041
'bzrlib.tests.test_source',
4042
'bzrlib.tests.test_ssh_transport',
4043
'bzrlib.tests.test_status',
4044
'bzrlib.tests.test_store',
4045
'bzrlib.tests.test_strace',
4046
'bzrlib.tests.test_subsume',
4047
'bzrlib.tests.test_switch',
4048
'bzrlib.tests.test_symbol_versioning',
4049
'bzrlib.tests.test_tag',
4050
'bzrlib.tests.test_test_server',
4051
'bzrlib.tests.test_testament',
4052
'bzrlib.tests.test_textfile',
4053
'bzrlib.tests.test_textmerge',
4054
'bzrlib.tests.test_cethread',
4055
'bzrlib.tests.test_timestamp',
4056
'bzrlib.tests.test_trace',
4057
'bzrlib.tests.test_transactions',
4058
'bzrlib.tests.test_transform',
4059
'bzrlib.tests.test_transport',
4060
'bzrlib.tests.test_transport_log',
4061
'bzrlib.tests.test_tree',
4062
'bzrlib.tests.test_treebuilder',
4063
'bzrlib.tests.test_treeshape',
4064
'bzrlib.tests.test_tsort',
4065
'bzrlib.tests.test_tuned_gzip',
4066
'bzrlib.tests.test_ui',
4067
'bzrlib.tests.test_uncommit',
4068
'bzrlib.tests.test_upgrade',
4069
'bzrlib.tests.test_upgrade_stacked',
4070
'bzrlib.tests.test_urlutils',
4071
'bzrlib.tests.test_utextwrap',
4072
'bzrlib.tests.test_version',
4073
'bzrlib.tests.test_version_info',
4074
'bzrlib.tests.test_versionedfile',
4075
'bzrlib.tests.test_weave',
4076
'bzrlib.tests.test_whitebox',
4077
'bzrlib.tests.test_win32utils',
4078
'bzrlib.tests.test_workingtree',
4079
'bzrlib.tests.test_workingtree_4',
4080
'bzrlib.tests.test_wsgi',
4081
'bzrlib.tests.test_xml',
4085
def _test_suite_modules_to_doctest():
4086
"""Return the list of modules to doctest."""
4088
# GZ 2009-03-31: No docstrings with -OO so there's nothing to doctest
4092
'bzrlib.branchbuilder',
4093
'bzrlib.decorators',
4095
'bzrlib.iterablefile',
4100
'bzrlib.symbol_versioning',
4102
'bzrlib.tests.fixtures',
4104
'bzrlib.transport.http',
4105
'bzrlib.version_info_formats.format_custom',
3341
4109
def test_suite(keep_only=None, starting_with=None):
3342
4110
"""Build and return TestSuite for the whole of bzrlib.
3349
4117
This function can be replaced if you need to change the default test
3350
4118
suite on a global basis, but it is not encouraged.
3354
'bzrlib.tests.blackbox',
3355
'bzrlib.tests.commands',
3356
'bzrlib.tests.per_branch',
3357
'bzrlib.tests.per_bzrdir',
3358
'bzrlib.tests.per_interrepository',
3359
'bzrlib.tests.per_intertree',
3360
'bzrlib.tests.per_inventory',
3361
'bzrlib.tests.per_interbranch',
3362
'bzrlib.tests.per_lock',
3363
'bzrlib.tests.per_transport',
3364
'bzrlib.tests.per_tree',
3365
'bzrlib.tests.per_repository',
3366
'bzrlib.tests.per_repository_chk',
3367
'bzrlib.tests.per_repository_reference',
3368
'bzrlib.tests.per_workingtree',
3369
'bzrlib.tests.test__annotator',
3370
'bzrlib.tests.test__chk_map',
3371
'bzrlib.tests.test__dirstate_helpers',
3372
'bzrlib.tests.test__groupcompress',
3373
'bzrlib.tests.test__known_graph',
3374
'bzrlib.tests.test__rio',
3375
'bzrlib.tests.test__walkdirs_win32',
3376
'bzrlib.tests.test_ancestry',
3377
'bzrlib.tests.test_annotate',
3378
'bzrlib.tests.test_api',
3379
'bzrlib.tests.test_atomicfile',
3380
'bzrlib.tests.test_bad_files',
3381
'bzrlib.tests.test_bencode',
3382
'bzrlib.tests.test_bisect_multi',
3383
'bzrlib.tests.test_branch',
3384
'bzrlib.tests.test_branchbuilder',
3385
'bzrlib.tests.test_btree_index',
3386
'bzrlib.tests.test_bugtracker',
3387
'bzrlib.tests.test_bundle',
3388
'bzrlib.tests.test_bzrdir',
3389
'bzrlib.tests.test__chunks_to_lines',
3390
'bzrlib.tests.test_cache_utf8',
3391
'bzrlib.tests.test_chk_map',
3392
'bzrlib.tests.test_chk_serializer',
3393
'bzrlib.tests.test_chunk_writer',
3394
'bzrlib.tests.test_clean_tree',
3395
'bzrlib.tests.test_commands',
3396
'bzrlib.tests.test_commit',
3397
'bzrlib.tests.test_commit_merge',
3398
'bzrlib.tests.test_config',
3399
'bzrlib.tests.test_conflicts',
3400
'bzrlib.tests.test_counted_lock',
3401
'bzrlib.tests.test_decorators',
3402
'bzrlib.tests.test_delta',
3403
'bzrlib.tests.test_debug',
3404
'bzrlib.tests.test_deprecated_graph',
3405
'bzrlib.tests.test_diff',
3406
'bzrlib.tests.test_directory_service',
3407
'bzrlib.tests.test_dirstate',
3408
'bzrlib.tests.test_email_message',
3409
'bzrlib.tests.test_eol_filters',
3410
'bzrlib.tests.test_errors',
3411
'bzrlib.tests.test_export',
3412
'bzrlib.tests.test_extract',
3413
'bzrlib.tests.test_fetch',
3414
'bzrlib.tests.test_fifo_cache',
3415
'bzrlib.tests.test_filters',
3416
'bzrlib.tests.test_ftp_transport',
3417
'bzrlib.tests.test_foreign',
3418
'bzrlib.tests.test_generate_docs',
3419
'bzrlib.tests.test_generate_ids',
3420
'bzrlib.tests.test_globbing',
3421
'bzrlib.tests.test_gpg',
3422
'bzrlib.tests.test_graph',
3423
'bzrlib.tests.test_groupcompress',
3424
'bzrlib.tests.test_hashcache',
3425
'bzrlib.tests.test_help',
3426
'bzrlib.tests.test_hooks',
3427
'bzrlib.tests.test_http',
3428
'bzrlib.tests.test_http_response',
3429
'bzrlib.tests.test_https_ca_bundle',
3430
'bzrlib.tests.test_identitymap',
3431
'bzrlib.tests.test_ignores',
3432
'bzrlib.tests.test_index',
3433
'bzrlib.tests.test_info',
3434
'bzrlib.tests.test_inv',
3435
'bzrlib.tests.test_inventory_delta',
3436
'bzrlib.tests.test_knit',
3437
'bzrlib.tests.test_lazy_import',
3438
'bzrlib.tests.test_lazy_regex',
3439
'bzrlib.tests.test_lockable_files',
3440
'bzrlib.tests.test_lockdir',
3441
'bzrlib.tests.test_log',
3442
'bzrlib.tests.test_lru_cache',
3443
'bzrlib.tests.test_lsprof',
3444
'bzrlib.tests.test_mail_client',
3445
'bzrlib.tests.test_memorytree',
3446
'bzrlib.tests.test_merge',
3447
'bzrlib.tests.test_merge3',
3448
'bzrlib.tests.test_merge_core',
3449
'bzrlib.tests.test_merge_directive',
3450
'bzrlib.tests.test_missing',
3451
'bzrlib.tests.test_msgeditor',
3452
'bzrlib.tests.test_multiparent',
3453
'bzrlib.tests.test_mutabletree',
3454
'bzrlib.tests.test_nonascii',
3455
'bzrlib.tests.test_options',
3456
'bzrlib.tests.test_osutils',
3457
'bzrlib.tests.test_osutils_encodings',
3458
'bzrlib.tests.test_pack',
3459
'bzrlib.tests.test_pack_repository',
3460
'bzrlib.tests.test_patch',
3461
'bzrlib.tests.test_patches',
3462
'bzrlib.tests.test_permissions',
3463
'bzrlib.tests.test_plugins',
3464
'bzrlib.tests.test_progress',
3465
'bzrlib.tests.test_read_bundle',
3466
'bzrlib.tests.test_reconcile',
3467
'bzrlib.tests.test_reconfigure',
3468
'bzrlib.tests.test_registry',
3469
'bzrlib.tests.test_remote',
3470
'bzrlib.tests.test_rename_map',
3471
'bzrlib.tests.test_repository',
3472
'bzrlib.tests.test_revert',
3473
'bzrlib.tests.test_revision',
3474
'bzrlib.tests.test_revisionspec',
3475
'bzrlib.tests.test_revisiontree',
3476
'bzrlib.tests.test_rio',
3477
'bzrlib.tests.test_rules',
3478
'bzrlib.tests.test_sampler',
3479
'bzrlib.tests.test_selftest',
3480
'bzrlib.tests.test_serializer',
3481
'bzrlib.tests.test_setup',
3482
'bzrlib.tests.test_sftp_transport',
3483
'bzrlib.tests.test_shelf',
3484
'bzrlib.tests.test_shelf_ui',
3485
'bzrlib.tests.test_smart',
3486
'bzrlib.tests.test_smart_add',
3487
'bzrlib.tests.test_smart_request',
3488
'bzrlib.tests.test_smart_transport',
3489
'bzrlib.tests.test_smtp_connection',
3490
'bzrlib.tests.test_source',
3491
'bzrlib.tests.test_ssh_transport',
3492
'bzrlib.tests.test_status',
3493
'bzrlib.tests.test_store',
3494
'bzrlib.tests.test_strace',
3495
'bzrlib.tests.test_subsume',
3496
'bzrlib.tests.test_switch',
3497
'bzrlib.tests.test_symbol_versioning',
3498
'bzrlib.tests.test_tag',
3499
'bzrlib.tests.test_testament',
3500
'bzrlib.tests.test_textfile',
3501
'bzrlib.tests.test_textmerge',
3502
'bzrlib.tests.test_timestamp',
3503
'bzrlib.tests.test_trace',
3504
'bzrlib.tests.test_transactions',
3505
'bzrlib.tests.test_transform',
3506
'bzrlib.tests.test_transport',
3507
'bzrlib.tests.test_transport_log',
3508
'bzrlib.tests.test_tree',
3509
'bzrlib.tests.test_treebuilder',
3510
'bzrlib.tests.test_tsort',
3511
'bzrlib.tests.test_tuned_gzip',
3512
'bzrlib.tests.test_ui',
3513
'bzrlib.tests.test_uncommit',
3514
'bzrlib.tests.test_upgrade',
3515
'bzrlib.tests.test_upgrade_stacked',
3516
'bzrlib.tests.test_urlutils',
3517
'bzrlib.tests.test_version',
3518
'bzrlib.tests.test_version_info',
3519
'bzrlib.tests.test_versionedfile',
3520
'bzrlib.tests.test_weave',
3521
'bzrlib.tests.test_whitebox',
3522
'bzrlib.tests.test_win32utils',
3523
'bzrlib.tests.test_workingtree',
3524
'bzrlib.tests.test_workingtree_4',
3525
'bzrlib.tests.test_wsgi',
3526
'bzrlib.tests.test_xml',
3529
4121
loader = TestUtil.TestLoader()
3531
4123
if keep_only is not None:
3532
4124
id_filter = TestIdList(keep_only)
3533
4125
if starting_with:
3534
starting_with = [test_prefix_alias_registry.resolve_alias(start)
3535
for start in starting_with]
3536
4126
# We take precedence over keep_only because *at loading time* using
3537
4127
# both options means we will load less tests for the same final result.
3538
4128
def interesting_module(name):
3728
4312
:param new_id: The id to assign to it.
3729
4313
:return: The new test.
3731
from copy import deepcopy
3732
new_test = deepcopy(test)
4315
new_test = copy.copy(test)
3733
4316
new_test.id = lambda: new_id
4317
# XXX: Workaround <https://bugs.launchpad.net/testtools/+bug/637725>, which
4318
# causes cloned tests to share the 'details' dict. This makes it hard to
4319
# read the test output for parameterized tests, because tracebacks will be
4320
# associated with irrelevant tests.
4322
details = new_test._TestCase__details
4323
except AttributeError:
4324
# must be a different version of testtools than expected. Do nothing.
4327
# Reset the '__details' dict.
4328
new_test._TestCase__details = {}
3734
4329
return new_test
3737
def _rmtree_temp_dir(dirname):
4332
def permute_tests_for_extension(standard_tests, loader, py_module_name,
4334
"""Helper for permutating tests against an extension module.
4336
This is meant to be used inside a modules 'load_tests()' function. It will
4337
create 2 scenarios, and cause all tests in the 'standard_tests' to be run
4338
against both implementations. Setting 'test.module' to the appropriate
4339
module. See bzrlib.tests.test__chk_map.load_tests as an example.
4341
:param standard_tests: A test suite to permute
4342
:param loader: A TestLoader
4343
:param py_module_name: The python path to a python module that can always
4344
be loaded, and will be considered the 'python' implementation. (eg
4345
'bzrlib._chk_map_py')
4346
:param ext_module_name: The python path to an extension module. If the
4347
module cannot be loaded, a single test will be added, which notes that
4348
the module is not available. If it can be loaded, all standard_tests
4349
will be run against that module.
4350
:return: (suite, feature) suite is a test-suite that has all the permuted
4351
tests. feature is the Feature object that can be used to determine if
4352
the module is available.
4355
from bzrlib.tests.features import ModuleAvailableFeature
4356
py_module = pyutils.get_named_object(py_module_name)
4358
('python', {'module': py_module}),
4360
suite = loader.suiteClass()
4361
feature = ModuleAvailableFeature(ext_module_name)
4362
if feature.available():
4363
scenarios.append(('C', {'module': feature.module}))
4365
# the compiled module isn't available, so we add a failing test
4366
class FailWithoutFeature(TestCase):
4367
def test_fail(self):
4368
self.requireFeature(feature)
4369
suite.addTest(loader.loadTestsFromTestCase(FailWithoutFeature))
4370
result = multiply_tests(standard_tests, scenarios, suite)
4371
return result, feature
4374
def _rmtree_temp_dir(dirname, test_id=None):
3738
4375
# If LANG=C we probably have created some bogus paths
3739
4376
# which rmtree(unicode) will fail to delete
3740
4377
# so make sure we are using rmtree(str) to delete everything
3750
4387
osutils.rmtree(dirname)
3751
4388
except OSError, e:
3752
if sys.platform == 'win32' and e.errno == errno.EACCES:
3753
sys.stderr.write('Permission denied: '
3754
'unable to remove testing dir '
3756
% (os.path.basename(dirname), e))
3761
class Feature(object):
3762
"""An operating system Feature."""
3765
self._available = None
3767
def available(self):
3768
"""Is the feature available?
3770
:return: True if the feature is available.
3772
if self._available is None:
3773
self._available = self._probe()
3774
return self._available
3777
"""Implement this method in concrete features.
3779
:return: True if the feature is available.
3781
raise NotImplementedError
3784
if getattr(self, 'feature_name', None):
3785
return self.feature_name()
3786
return self.__class__.__name__
3789
class _SymlinkFeature(Feature):
3792
return osutils.has_symlinks()
3794
def feature_name(self):
3797
SymlinkFeature = _SymlinkFeature()
3800
class _HardlinkFeature(Feature):
3803
return osutils.has_hardlinks()
3805
def feature_name(self):
3808
HardlinkFeature = _HardlinkFeature()
3811
class _OsFifoFeature(Feature):
3814
return getattr(os, 'mkfifo', None)
3816
def feature_name(self):
3817
return 'filesystem fifos'
3819
OsFifoFeature = _OsFifoFeature()
3822
class _UnicodeFilenameFeature(Feature):
3823
"""Does the filesystem support Unicode filenames?"""
3827
# Check for character combinations unlikely to be covered by any
3828
# single non-unicode encoding. We use the characters
3829
# - greek small letter alpha (U+03B1) and
3830
# - braille pattern dots-123456 (U+283F).
3831
os.stat(u'\u03b1\u283f')
3832
except UnicodeEncodeError:
3834
except (IOError, OSError):
3835
# The filesystem allows the Unicode filename but the file doesn't
3839
# The filesystem allows the Unicode filename and the file exists,
3843
UnicodeFilenameFeature = _UnicodeFilenameFeature()
4389
# We don't want to fail here because some useful display will be lost
4390
# otherwise. Polluting the tmp dir is bad, but not giving all the
4391
# possible info to the test runner is even worse.
4393
ui.ui_factory.clear_term()
4394
sys.stderr.write('\nWhile running: %s\n' % (test_id,))
4395
# Ugly, but the last thing we want here is fail, so bear with it.
4396
printable_e = str(e).decode(osutils.get_user_encoding(), 'replace'
4397
).encode('ascii', 'replace')
4398
sys.stderr.write('Unable to remove testing dir %s\n%s'
4399
% (os.path.basename(dirname), printable_e))
3846
4402
def probe_unicode_in_user_encoding():
3879
class _HTTPSServerFeature(Feature):
3880
"""Some tests want an https Server, check if one is available.
3882
Right now, the only way this is available is under python2.6 which provides
3893
def feature_name(self):
3894
return 'HTTPSServer'
3897
HTTPSServerFeature = _HTTPSServerFeature()
3900
class _UnicodeFilename(Feature):
3901
"""Does the filesystem support Unicode filenames?"""
3906
except UnicodeEncodeError:
3908
except (IOError, OSError):
3909
# The filesystem allows the Unicode filename but the file doesn't
3913
# The filesystem allows the Unicode filename and the file exists,
3917
UnicodeFilename = _UnicodeFilename()
3920
class _UTF8Filesystem(Feature):
3921
"""Is the filesystem UTF-8?"""
3924
if osutils._fs_enc.upper() in ('UTF-8', 'UTF8'):
3928
UTF8Filesystem = _UTF8Filesystem()
3931
class _CaseInsCasePresFilenameFeature(Feature):
3932
"""Is the file-system case insensitive, but case-preserving?"""
3935
fileno, name = tempfile.mkstemp(prefix='MixedCase')
3937
# first check truly case-preserving for created files, then check
3938
# case insensitive when opening existing files.
3939
name = osutils.normpath(name)
3940
base, rel = osutils.split(name)
3941
found_rel = osutils.canonical_relpath(base, name)
3942
return (found_rel == rel
3943
and os.path.isfile(name.upper())
3944
and os.path.isfile(name.lower()))
3949
def feature_name(self):
3950
return "case-insensitive case-preserving filesystem"
3952
CaseInsCasePresFilenameFeature = _CaseInsCasePresFilenameFeature()
3955
class _CaseInsensitiveFilesystemFeature(Feature):
3956
"""Check if underlying filesystem is case-insensitive but *not* case
3959
# Note that on Windows, Cygwin, MacOS etc, the file-systems are far
3960
# more likely to be case preserving, so this case is rare.
3963
if CaseInsCasePresFilenameFeature.available():
3966
if TestCaseWithMemoryTransport.TEST_ROOT is None:
3967
root = osutils.mkdtemp(prefix='testbzr-', suffix='.tmp')
3968
TestCaseWithMemoryTransport.TEST_ROOT = root
3970
root = TestCaseWithMemoryTransport.TEST_ROOT
3971
tdir = osutils.mkdtemp(prefix='case-sensitive-probe-', suffix='',
3973
name_a = osutils.pathjoin(tdir, 'a')
3974
name_A = osutils.pathjoin(tdir, 'A')
3976
result = osutils.isdir(name_A)
3977
_rmtree_temp_dir(tdir)
3980
def feature_name(self):
3981
return 'case-insensitive filesystem'
3983
CaseInsensitiveFilesystemFeature = _CaseInsensitiveFilesystemFeature()
3986
class _SubUnitFeature(Feature):
3987
"""Check if subunit is available."""
3996
def feature_name(self):
3999
SubUnitFeature = _SubUnitFeature()
4000
4435
# Only define SubUnitBzrRunner if subunit is available.
4002
4437
from subunit import TestProtocolClient
4004
from subunit.test_results import AutoTimingTestResultDecorator
4006
AutoTimingTestResultDecorator = lambda x:x
4438
from subunit.test_results import AutoTimingTestResultDecorator
4439
class SubUnitBzrProtocolClient(TestProtocolClient):
4441
def stopTest(self, test):
4442
super(SubUnitBzrProtocolClient, self).stopTest(test)
4443
_clear__type_equality_funcs(test)
4445
def addSuccess(self, test, details=None):
4446
# The subunit client always includes the details in the subunit
4447
# stream, but we don't want to include it in ours.
4448
if details is not None and 'log' in details:
4450
return super(SubUnitBzrProtocolClient, self).addSuccess(
4007
4453
class SubUnitBzrRunner(TextTestRunner):
4008
4454
def run(self, test):
4009
4455
result = AutoTimingTestResultDecorator(
4010
TestProtocolClient(self.stream))
4456
SubUnitBzrProtocolClient(self.stream))
4011
4457
test.run(result)
4013
4459
except ImportError:
4463
@deprecated_function(deprecated_in((2, 5, 0)))
4464
def ModuleAvailableFeature(name):
4465
from bzrlib.tests import features
4466
return features.ModuleAvailableFeature(name)