485
528
create.new_file('vfile', root, 'myfile-text', 'myfile-id')
486
529
create.new_file('uvfile', root, 'othertext')
488
self.assertEqual(find_interesting(wt, wt, ['vfile']),
490
self.assertRaises(NotVersionedError, find_interesting, wt, wt,
531
result = self.applyDeprecated(symbol_versioning.zero_fifteen,
532
find_interesting, wt, wt, ['vfile'])
533
self.assertEqual(result, set(['myfile-id']))
535
def test_set_executability_order(self):
536
"""Ensure that executability behaves the same, no matter what order.
538
- create file and set executability simultaneously
539
- create file and set executability afterward
540
- unsetting the executability of a file whose executability has not been
541
declared should throw an exception (this may happen when a
542
merge attempts to create a file with a duplicate ID)
544
transform, root = self.get_transform()
546
transform.new_file('set_on_creation', root, 'Set on creation', 'soc',
548
sac = transform.new_file('set_after_creation', root, 'Set after creation', 'sac')
549
transform.set_executability(True, sac)
550
uws = transform.new_file('unset_without_set', root, 'Unset badly', 'uws')
551
self.assertRaises(KeyError, transform.set_executability, None, uws)
553
self.assertTrue(wt.is_executable('soc'))
554
self.assertTrue(wt.is_executable('sac'))
556
def test_preserve_mode(self):
557
"""File mode is preserved when replacing content"""
558
if sys.platform == 'win32':
559
raise TestSkipped('chmod has no effect on win32')
560
transform, root = self.get_transform()
561
transform.new_file('file1', root, 'contents', 'file1-id', True)
563
self.assertTrue(self.wt.is_executable('file1-id'))
564
transform, root = self.get_transform()
565
file1_id = transform.trans_id_tree_file_id('file1-id')
566
transform.delete_contents(file1_id)
567
transform.create_file('contents2', file1_id)
569
self.assertTrue(self.wt.is_executable('file1-id'))
571
def test__set_mode_stats_correctly(self):
572
"""_set_mode stats to determine file mode."""
573
if sys.platform == 'win32':
574
raise TestSkipped('chmod has no effect on win32')
578
def instrumented_stat(path):
579
stat_paths.append(path)
580
return real_stat(path)
582
transform, root = self.get_transform()
584
bar1_id = transform.new_file('bar', root, 'bar contents 1\n',
585
file_id='bar-id-1', executable=False)
588
transform, root = self.get_transform()
589
bar1_id = transform.trans_id_tree_path('bar')
590
bar2_id = transform.trans_id_tree_path('bar2')
592
os.stat = instrumented_stat
593
transform.create_file('bar2 contents\n', bar2_id, mode_id=bar1_id)
598
bar1_abspath = self.wt.abspath('bar')
599
self.assertEqual([bar1_abspath], stat_paths)
601
def test_iter_changes(self):
602
self.wt.set_root_id('eert_toor')
603
transform, root = self.get_transform()
604
transform.new_file('old', root, 'blah', 'id-1', True)
606
transform, root = self.get_transform()
608
self.assertEqual([], list(transform._iter_changes()))
609
old = transform.trans_id_tree_file_id('id-1')
610
transform.unversion_file(old)
611
self.assertEqual([('id-1', ('old', None), False, (True, False),
612
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
613
(True, True))], list(transform._iter_changes()))
614
transform.new_directory('new', root, 'id-1')
615
self.assertEqual([('id-1', ('old', 'new'), True, (True, True),
616
('eert_toor', 'eert_toor'), ('old', 'new'),
617
('file', 'directory'),
618
(True, False))], list(transform._iter_changes()))
622
def test_iter_changes_new(self):
623
self.wt.set_root_id('eert_toor')
624
transform, root = self.get_transform()
625
transform.new_file('old', root, 'blah')
627
transform, root = self.get_transform()
629
old = transform.trans_id_tree_path('old')
630
transform.version_file('id-1', old)
631
self.assertEqual([('id-1', (None, 'old'), False, (False, True),
632
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
633
(False, False))], list(transform._iter_changes()))
637
def test_iter_changes_modifications(self):
638
self.wt.set_root_id('eert_toor')
639
transform, root = self.get_transform()
640
transform.new_file('old', root, 'blah', 'id-1')
641
transform.new_file('new', root, 'blah')
642
transform.new_directory('subdir', root, 'subdir-id')
644
transform, root = self.get_transform()
646
old = transform.trans_id_tree_path('old')
647
subdir = transform.trans_id_tree_file_id('subdir-id')
648
new = transform.trans_id_tree_path('new')
649
self.assertEqual([], list(transform._iter_changes()))
652
transform.delete_contents(old)
653
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
654
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', None),
655
(False, False))], list(transform._iter_changes()))
658
transform.create_file('blah', old)
659
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
660
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
661
(False, False))], list(transform._iter_changes()))
662
transform.cancel_deletion(old)
663
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
664
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
665
(False, False))], list(transform._iter_changes()))
666
transform.cancel_creation(old)
668
# move file_id to a different file
669
self.assertEqual([], list(transform._iter_changes()))
670
transform.unversion_file(old)
671
transform.version_file('id-1', new)
672
transform.adjust_path('old', root, new)
673
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
674
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
675
(False, False))], list(transform._iter_changes()))
676
transform.cancel_versioning(new)
677
transform._removed_id = set()
680
self.assertEqual([], list(transform._iter_changes()))
681
transform.set_executability(True, old)
682
self.assertEqual([('id-1', ('old', 'old'), False, (True, True),
683
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
684
(False, True))], list(transform._iter_changes()))
685
transform.set_executability(None, old)
688
self.assertEqual([], list(transform._iter_changes()))
689
transform.adjust_path('new', root, old)
690
transform._new_parent = {}
691
self.assertEqual([('id-1', ('old', 'new'), False, (True, True),
692
('eert_toor', 'eert_toor'), ('old', 'new'), ('file', 'file'),
693
(False, False))], list(transform._iter_changes()))
694
transform._new_name = {}
697
self.assertEqual([], list(transform._iter_changes()))
698
transform.adjust_path('new', subdir, old)
699
transform._new_name = {}
700
self.assertEqual([('id-1', ('old', 'subdir/old'), False,
701
(True, True), ('eert_toor', 'subdir-id'), ('old', 'old'),
702
('file', 'file'), (False, False))],
703
list(transform._iter_changes()))
704
transform._new_path = {}
709
def test_iter_changes_modified_bleed(self):
710
self.wt.set_root_id('eert_toor')
711
"""Modified flag should not bleed from one change to another"""
712
# unfortunately, we have no guarantee that file1 (which is modified)
713
# will be applied before file2. And if it's applied after file2, it
714
# obviously can't bleed into file2's change output. But for now, it
716
transform, root = self.get_transform()
717
transform.new_file('file1', root, 'blah', 'id-1')
718
transform.new_file('file2', root, 'blah', 'id-2')
720
transform, root = self.get_transform()
722
transform.delete_contents(transform.trans_id_file_id('id-1'))
723
transform.set_executability(True,
724
transform.trans_id_file_id('id-2'))
725
self.assertEqual([('id-1', (u'file1', u'file1'), True, (True, True),
726
('eert_toor', 'eert_toor'), ('file1', u'file1'),
727
('file', None), (False, False)),
728
('id-2', (u'file2', u'file2'), False, (True, True),
729
('eert_toor', 'eert_toor'), ('file2', u'file2'),
730
('file', 'file'), (False, True))],
731
list(transform._iter_changes()))
735
def test_iter_changes_pointless(self):
736
"""Ensure that no-ops are not treated as modifications"""
737
self.wt.set_root_id('eert_toor')
738
transform, root = self.get_transform()
739
transform.new_file('old', root, 'blah', 'id-1')
740
transform.new_directory('subdir', root, 'subdir-id')
742
transform, root = self.get_transform()
744
old = transform.trans_id_tree_path('old')
745
subdir = transform.trans_id_tree_file_id('subdir-id')
746
self.assertEqual([], list(transform._iter_changes()))
747
transform.delete_contents(subdir)
748
transform.create_directory(subdir)
749
transform.set_executability(False, old)
750
transform.unversion_file(old)
751
transform.version_file('id-1', old)
752
transform.adjust_path('old', root, old)
753
self.assertEqual([], list(transform._iter_changes()))
494
757
class TransformGroup(object):
495
def __init__(self, dirname):
758
def __init__(self, dirname, root_id):
496
759
self.name = dirname
497
760
os.mkdir(dirname)
498
761
self.wt = BzrDir.create_standalone_workingtree(dirname)
762
self.wt.set_root_id(root_id)
499
763
self.b = self.wt.branch
500
764
self.tt = TreeTransform(self.wt)
501
765
self.root = self.tt.trans_id_tree_file_id(self.wt.get_root_id())
503
768
def conflict_text(tree, merge):
504
769
template = '%s TREE\n%s%s\n%s%s MERGE-SOURCE\n'
505
770
return template % ('<' * 7, tree, '=' * 7, merge, '>' * 7)
698
969
a.add(['foo', 'foo/bar', 'foo/baz'])
699
970
a.commit('initial commit')
700
971
b = BzrDir.create_standalone_workingtree('b')
701
build_tree(a.basis_tree(), b)
972
basis = a.basis_tree()
974
self.addCleanup(basis.unlock)
702
976
self.assertIs(os.path.isdir('b/foo'), True)
703
977
self.assertEqual(file('b/foo/bar', 'rb').read(), "contents")
704
978
self.assertEqual(os.readlink('b/foo/baz'), 'a/foo/bar')
980
def test_build_with_references(self):
981
tree = self.make_branch_and_tree('source',
982
format='dirstate-with-subtree')
983
subtree = self.make_branch_and_tree('source/subtree',
984
format='dirstate-with-subtree')
985
tree.add_reference(subtree)
986
tree.commit('a revision')
987
tree.branch.create_checkout('target')
988
self.failUnlessExists('target')
989
self.failUnlessExists('target/subtree')
991
def test_file_conflict_handling(self):
992
"""Ensure that when building trees, conflict handling is done"""
993
source = self.make_branch_and_tree('source')
994
target = self.make_branch_and_tree('target')
995
self.build_tree(['source/file', 'target/file'])
996
source.add('file', 'new-file')
997
source.commit('added file')
998
build_tree(source.basis_tree(), target)
999
self.assertEqual([DuplicateEntry('Moved existing file to',
1000
'file.moved', 'file', None, 'new-file')],
1002
target2 = self.make_branch_and_tree('target2')
1003
target_file = file('target2/file', 'wb')
1005
source_file = file('source/file', 'rb')
1007
target_file.write(source_file.read())
1012
build_tree(source.basis_tree(), target2)
1013
self.assertEqual([], target2.conflicts())
1015
def test_symlink_conflict_handling(self):
1016
"""Ensure that when building trees, conflict handling is done"""
1017
if not has_symlinks():
1018
raise TestSkipped('Test requires symlink support')
1019
source = self.make_branch_and_tree('source')
1020
os.symlink('foo', 'source/symlink')
1021
source.add('symlink', 'new-symlink')
1022
source.commit('added file')
1023
target = self.make_branch_and_tree('target')
1024
os.symlink('bar', 'target/symlink')
1025
build_tree(source.basis_tree(), target)
1026
self.assertEqual([DuplicateEntry('Moved existing file to',
1027
'symlink.moved', 'symlink', None, 'new-symlink')],
1029
target = self.make_branch_and_tree('target2')
1030
os.symlink('foo', 'target2/symlink')
1031
build_tree(source.basis_tree(), target)
1032
self.assertEqual([], target.conflicts())
1034
def test_directory_conflict_handling(self):
1035
"""Ensure that when building trees, conflict handling is done"""
1036
source = self.make_branch_and_tree('source')
1037
target = self.make_branch_and_tree('target')
1038
self.build_tree(['source/dir1/', 'source/dir1/file', 'target/dir1/'])
1039
source.add(['dir1', 'dir1/file'], ['new-dir1', 'new-file'])
1040
source.commit('added file')
1041
build_tree(source.basis_tree(), target)
1042
self.assertEqual([], target.conflicts())
1043
self.failUnlessExists('target/dir1/file')
1045
# Ensure contents are merged
1046
target = self.make_branch_and_tree('target2')
1047
self.build_tree(['target2/dir1/', 'target2/dir1/file2'])
1048
build_tree(source.basis_tree(), target)
1049
self.assertEqual([], target.conflicts())
1050
self.failUnlessExists('target2/dir1/file2')
1051
self.failUnlessExists('target2/dir1/file')
1053
# Ensure new contents are suppressed for existing branches
1054
target = self.make_branch_and_tree('target3')
1055
self.make_branch('target3/dir1')
1056
self.build_tree(['target3/dir1/file2'])
1057
build_tree(source.basis_tree(), target)
1058
self.failIfExists('target3/dir1/file')
1059
self.failUnlessExists('target3/dir1/file2')
1060
self.failUnlessExists('target3/dir1.diverted/file')
1061
self.assertEqual([DuplicateEntry('Diverted to',
1062
'dir1.diverted', 'dir1', 'new-dir1', None)],
1065
target = self.make_branch_and_tree('target4')
1066
self.build_tree(['target4/dir1/'])
1067
self.make_branch('target4/dir1/file')
1068
build_tree(source.basis_tree(), target)
1069
self.failUnlessExists('target4/dir1/file')
1070
self.assertEqual('directory', file_kind('target4/dir1/file'))
1071
self.failUnlessExists('target4/dir1/file.diverted')
1072
self.assertEqual([DuplicateEntry('Diverted to',
1073
'dir1/file.diverted', 'dir1/file', 'new-file', None)],
1076
def test_mixed_conflict_handling(self):
1077
"""Ensure that when building trees, conflict handling is done"""
1078
source = self.make_branch_and_tree('source')
1079
target = self.make_branch_and_tree('target')
1080
self.build_tree(['source/name', 'target/name/'])
1081
source.add('name', 'new-name')
1082
source.commit('added file')
1083
build_tree(source.basis_tree(), target)
1084
self.assertEqual([DuplicateEntry('Moved existing file to',
1085
'name.moved', 'name', None, 'new-name')], target.conflicts())
1087
def test_raises_in_populated(self):
1088
source = self.make_branch_and_tree('source')
1089
self.build_tree(['source/name'])
1091
source.commit('added name')
1092
target = self.make_branch_and_tree('target')
1093
self.build_tree(['target/name'])
1095
self.assertRaises(errors.WorkingTreeAlreadyPopulated,
1096
build_tree, source.basis_tree(), target)
706
1099
class MockTransform(object):
708
1101
def has_named_child(self, by_parent, parent_id, name):