~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_selftest.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2009-11-30 22:04:45 UTC
  • mfrom: (4789.28.4 2.1.0b4-builder-no-keys)
  • Revision ID: pqm@pqm.ubuntu.com-20091130220445-vbfmmgocbgcs195q
(jam) Update BTreeBuilder to remove ._keys and use StaticTuple

Show diffs side-by-side

added added

removed removed

Lines of Context:
38
38
    repository,
39
39
    symbol_versioning,
40
40
    tests,
 
41
    transport,
41
42
    workingtree,
42
43
    )
43
44
from bzrlib.repofmt import (
489
490
        self.assertEqualStat(real, fake)
490
491
 
491
492
    def test_assertEqualStat_notequal(self):
492
 
        self.build_tree(["foo", "bar"])
 
493
        self.build_tree(["foo", "longname"])
493
494
        self.assertRaises(AssertionError, self.assertEqualStat,
494
 
            os.lstat("foo"), os.lstat("bar"))
 
495
            os.lstat("foo"), os.lstat("longname"))
495
496
 
496
497
 
497
498
class TestTestCaseWithMemoryTransport(tests.TestCaseWithMemoryTransport):
515
516
        cwd = osutils.getcwd()
516
517
        self.assertIsSameRealPath(self.test_dir, cwd)
517
518
 
 
519
    def test_BZR_HOME_and_HOME_are_bytestrings(self):
 
520
        """The $BZR_HOME and $HOME environment variables should not be unicode.
 
521
 
 
522
        See https://bugs.launchpad.net/bzr/+bug/464174
 
523
        """
 
524
        self.assertIsInstance(os.environ['BZR_HOME'], str)
 
525
        self.assertIsInstance(os.environ['HOME'], str)
 
526
 
518
527
    def test_make_branch_and_memory_tree(self):
519
528
        """In TestCaseWithMemoryTransport we should not make the branch on disk.
520
529
 
576
585
                         self.get_transport().get_bytes(
577
586
                            'dir/.bzr/repository/format'))
578
587
 
579
 
    def test_safety_net(self):
580
 
        """No test should modify the safety .bzr directory.
581
 
 
582
 
        We just test that the _check_safety_net private method raises
583
 
        AssertionError, it's easier than building a test suite with the same
584
 
        test.
585
 
        """
586
 
        # Oops, a commit in the current directory (i.e. without local .bzr
587
 
        # directory) will crawl up the hierarchy to find a .bzr directory.
588
 
        self.run_bzr(['commit', '-mfoo', '--unchanged'])
589
 
        # But we have a safety net in place.
590
 
        self.assertRaises(AssertionError, self._check_safety_net)
591
 
 
592
588
    def test_dangling_locks_cause_failures(self):
593
589
        class TestDanglingLock(tests.TestCaseWithMemoryTransport):
594
590
            def test_function(self):
687
683
        self.assertEqual(url, t.clone('..').base)
688
684
 
689
685
 
 
686
class TestProfileResult(tests.TestCase):
 
687
 
 
688
    def test_profiles_tests(self):
 
689
        self.requireFeature(test_lsprof.LSProfFeature)
 
690
        terminal = unittest.TestResult()
 
691
        result = tests.ProfileResult(terminal)
 
692
        class Sample(tests.TestCase):
 
693
            def a(self):
 
694
                self.sample_function()
 
695
            def sample_function(self):
 
696
                pass
 
697
        test = Sample("a")
 
698
        test.attrs_to_keep = test.attrs_to_keep + ('_benchcalls',)
 
699
        test.run(result)
 
700
        self.assertLength(1, test._benchcalls)
 
701
        # We must be able to unpack it as the test reporting code wants
 
702
        (_, _, _), stats = test._benchcalls[0]
 
703
        self.assertTrue(callable(stats.pprint))
 
704
 
 
705
 
690
706
class TestTestResult(tests.TestCase):
691
707
 
692
708
    def check_timing(self, test_case, expected_re):
719
735
        self.check_timing(ShortDelayTestCase('test_short_delay'),
720
736
                          r"^ +[0-9]+ms$")
721
737
 
 
738
    def _patch_get_bzr_source_tree(self):
 
739
        # Reading from the actual source tree breaks isolation, but we don't
 
740
        # want to assume that thats *all* that would happen.
 
741
        def _get_bzr_source_tree():
 
742
            return None
 
743
        orig_get_bzr_source_tree = bzrlib.version._get_bzr_source_tree
 
744
        bzrlib.version._get_bzr_source_tree = _get_bzr_source_tree
 
745
        def restore():
 
746
            bzrlib.version._get_bzr_source_tree = orig_get_bzr_source_tree
 
747
        self.addCleanup(restore)
 
748
 
722
749
    def test_assigned_benchmark_file_stores_date(self):
 
750
        self._patch_get_bzr_source_tree()
723
751
        output = StringIO()
724
752
        result = bzrlib.tests.TextTestResult(self._log_file,
725
753
                                        descriptions=0,
733
761
        self.assertContainsRe(output_string, "--date [0-9.]+")
734
762
 
735
763
    def test_benchhistory_records_test_times(self):
 
764
        self._patch_get_bzr_source_tree()
736
765
        result_stream = StringIO()
737
766
        result = bzrlib.tests.TextTestResult(
738
767
            self._log_file,
800
829
    def test_known_failure(self):
801
830
        """A KnownFailure being raised should trigger several result actions."""
802
831
        class InstrumentedTestResult(tests.ExtendedTestResult):
803
 
            def done(self): pass
 
832
            def stopTestRun(self): pass
804
833
            def startTests(self): pass
805
834
            def report_test_start(self, test): pass
806
835
            def report_known_failure(self, test, err):
807
836
                self._call = test, err
808
837
        result = InstrumentedTestResult(None, None, None, None)
809
 
        def test_function():
810
 
            raise tests.KnownFailure('failed!')
811
 
        test = unittest.FunctionTestCase(test_function)
 
838
        class Test(tests.TestCase):
 
839
            def test_function(self):
 
840
                raise tests.KnownFailure('failed!')
 
841
        test = Test("test_function")
812
842
        test.run(result)
813
843
        # it should invoke 'report_known_failure'.
814
844
        self.assertEqual(2, len(result._call))
854
884
    def test_add_not_supported(self):
855
885
        """Test the behaviour of invoking addNotSupported."""
856
886
        class InstrumentedTestResult(tests.ExtendedTestResult):
857
 
            def done(self): pass
 
887
            def stopTestRun(self): pass
858
888
            def startTests(self): pass
859
889
            def report_test_start(self, test): pass
860
890
            def report_unsupported(self, test, feature):
898
928
    def test_unavailable_exception(self):
899
929
        """An UnavailableFeature being raised should invoke addNotSupported."""
900
930
        class InstrumentedTestResult(tests.ExtendedTestResult):
901
 
            def done(self): pass
 
931
            def stopTestRun(self): pass
902
932
            def startTests(self): pass
903
933
            def report_test_start(self, test): pass
904
934
            def addNotSupported(self, test, feature):
905
935
                self._call = test, feature
906
936
        result = InstrumentedTestResult(None, None, None, None)
907
937
        feature = tests.Feature()
908
 
        def test_function():
909
 
            raise tests.UnavailableFeature(feature)
910
 
        test = unittest.FunctionTestCase(test_function)
 
938
        class Test(tests.TestCase):
 
939
            def test_function(self):
 
940
                raise tests.UnavailableFeature(feature)
 
941
        test = Test("test_function")
911
942
        test.run(result)
912
943
        # it should invoke 'addNotSupported'.
913
944
        self.assertEqual(2, len(result._call))
930
961
                                             verbosity=1)
931
962
        test = self.get_passing_test()
932
963
        err = (tests.KnownFailure, tests.KnownFailure('foo'), None)
933
 
        result._addKnownFailure(test, err)
 
964
        result.addExpectedFailure(test, err)
934
965
        self.assertFalse(result.wasStrictlySuccessful())
935
966
        self.assertEqual(None, result._extractBenchmarkTime(test))
936
967
 
981
1012
        because of our use of global state.
982
1013
        """
983
1014
        old_root = tests.TestCaseInTempDir.TEST_ROOT
 
1015
        old_leak = tests.TestCase._first_thread_leaker_id
984
1016
        try:
985
1017
            tests.TestCaseInTempDir.TEST_ROOT = None
 
1018
            tests.TestCase._first_thread_leaker_id = None
986
1019
            return testrunner.run(test)
987
1020
        finally:
988
1021
            tests.TestCaseInTempDir.TEST_ROOT = old_root
 
1022
            tests.TestCase._first_thread_leaker_id = old_leak
989
1023
 
990
1024
    def test_known_failure_failed_run(self):
991
1025
        # run a test that generates a known failure which should be printed in
992
1026
        # the final output when real failures occur.
993
 
        def known_failure_test():
994
 
            raise tests.KnownFailure('failed')
 
1027
        class Test(tests.TestCase):
 
1028
            def known_failure_test(self):
 
1029
                raise tests.KnownFailure('failed')
995
1030
        test = unittest.TestSuite()
996
 
        test.addTest(unittest.FunctionTestCase(known_failure_test))
 
1031
        test.addTest(Test("known_failure_test"))
997
1032
        def failing_test():
998
1033
            raise AssertionError('foo')
999
1034
        test.addTest(unittest.FunctionTestCase(failing_test))
1017
1052
            )
1018
1053
 
1019
1054
    def test_known_failure_ok_run(self):
1020
 
        # run a test that generates a known failure which should be printed in the final output.
1021
 
        def known_failure_test():
1022
 
            raise tests.KnownFailure('failed')
1023
 
        test = unittest.FunctionTestCase(known_failure_test)
 
1055
        # run a test that generates a known failure which should be printed in
 
1056
        # the final output.
 
1057
        class Test(tests.TestCase):
 
1058
            def known_failure_test(self):
 
1059
                raise tests.KnownFailure('failed')
 
1060
        test = Test("known_failure_test")
1024
1061
        stream = StringIO()
1025
1062
        runner = tests.TextTestRunner(stream=stream)
1026
1063
        result = self.run_test_runner(runner, test)
1031
1068
            '\n'
1032
1069
            'OK \\(known_failures=1\\)\n')
1033
1070
 
 
1071
    def test_result_decorator(self):
 
1072
        # decorate results
 
1073
        calls = []
 
1074
        class LoggingDecorator(tests.ForwardingResult):
 
1075
            def startTest(self, test):
 
1076
                tests.ForwardingResult.startTest(self, test)
 
1077
                calls.append('start')
 
1078
        test = unittest.FunctionTestCase(lambda:None)
 
1079
        stream = StringIO()
 
1080
        runner = tests.TextTestRunner(stream=stream,
 
1081
            result_decorators=[LoggingDecorator])
 
1082
        result = self.run_test_runner(runner, test)
 
1083
        self.assertLength(1, calls)
 
1084
 
1034
1085
    def test_skipped_test(self):
1035
1086
        # run a test that is skipped, and check the suite as a whole still
1036
1087
        # succeeds.
1089
1140
 
1090
1141
    def test_not_applicable(self):
1091
1142
        # run a test that is skipped because it's not applicable
1092
 
        def not_applicable_test():
1093
 
            raise tests.TestNotApplicable('this test never runs')
 
1143
        class Test(tests.TestCase):
 
1144
            def not_applicable_test(self):
 
1145
                raise tests.TestNotApplicable('this test never runs')
1094
1146
        out = StringIO()
1095
1147
        runner = tests.TextTestRunner(stream=out, verbosity=2)
1096
 
        test = unittest.FunctionTestCase(not_applicable_test)
 
1148
        test = Test("not_applicable_test")
1097
1149
        result = self.run_test_runner(runner, test)
1098
1150
        self._log_file.write(out.getvalue())
1099
1151
        self.assertTrue(result.wasSuccessful())
1103
1155
        self.assertContainsRe(out.getvalue(),
1104
1156
                r'(?m)^    this test never runs')
1105
1157
 
1106
 
    def test_not_applicable_demo(self):
1107
 
        # just so you can see it in the test output
1108
 
        raise tests.TestNotApplicable('this test is just a demonstation')
1109
 
 
1110
1158
    def test_unsupported_features_listed(self):
1111
1159
        """When unsupported features are encountered they are detailed."""
1112
1160
        class Feature1(tests.Feature):
1132
1180
            ],
1133
1181
            lines[-3:])
1134
1182
 
 
1183
    def _patch_get_bzr_source_tree(self):
 
1184
        # Reading from the actual source tree breaks isolation, but we don't
 
1185
        # want to assume that thats *all* that would happen.
 
1186
        self._get_source_tree_calls = []
 
1187
        def _get_bzr_source_tree():
 
1188
            self._get_source_tree_calls.append("called")
 
1189
            return None
 
1190
        orig_get_bzr_source_tree = bzrlib.version._get_bzr_source_tree
 
1191
        bzrlib.version._get_bzr_source_tree = _get_bzr_source_tree
 
1192
        def restore():
 
1193
            bzrlib.version._get_bzr_source_tree = orig_get_bzr_source_tree
 
1194
        self.addCleanup(restore)
 
1195
 
1135
1196
    def test_bench_history(self):
1136
 
        # tests that the running the benchmark produces a history file
1137
 
        # containing a timestamp and the revision id of the bzrlib source which
1138
 
        # was tested.
1139
 
        workingtree = _get_bzr_source_tree()
 
1197
        # tests that the running the benchmark passes bench_history into
 
1198
        # the test result object. We can tell that happens if
 
1199
        # _get_bzr_source_tree is called.
 
1200
        self._patch_get_bzr_source_tree()
1140
1201
        test = TestRunner('dummy_test')
1141
1202
        output = StringIO()
1142
1203
        runner = tests.TextTestRunner(stream=self._log_file,
1144
1205
        result = self.run_test_runner(runner, test)
1145
1206
        output_string = output.getvalue()
1146
1207
        self.assertContainsRe(output_string, "--date [0-9.]+")
1147
 
        if workingtree is not None:
1148
 
            revision_id = workingtree.get_parent_ids()[0]
1149
 
            self.assertEndsWith(output_string.rstrip(), revision_id)
 
1208
        self.assertLength(1, self._get_source_tree_calls)
1150
1209
 
1151
1210
    def assertLogDeleted(self, test):
1152
1211
        log = test._get_log()
1261
1320
        self.assertContainsRe(log, 'this will be kept')
1262
1321
        self.assertEqual(log, test._log_contents)
1263
1322
 
 
1323
    def test_startTestRun(self):
 
1324
        """run should call result.startTestRun()"""
 
1325
        calls = []
 
1326
        class LoggingDecorator(tests.ForwardingResult):
 
1327
            def startTestRun(self):
 
1328
                tests.ForwardingResult.startTestRun(self)
 
1329
                calls.append('startTestRun')
 
1330
        test = unittest.FunctionTestCase(lambda:None)
 
1331
        stream = StringIO()
 
1332
        runner = tests.TextTestRunner(stream=stream,
 
1333
            result_decorators=[LoggingDecorator])
 
1334
        result = self.run_test_runner(runner, test)
 
1335
        self.assertLength(1, calls)
 
1336
 
 
1337
    def test_stopTestRun(self):
 
1338
        """run should call result.stopTestRun()"""
 
1339
        calls = []
 
1340
        class LoggingDecorator(tests.ForwardingResult):
 
1341
            def stopTestRun(self):
 
1342
                tests.ForwardingResult.stopTestRun(self)
 
1343
                calls.append('stopTestRun')
 
1344
        test = unittest.FunctionTestCase(lambda:None)
 
1345
        stream = StringIO()
 
1346
        runner = tests.TextTestRunner(stream=stream,
 
1347
            result_decorators=[LoggingDecorator])
 
1348
        result = self.run_test_runner(runner, test)
 
1349
        self.assertLength(1, calls)
 
1350
 
1264
1351
 
1265
1352
class SampleTestCase(tests.TestCase):
1266
1353
 
1432
1519
        outer_test = TestTestCase("outer_child")
1433
1520
        result = self.make_test_result()
1434
1521
        outer_test.run(result)
 
1522
        self.addCleanup(osutils.delete_any, outer_test._log_file_name)
1435
1523
        self.assertEqual(original_trace, bzrlib.trace._trace_file)
1436
1524
 
1437
1525
    def method_that_times_a_bit_twice(self):
1480
1568
        self.assertEqual((time.sleep, (0.003,), {}), self._benchcalls[1][0])
1481
1569
        self.assertIsInstance(self._benchcalls[0][1], bzrlib.lsprof.Stats)
1482
1570
        self.assertIsInstance(self._benchcalls[1][1], bzrlib.lsprof.Stats)
 
1571
        del self._benchcalls[:]
1483
1572
 
1484
1573
    def test_knownFailure(self):
1485
1574
        """Self.knownFailure() should raise a KnownFailure exception."""
1486
1575
        self.assertRaises(tests.KnownFailure, self.knownFailure, "A Failure")
1487
1576
 
 
1577
    def test_open_bzrdir_safe_roots(self):
 
1578
        # even a memory transport should fail to open when its url isn't 
 
1579
        # permitted.
 
1580
        # Manually set one up (TestCase doesn't and shouldn't provide magic
 
1581
        # machinery)
 
1582
        transport_server = MemoryServer()
 
1583
        transport_server.setUp()
 
1584
        self.addCleanup(transport_server.tearDown)
 
1585
        t = transport.get_transport(transport_server.get_url())
 
1586
        bzrdir.BzrDir.create(t.base)
 
1587
        self.assertRaises(errors.BzrError,
 
1588
            bzrdir.BzrDir.open_from_transport, t)
 
1589
        # But if we declare this as safe, we can open the bzrdir.
 
1590
        self.permit_url(t.base)
 
1591
        self._bzr_selftest_roots.append(t.base)
 
1592
        bzrdir.BzrDir.open_from_transport(t)
 
1593
 
1488
1594
    def test_requireFeature_available(self):
1489
1595
        """self.requireFeature(available) is a no-op."""
1490
1596
        class Available(tests.Feature):
1557
1663
            ],
1558
1664
            result.calls)
1559
1665
 
 
1666
    def test_start_server_registers_url(self):
 
1667
        transport_server = MemoryServer()
 
1668
        # A little strict, but unlikely to be changed soon.
 
1669
        self.assertEqual([], self._bzr_selftest_roots)
 
1670
        self.start_server(transport_server)
 
1671
        self.assertSubset([transport_server.get_url()],
 
1672
            self._bzr_selftest_roots)
 
1673
 
1560
1674
    def test_assert_list_raises_on_generator(self):
1561
1675
        def generator_which_will_raise():
1562
1676
            # This will not raise until after the first yield
1660
1774
        self.assertEndsWith('foo', 'oo')
1661
1775
        self.assertRaises(AssertionError, self.assertEndsWith, 'o', 'oo')
1662
1776
 
 
1777
    def test_assertEqualDiff(self):
 
1778
        e = self.assertRaises(AssertionError,
 
1779
                              self.assertEqualDiff, '', '\n')
 
1780
        self.assertEquals(str(e),
 
1781
                          # Don't blink ! The '+' applies to the second string
 
1782
                          'first string is missing a final newline.\n+ \n')
 
1783
        e = self.assertRaises(AssertionError,
 
1784
                              self.assertEqualDiff, '\n', '')
 
1785
        self.assertEquals(str(e),
 
1786
                          # Don't blink ! The '-' applies to the second string
 
1787
                          'second string is missing a final newline.\n- \n')
 
1788
 
 
1789
 
 
1790
class TestDeprecations(tests.TestCase):
 
1791
 
1663
1792
    def test_applyDeprecated_not_deprecated(self):
1664
1793
        sample_object = ApplyDeprecatedHelper()
1665
1794
        # calling an undeprecated callable raises an assertion
1742
1871
        tree = self.make_branch_and_memory_tree('a')
1743
1872
        self.assertIsInstance(tree, bzrlib.memorytree.MemoryTree)
1744
1873
 
1745
 
 
1746
 
class TestSFTPMakeBranchAndTree(test_sftp_transport.TestCaseWithSFTPServer):
1747
 
 
1748
 
    def test_make_tree_for_sftp_branch(self):
1749
 
        """Transports backed by local directories create local trees."""
1750
 
        # NB: This is arguably a bug in the definition of make_branch_and_tree.
 
1874
    def test_make_tree_for_local_vfs_backed_transport(self):
 
1875
        # make_branch_and_tree has to use local branch and repositories
 
1876
        # when the vfs transport and local disk are colocated, even if
 
1877
        # a different transport is in use for url generation.
 
1878
        from bzrlib.transport.fakevfat import FakeVFATServer
 
1879
        self.transport_server = FakeVFATServer
 
1880
        self.assertFalse(self.get_url('t1').startswith('file://'))
1751
1881
        tree = self.make_branch_and_tree('t1')
1752
1882
        base = tree.bzrdir.root_transport.base
1753
 
        self.failIf(base.startswith('sftp'),
1754
 
                'base %r is on sftp but should be local' % base)
 
1883
        self.assertStartsWith(base, 'file://')
1755
1884
        self.assertEquals(tree.bzrdir.root_transport,
1756
1885
                tree.branch.bzrdir.root_transport)
1757
1886
        self.assertEquals(tree.bzrdir.root_transport,
1817
1946
        self.assertNotContainsRe("Test.b", output.getvalue())
1818
1947
        self.assertLength(2, output.readlines())
1819
1948
 
 
1949
    def test_lsprof_tests(self):
 
1950
        self.requireFeature(test_lsprof.LSProfFeature)
 
1951
        calls = []
 
1952
        class Test(object):
 
1953
            def __call__(test, result):
 
1954
                test.run(result)
 
1955
            def run(test, result):
 
1956
                self.assertIsInstance(result, tests.ForwardingResult)
 
1957
                calls.append("called")
 
1958
            def countTestCases(self):
 
1959
                return 1
 
1960
        self.run_selftest(test_suite_factory=Test, lsprof_tests=True)
 
1961
        self.assertLength(1, calls)
 
1962
 
1820
1963
    def test_random(self):
1821
1964
        # test randomising by listing a number of tests.
1822
1965
        output_123 = self.run_selftest(test_suite_factory=self.factory,
1877
2020
    def test_transport_sftp(self):
1878
2021
        try:
1879
2022
            import bzrlib.transport.sftp
1880
 
        except ParamikoNotPresent:
1881
 
            raise TestSkipped("Paramiko not present")
 
2023
        except errors.ParamikoNotPresent:
 
2024
            raise tests.TestSkipped("Paramiko not present")
1882
2025
        self.check_transport_set(bzrlib.transport.sftp.SFTPAbsoluteServer)
1883
2026
 
1884
2027
    def test_transport_memory(self):
1914
2057
 
1915
2058
        Attempts to run bzr from inside this class don't actually run it.
1916
2059
 
1917
 
        We test how run_bzr actually invokes bzr in another location.
1918
 
        Here we only need to test that it is run_bzr passes the right
1919
 
        parameters to run_bzr.
 
2060
        We test how run_bzr actually invokes bzr in another location.  Here we
 
2061
        only need to test that it passes the right parameters to run_bzr.
1920
2062
        """
1921
2063
        self.argv = list(argv)
1922
2064
        self.retcode = retcode
1923
2065
        self.encoding = encoding
1924
2066
        self.stdin = stdin
1925
2067
        self.working_dir = working_dir
1926
 
        return self.out, self.err
 
2068
        return self.retcode, self.out, self.err
1927
2069
 
1928
2070
    def test_run_bzr_error(self):
1929
2071
        self.out = "It sure does!\n"
1930
2072
        out, err = self.run_bzr_error(['^$'], ['rocks'], retcode=34)
1931
2073
        self.assertEqual(['rocks'], self.argv)
1932
2074
        self.assertEqual(34, self.retcode)
1933
 
        self.assertEqual(out, 'It sure does!\n')
 
2075
        self.assertEqual('It sure does!\n', out)
 
2076
        self.assertEquals(out, self.out)
 
2077
        self.assertEqual('', err)
 
2078
        self.assertEquals(err, self.err)
1934
2079
 
1935
2080
    def test_run_bzr_error_regexes(self):
1936
2081
        self.out = ''
1937
2082
        self.err = "bzr: ERROR: foobarbaz is not versioned"
1938
2083
        out, err = self.run_bzr_error(
1939
 
                ["bzr: ERROR: foobarbaz is not versioned"],
1940
 
                ['file-id', 'foobarbaz'])
 
2084
            ["bzr: ERROR: foobarbaz is not versioned"],
 
2085
            ['file-id', 'foobarbaz'])
1941
2086
 
1942
2087
    def test_encoding(self):
1943
2088
        """Test that run_bzr passes encoding to _run_bzr_core"""
2072
2217
        return self.out, self.err
2073
2218
 
2074
2219
 
2075
 
class TestRunBzrSubprocess(tests.TestCaseWithTransport):
 
2220
class TestWithFakedStartBzrSubprocess(tests.TestCaseWithTransport):
 
2221
    """Base class for tests testing how we might run bzr."""
2076
2222
 
2077
2223
    def setUp(self):
2078
2224
        tests.TestCaseWithTransport.setUp(self)
2089
2235
            'working_dir':working_dir, 'allow_plugins':allow_plugins})
2090
2236
        return self.next_subprocess
2091
2237
 
 
2238
 
 
2239
class TestRunBzrSubprocess(TestWithFakedStartBzrSubprocess):
 
2240
 
2092
2241
    def assertRunBzrSubprocess(self, expected_args, process, *args, **kwargs):
2093
2242
        """Run run_bzr_subprocess with args and kwargs using a stubbed process.
2094
2243
 
2157
2306
            StubProcess(), '', allow_plugins=True)
2158
2307
 
2159
2308
 
 
2309
class TestFinishBzrSubprocess(TestWithFakedStartBzrSubprocess):
 
2310
 
 
2311
    def test_finish_bzr_subprocess_with_error(self):
 
2312
        """finish_bzr_subprocess allows specification of the desired exit code.
 
2313
        """
 
2314
        process = StubProcess(err="unknown command", retcode=3)
 
2315
        result = self.finish_bzr_subprocess(process, retcode=3)
 
2316
        self.assertEqual('', result[0])
 
2317
        self.assertContainsRe(result[1], 'unknown command')
 
2318
 
 
2319
    def test_finish_bzr_subprocess_ignoring_retcode(self):
 
2320
        """finish_bzr_subprocess allows the exit code to be ignored."""
 
2321
        process = StubProcess(err="unknown command", retcode=3)
 
2322
        result = self.finish_bzr_subprocess(process, retcode=None)
 
2323
        self.assertEqual('', result[0])
 
2324
        self.assertContainsRe(result[1], 'unknown command')
 
2325
 
 
2326
    def test_finish_subprocess_with_unexpected_retcode(self):
 
2327
        """finish_bzr_subprocess raises self.failureException if the retcode is
 
2328
        not the expected one.
 
2329
        """
 
2330
        process = StubProcess(err="unknown command", retcode=3)
 
2331
        self.assertRaises(self.failureException, self.finish_bzr_subprocess,
 
2332
                          process)
 
2333
 
 
2334
 
2160
2335
class _DontSpawnProcess(Exception):
2161
2336
    """A simple exception which just allows us to skip unnecessary steps"""
2162
2337
 
2240
2415
        self.assertEqual(['foo', 'current'], chdirs)
2241
2416
 
2242
2417
 
2243
 
class TestBzrSubprocess(tests.TestCaseWithTransport):
2244
 
 
2245
 
    def test_start_and_stop_bzr_subprocess(self):
2246
 
        """We can start and perform other test actions while that process is
2247
 
        still alive.
2248
 
        """
2249
 
        process = self.start_bzr_subprocess(['--version'])
2250
 
        result = self.finish_bzr_subprocess(process)
2251
 
        self.assertContainsRe(result[0], 'is free software')
2252
 
        self.assertEqual('', result[1])
2253
 
 
2254
 
    def test_start_and_stop_bzr_subprocess_with_error(self):
2255
 
        """finish_bzr_subprocess allows specification of the desired exit code.
2256
 
        """
2257
 
        process = self.start_bzr_subprocess(['--versionn'])
2258
 
        result = self.finish_bzr_subprocess(process, retcode=3)
2259
 
        self.assertEqual('', result[0])
2260
 
        self.assertContainsRe(result[1], 'unknown command')
2261
 
 
2262
 
    def test_start_and_stop_bzr_subprocess_ignoring_retcode(self):
2263
 
        """finish_bzr_subprocess allows the exit code to be ignored."""
2264
 
        process = self.start_bzr_subprocess(['--versionn'])
2265
 
        result = self.finish_bzr_subprocess(process, retcode=None)
2266
 
        self.assertEqual('', result[0])
2267
 
        self.assertContainsRe(result[1], 'unknown command')
2268
 
 
2269
 
    def test_start_and_stop_bzr_subprocess_with_unexpected_retcode(self):
2270
 
        """finish_bzr_subprocess raises self.failureException if the retcode is
2271
 
        not the expected one.
2272
 
        """
2273
 
        process = self.start_bzr_subprocess(['--versionn'])
2274
 
        self.assertRaises(self.failureException, self.finish_bzr_subprocess,
2275
 
                          process)
 
2418
class TestActuallyStartBzrSubprocess(tests.TestCaseWithTransport):
 
2419
    """Tests that really need to do things with an external bzr."""
2276
2420
 
2277
2421
    def test_start_and_stop_bzr_subprocess_send_signal(self):
2278
2422
        """finish_bzr_subprocess raises self.failureException if the retcode is
2279
2423
        not the expected one.
2280
2424
        """
 
2425
        self.disable_missing_extensions_warning()
2281
2426
        process = self.start_bzr_subprocess(['wait-until-signalled'],
2282
2427
                                            skip_if_plan_to_signal=True)
2283
2428
        self.assertEqual('running\n', process.stdout.readline())
2286
2431
        self.assertEqual('', result[0])
2287
2432
        self.assertEqual('bzr: interrupted\n', result[1])
2288
2433
 
2289
 
    def test_start_and_stop_working_dir(self):
2290
 
        cwd = osutils.getcwd()
2291
 
        self.make_branch_and_tree('one')
2292
 
        process = self.start_bzr_subprocess(['root'], working_dir='one')
2293
 
        result = self.finish_bzr_subprocess(process, universal_newlines=True)
2294
 
        self.assertEndsWith(result[0], 'one\n')
2295
 
        self.assertEqual('', result[1])
2296
 
 
2297
2434
 
2298
2435
class TestKnownFailure(tests.TestCase):
2299
2436
 
2549
2686
        # Running bzr in blackbox mode, normal/expected/user errors should be
2550
2687
        # caught in the regular way and turned into an error message plus exit
2551
2688
        # code.
2552
 
        out, err = self.run_bzr(["log", "/nonexistantpath"], retcode=3)
 
2689
        transport_server = MemoryServer()
 
2690
        transport_server.setUp()
 
2691
        self.addCleanup(transport_server.tearDown)
 
2692
        url = transport_server.get_url()
 
2693
        self.permit_url(url)
 
2694
        out, err = self.run_bzr(["log", "%s/nonexistantpath" % url], retcode=3)
2553
2695
        self.assertEqual(out, '')
2554
2696
        self.assertContainsRe(err,
2555
2697
            'bzr: ERROR: Not a branch: ".*nonexistantpath/".\n')
2681
2823
 
2682
2824
class TestTestSuite(tests.TestCase):
2683
2825
 
 
2826
    def test__test_suite_testmod_names(self):
 
2827
        # Test that a plausible list of test module names are returned
 
2828
        # by _test_suite_testmod_names.
 
2829
        test_list = tests._test_suite_testmod_names()
 
2830
        self.assertSubset([
 
2831
            'bzrlib.tests.blackbox',
 
2832
            'bzrlib.tests.per_transport',
 
2833
            'bzrlib.tests.test_selftest',
 
2834
            ],
 
2835
            test_list)
 
2836
 
 
2837
    def test__test_suite_modules_to_doctest(self):
 
2838
        # Test that a plausible list of modules to doctest is returned
 
2839
        # by _test_suite_modules_to_doctest.
 
2840
        test_list = tests._test_suite_modules_to_doctest()
 
2841
        self.assertSubset([
 
2842
            'bzrlib.timestamp',
 
2843
            ],
 
2844
            test_list)
 
2845
 
2684
2846
    def test_test_suite(self):
2685
 
        # This test is slow - it loads the entire test suite to operate, so we
2686
 
        # do a single test with one test in each category
2687
 
        test_list = [
 
2847
        # test_suite() loads the entire test suite to operate. To avoid this
 
2848
        # overhead, and yet still be confident that things are happening,
 
2849
        # we temporarily replace two functions used by test_suite with 
 
2850
        # test doubles that supply a few sample tests to load, and check they
 
2851
        # are loaded.
 
2852
        calls = []
 
2853
        def _test_suite_testmod_names():
 
2854
            calls.append("testmod_names")
 
2855
            return [
 
2856
                'bzrlib.tests.blackbox.test_branch',
 
2857
                'bzrlib.tests.per_transport',
 
2858
                'bzrlib.tests.test_selftest',
 
2859
                ]
 
2860
        original_testmod_names = tests._test_suite_testmod_names
 
2861
        def _test_suite_modules_to_doctest():
 
2862
            calls.append("modules_to_doctest")
 
2863
            return ['bzrlib.timestamp']
 
2864
        orig_modules_to_doctest = tests._test_suite_modules_to_doctest
 
2865
        def restore_names():
 
2866
            tests._test_suite_testmod_names = original_testmod_names
 
2867
            tests._test_suite_modules_to_doctest = orig_modules_to_doctest
 
2868
        self.addCleanup(restore_names)
 
2869
        tests._test_suite_testmod_names = _test_suite_testmod_names
 
2870
        tests._test_suite_modules_to_doctest = _test_suite_modules_to_doctest
 
2871
        expected_test_list = [
2688
2872
            # testmod_names
2689
2873
            'bzrlib.tests.blackbox.test_branch.TestBranch.test_branch',
2690
2874
            ('bzrlib.tests.per_transport.TransportTests'
2691
 
             '.test_abspath(LocalURLServer)'),
 
2875
             '.test_abspath(LocalTransport,LocalURLServer)'),
2692
2876
            'bzrlib.tests.test_selftest.TestTestSuite.test_test_suite',
2693
2877
            # modules_to_doctest
2694
2878
            'bzrlib.timestamp.format_highres_date',
2695
2879
            # plugins can't be tested that way since selftest may be run with
2696
2880
            # --no-plugins
2697
2881
            ]
2698
 
        suite = tests.test_suite(test_list)
2699
 
        self.assertEquals(test_list, _test_ids(suite))
 
2882
        suite = tests.test_suite()
 
2883
        self.assertEqual(set(["testmod_names", "modules_to_doctest"]),
 
2884
            set(calls))
 
2885
        self.assertSubset(expected_test_list, _test_ids(suite))
2700
2886
 
2701
2887
    def test_test_suite_list_and_start(self):
2702
2888
        # We cannot test this at the same time as the main load, because we want
2703
 
        # to know that starting_with == None works. So a second full load is
2704
 
        # incurred.
 
2889
        # to know that starting_with == None works. So a second load is
 
2890
        # incurred - note that the starting_with parameter causes a partial load
 
2891
        # rather than a full load so this test should be pretty quick.
2705
2892
        test_list = ['bzrlib.tests.test_selftest.TestTestSuite.test_test_suite']
2706
2893
        suite = tests.test_suite(test_list,
2707
2894
                                 ['bzrlib.tests.test_selftest.TestTestSuite'])
2853
3040
                                                self.verbosity)
2854
3041
        tests.run_suite(suite, runner_class=MyRunner, stream=StringIO())
2855
3042
        self.assertLength(1, calls)
2856
 
 
2857
 
    def test_done(self):
2858
 
        """run_suite should call result.done()"""
2859
 
        self.calls = 0
2860
 
        def one_more_call(): self.calls += 1
2861
 
        def test_function():
2862
 
            pass
2863
 
        test = unittest.FunctionTestCase(test_function)
2864
 
        class InstrumentedTestResult(tests.ExtendedTestResult):
2865
 
            def done(self): one_more_call()
2866
 
        class MyRunner(tests.TextTestRunner):
2867
 
            def run(self, test):
2868
 
                return InstrumentedTestResult(self.stream, self.descriptions,
2869
 
                                              self.verbosity)
2870
 
        tests.run_suite(test, runner_class=MyRunner, stream=StringIO())
2871
 
        self.assertEquals(1, self.calls)