~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_dirstate.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2008-04-07 07:52:50 UTC
  • mfrom: (3340.1.1 208418-1.4)
  • Revision ID: pqm@pqm.ubuntu.com-20080407075250-phs53xnslo8boaeo
Return the correct knit serialisation method in _StreamAccess.
        (Andrew Bennetts, Martin Pool, Robert Collins)

Show diffs side-by-side

added added

removed removed

Lines of Context:
26
26
    osutils,
27
27
    )
28
28
from bzrlib.memorytree import MemoryTree
29
 
from bzrlib.osutils import has_symlinks
30
29
from bzrlib.tests import (
 
30
        SymlinkFeature,
31
31
        TestCase,
32
32
        TestCaseWithTransport,
33
 
        TestSkipped,
34
33
        )
35
34
 
36
35
 
189
188
              c
190
189
              d/
191
190
                e
 
191
            b-c
192
192
            f
193
193
        """
194
194
        tree = self.make_branch_and_tree('tree')
195
 
        paths = ['a', 'b/', 'b/c', 'b/d/', 'b/d/e', 'f']
196
 
        file_ids = ['a-id', 'b-id', 'c-id', 'd-id', 'e-id', 'f-id']
 
195
        paths = ['a', 'b/', 'b/c', 'b/d/', 'b/d/e', 'b-c', 'f']
 
196
        file_ids = ['a-id', 'b-id', 'c-id', 'd-id', 'e-id', 'b-c-id', 'f-id']
197
197
        self.build_tree(['tree/' + p for p in paths])
198
198
        tree.set_root_id('TREE_ROOT')
199
199
        tree.add([p.rstrip('/') for p in paths], file_ids)
200
200
        tree.commit('initial', rev_id='rev-1')
201
201
        revision_id = 'rev-1'
202
202
        # a_packed_stat = dirstate.pack_stat(os.stat('tree/a'))
203
 
        t = self.get_transport().clone('tree')
 
203
        t = self.get_transport('tree')
204
204
        a_text = t.get_bytes('a')
205
205
        a_sha = osutils.sha_string(a_text)
206
206
        a_len = len(a_text)
214
214
        e_text = t.get_bytes('b/d/e')
215
215
        e_sha = osutils.sha_string(e_text)
216
216
        e_len = len(e_text)
 
217
        b_c_text = t.get_bytes('b-c')
 
218
        b_c_sha = osutils.sha_string(b_c_text)
 
219
        b_c_len = len(b_c_text)
217
220
        # f_packed_stat = dirstate.pack_stat(os.stat('tree/f'))
218
221
        f_text = t.get_bytes('f')
219
222
        f_sha = osutils.sha_string(f_text)
244
247
                      ('f', '', 0, False, null_stat),
245
248
                      ('f', e_sha, e_len, False, revision_id),
246
249
                     ]),
 
250
            'b-c':(('', 'b-c', 'b-c-id'), [
 
251
                      ('f', '', 0, False, null_stat),
 
252
                      ('f', b_c_sha, b_c_len, False, revision_id),
 
253
                     ]),
247
254
            'f':(('', 'f', 'f-id'), [
248
255
                  ('f', '', 0, False, null_stat),
249
256
                  ('f', f_sha, f_len, False, revision_id),
276
283
        tree, state, expected = self.create_basic_dirstate()
277
284
        # Now we will just remove and add every file so we get an extra entry
278
285
        # per entry. Unversion in reverse order so we handle subdirs
279
 
        tree.unversion(['f-id', 'e-id', 'd-id', 'c-id', 'b-id', 'a-id'])
280
 
        tree.add(['a', 'b', 'b/c', 'b/d', 'b/d/e', 'f'],
281
 
                 ['a-id2', 'b-id2', 'c-id2', 'd-id2', 'e-id2', 'f-id2'])
 
286
        tree.unversion(['f-id', 'b-c-id', 'e-id', 'd-id', 'c-id', 'b-id', 'a-id'])
 
287
        tree.add(['a', 'b', 'b/c', 'b/d', 'b/d/e', 'b-c', 'f'],
 
288
                 ['a-id2', 'b-id2', 'c-id2', 'd-id2', 'e-id2', 'b-c-id2', 'f-id2'])
282
289
 
283
290
        # Update the expected dictionary.
284
 
        for path in ['a', 'b', 'b/c', 'b/d', 'b/d/e', 'f']:
 
291
        for path in ['a', 'b', 'b/c', 'b/d', 'b/d/e', 'b-c', 'f']:
285
292
            orig = expected[path]
286
293
            path2 = path + '2'
287
294
            # This record was deleted in the current tree
344
351
            state.lock_read()
345
352
        return tree, state, expected
346
353
 
 
354
 
347
355
class TestTreeToDirState(TestCaseWithDirState):
348
356
 
349
357
    def test_empty_to_dirstate(self):
351
359
        # There are no files on disk and no parents
352
360
        tree = self.make_branch_and_tree('tree')
353
361
        expected_result = ([], [
354
 
            (('', '', tree.path2id('')), # common details
 
362
            (('', '', tree.get_root_id()), # common details
355
363
             [('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
356
364
             ])])
357
365
        state = dirstate.DirState.from_tree(tree, 'dirstate')
364
372
        rev_id = tree.commit('first post').encode('utf8')
365
373
        root_stat_pack = dirstate.pack_stat(os.stat(tree.basedir))
366
374
        expected_result = ([rev_id], [
367
 
            (('', '', tree.path2id('')), # common details
 
375
            (('', '', tree.get_root_id()), # common details
368
376
             [('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
369
377
              ('d', '', 0, False, rev_id), # first parent details
370
378
             ])])
384
392
        rev_id2 = tree2.commit('second post', allow_pointless=True)
385
393
        tree.merge_from_branch(tree2.branch)
386
394
        expected_result = ([rev_id, rev_id2], [
387
 
            (('', '', tree.path2id('')), # common details
 
395
            (('', '', tree.get_root_id()), # common details
388
396
             [('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
389
397
              ('d', '', 0, False, rev_id), # first parent details
390
398
              ('d', '', 0, False, rev_id2), # second parent details
403
411
        tree = self.make_branch_and_tree('tree')
404
412
        self.build_tree(['tree/unknown'])
405
413
        expected_result = ([], [
406
 
            (('', '', tree.path2id('')), # common details
 
414
            (('', '', tree.get_root_id()), # common details
407
415
             [('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
408
416
             ])])
409
417
        state = dirstate.DirState.from_tree(tree, 'dirstate')
412
420
    def get_tree_with_a_file(self):
413
421
        tree = self.make_branch_and_tree('tree')
414
422
        self.build_tree(['tree/a file'])
415
 
        tree.add('a file', 'a file id')
 
423
        tree.add('a file', 'a-file-id')
416
424
        return tree
417
425
 
418
426
    def test_non_empty_no_parents_to_dirstate(self):
420
428
        # There are files on disk and no parents
421
429
        tree = self.get_tree_with_a_file()
422
430
        expected_result = ([], [
423
 
            (('', '', tree.path2id('')), # common details
 
431
            (('', '', tree.get_root_id()), # common details
424
432
             [('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
425
433
             ]),
426
 
            (('', 'a file', 'a file id'), # common
 
434
            (('', 'a file', 'a-file-id'), # common
427
435
             [('f', '', 0, False, dirstate.DirState.NULLSTAT), # current
428
436
             ]),
429
437
            ])
438
446
        # and length:
439
447
        self.build_tree_contents([('tree/a file', 'new content\n')])
440
448
        expected_result = ([rev_id], [
441
 
            (('', '', tree.path2id('')), # common details
 
449
            (('', '', tree.get_root_id()), # common details
442
450
             [('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
443
451
              ('d', '', 0, False, rev_id), # first parent details
444
452
             ]),
445
 
            (('', 'a file', 'a file id'), # common
 
453
            (('', 'a file', 'a-file-id'), # common
446
454
             [('f', '', 0, False, dirstate.DirState.NULLSTAT), # current
447
455
              ('f', 'c3ed76e4bfd45ff1763ca206055bca8e9fc28aa8', 24, False,
448
456
               rev_id), # first parent
465
473
        # and length again, giving us three distinct values:
466
474
        self.build_tree_contents([('tree/a file', 'new content\n')])
467
475
        expected_result = ([rev_id, rev_id2], [
468
 
            (('', '', tree.path2id('')), # common details
 
476
            (('', '', tree.get_root_id()), # common details
469
477
             [('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
470
478
              ('d', '', 0, False, rev_id), # first parent details
471
479
              ('d', '', 0, False, rev_id2), # second parent details
472
480
             ]),
473
 
            (('', 'a file', 'a file id'), # common
 
481
            (('', 'a file', 'a-file-id'), # common
474
482
             [('f', '', 0, False, dirstate.DirState.NULLSTAT), # current
475
483
              ('f', 'c3ed76e4bfd45ff1763ca206055bca8e9fc28aa8', 24, False,
476
484
               rev_id), # first parent
517
525
        # get a state object
518
526
        # no parents, default tree content
519
527
        expected_result = ([], [
520
 
            (('', '', tree.path2id('')), # common details
 
528
            (('', '', tree.get_root_id()), # common details
521
529
             # current tree details, but new from_tree skips statting, it
522
530
             # uses set_state_from_inventory, and thus depends on the
523
531
             # inventory state.
558
566
            self.assertEqual('', entry[1][0][1])
559
567
            # We should have a real entry.
560
568
            self.assertNotEqual((None, None), entry)
 
569
            # Make sure everything is old enough
 
570
            state._sha_cutoff_time()
 
571
            state._cutoff_time += 10
561
572
            sha1sum = state.update_entry(entry, 'a-file', os.lstat('a-file'))
562
573
            # We should have gotten a real sha1
563
574
            self.assertEqual('ecc5374e9ed82ad3ea3b4d452ea995a5fd3e70e3',
635
646
        finally:
636
647
            state.unlock()
637
648
 
 
649
    def test_save_refuses_if_changes_aborted(self):
 
650
        self.build_tree(['a-file', 'a-dir/'])
 
651
        state = dirstate.DirState.initialize('dirstate')
 
652
        try:
 
653
            # No stat and no sha1 sum.
 
654
            state.add('a-file', 'a-file-id', 'file', None, '')
 
655
            state.save()
 
656
        finally:
 
657
            state.unlock()
 
658
 
 
659
        # The dirstate should include TREE_ROOT and 'a-file' and nothing else
 
660
        expected_blocks = [
 
661
            ('', [(('', '', 'TREE_ROOT'),
 
662
                   [('d', '', 0, False, dirstate.DirState.NULLSTAT)])]),
 
663
            ('', [(('', 'a-file', 'a-file-id'),
 
664
                   [('f', '', 0, False, dirstate.DirState.NULLSTAT)])]),
 
665
        ]
 
666
 
 
667
        state = dirstate.DirState.on_file('dirstate')
 
668
        state.lock_write()
 
669
        try:
 
670
            state._read_dirblocks_if_needed()
 
671
            self.assertEqual(expected_blocks, state._dirblocks)
 
672
 
 
673
            # Now modify the state, but mark it as inconsistent
 
674
            state.add('a-dir', 'a-dir-id', 'directory', None, '')
 
675
            state._changes_aborted = True
 
676
            state.save()
 
677
        finally:
 
678
            state.unlock()
 
679
 
 
680
        state = dirstate.DirState.on_file('dirstate')
 
681
        state.lock_read()
 
682
        try:
 
683
            state._read_dirblocks_if_needed()
 
684
            self.assertEqual(expected_blocks, state._dirblocks)
 
685
        finally:
 
686
            state.unlock()
 
687
 
638
688
 
639
689
class TestDirStateInitialize(TestCaseWithDirState):
640
690
 
667
717
        try:
668
718
            tree1.add('')
669
719
            revid1 = tree1.commit('foo').encode('utf8')
670
 
            root_id = tree1.inventory.root.file_id
 
720
            root_id = tree1.get_root_id()
671
721
            inv = tree1.inventory
672
722
        finally:
673
723
            tree1.unlock()
688
738
            # This will unlock it
689
739
            self.check_state_with_reopen(expected_result, state)
690
740
 
 
741
    def test_set_state_from_inventory_preserves_hashcache(self):
 
742
        # https://bugs.launchpad.net/bzr/+bug/146176
 
743
        # set_state_from_inventory should preserve the stat and hash value for
 
744
        # workingtree files that are not changed by the inventory.
 
745
       
 
746
        tree = self.make_branch_and_tree('.')
 
747
        # depends on the default format using dirstate...
 
748
        tree.lock_write()
 
749
        try:
 
750
            # make a dirstate with some valid hashcache data 
 
751
            # file on disk, but that's not needed for this test
 
752
            foo_contents = 'contents of foo'
 
753
            self.build_tree_contents([('foo', foo_contents)])
 
754
            tree.add('foo', 'foo-id')
 
755
 
 
756
            foo_stat = os.stat('foo')
 
757
            foo_packed = dirstate.pack_stat(foo_stat)
 
758
            foo_sha = osutils.sha_string(foo_contents)
 
759
            foo_size = len(foo_contents)
 
760
 
 
761
            # should not be cached yet, because the file's too fresh
 
762
            self.assertEqual(
 
763
                (('', 'foo', 'foo-id',),
 
764
                 [('f', '', 0, False, dirstate.DirState.NULLSTAT)]),
 
765
                tree._dirstate._get_entry(0, 'foo-id'))
 
766
            # poke in some hashcache information - it wouldn't normally be
 
767
            # stored because it's too fresh
 
768
            tree._dirstate.update_minimal(
 
769
                ('', 'foo', 'foo-id'),
 
770
                'f', False, foo_sha, foo_packed, foo_size, 'foo')
 
771
            # now should be cached
 
772
            self.assertEqual(
 
773
                (('', 'foo', 'foo-id',),
 
774
                 [('f', foo_sha, foo_size, False, foo_packed)]),
 
775
                tree._dirstate._get_entry(0, 'foo-id'))
 
776
           
 
777
            # extract the inventory, and add something to it
 
778
            inv = tree._get_inventory()
 
779
            # should see the file we poked in...
 
780
            self.assertTrue(inv.has_id('foo-id'))
 
781
            self.assertTrue(inv.has_filename('foo'))
 
782
            inv.add_path('bar', 'file', 'bar-id')
 
783
            tree._dirstate._validate()
 
784
            # this used to cause it to lose its hashcache
 
785
            tree._dirstate.set_state_from_inventory(inv)
 
786
            tree._dirstate._validate()
 
787
        finally:
 
788
            tree.unlock()
 
789
 
 
790
        tree.lock_read()
 
791
        try:
 
792
            # now check that the state still has the original hashcache value
 
793
            state = tree._dirstate
 
794
            state._validate()
 
795
            foo_tuple = state._get_entry(0, path_utf8='foo')
 
796
            self.assertEqual(
 
797
                (('', 'foo', 'foo-id',),
 
798
                 [('f', foo_sha, len(foo_contents), False,
 
799
                   dirstate.pack_stat(foo_stat))]),
 
800
                foo_tuple)
 
801
        finally:
 
802
            tree.unlock()
 
803
 
 
804
 
 
805
    def test_set_state_from_inventory_mixed_paths(self):
 
806
        tree1 = self.make_branch_and_tree('tree1')
 
807
        self.build_tree(['tree1/a/', 'tree1/a/b/', 'tree1/a-b/',
 
808
                         'tree1/a/b/foo', 'tree1/a-b/bar'])
 
809
        tree1.lock_write()
 
810
        try:
 
811
            tree1.add(['a', 'a/b', 'a-b', 'a/b/foo', 'a-b/bar'],
 
812
                      ['a-id', 'b-id', 'a-b-id', 'foo-id', 'bar-id'])
 
813
            tree1.commit('rev1', rev_id='rev1')
 
814
            root_id = tree1.get_root_id()
 
815
            inv = tree1.inventory
 
816
        finally:
 
817
            tree1.unlock()
 
818
        expected_result1 = [('', '', root_id, 'd'),
 
819
                            ('', 'a', 'a-id', 'd'),
 
820
                            ('', 'a-b', 'a-b-id', 'd'),
 
821
                            ('a', 'b', 'b-id', 'd'),
 
822
                            ('a/b', 'foo', 'foo-id', 'f'),
 
823
                            ('a-b', 'bar', 'bar-id', 'f'),
 
824
                           ]
 
825
        expected_result2 = [('', '', root_id, 'd'),
 
826
                            ('', 'a', 'a-id', 'd'),
 
827
                            ('', 'a-b', 'a-b-id', 'd'),
 
828
                            ('a-b', 'bar', 'bar-id', 'f'),
 
829
                           ]
 
830
        state = dirstate.DirState.initialize('dirstate')
 
831
        try:
 
832
            state.set_state_from_inventory(inv)
 
833
            values = []
 
834
            for entry in state._iter_entries():
 
835
                values.append(entry[0] + entry[1][0][:1])
 
836
            self.assertEqual(expected_result1, values)
 
837
            del inv['b-id']
 
838
            state.set_state_from_inventory(inv)
 
839
            values = []
 
840
            for entry in state._iter_entries():
 
841
                values.append(entry[0] + entry[1][0][:1])
 
842
            self.assertEqual(expected_result2, values)
 
843
        finally:
 
844
            state.unlock()
 
845
 
691
846
    def test_set_path_id_no_parents(self):
692
847
        """The id of a path can be changed trivally with no parents."""
693
848
        state = dirstate.DirState.initialize('dirstate')
776
931
        tree2.lock_write()
777
932
        try:
778
933
            revid2 = tree2.commit('foo')
779
 
            root_id = tree2.inventory.root.file_id
 
934
            root_id = tree2.get_root_id()
780
935
        finally:
781
936
            tree2.unlock()
782
937
        state = dirstate.DirState.initialize('dirstate')
846
1001
        try:
847
1002
            tree2.put_file_bytes_non_atomic('file-id', 'new file-content')
848
1003
            revid2 = tree2.commit('foo')
849
 
            root_id = tree2.inventory.root.file_id
 
1004
            root_id = tree2.get_root_id()
850
1005
        finally:
851
1006
            tree2.unlock()
852
1007
        # check the layout in memory
892
1047
            (('', '', 'TREE_ROOT'), [
893
1048
             ('d', '', 0, False, dirstate.DirState.NULLSTAT), # current tree
894
1049
             ]),
895
 
            (('', 'a file', 'a file id'), [
 
1050
            (('', 'a file', 'a-file-id'), [
896
1051
             ('f', '1'*20, 19, False, dirstate.pack_stat(stat)), # current tree
897
1052
             ]),
898
1053
            ]
899
1054
        try:
900
 
            state.add('a file', 'a file id', 'file', stat, '1'*20)
 
1055
            state.add('a file', 'a-file-id', 'file', stat, '1'*20)
901
1056
            # having added it, it should be in the output of iter_entries.
902
1057
            self.assertEqual(expected_entries, list(state._iter_entries()))
903
1058
            # saving and reloading should not affect this.
922
1077
        state = dirstate.DirState.initialize('dirstate')
923
1078
        try:
924
1079
            self.assertRaises(errors.NotVersionedError, state.add,
925
 
                'unversioned/a file', 'a file id', 'file', None, None)
 
1080
                'unversioned/a file', 'a-file-id', 'file', None, None)
926
1081
        finally:
927
1082
            state.unlock()
928
1083
 
960
1115
        # The most trivial addition of a symlink when there are no parents and
961
1116
        # its in the root and all data about the file is supplied
962
1117
        # bzr doesn't support fake symlinks on windows, yet.
963
 
        if not has_symlinks():
964
 
            raise TestSkipped("No symlink support")
 
1118
        self.requireFeature(SymlinkFeature)
965
1119
        os.symlink('target', 'a link')
966
1120
        stat = os.lstat('a link')
967
1121
        expected_entries = [
1000
1154
            (('', 'a dir', 'a dir id'), [
1001
1155
             ('d', '', 0, False, dirstate.pack_stat(dirstat)), # current tree
1002
1156
             ]),
1003
 
            (('a dir', 'a file', 'a file id'), [
 
1157
            (('a dir', 'a file', 'a-file-id'), [
1004
1158
             ('f', '1'*20, 25, False,
1005
1159
              dirstate.pack_stat(filestat)), # current tree details
1006
1160
             ]),
1008
1162
        state = dirstate.DirState.initialize('dirstate')
1009
1163
        try:
1010
1164
            state.add('a dir', 'a dir id', 'directory', dirstat, None)
1011
 
            state.add('a dir/a file', 'a file id', 'file', filestat, '1'*20)
 
1165
            state.add('a dir/a file', 'a-file-id', 'file', filestat, '1'*20)
1012
1166
            # added it, it should be in the output of iter_entries.
1013
1167
            self.assertEqual(expected_entries, list(state._iter_entries()))
1014
1168
            # saving and reloading should not affect this.
1277
1431
            state.unlock()
1278
1432
 
1279
1433
 
 
1434
class TestIterChildEntries(TestCaseWithDirState):
 
1435
 
 
1436
    def create_dirstate_with_two_trees(self):
 
1437
        """This dirstate contains multiple files and directories.
 
1438
 
 
1439
         /        a-root-value
 
1440
         a/       a-dir
 
1441
         b/       b-dir
 
1442
         c        c-file
 
1443
         d        d-file
 
1444
         a/e/     e-dir
 
1445
         a/f      f-file
 
1446
         b/g      g-file
 
1447
         b/h\xc3\xa5  h-\xc3\xa5-file  #This is u'\xe5' encoded into utf-8
 
1448
 
 
1449
        Notice that a/e is an empty directory.
 
1450
 
 
1451
        There is one parent tree, which has the same shape with the following variations:
 
1452
        b/g in the parent is gone.
 
1453
        b/h in the parent has a different id
 
1454
        b/i is new in the parent 
 
1455
        c is renamed to b/j in the parent
 
1456
 
 
1457
        :return: The dirstate, still write-locked.
 
1458
        """
 
1459
        packed_stat = 'AAAAREUHaIpFB2iKAAADAQAtkqUAAIGk'
 
1460
        null_sha = 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'
 
1461
        NULL_PARENT_DETAILS = dirstate.DirState.NULL_PARENT_DETAILS
 
1462
        root_entry = ('', '', 'a-root-value'), [
 
1463
            ('d', '', 0, False, packed_stat),
 
1464
            ('d', '', 0, False, 'parent-revid'),
 
1465
            ]
 
1466
        a_entry = ('', 'a', 'a-dir'), [
 
1467
            ('d', '', 0, False, packed_stat),
 
1468
            ('d', '', 0, False, 'parent-revid'),
 
1469
            ]
 
1470
        b_entry = ('', 'b', 'b-dir'), [
 
1471
            ('d', '', 0, False, packed_stat),
 
1472
            ('d', '', 0, False, 'parent-revid'),
 
1473
            ]
 
1474
        c_entry = ('', 'c', 'c-file'), [
 
1475
            ('f', null_sha, 10, False, packed_stat),
 
1476
            ('r', 'b/j', 0, False, ''),
 
1477
            ]
 
1478
        d_entry = ('', 'd', 'd-file'), [
 
1479
            ('f', null_sha, 20, False, packed_stat),
 
1480
            ('f', 'd', 20, False, 'parent-revid'),
 
1481
            ]
 
1482
        e_entry = ('a', 'e', 'e-dir'), [
 
1483
            ('d', '', 0, False, packed_stat),
 
1484
            ('d', '', 0, False, 'parent-revid'),
 
1485
            ]
 
1486
        f_entry = ('a', 'f', 'f-file'), [
 
1487
            ('f', null_sha, 30, False, packed_stat),
 
1488
            ('f', 'f', 20, False, 'parent-revid'),
 
1489
            ]
 
1490
        g_entry = ('b', 'g', 'g-file'), [
 
1491
            ('f', null_sha, 30, False, packed_stat),
 
1492
            NULL_PARENT_DETAILS,
 
1493
            ]
 
1494
        h_entry1 = ('b', 'h\xc3\xa5', 'h-\xc3\xa5-file1'), [
 
1495
            ('f', null_sha, 40, False, packed_stat),
 
1496
            NULL_PARENT_DETAILS,
 
1497
            ]
 
1498
        h_entry2 = ('b', 'h\xc3\xa5', 'h-\xc3\xa5-file2'), [
 
1499
            NULL_PARENT_DETAILS,
 
1500
            ('f', 'h', 20, False, 'parent-revid'),
 
1501
            ]
 
1502
        i_entry = ('b', 'i', 'i-file'), [
 
1503
            NULL_PARENT_DETAILS,
 
1504
            ('f', 'h', 20, False, 'parent-revid'),
 
1505
            ]
 
1506
        j_entry = ('b', 'j', 'c-file'), [
 
1507
            ('r', 'c', 0, False, ''),
 
1508
            ('f', 'j', 20, False, 'parent-revid'),
 
1509
            ]
 
1510
        dirblocks = []
 
1511
        dirblocks.append(('', [root_entry]))
 
1512
        dirblocks.append(('', [a_entry, b_entry, c_entry, d_entry]))
 
1513
        dirblocks.append(('a', [e_entry, f_entry]))
 
1514
        dirblocks.append(('b', [g_entry, h_entry1, h_entry2, i_entry, j_entry]))
 
1515
        state = dirstate.DirState.initialize('dirstate')
 
1516
        state._validate()
 
1517
        try:
 
1518
            state._set_data(['parent'], dirblocks)
 
1519
        except:
 
1520
            state.unlock()
 
1521
            raise
 
1522
        return state, dirblocks
 
1523
 
 
1524
    def test_iter_children_b(self):
 
1525
        state, dirblocks = self.create_dirstate_with_two_trees()
 
1526
        self.addCleanup(state.unlock)
 
1527
        expected_result = []
 
1528
        expected_result.append(dirblocks[3][1][2]) # h2
 
1529
        expected_result.append(dirblocks[3][1][3]) # i
 
1530
        expected_result.append(dirblocks[3][1][4]) # j
 
1531
        self.assertEqual(expected_result,
 
1532
            list(state._iter_child_entries(1, 'b')))
 
1533
 
 
1534
    def test_iter_child_root(self):
 
1535
        state, dirblocks = self.create_dirstate_with_two_trees()
 
1536
        self.addCleanup(state.unlock)
 
1537
        expected_result = []
 
1538
        expected_result.append(dirblocks[1][1][0]) # a
 
1539
        expected_result.append(dirblocks[1][1][1]) # b
 
1540
        expected_result.append(dirblocks[1][1][3]) # d
 
1541
        expected_result.append(dirblocks[2][1][0]) # e
 
1542
        expected_result.append(dirblocks[2][1][1]) # f
 
1543
        expected_result.append(dirblocks[3][1][2]) # h2
 
1544
        expected_result.append(dirblocks[3][1][3]) # i
 
1545
        expected_result.append(dirblocks[3][1][4]) # j
 
1546
        self.assertEqual(expected_result,
 
1547
            list(state._iter_child_entries(1, '')))
 
1548
 
 
1549
 
1280
1550
class TestDirstateSortOrder(TestCaseWithTransport):
1281
1551
    """Test that DirState adds entries in the right order."""
1282
1552
 
1346
1616
        super(InstrumentedDirState, self).__init__(path)
1347
1617
        self._time_offset = 0
1348
1618
        self._log = []
 
1619
        # member is dynamically set in DirState.__init__ to turn on trace
 
1620
        self._sha1_file = self._sha1_file_and_log
1349
1621
 
1350
1622
    def _sha_cutoff_time(self):
1351
1623
        timestamp = super(InstrumentedDirState, self)._sha_cutoff_time()
1352
1624
        self._cutoff_time = timestamp + self._time_offset
1353
1625
 
1354
 
    def _sha1_file(self, abspath, entry):
 
1626
    def _sha1_file_and_log(self, abspath):
1355
1627
        self._log.append(('sha1', abspath))
1356
 
        return super(InstrumentedDirState, self)._sha1_file(abspath, entry)
 
1628
        return osutils.sha_file_by_name(abspath)
1357
1629
 
1358
1630
    def _read_link(self, abspath, old_link):
1359
1631
        self._log.append(('read_link', abspath, old_link))
1421
1693
        self.assertEqual('b50e5406bb5e153ebbeb20268fcf37c87e1ecfb6',
1422
1694
                         link_or_sha1)
1423
1695
 
1424
 
        # The dirblock entry should be updated with the new info
1425
 
        self.assertEqual([('f', link_or_sha1, 14, False, packed_stat)],
 
1696
        # The dirblock entry should not cache the file's sha1
 
1697
        self.assertEqual([('f', '', 14, False, dirstate.DirState.NULLSTAT)],
1426
1698
                         entry[1])
1427
1699
        self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
1428
1700
                         state._dirblock_state)
1447
1719
                         link_or_sha1)
1448
1720
        self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
1449
1721
                         state._dirblock_state)
 
1722
        self.assertEqual([('f', '', 14, False, dirstate.DirState.NULLSTAT)],
 
1723
                         entry[1])
1450
1724
        state.save()
1451
1725
 
1452
1726
        # However, if we move the clock forward so the file is considered
1453
 
        # "stable", it should just returned the cached value.
1454
 
        state.adjust_time(20)
 
1727
        # "stable", it should just cache the value.
 
1728
        state.adjust_time(+20)
1455
1729
        link_or_sha1 = state.update_entry(entry, abspath='a',
1456
1730
                                          stat_value=stat_value)
1457
1731
        self.assertEqual('b50e5406bb5e153ebbeb20268fcf37c87e1ecfb6',
1458
1732
                         link_or_sha1)
1459
1733
        self.assertEqual([('sha1', 'a'), ('is_exec', mode, False),
1460
1734
                          ('sha1', 'a'), ('is_exec', mode, False),
 
1735
                          ('sha1', 'a'), ('is_exec', mode, False),
1461
1736
                         ], state._log)
 
1737
        self.assertEqual([('f', link_or_sha1, 14, False, packed_stat)],
 
1738
                         entry[1])
1462
1739
 
1463
 
    def test_update_entry_no_stat_value(self):
1464
 
        """Passing the stat_value is optional."""
1465
 
        state, entry = self.get_state_with_a()
1466
 
        state.adjust_time(-10) # Make sure the file looks new
1467
 
        self.build_tree(['a'])
1468
 
        # Add one where we don't provide the stat or sha already
1469
 
        link_or_sha1 = state.update_entry(entry, abspath='a')
 
1740
        # Subsequent calls will just return the cached value
 
1741
        link_or_sha1 = state.update_entry(entry, abspath='a',
 
1742
                                          stat_value=stat_value)
1470
1743
        self.assertEqual('b50e5406bb5e153ebbeb20268fcf37c87e1ecfb6',
1471
1744
                         link_or_sha1)
1472
 
        stat_value = os.lstat('a')
1473
 
        self.assertEqual([('lstat', 'a'), ('sha1', 'a'),
1474
 
                          ('is_exec', stat_value.st_mode, False),
 
1745
        self.assertEqual([('sha1', 'a'), ('is_exec', mode, False),
 
1746
                          ('sha1', 'a'), ('is_exec', mode, False),
 
1747
                          ('sha1', 'a'), ('is_exec', mode, False),
1475
1748
                         ], state._log)
 
1749
        self.assertEqual([('f', link_or_sha1, 14, False, packed_stat)],
 
1750
                         entry[1])
1476
1751
 
1477
1752
    def test_update_entry_symlink(self):
1478
1753
        """Update entry should read symlinks."""
1479
 
        if not osutils.has_symlinks():
1480
 
            # PlatformDeficiency / TestSkipped
1481
 
            raise TestSkipped("No symlink support")
 
1754
        self.requireFeature(SymlinkFeature)
1482
1755
        state, entry = self.get_state_with_a()
1483
1756
        state.save()
1484
1757
        self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
1492
1765
                                          stat_value=stat_value)
1493
1766
        self.assertEqual('target', link_or_sha1)
1494
1767
        self.assertEqual([('read_link', 'a', '')], state._log)
1495
 
        # Dirblock is updated
1496
 
        self.assertEqual([('l', link_or_sha1, 6, False, packed_stat)],
 
1768
        # Dirblock is not updated (the link is too new)
 
1769
        self.assertEqual([('l', '', 6, False, dirstate.DirState.NULLSTAT)],
1497
1770
                         entry[1])
1498
1771
        self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
1499
1772
                         state._dirblock_state)
1503
1776
                                          stat_value=stat_value)
1504
1777
        self.assertEqual('target', link_or_sha1)
1505
1778
        self.assertEqual([('read_link', 'a', ''),
1506
 
                          ('read_link', 'a', 'target'),
 
1779
                          ('read_link', 'a', ''),
1507
1780
                         ], state._log)
 
1781
        self.assertEqual([('l', '', 6, False, dirstate.DirState.NULLSTAT)],
 
1782
                         entry[1])
1508
1783
        state.adjust_time(+20) # Skip into the future, all files look old
1509
1784
        link_or_sha1 = state.update_entry(entry, abspath='a',
1510
1785
                                          stat_value=stat_value)
1511
1786
        self.assertEqual('target', link_or_sha1)
1512
 
        # There should not be a new read_link call.
1513
 
        # (this is a weak assertion, because read_link is fairly inexpensive,
1514
 
        # versus the number of symlinks that we would have)
1515
 
        self.assertEqual([('read_link', 'a', ''),
1516
 
                          ('read_link', 'a', 'target'),
1517
 
                         ], state._log)
 
1787
        # We need to re-read the link because only now can we cache it
 
1788
        self.assertEqual([('read_link', 'a', ''),
 
1789
                          ('read_link', 'a', ''),
 
1790
                          ('read_link', 'a', ''),
 
1791
                         ], state._log)
 
1792
        self.assertEqual([('l', 'target', 6, False, packed_stat)],
 
1793
                         entry[1])
 
1794
 
 
1795
        # Another call won't re-read the link
 
1796
        self.assertEqual([('read_link', 'a', ''),
 
1797
                          ('read_link', 'a', ''),
 
1798
                          ('read_link', 'a', ''),
 
1799
                         ], state._log)
 
1800
        link_or_sha1 = state.update_entry(entry, abspath='a',
 
1801
                                          stat_value=stat_value)
 
1802
        self.assertEqual('target', link_or_sha1)
 
1803
        self.assertEqual([('l', 'target', 6, False, packed_stat)],
 
1804
                         entry[1])
 
1805
 
 
1806
    def do_update_entry(self, state, entry, abspath):
 
1807
        stat_value = os.lstat(abspath)
 
1808
        return state.update_entry(entry, abspath, stat_value)
1518
1809
 
1519
1810
    def test_update_entry_dir(self):
1520
1811
        state, entry = self.get_state_with_a()
1521
1812
        self.build_tree(['a/'])
1522
 
        self.assertIs(None, state.update_entry(entry, 'a'))
 
1813
        self.assertIs(None, self.do_update_entry(state, entry, 'a'))
 
1814
 
 
1815
    def test_update_entry_dir_unchanged(self):
 
1816
        state, entry = self.get_state_with_a()
 
1817
        self.build_tree(['a/'])
 
1818
        state.adjust_time(+20)
 
1819
        self.assertIs(None, self.do_update_entry(state, entry, 'a'))
 
1820
        self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
 
1821
                         state._dirblock_state)
 
1822
        state.save()
 
1823
        self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
 
1824
                         state._dirblock_state)
 
1825
        self.assertIs(None, self.do_update_entry(state, entry, 'a'))
 
1826
        self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
 
1827
                         state._dirblock_state)
 
1828
 
 
1829
    def test_update_entry_file_unchanged(self):
 
1830
        state, entry = self.get_state_with_a()
 
1831
        self.build_tree(['a'])
 
1832
        sha1sum = 'b50e5406bb5e153ebbeb20268fcf37c87e1ecfb6'
 
1833
        state.adjust_time(+20)
 
1834
        self.assertEqual(sha1sum, self.do_update_entry(state, entry, 'a'))
 
1835
        self.assertEqual(dirstate.DirState.IN_MEMORY_MODIFIED,
 
1836
                         state._dirblock_state)
 
1837
        state.save()
 
1838
        self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
 
1839
                         state._dirblock_state)
 
1840
        self.assertEqual(sha1sum, self.do_update_entry(state, entry, 'a'))
 
1841
        self.assertEqual(dirstate.DirState.IN_MEMORY_UNMODIFIED,
 
1842
                         state._dirblock_state)
1523
1843
 
1524
1844
    def create_and_test_file(self, state, entry):
1525
1845
        """Create a file at 'a' and verify the state finds it.
1531
1851
        stat_value = os.lstat('a')
1532
1852
        packed_stat = dirstate.pack_stat(stat_value)
1533
1853
 
1534
 
        link_or_sha1 = state.update_entry(entry, abspath='a')
 
1854
        link_or_sha1 = self.do_update_entry(state, entry, abspath='a')
1535
1855
        self.assertEqual('b50e5406bb5e153ebbeb20268fcf37c87e1ecfb6',
1536
1856
                         link_or_sha1)
1537
1857
        self.assertEqual([('f', link_or_sha1, 14, False, packed_stat)],
1548
1868
        stat_value = os.lstat('a')
1549
1869
        packed_stat = dirstate.pack_stat(stat_value)
1550
1870
 
1551
 
        link_or_sha1 = state.update_entry(entry, abspath='a')
 
1871
        link_or_sha1 = self.do_update_entry(state, entry, abspath='a')
1552
1872
        self.assertIs(None, link_or_sha1)
1553
1873
        self.assertEqual([('d', '', 0, False, packed_stat)], entry[1])
1554
1874
 
1569
1889
        stat_value = os.lstat('a')
1570
1890
        packed_stat = dirstate.pack_stat(stat_value)
1571
1891
 
1572
 
        link_or_sha1 = state.update_entry(entry, abspath='a')
 
1892
        link_or_sha1 = self.do_update_entry(state, entry, abspath='a')
1573
1893
        self.assertEqual('path/to/foo', link_or_sha1)
1574
1894
        self.assertEqual([('l', 'path/to/foo', 11, False, packed_stat)],
1575
1895
                         entry[1])
1576
1896
        return packed_stat
1577
1897
 
1578
 
    def test_update_missing_file(self):
1579
 
        state, entry = self.get_state_with_a()
1580
 
        packed_stat = self.create_and_test_file(state, entry)
1581
 
        # Now if we delete the file, update_entry should recover and
1582
 
        # return None.
1583
 
        os.remove('a')
1584
 
        self.assertIs(None, state.update_entry(entry, abspath='a'))
1585
 
        # And the record shouldn't be changed.
1586
 
        digest = 'b50e5406bb5e153ebbeb20268fcf37c87e1ecfb6'
1587
 
        self.assertEqual([('f', digest, 14, False, packed_stat)],
1588
 
                         entry[1])
1589
 
 
1590
 
    def test_update_missing_dir(self):
1591
 
        state, entry = self.get_state_with_a()
1592
 
        packed_stat = self.create_and_test_dir(state, entry)
1593
 
        # Now if we delete the directory, update_entry should recover and
1594
 
        # return None.
1595
 
        os.rmdir('a')
1596
 
        self.assertIs(None, state.update_entry(entry, abspath='a'))
1597
 
        self.assertEqual([('d', '', 0, False, packed_stat)], entry[1])
1598
 
 
1599
 
    def test_update_missing_symlink(self):
1600
 
        if not osutils.has_symlinks():
1601
 
            # PlatformDeficiency / TestSkipped
1602
 
            raise TestSkipped("No symlink support")
1603
 
        state, entry = self.get_state_with_a()
1604
 
        packed_stat = self.create_and_test_symlink(state, entry)
1605
 
        os.remove('a')
1606
 
        self.assertIs(None, state.update_entry(entry, abspath='a'))
1607
 
        # And the record shouldn't be changed.
1608
 
        self.assertEqual([('l', 'path/to/foo', 11, False, packed_stat)],
1609
 
                         entry[1])
1610
 
 
1611
1898
    def test_update_file_to_dir(self):
1612
1899
        """If a file changes to a directory we return None for the sha.
1613
1900
        We also update the inventory record.
1614
1901
        """
1615
1902
        state, entry = self.get_state_with_a()
 
1903
        # The file sha1 won't be cached unless the file is old
 
1904
        state.adjust_time(+10)
1616
1905
        self.create_and_test_file(state, entry)
1617
1906
        os.remove('a')
1618
1907
        self.create_and_test_dir(state, entry)
1619
1908
 
1620
1909
    def test_update_file_to_symlink(self):
1621
1910
        """File becomes a symlink"""
1622
 
        if not osutils.has_symlinks():
1623
 
            # PlatformDeficiency / TestSkipped
1624
 
            raise TestSkipped("No symlink support")
 
1911
        self.requireFeature(SymlinkFeature)
1625
1912
        state, entry = self.get_state_with_a()
 
1913
        # The file sha1 won't be cached unless the file is old
 
1914
        state.adjust_time(+10)
1626
1915
        self.create_and_test_file(state, entry)
1627
1916
        os.remove('a')
1628
1917
        self.create_and_test_symlink(state, entry)
1630
1919
    def test_update_dir_to_file(self):
1631
1920
        """Directory becoming a file updates the entry."""
1632
1921
        state, entry = self.get_state_with_a()
 
1922
        # The file sha1 won't be cached unless the file is old
 
1923
        state.adjust_time(+10)
1633
1924
        self.create_and_test_dir(state, entry)
1634
1925
        os.rmdir('a')
1635
1926
        self.create_and_test_file(state, entry)
1636
1927
 
1637
1928
    def test_update_dir_to_symlink(self):
1638
1929
        """Directory becomes a symlink"""
1639
 
        if not osutils.has_symlinks():
1640
 
            # PlatformDeficiency / TestSkipped
1641
 
            raise TestSkipped("No symlink support")
 
1930
        self.requireFeature(SymlinkFeature)
1642
1931
        state, entry = self.get_state_with_a()
 
1932
        # The symlink target won't be cached if it isn't old
 
1933
        state.adjust_time(+10)
1643
1934
        self.create_and_test_dir(state, entry)
1644
1935
        os.rmdir('a')
1645
1936
        self.create_and_test_symlink(state, entry)
1646
1937
 
1647
1938
    def test_update_symlink_to_file(self):
1648
1939
        """Symlink becomes a file"""
1649
 
        if not has_symlinks():
1650
 
            raise TestSkipped("No symlink support")
 
1940
        self.requireFeature(SymlinkFeature)
1651
1941
        state, entry = self.get_state_with_a()
 
1942
        # The symlink and file info won't be cached unless old
 
1943
        state.adjust_time(+10)
1652
1944
        self.create_and_test_symlink(state, entry)
1653
1945
        os.remove('a')
1654
1946
        self.create_and_test_file(state, entry)
1655
1947
 
1656
1948
    def test_update_symlink_to_dir(self):
1657
1949
        """Symlink becomes a directory"""
1658
 
        if not has_symlinks():
1659
 
            raise TestSkipped("No symlink support")
 
1950
        self.requireFeature(SymlinkFeature)
1660
1951
        state, entry = self.get_state_with_a()
 
1952
        # The symlink target won't be cached if it isn't old
 
1953
        state.adjust_time(+10)
1661
1954
        self.create_and_test_symlink(state, entry)
1662
1955
        os.remove('a')
1663
1956
        self.create_and_test_dir(state, entry)
1677
1970
        packed_stat = dirstate.pack_stat(stat_value)
1678
1971
 
1679
1972
        state.adjust_time(-10) # Make sure everything is new
1680
 
        # Make sure it wants to kkkkkkkk
1681
1973
        state.update_entry(entry, abspath='a', stat_value=stat_value)
1682
1974
 
1683
1975
        # The row is updated, but the executable bit stays set.
 
1976
        self.assertEqual([('f', '', 14, True, dirstate.DirState.NULLSTAT)],
 
1977
                         entry[1])
 
1978
 
 
1979
        # Make the disk object look old enough to cache
 
1980
        state.adjust_time(+20)
1684
1981
        digest = 'b50e5406bb5e153ebbeb20268fcf37c87e1ecfb6'
 
1982
        state.update_entry(entry, abspath='a', stat_value=stat_value)
1685
1983
        self.assertEqual([('f', digest, 14, True, packed_stat)], entry[1])
1686
1984
 
1687
1985
 
1750
2048
                      (dir, name) tuples, and sorted according to how _bisect
1751
2049
                      requires.
1752
2050
        """
1753
 
        dir_names = sorted(osutils.split(p) for p in paths)
1754
 
        result = state._bisect(dir_names)
 
2051
        result = state._bisect(paths)
1755
2052
        # For now, results are just returned in whatever order we read them.
1756
2053
        # We could sort by (dir, name, file_id) or something like that, but in
1757
2054
        # the end it would still be fairly arbitrary, and we don't want the
1758
2055
        # extra overhead if we can avoid it. So sort everything to make sure
1759
2056
        # equality is true
1760
 
        assert len(map_keys) == len(dir_names)
 
2057
        assert len(map_keys) == len(paths)
1761
2058
        expected = {}
1762
 
        for dir_name, keys in zip(dir_names, map_keys):
 
2059
        for path, keys in zip(paths, map_keys):
1763
2060
            if keys is None:
1764
2061
                # This should not be present in the output
1765
2062
                continue
1766
 
            expected[dir_name] = sorted(expected_map[k] for k in keys)
 
2063
            expected[path] = sorted(expected_map[k] for k in keys)
1767
2064
 
1768
 
        for dir_name in result:
1769
 
            result[dir_name].sort()
 
2065
        # The returned values are just arranged randomly based on when they
 
2066
        # were read, for testing, make sure it is properly sorted.
 
2067
        for path in result:
 
2068
            result[path].sort()
1770
2069
 
1771
2070
        self.assertEqual(expected, result)
1772
2071
 
1809
2108
            dir_name_id, trees_info = entry
1810
2109
            expected[dir_name_id] = trees_info
1811
2110
 
1812
 
        dir_names = sorted(osutils.split(p) for p in paths)
1813
 
        result = state._bisect_recursive(dir_names)
 
2111
        result = state._bisect_recursive(paths)
1814
2112
 
1815
2113
        self.assertEqual(expected, result)
1816
2114
 
1825
2123
        self.assertBisect(expected, [['b/c']], state, ['b/c'])
1826
2124
        self.assertBisect(expected, [['b/d']], state, ['b/d'])
1827
2125
        self.assertBisect(expected, [['b/d/e']], state, ['b/d/e'])
 
2126
        self.assertBisect(expected, [['b-c']], state, ['b-c'])
1828
2127
        self.assertBisect(expected, [['f']], state, ['f'])
1829
2128
 
1830
2129
    def test_bisect_multi(self):
1833
2132
        # Bisect should be capable of finding multiple entries at the same time
1834
2133
        self.assertBisect(expected, [['a'], ['b'], ['f']],
1835
2134
                          state, ['a', 'b', 'f'])
1836
 
        # ('', 'f') sorts before the others
1837
2135
        self.assertBisect(expected, [['f'], ['b/d'], ['b/d/e']],
1838
 
                          state, ['b/d', 'b/d/e', 'f'])
 
2136
                          state, ['f', 'b/d', 'b/d/e'])
 
2137
        self.assertBisect(expected, [['b'], ['b-c'], ['b/c']],
 
2138
                          state, ['b', 'b-c', 'b/c'])
1839
2139
 
1840
2140
    def test_bisect_one_page(self):
1841
2141
        """Test bisect when there is only 1 page to read"""
1847
2147
        self.assertBisect(expected,[['b/c']], state, ['b/c'])
1848
2148
        self.assertBisect(expected,[['b/d']], state, ['b/d'])
1849
2149
        self.assertBisect(expected,[['b/d/e']], state, ['b/d/e'])
 
2150
        self.assertBisect(expected,[['b-c']], state, ['b-c'])
1850
2151
        self.assertBisect(expected,[['f']], state, ['f'])
1851
2152
        self.assertBisect(expected,[['a'], ['b'], ['f']],
1852
2153
                          state, ['a', 'b', 'f'])
1853
 
        # ('', 'f') sorts before the others
1854
 
        self.assertBisect(expected, [['f'], ['b/d'], ['b/d/e']],
 
2154
        self.assertBisect(expected, [['b/d'], ['b/d/e'], ['f']],
1855
2155
                          state, ['b/d', 'b/d/e', 'f'])
 
2156
        self.assertBisect(expected, [['b'], ['b/c'], ['b-c']],
 
2157
                          state, ['b', 'b/c', 'b-c'])
1856
2158
 
1857
2159
    def test_bisect_duplicate_paths(self):
1858
2160
        """When bisecting for a path, handle multiple entries."""
1866
2168
        self.assertBisect(expected, [['b/d', 'b/d2']], state, ['b/d'])
1867
2169
        self.assertBisect(expected, [['b/d/e', 'b/d/e2']],
1868
2170
                          state, ['b/d/e'])
 
2171
        self.assertBisect(expected, [['b-c', 'b-c2']], state, ['b-c'])
1869
2172
        self.assertBisect(expected, [['f', 'f2']], state, ['f'])
1870
2173
 
1871
2174
    def test_bisect_page_size_too_small(self):
1878
2181
        self.assertBisect(expected, [['b/c']], state, ['b/c'])
1879
2182
        self.assertBisect(expected, [['b/d']], state, ['b/d'])
1880
2183
        self.assertBisect(expected, [['b/d/e']], state, ['b/d/e'])
 
2184
        self.assertBisect(expected, [['b-c']], state, ['b-c'])
1881
2185
        self.assertBisect(expected, [['f']], state, ['f'])
1882
2186
 
1883
2187
    def test_bisect_missing(self):
1886
2190
        self.assertBisect(expected, [None], state, ['foo'])
1887
2191
        self.assertBisect(expected, [None], state, ['b/foo'])
1888
2192
        self.assertBisect(expected, [None], state, ['bar/foo'])
 
2193
        self.assertBisect(expected, [None], state, ['b-c/foo'])
1889
2194
 
1890
2195
        self.assertBisect(expected, [['a'], None, ['b/d']],
1891
2196
                          state, ['a', 'foo', 'b/d'])
1907
2212
    def test_bisect_dirblocks(self):
1908
2213
        tree, state, expected = self.create_duplicated_dirstate()
1909
2214
        self.assertBisectDirBlocks(expected,
1910
 
            [['', 'a', 'a2', 'b', 'b2', 'f', 'f2']], state, [''])
 
2215
            [['', 'a', 'a2', 'b', 'b2', 'b-c', 'b-c2', 'f', 'f2']],
 
2216
            state, [''])
1911
2217
        self.assertBisectDirBlocks(expected,
1912
2218
            [['b/c', 'b/c2', 'b/d', 'b/d2']], state, ['b'])
1913
2219
        self.assertBisectDirBlocks(expected,
1914
2220
            [['b/d/e', 'b/d/e2']], state, ['b/d'])
1915
2221
        self.assertBisectDirBlocks(expected,
1916
 
            [['', 'a', 'a2', 'b', 'b2', 'f', 'f2'],
 
2222
            [['', 'a', 'a2', 'b', 'b2', 'b-c', 'b-c2', 'f', 'f2'],
1917
2223
             ['b/c', 'b/c2', 'b/d', 'b/d2'],
1918
2224
             ['b/d/e', 'b/d/e2'],
1919
2225
            ], state, ['', 'b', 'b/d'])
1934
2240
        self.assertBisectRecursive(expected, ['a'], state, ['a'])
1935
2241
        self.assertBisectRecursive(expected, ['b/c'], state, ['b/c'])
1936
2242
        self.assertBisectRecursive(expected, ['b/d/e'], state, ['b/d/e'])
 
2243
        self.assertBisectRecursive(expected, ['b-c'], state, ['b-c'])
1937
2244
        self.assertBisectRecursive(expected, ['b/d', 'b/d/e'],
1938
2245
                                   state, ['b/d'])
1939
2246
        self.assertBisectRecursive(expected, ['b', 'b/c', 'b/d', 'b/d/e'],
1940
2247
                                   state, ['b'])
1941
 
        self.assertBisectRecursive(expected, ['', 'a', 'b', 'f', 'b/c',
 
2248
        self.assertBisectRecursive(expected, ['', 'a', 'b', 'b-c', 'f', 'b/c',
1942
2249
                                              'b/d', 'b/d/e'],
1943
2250
                                   state, [''])
1944
2251
 
1968
2275
                                   state, ['b'])
1969
2276
 
1970
2277
 
1971
 
class TestBisectDirblock(TestCase):
1972
 
    """Test that bisect_dirblock() returns the expected values.
1973
 
 
1974
 
    bisect_dirblock is intended to work like bisect.bisect_left() except it
1975
 
    knows it is working on dirblocks and that dirblocks are sorted by ('path',
1976
 
    'to', 'foo') chunks rather than by raw 'path/to/foo'.
1977
 
    """
1978
 
 
1979
 
    def setUp(self):
1980
 
        super(TestBisectDirblock, self).setUp()
1981
 
        # We have to set this here, because if we set it at the class variable
1982
 
        # level, Python interprets it as a member function, and passes 'self'
1983
 
        # as the first argument.
1984
 
        self.bisect_dirblock_func = dirstate.py_bisect_dirblock
1985
 
 
1986
 
    def assertBisect(self, dirblocks, split_dirblocks, path, *args, **kwargs):
1987
 
        """Assert that bisect_split works like bisect_left on the split paths.
1988
 
 
1989
 
        :param dirblocks: A list of (path, [info]) pairs.
1990
 
        :param split_dirblocks: A list of ((split, path), [info]) pairs.
1991
 
        :param path: The path we are indexing.
1992
 
 
1993
 
        All other arguments will be passed along.
1994
 
        """
1995
 
        self.assertIsInstance(dirblocks, list)
1996
 
        bisect_split_idx = self.bisect_dirblock_func(dirblocks, path,
1997
 
                                                     *args, **kwargs)
1998
 
        split_dirblock = (path.split('/'), [])
1999
 
        bisect_left_idx = bisect.bisect_left(split_dirblocks, split_dirblock,
2000
 
                                             *args)
2001
 
        self.assertEqual(bisect_left_idx, bisect_split_idx,
2002
 
                         'bisect_split disagreed. %s != %s'
2003
 
                         ' for key %s'
2004
 
                         % (bisect_left_idx, bisect_split_idx, path)
2005
 
                         )
2006
 
 
2007
 
    def paths_to_dirblocks(self, paths):
2008
 
        """Convert a list of paths into dirblock form.
2009
 
 
2010
 
        Also, ensure that the paths are in proper sorted order.
2011
 
        """
2012
 
        dirblocks = [(path, []) for path in paths]
2013
 
        split_dirblocks = [(path.split('/'), []) for path in paths]
2014
 
        self.assertEqual(sorted(split_dirblocks), split_dirblocks)
2015
 
        return dirblocks, split_dirblocks
2016
 
 
2017
 
    def test_simple(self):
2018
 
        """In the simple case it works just like bisect_left"""
2019
 
        paths = ['', 'a', 'b', 'c', 'd']
2020
 
        dirblocks, split_dirblocks = self.paths_to_dirblocks(paths)
2021
 
        for path in paths:
2022
 
            self.assertBisect(dirblocks, split_dirblocks, path)
2023
 
        self.assertBisect(dirblocks, split_dirblocks, '_')
2024
 
        self.assertBisect(dirblocks, split_dirblocks, 'aa')
2025
 
        self.assertBisect(dirblocks, split_dirblocks, 'bb')
2026
 
        self.assertBisect(dirblocks, split_dirblocks, 'cc')
2027
 
        self.assertBisect(dirblocks, split_dirblocks, 'dd')
2028
 
        self.assertBisect(dirblocks, split_dirblocks, 'a/a')
2029
 
        self.assertBisect(dirblocks, split_dirblocks, 'b/b')
2030
 
        self.assertBisect(dirblocks, split_dirblocks, 'c/c')
2031
 
        self.assertBisect(dirblocks, split_dirblocks, 'd/d')
2032
 
 
2033
 
    def test_involved(self):
2034
 
        """This is where bisect_left diverges slightly."""
2035
 
        paths = ['', 'a',
2036
 
                 'a/a', 'a/a/a', 'a/a/z', 'a/a-a', 'a/a-z',
2037
 
                 'a/z', 'a/z/a', 'a/z/z', 'a/z-a', 'a/z-z',
2038
 
                 'a-a', 'a-z',
2039
 
                 'z', 'z/a/a', 'z/a/z', 'z/a-a', 'z/a-z',
2040
 
                 'z/z', 'z/z/a', 'z/z/z', 'z/z-a', 'z/z-z',
2041
 
                 'z-a', 'z-z',
2042
 
                ]
2043
 
        dirblocks, split_dirblocks = self.paths_to_dirblocks(paths)
2044
 
        for path in paths:
2045
 
            self.assertBisect(dirblocks, split_dirblocks, path)
2046
 
 
2047
 
    def test_involved_cached(self):
2048
 
        """This is where bisect_left diverges slightly."""
2049
 
        paths = ['', 'a',
2050
 
                 'a/a', 'a/a/a', 'a/a/z', 'a/a-a', 'a/a-z',
2051
 
                 'a/z', 'a/z/a', 'a/z/z', 'a/z-a', 'a/z-z',
2052
 
                 'a-a', 'a-z',
2053
 
                 'z', 'z/a/a', 'z/a/z', 'z/a-a', 'z/a-z',
2054
 
                 'z/z', 'z/z/a', 'z/z/z', 'z/z-a', 'z/z-z',
2055
 
                 'z-a', 'z-z',
2056
 
                ]
2057
 
        cache = {}
2058
 
        dirblocks, split_dirblocks = self.paths_to_dirblocks(paths)
2059
 
        for path in paths:
2060
 
            self.assertBisect(dirblocks, split_dirblocks, path, cache=cache)
2061
 
 
2062
 
 
2063
2278
class TestDirstateValidation(TestCaseWithDirState):
2064
2279
 
2065
2280
    def test_validate_correct_dirstate(self):
2115
2330
            state._validate)
2116
2331
        self.assertContainsRe(str(e),
2117
2332
            'file a-id is absent in row')
 
2333
 
 
2334
 
 
2335
class TestDirstateTreeReference(TestCaseWithDirState):
 
2336
 
 
2337
    def test_reference_revision_is_none(self):
 
2338
        tree = self.make_branch_and_tree('tree', format='dirstate-with-subtree')
 
2339
        subtree = self.make_branch_and_tree('tree/subtree',
 
2340
                            format='dirstate-with-subtree')
 
2341
        subtree.set_root_id('subtree')
 
2342
        tree.add_reference(subtree)
 
2343
        tree.add('subtree')
 
2344
        state = dirstate.DirState.from_tree(tree, 'dirstate')
 
2345
        key = ('', 'subtree', 'subtree')
 
2346
        expected = ('', [(key,
 
2347
            [('t', '', 0, False, 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx')])])
 
2348
 
 
2349
        try:
 
2350
            self.assertEqual(expected, state._find_block(key))
 
2351
        finally:
 
2352
            state.unlock()
 
2353
 
 
2354
 
 
2355
class TestDiscardMergeParents(TestCaseWithDirState):
 
2356
 
 
2357
    def test_discard_no_parents(self):
 
2358
        # This should be a no-op
 
2359
        state = self.create_empty_dirstate()
 
2360
        self.addCleanup(state.unlock)
 
2361
        state._discard_merge_parents()
 
2362
        state._validate()
 
2363
 
 
2364
    def test_discard_one_parent(self):
 
2365
        # No-op
 
2366
        packed_stat = 'AAAAREUHaIpFB2iKAAADAQAtkqUAAIGk'
 
2367
        root_entry_direntry = ('', '', 'a-root-value'), [
 
2368
            ('d', '', 0, False, packed_stat),
 
2369
            ('d', '', 0, False, packed_stat),
 
2370
            ]
 
2371
        dirblocks = []
 
2372
        dirblocks.append(('', [root_entry_direntry]))
 
2373
        dirblocks.append(('', []))
 
2374
 
 
2375
        state = self.create_empty_dirstate()
 
2376
        self.addCleanup(state.unlock)
 
2377
        state._set_data(['parent-id'], dirblocks[:])
 
2378
        state._validate()
 
2379
 
 
2380
        state._discard_merge_parents()
 
2381
        state._validate()
 
2382
        self.assertEqual(dirblocks, state._dirblocks)
 
2383
 
 
2384
    def test_discard_simple(self):
 
2385
        # No-op
 
2386
        packed_stat = 'AAAAREUHaIpFB2iKAAADAQAtkqUAAIGk'
 
2387
        root_entry_direntry = ('', '', 'a-root-value'), [
 
2388
            ('d', '', 0, False, packed_stat),
 
2389
            ('d', '', 0, False, packed_stat),
 
2390
            ('d', '', 0, False, packed_stat),
 
2391
            ]
 
2392
        expected_root_entry_direntry = ('', '', 'a-root-value'), [
 
2393
            ('d', '', 0, False, packed_stat),
 
2394
            ('d', '', 0, False, packed_stat),
 
2395
            ]
 
2396
        dirblocks = []
 
2397
        dirblocks.append(('', [root_entry_direntry]))
 
2398
        dirblocks.append(('', []))
 
2399
 
 
2400
        state = self.create_empty_dirstate()
 
2401
        self.addCleanup(state.unlock)
 
2402
        state._set_data(['parent-id', 'merged-id'], dirblocks[:])
 
2403
        state._validate()
 
2404
 
 
2405
        # This should strip of the extra column
 
2406
        state._discard_merge_parents()
 
2407
        state._validate()
 
2408
        expected_dirblocks = [('', [expected_root_entry_direntry]), ('', [])]
 
2409
        self.assertEqual(expected_dirblocks, state._dirblocks)
 
2410
 
 
2411
    def test_discard_absent(self):
 
2412
        """If entries are only in a merge, discard should remove the entries"""
 
2413
        null_stat = dirstate.DirState.NULLSTAT
 
2414
        present_dir = ('d', '', 0, False, null_stat)
 
2415
        present_file = ('f', '', 0, False, null_stat)
 
2416
        absent = dirstate.DirState.NULL_PARENT_DETAILS
 
2417
        root_key = ('', '', 'a-root-value')
 
2418
        file_in_root_key = ('', 'file-in-root', 'a-file-id')
 
2419
        file_in_merged_key = ('', 'file-in-merged', 'b-file-id')
 
2420
        dirblocks = [('', [(root_key, [present_dir, present_dir, present_dir])]),
 
2421
                     ('', [(file_in_merged_key,
 
2422
                            [absent, absent, present_file]),
 
2423
                           (file_in_root_key,
 
2424
                            [present_file, present_file, present_file]),
 
2425
                          ]),
 
2426
                    ]
 
2427
 
 
2428
        state = self.create_empty_dirstate()
 
2429
        self.addCleanup(state.unlock)
 
2430
        state._set_data(['parent-id', 'merged-id'], dirblocks[:])
 
2431
        state._validate()
 
2432
 
 
2433
        exp_dirblocks = [('', [(root_key, [present_dir, present_dir])]),
 
2434
                         ('', [(file_in_root_key,
 
2435
                                [present_file, present_file]),
 
2436
                              ]),
 
2437
                        ]
 
2438
        state._discard_merge_parents()
 
2439
        state._validate()
 
2440
        self.assertEqual(exp_dirblocks, state._dirblocks)
 
2441
 
 
2442
    def test_discard_renamed(self):
 
2443
        null_stat = dirstate.DirState.NULLSTAT
 
2444
        present_dir = ('d', '', 0, False, null_stat)
 
2445
        present_file = ('f', '', 0, False, null_stat)
 
2446
        absent = dirstate.DirState.NULL_PARENT_DETAILS
 
2447
        root_key = ('', '', 'a-root-value')
 
2448
        file_in_root_key = ('', 'file-in-root', 'a-file-id')
 
2449
        # Renamed relative to parent
 
2450
        file_rename_s_key = ('', 'file-s', 'b-file-id')
 
2451
        file_rename_t_key = ('', 'file-t', 'b-file-id')
 
2452
        # And one that is renamed between the parents, but absent in this
 
2453
        key_in_1 = ('', 'file-in-1', 'c-file-id')
 
2454
        key_in_2 = ('', 'file-in-2', 'c-file-id')
 
2455
 
 
2456
        dirblocks = [
 
2457
            ('', [(root_key, [present_dir, present_dir, present_dir])]),
 
2458
            ('', [(key_in_1,
 
2459
                   [absent, present_file, ('r', 'file-in-2', 'c-file-id')]),
 
2460
                  (key_in_2,
 
2461
                   [absent, ('r', 'file-in-1', 'c-file-id'), present_file]),
 
2462
                  (file_in_root_key,
 
2463
                   [present_file, present_file, present_file]),
 
2464
                  (file_rename_s_key,
 
2465
                   [('r', 'file-t', 'b-file-id'), absent, present_file]),
 
2466
                  (file_rename_t_key,
 
2467
                   [present_file, absent, ('r', 'file-s', 'b-file-id')]),
 
2468
                 ]),
 
2469
        ]
 
2470
        exp_dirblocks = [
 
2471
            ('', [(root_key, [present_dir, present_dir])]),
 
2472
            ('', [(key_in_1, [absent, present_file]),
 
2473
                  (file_in_root_key, [present_file, present_file]),
 
2474
                  (file_rename_t_key, [present_file, absent]),
 
2475
                 ]),
 
2476
        ]
 
2477
        state = self.create_empty_dirstate()
 
2478
        self.addCleanup(state.unlock)
 
2479
        state._set_data(['parent-id', 'merged-id'], dirblocks[:])
 
2480
        state._validate()
 
2481
 
 
2482
        state._discard_merge_parents()
 
2483
        state._validate()
 
2484
        self.assertEqual(exp_dirblocks, state._dirblocks)
 
2485
 
 
2486
    def test_discard_all_subdir(self):
 
2487
        null_stat = dirstate.DirState.NULLSTAT
 
2488
        present_dir = ('d', '', 0, False, null_stat)
 
2489
        present_file = ('f', '', 0, False, null_stat)
 
2490
        absent = dirstate.DirState.NULL_PARENT_DETAILS
 
2491
        root_key = ('', '', 'a-root-value')
 
2492
        subdir_key = ('', 'sub', 'dir-id')
 
2493
        child1_key = ('sub', 'child1', 'child1-id')
 
2494
        child2_key = ('sub', 'child2', 'child2-id')
 
2495
        child3_key = ('sub', 'child3', 'child3-id')
 
2496
 
 
2497
        dirblocks = [
 
2498
            ('', [(root_key, [present_dir, present_dir, present_dir])]),
 
2499
            ('', [(subdir_key, [present_dir, present_dir, present_dir])]),
 
2500
            ('sub', [(child1_key, [absent, absent, present_file]),
 
2501
                     (child2_key, [absent, absent, present_file]),
 
2502
                     (child3_key, [absent, absent, present_file]),
 
2503
                    ]),
 
2504
        ]
 
2505
        exp_dirblocks = [
 
2506
            ('', [(root_key, [present_dir, present_dir])]),
 
2507
            ('', [(subdir_key, [present_dir, present_dir])]),
 
2508
            ('sub', []),
 
2509
        ]
 
2510
        state = self.create_empty_dirstate()
 
2511
        self.addCleanup(state.unlock)
 
2512
        state._set_data(['parent-id', 'merged-id'], dirblocks[:])
 
2513
        state._validate()
 
2514
 
 
2515
        state._discard_merge_parents()
 
2516
        state._validate()
 
2517
        self.assertEqual(exp_dirblocks, state._dirblocks)