830
467
rename.set_executability(True, myfile)
833
def test_set_executability_order(self):
834
"""Ensure that executability behaves the same, no matter what order.
836
- create file and set executability simultaneously
837
- create file and set executability afterward
838
- unsetting the executability of a file whose executability has not been
839
declared should throw an exception (this may happen when a
840
merge attempts to create a file with a duplicate ID)
842
transform, root = self.get_transform()
845
self.addCleanup(wt.unlock)
846
transform.new_file('set_on_creation', root, 'Set on creation', 'soc',
848
sac = transform.new_file('set_after_creation', root,
849
'Set after creation', 'sac')
850
transform.set_executability(True, sac)
851
uws = transform.new_file('unset_without_set', root, 'Unset badly',
853
self.assertRaises(KeyError, transform.set_executability, None, uws)
855
self.assertTrue(wt.is_executable('soc'))
856
self.assertTrue(wt.is_executable('sac'))
858
def test_preserve_mode(self):
859
"""File mode is preserved when replacing content"""
860
if sys.platform == 'win32':
861
raise TestSkipped('chmod has no effect on win32')
862
transform, root = self.get_transform()
863
transform.new_file('file1', root, 'contents', 'file1-id', True)
866
self.addCleanup(self.wt.unlock)
867
self.assertTrue(self.wt.is_executable('file1-id'))
868
transform, root = self.get_transform()
869
file1_id = transform.trans_id_tree_file_id('file1-id')
870
transform.delete_contents(file1_id)
871
transform.create_file('contents2', file1_id)
873
self.assertTrue(self.wt.is_executable('file1-id'))
875
def test__set_mode_stats_correctly(self):
876
"""_set_mode stats to determine file mode."""
877
if sys.platform == 'win32':
878
raise TestSkipped('chmod has no effect on win32')
882
def instrumented_stat(path):
883
stat_paths.append(path)
884
return real_stat(path)
886
transform, root = self.get_transform()
888
bar1_id = transform.new_file('bar', root, 'bar contents 1\n',
889
file_id='bar-id-1', executable=False)
892
transform, root = self.get_transform()
893
bar1_id = transform.trans_id_tree_path('bar')
894
bar2_id = transform.trans_id_tree_path('bar2')
896
os.stat = instrumented_stat
897
transform.create_file('bar2 contents\n', bar2_id, mode_id=bar1_id)
902
bar1_abspath = self.wt.abspath('bar')
903
self.assertEqual([bar1_abspath], stat_paths)
905
def test_iter_changes(self):
906
self.wt.set_root_id('eert_toor')
907
transform, root = self.get_transform()
908
transform.new_file('old', root, 'blah', 'id-1', True)
910
transform, root = self.get_transform()
912
self.assertEqual([], list(transform.iter_changes()))
913
old = transform.trans_id_tree_file_id('id-1')
914
transform.unversion_file(old)
915
self.assertEqual([('id-1', ('old', None), False, (True, False),
916
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
917
(True, True))], list(transform.iter_changes()))
918
transform.new_directory('new', root, 'id-1')
919
self.assertEqual([('id-1', ('old', 'new'), True, (True, True),
920
('eert_toor', 'eert_toor'), ('old', 'new'),
921
('file', 'directory'),
922
(True, False))], list(transform.iter_changes()))
926
def test_iter_changes_new(self):
927
self.wt.set_root_id('eert_toor')
928
transform, root = self.get_transform()
929
transform.new_file('old', root, 'blah')
931
transform, root = self.get_transform()
933
old = transform.trans_id_tree_path('old')
934
transform.version_file('id-1', old)
935
self.assertEqual([('id-1', (None, 'old'), False, (False, True),
936
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
937
(False, False))], list(transform.iter_changes()))
941
def test_iter_changes_modifications(self):
942
self.wt.set_root_id('eert_toor')
943
transform, root = self.get_transform()
944
transform.new_file('old', root, 'blah', 'id-1')
945
transform.new_file('new', root, 'blah')
946
transform.new_directory('subdir', root, 'subdir-id')
948
transform, root = self.get_transform()
950
old = transform.trans_id_tree_path('old')
951
subdir = transform.trans_id_tree_file_id('subdir-id')
952
new = transform.trans_id_tree_path('new')
953
self.assertEqual([], list(transform.iter_changes()))
956
transform.delete_contents(old)
957
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
958
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', None),
959
(False, False))], list(transform.iter_changes()))
962
transform.create_file('blah', old)
963
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
964
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
965
(False, False))], list(transform.iter_changes()))
966
transform.cancel_deletion(old)
967
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
968
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
969
(False, False))], list(transform.iter_changes()))
970
transform.cancel_creation(old)
972
# move file_id to a different file
973
self.assertEqual([], list(transform.iter_changes()))
974
transform.unversion_file(old)
975
transform.version_file('id-1', new)
976
transform.adjust_path('old', root, new)
977
self.assertEqual([('id-1', ('old', 'old'), True, (True, True),
978
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
979
(False, False))], list(transform.iter_changes()))
980
transform.cancel_versioning(new)
981
transform._removed_id = set()
984
self.assertEqual([], list(transform.iter_changes()))
985
transform.set_executability(True, old)
986
self.assertEqual([('id-1', ('old', 'old'), False, (True, True),
987
('eert_toor', 'eert_toor'), ('old', 'old'), ('file', 'file'),
988
(False, True))], list(transform.iter_changes()))
989
transform.set_executability(None, old)
992
self.assertEqual([], list(transform.iter_changes()))
993
transform.adjust_path('new', root, old)
994
transform._new_parent = {}
995
self.assertEqual([('id-1', ('old', 'new'), False, (True, True),
996
('eert_toor', 'eert_toor'), ('old', 'new'), ('file', 'file'),
997
(False, False))], list(transform.iter_changes()))
998
transform._new_name = {}
1001
self.assertEqual([], list(transform.iter_changes()))
1002
transform.adjust_path('new', subdir, old)
1003
transform._new_name = {}
1004
self.assertEqual([('id-1', ('old', 'subdir/old'), False,
1005
(True, True), ('eert_toor', 'subdir-id'), ('old', 'old'),
1006
('file', 'file'), (False, False))],
1007
list(transform.iter_changes()))
1008
transform._new_path = {}
1011
transform.finalize()
1013
def test_iter_changes_modified_bleed(self):
1014
self.wt.set_root_id('eert_toor')
1015
"""Modified flag should not bleed from one change to another"""
1016
# unfortunately, we have no guarantee that file1 (which is modified)
1017
# will be applied before file2. And if it's applied after file2, it
1018
# obviously can't bleed into file2's change output. But for now, it
1020
transform, root = self.get_transform()
1021
transform.new_file('file1', root, 'blah', 'id-1')
1022
transform.new_file('file2', root, 'blah', 'id-2')
1024
transform, root = self.get_transform()
1026
transform.delete_contents(transform.trans_id_file_id('id-1'))
1027
transform.set_executability(True,
1028
transform.trans_id_file_id('id-2'))
1029
self.assertEqual([('id-1', (u'file1', u'file1'), True, (True, True),
1030
('eert_toor', 'eert_toor'), ('file1', u'file1'),
1031
('file', None), (False, False)),
1032
('id-2', (u'file2', u'file2'), False, (True, True),
1033
('eert_toor', 'eert_toor'), ('file2', u'file2'),
1034
('file', 'file'), (False, True))],
1035
list(transform.iter_changes()))
1037
transform.finalize()
1039
def test_iter_changes_move_missing(self):
1040
"""Test moving ids with no files around"""
1041
self.wt.set_root_id('toor_eert')
1042
# Need two steps because versioning a non-existant file is a conflict.
1043
transform, root = self.get_transform()
1044
transform.new_directory('floater', root, 'floater-id')
1046
transform, root = self.get_transform()
1047
transform.delete_contents(transform.trans_id_tree_path('floater'))
1049
transform, root = self.get_transform()
1050
floater = transform.trans_id_tree_path('floater')
1052
transform.adjust_path('flitter', root, floater)
1053
self.assertEqual([('floater-id', ('floater', 'flitter'), False,
1054
(True, True), ('toor_eert', 'toor_eert'), ('floater', 'flitter'),
1055
(None, None), (False, False))], list(transform.iter_changes()))
1057
transform.finalize()
1059
def test_iter_changes_pointless(self):
1060
"""Ensure that no-ops are not treated as modifications"""
1061
self.wt.set_root_id('eert_toor')
1062
transform, root = self.get_transform()
1063
transform.new_file('old', root, 'blah', 'id-1')
1064
transform.new_directory('subdir', root, 'subdir-id')
1066
transform, root = self.get_transform()
1068
old = transform.trans_id_tree_path('old')
1069
subdir = transform.trans_id_tree_file_id('subdir-id')
1070
self.assertEqual([], list(transform.iter_changes()))
1071
transform.delete_contents(subdir)
1072
transform.create_directory(subdir)
1073
transform.set_executability(False, old)
1074
transform.unversion_file(old)
1075
transform.version_file('id-1', old)
1076
transform.adjust_path('old', root, old)
1077
self.assertEqual([], list(transform.iter_changes()))
1079
transform.finalize()
1081
def test_rename_count(self):
1082
transform, root = self.get_transform()
1083
transform.new_file('name1', root, 'contents')
1084
self.assertEqual(transform.rename_count, 0)
1086
self.assertEqual(transform.rename_count, 1)
1087
transform2, root = self.get_transform()
1088
transform2.adjust_path('name2', root,
1089
transform2.trans_id_tree_path('name1'))
1090
self.assertEqual(transform2.rename_count, 0)
1092
self.assertEqual(transform2.rename_count, 2)
1094
def test_change_parent(self):
1095
"""Ensure that after we change a parent, the results are still right.
1097
Renames and parent changes on pending transforms can happen as part
1098
of conflict resolution, and are explicitly permitted by the
1101
This test ensures they work correctly with the rename-avoidance
1104
transform, root = self.get_transform()
1105
parent1 = transform.new_directory('parent1', root)
1106
child1 = transform.new_file('child1', parent1, 'contents')
1107
parent2 = transform.new_directory('parent2', root)
1108
transform.adjust_path('child1', parent2, child1)
1110
self.failIfExists(self.wt.abspath('parent1/child1'))
1111
self.failUnlessExists(self.wt.abspath('parent2/child1'))
1112
# rename limbo/new-1 => parent1, rename limbo/new-3 => parent2
1113
# no rename for child1 (counting only renames during apply)
1114
self.failUnlessEqual(2, transform.rename_count)
1116
def test_cancel_parent(self):
1117
"""Cancelling a parent doesn't cause deletion of a non-empty directory
1119
This is like the test_change_parent, except that we cancel the parent
1120
before adjusting the path. The transform must detect that the
1121
directory is non-empty, and move children to safe locations.
1123
transform, root = self.get_transform()
1124
parent1 = transform.new_directory('parent1', root)
1125
child1 = transform.new_file('child1', parent1, 'contents')
1126
child2 = transform.new_file('child2', parent1, 'contents')
1128
transform.cancel_creation(parent1)
1130
self.fail('Failed to move child1 before deleting parent1')
1131
transform.cancel_creation(child2)
1132
transform.create_directory(parent1)
1134
transform.cancel_creation(parent1)
1135
# If the transform incorrectly believes that child2 is still in
1136
# parent1's limbo directory, it will try to rename it and fail
1137
# because was already moved by the first cancel_creation.
1139
self.fail('Transform still thinks child2 is a child of parent1')
1140
parent2 = transform.new_directory('parent2', root)
1141
transform.adjust_path('child1', parent2, child1)
1143
self.failIfExists(self.wt.abspath('parent1'))
1144
self.failUnlessExists(self.wt.abspath('parent2/child1'))
1145
# rename limbo/new-3 => parent2, rename limbo/new-2 => child1
1146
self.failUnlessEqual(2, transform.rename_count)
1148
def test_adjust_and_cancel(self):
1149
"""Make sure adjust_path keeps track of limbo children properly"""
1150
transform, root = self.get_transform()
1151
parent1 = transform.new_directory('parent1', root)
1152
child1 = transform.new_file('child1', parent1, 'contents')
1153
parent2 = transform.new_directory('parent2', root)
1154
transform.adjust_path('child1', parent2, child1)
1155
transform.cancel_creation(child1)
1157
transform.cancel_creation(parent1)
1158
# if the transform thinks child1 is still in parent1's limbo
1159
# directory, it will attempt to move it and fail.
1161
self.fail('Transform still thinks child1 is a child of parent1')
1162
transform.finalize()
1164
def test_noname_contents(self):
1165
"""TreeTransform should permit deferring naming files."""
1166
transform, root = self.get_transform()
1167
parent = transform.trans_id_file_id('parent-id')
1169
transform.create_directory(parent)
1171
self.fail("Can't handle contents with no name")
1172
transform.finalize()
1174
def test_noname_contents_nested(self):
1175
"""TreeTransform should permit deferring naming files."""
1176
transform, root = self.get_transform()
1177
parent = transform.trans_id_file_id('parent-id')
1179
transform.create_directory(parent)
1181
self.fail("Can't handle contents with no name")
1182
child = transform.new_directory('child', parent)
1183
transform.adjust_path('parent', root, parent)
1185
self.failUnlessExists(self.wt.abspath('parent/child'))
1186
self.assertEqual(1, transform.rename_count)
1188
def test_reuse_name(self):
1189
"""Avoid reusing the same limbo name for different files"""
1190
transform, root = self.get_transform()
1191
parent = transform.new_directory('parent', root)
1192
child1 = transform.new_directory('child', parent)
1194
child2 = transform.new_directory('child', parent)
1196
self.fail('Tranform tried to use the same limbo name twice')
1197
transform.adjust_path('child2', parent, child2)
1199
# limbo/new-1 => parent, limbo/new-3 => parent/child2
1200
# child2 is put into top-level limbo because child1 has already
1201
# claimed the direct limbo path when child2 is created. There is no
1202
# advantage in renaming files once they're in top-level limbo, except
1204
self.assertEqual(2, transform.rename_count)
1206
def test_reuse_when_first_moved(self):
1207
"""Don't avoid direct paths when it is safe to use them"""
1208
transform, root = self.get_transform()
1209
parent = transform.new_directory('parent', root)
1210
child1 = transform.new_directory('child', parent)
1211
transform.adjust_path('child1', parent, child1)
1212
child2 = transform.new_directory('child', parent)
1214
# limbo/new-1 => parent
1215
self.assertEqual(1, transform.rename_count)
1217
def test_reuse_after_cancel(self):
1218
"""Don't avoid direct paths when it is safe to use them"""
1219
transform, root = self.get_transform()
1220
parent2 = transform.new_directory('parent2', root)
1221
child1 = transform.new_directory('child1', parent2)
1222
transform.cancel_creation(parent2)
1223
transform.create_directory(parent2)
1224
child2 = transform.new_directory('child1', parent2)
1225
transform.adjust_path('child2', parent2, child1)
1227
# limbo/new-1 => parent2, limbo/new-2 => parent2/child1
1228
self.assertEqual(2, transform.rename_count)
1230
def test_finalize_order(self):
1231
"""Finalize must be done in child-to-parent order"""
1232
transform, root = self.get_transform()
1233
parent = transform.new_directory('parent', root)
1234
child = transform.new_directory('child', parent)
1236
transform.finalize()
1238
self.fail('Tried to remove parent before child1')
1240
def test_cancel_with_cancelled_child_should_succeed(self):
1241
transform, root = self.get_transform()
1242
parent = transform.new_directory('parent', root)
1243
child = transform.new_directory('child', parent)
1244
transform.cancel_creation(child)
1245
transform.cancel_creation(parent)
1246
transform.finalize()
1248
def test_rollback_on_directory_clash(self):
1250
wt = self.make_branch_and_tree('.')
1251
tt = TreeTransform(wt) # TreeTransform obtains write lock
1253
foo = tt.new_directory('foo', tt.root)
1254
tt.new_file('bar', foo, 'foobar')
1255
baz = tt.new_directory('baz', tt.root)
1256
tt.new_file('qux', baz, 'quux')
1257
# Ask for a rename 'foo' -> 'baz'
1258
tt.adjust_path('baz', tt.root, foo)
1259
# Lie to tt that we've already resolved all conflicts.
1260
tt.apply(no_conflicts=True)
1264
# The rename will fail because the target directory is not empty (but
1265
# raises FileExists anyway).
1266
err = self.assertRaises(errors.FileExists, tt_helper)
1267
self.assertContainsRe(str(err),
1268
"^File exists: .+/baz")
1270
def test_two_directories_clash(self):
1272
wt = self.make_branch_and_tree('.')
1273
tt = TreeTransform(wt) # TreeTransform obtains write lock
1275
foo_1 = tt.new_directory('foo', tt.root)
1276
tt.new_directory('bar', foo_1)
1277
# Adding the same directory with a different content
1278
foo_2 = tt.new_directory('foo', tt.root)
1279
tt.new_directory('baz', foo_2)
1280
# Lie to tt that we've already resolved all conflicts.
1281
tt.apply(no_conflicts=True)
1285
err = self.assertRaises(errors.FileExists, tt_helper)
1286
self.assertContainsRe(str(err),
1287
"^File exists: .+/foo")
1289
def test_two_directories_clash_finalize(self):
1291
wt = self.make_branch_and_tree('.')
1292
tt = TreeTransform(wt) # TreeTransform obtains write lock
1294
foo_1 = tt.new_directory('foo', tt.root)
1295
tt.new_directory('bar', foo_1)
1296
# Adding the same directory with a different content
1297
foo_2 = tt.new_directory('foo', tt.root)
1298
tt.new_directory('baz', foo_2)
1299
# Lie to tt that we've already resolved all conflicts.
1300
tt.apply(no_conflicts=True)
1304
err = self.assertRaises(errors.FileExists, tt_helper)
1305
self.assertContainsRe(str(err),
1306
"^File exists: .+/foo")
1308
def test_file_to_directory(self):
1309
wt = self.make_branch_and_tree('.')
1310
self.build_tree(['foo'])
1313
tt = TreeTransform(wt)
1314
self.addCleanup(tt.finalize)
1315
foo_trans_id = tt.trans_id_tree_path("foo")
1316
tt.delete_contents(foo_trans_id)
1317
tt.create_directory(foo_trans_id)
1318
bar_trans_id = tt.trans_id_tree_path("foo/bar")
1319
tt.create_file(["aa\n"], bar_trans_id)
1320
tt.version_file("bar-1", bar_trans_id)
1322
self.failUnlessExists("foo/bar")
1325
self.assertEqual(wt.inventory.get_file_kind(wt.path2id("foo")),
1330
changes = wt.changes_from(wt.basis_tree())
1331
self.assertFalse(changes.has_changed(), changes)
1333
def test_file_to_symlink(self):
1334
self.requireFeature(SymlinkFeature)
1335
wt = self.make_branch_and_tree('.')
1336
self.build_tree(['foo'])
1339
tt = TreeTransform(wt)
1340
self.addCleanup(tt.finalize)
1341
foo_trans_id = tt.trans_id_tree_path("foo")
1342
tt.delete_contents(foo_trans_id)
1343
tt.create_symlink("bar", foo_trans_id)
1345
self.failUnlessExists("foo")
1347
self.addCleanup(wt.unlock)
1348
self.assertEqual(wt.inventory.get_file_kind(wt.path2id("foo")),
1351
def test_dir_to_file(self):
1352
wt = self.make_branch_and_tree('.')
1353
self.build_tree(['foo/', 'foo/bar'])
1354
wt.add(['foo', 'foo/bar'])
1356
tt = TreeTransform(wt)
1357
self.addCleanup(tt.finalize)
1358
foo_trans_id = tt.trans_id_tree_path("foo")
1359
bar_trans_id = tt.trans_id_tree_path("foo/bar")
1360
tt.delete_contents(foo_trans_id)
1361
tt.delete_versioned(bar_trans_id)
1362
tt.create_file(["aa\n"], foo_trans_id)
1364
self.failUnlessExists("foo")
1366
self.addCleanup(wt.unlock)
1367
self.assertEqual(wt.inventory.get_file_kind(wt.path2id("foo")),
1370
def test_dir_to_hardlink(self):
1371
self.requireFeature(HardlinkFeature)
1372
wt = self.make_branch_and_tree('.')
1373
self.build_tree(['foo/', 'foo/bar'])
1374
wt.add(['foo', 'foo/bar'])
1376
tt = TreeTransform(wt)
1377
self.addCleanup(tt.finalize)
1378
foo_trans_id = tt.trans_id_tree_path("foo")
1379
bar_trans_id = tt.trans_id_tree_path("foo/bar")
1380
tt.delete_contents(foo_trans_id)
1381
tt.delete_versioned(bar_trans_id)
1382
self.build_tree(['baz'])
1383
tt.create_hardlink("baz", foo_trans_id)
1385
self.failUnlessExists("foo")
1386
self.failUnlessExists("baz")
1388
self.addCleanup(wt.unlock)
1389
self.assertEqual(wt.inventory.get_file_kind(wt.path2id("foo")),
1392
def test_no_final_path(self):
1393
transform, root = self.get_transform()
1394
trans_id = transform.trans_id_file_id('foo')
1395
transform.create_file('bar', trans_id)
1396
transform.cancel_creation(trans_id)
1399
def test_create_from_tree(self):
1400
tree1 = self.make_branch_and_tree('tree1')
1401
self.build_tree_contents([('tree1/foo/',), ('tree1/bar', 'baz')])
1402
tree1.add(['foo', 'bar'], ['foo-id', 'bar-id'])
1403
tree2 = self.make_branch_and_tree('tree2')
1404
tt = TreeTransform(tree2)
1405
foo_trans_id = tt.create_path('foo', tt.root)
1406
create_from_tree(tt, foo_trans_id, tree1, 'foo-id')
1407
bar_trans_id = tt.create_path('bar', tt.root)
1408
create_from_tree(tt, bar_trans_id, tree1, 'bar-id')
1410
self.assertEqual('directory', osutils.file_kind('tree2/foo'))
1411
self.assertFileEqual('baz', 'tree2/bar')
1413
def test_create_from_tree_bytes(self):
1414
"""Provided lines are used instead of tree content."""
1415
tree1 = self.make_branch_and_tree('tree1')
1416
self.build_tree_contents([('tree1/foo', 'bar'),])
1417
tree1.add('foo', 'foo-id')
1418
tree2 = self.make_branch_and_tree('tree2')
1419
tt = TreeTransform(tree2)
1420
foo_trans_id = tt.create_path('foo', tt.root)
1421
create_from_tree(tt, foo_trans_id, tree1, 'foo-id', bytes='qux')
1423
self.assertFileEqual('qux', 'tree2/foo')
1425
def test_create_from_tree_symlink(self):
1426
self.requireFeature(SymlinkFeature)
1427
tree1 = self.make_branch_and_tree('tree1')
1428
os.symlink('bar', 'tree1/foo')
1429
tree1.add('foo', 'foo-id')
1430
tt = TreeTransform(self.make_branch_and_tree('tree2'))
1431
foo_trans_id = tt.create_path('foo', tt.root)
1432
create_from_tree(tt, foo_trans_id, tree1, 'foo-id')
1434
self.assertEqual('bar', os.readlink('tree2/foo'))
470
def test_find_interesting(self):
471
create, root = self.get_transform()
473
create.new_file('vfile', root, 'myfile-text', 'myfile-id')
474
create.new_file('uvfile', root, 'othertext')
476
self.assertEqual(find_interesting(wt, wt, ['vfile']),
478
self.assertRaises(NotVersionedError, find_interesting, wt, wt,
1437
482
class TransformGroup(object):
1439
def __init__(self, dirname, root_id):
483
def __init__(self, dirname):
1440
484
self.name = dirname
1441
485
os.mkdir(dirname)
1442
486
self.wt = BzrDir.create_standalone_workingtree(dirname)
1443
self.wt.set_root_id(root_id)
1444
487
self.b = self.wt.branch
1445
488
self.tt = TreeTransform(self.wt)
1446
489
self.root = self.tt.trans_id_tree_file_id(self.wt.get_root_id())
1449
491
def conflict_text(tree, merge):
1450
492
template = '%s TREE\n%s%s\n%s%s MERGE-SOURCE\n'
1451
493
return template % ('<' * 7, tree, '=' * 7, merge, '>' * 7)
1454
496
class TestTransformMerge(TestCaseInTempDir):
1456
497
def test_text_merge(self):
1457
root_id = generate_ids.gen_root_id()
1458
base = TransformGroup("base", root_id)
498
base = TransformGroup("base")
1459
499
base.tt.new_file('a', base.root, 'a\nb\nc\nd\be\n', 'a')
1460
500
base.tt.new_file('b', base.root, 'b1', 'b')
1461
501
base.tt.new_file('c', base.root, 'c', 'c')
1650
684
a.add(['foo', 'foo/bar', 'foo/baz'])
1651
685
a.commit('initial commit')
1652
686
b = BzrDir.create_standalone_workingtree('b')
1653
basis = a.basis_tree()
1655
self.addCleanup(basis.unlock)
1656
build_tree(basis, b)
687
build_tree(a.basis_tree(), b)
1657
688
self.assertIs(os.path.isdir('b/foo'), True)
1658
689
self.assertEqual(file('b/foo/bar', 'rb').read(), "contents")
1659
690
self.assertEqual(os.readlink('b/foo/baz'), 'a/foo/bar')
1661
def test_build_with_references(self):
1662
tree = self.make_branch_and_tree('source',
1663
format='dirstate-with-subtree')
1664
subtree = self.make_branch_and_tree('source/subtree',
1665
format='dirstate-with-subtree')
1666
tree.add_reference(subtree)
1667
tree.commit('a revision')
1668
tree.branch.create_checkout('target')
1669
self.failUnlessExists('target')
1670
self.failUnlessExists('target/subtree')
1672
def test_file_conflict_handling(self):
1673
"""Ensure that when building trees, conflict handling is done"""
1674
source = self.make_branch_and_tree('source')
1675
target = self.make_branch_and_tree('target')
1676
self.build_tree(['source/file', 'target/file'])
1677
source.add('file', 'new-file')
1678
source.commit('added file')
1679
build_tree(source.basis_tree(), target)
1680
self.assertEqual([DuplicateEntry('Moved existing file to',
1681
'file.moved', 'file', None, 'new-file')],
1683
target2 = self.make_branch_and_tree('target2')
1684
target_file = file('target2/file', 'wb')
1686
source_file = file('source/file', 'rb')
1688
target_file.write(source_file.read())
1693
build_tree(source.basis_tree(), target2)
1694
self.assertEqual([], target2.conflicts())
1696
def test_symlink_conflict_handling(self):
1697
"""Ensure that when building trees, conflict handling is done"""
1698
self.requireFeature(SymlinkFeature)
1699
source = self.make_branch_and_tree('source')
1700
os.symlink('foo', 'source/symlink')
1701
source.add('symlink', 'new-symlink')
1702
source.commit('added file')
1703
target = self.make_branch_and_tree('target')
1704
os.symlink('bar', 'target/symlink')
1705
build_tree(source.basis_tree(), target)
1706
self.assertEqual([DuplicateEntry('Moved existing file to',
1707
'symlink.moved', 'symlink', None, 'new-symlink')],
1709
target = self.make_branch_and_tree('target2')
1710
os.symlink('foo', 'target2/symlink')
1711
build_tree(source.basis_tree(), target)
1712
self.assertEqual([], target.conflicts())
1714
def test_directory_conflict_handling(self):
1715
"""Ensure that when building trees, conflict handling is done"""
1716
source = self.make_branch_and_tree('source')
1717
target = self.make_branch_and_tree('target')
1718
self.build_tree(['source/dir1/', 'source/dir1/file', 'target/dir1/'])
1719
source.add(['dir1', 'dir1/file'], ['new-dir1', 'new-file'])
1720
source.commit('added file')
1721
build_tree(source.basis_tree(), target)
1722
self.assertEqual([], target.conflicts())
1723
self.failUnlessExists('target/dir1/file')
1725
# Ensure contents are merged
1726
target = self.make_branch_and_tree('target2')
1727
self.build_tree(['target2/dir1/', 'target2/dir1/file2'])
1728
build_tree(source.basis_tree(), target)
1729
self.assertEqual([], target.conflicts())
1730
self.failUnlessExists('target2/dir1/file2')
1731
self.failUnlessExists('target2/dir1/file')
1733
# Ensure new contents are suppressed for existing branches
1734
target = self.make_branch_and_tree('target3')
1735
self.make_branch('target3/dir1')
1736
self.build_tree(['target3/dir1/file2'])
1737
build_tree(source.basis_tree(), target)
1738
self.failIfExists('target3/dir1/file')
1739
self.failUnlessExists('target3/dir1/file2')
1740
self.failUnlessExists('target3/dir1.diverted/file')
1741
self.assertEqual([DuplicateEntry('Diverted to',
1742
'dir1.diverted', 'dir1', 'new-dir1', None)],
1745
target = self.make_branch_and_tree('target4')
1746
self.build_tree(['target4/dir1/'])
1747
self.make_branch('target4/dir1/file')
1748
build_tree(source.basis_tree(), target)
1749
self.failUnlessExists('target4/dir1/file')
1750
self.assertEqual('directory', file_kind('target4/dir1/file'))
1751
self.failUnlessExists('target4/dir1/file.diverted')
1752
self.assertEqual([DuplicateEntry('Diverted to',
1753
'dir1/file.diverted', 'dir1/file', 'new-file', None)],
1756
def test_mixed_conflict_handling(self):
1757
"""Ensure that when building trees, conflict handling is done"""
1758
source = self.make_branch_and_tree('source')
1759
target = self.make_branch_and_tree('target')
1760
self.build_tree(['source/name', 'target/name/'])
1761
source.add('name', 'new-name')
1762
source.commit('added file')
1763
build_tree(source.basis_tree(), target)
1764
self.assertEqual([DuplicateEntry('Moved existing file to',
1765
'name.moved', 'name', None, 'new-name')], target.conflicts())
1767
def test_raises_in_populated(self):
1768
source = self.make_branch_and_tree('source')
1769
self.build_tree(['source/name'])
1771
source.commit('added name')
1772
target = self.make_branch_and_tree('target')
1773
self.build_tree(['target/name'])
1775
self.assertRaises(errors.WorkingTreeAlreadyPopulated,
1776
build_tree, source.basis_tree(), target)
1778
def test_build_tree_rename_count(self):
1779
source = self.make_branch_and_tree('source')
1780
self.build_tree(['source/file1', 'source/dir1/'])
1781
source.add(['file1', 'dir1'])
1782
source.commit('add1')
1783
target1 = self.make_branch_and_tree('target1')
1784
transform_result = build_tree(source.basis_tree(), target1)
1785
self.assertEqual(2, transform_result.rename_count)
1787
self.build_tree(['source/dir1/file2'])
1788
source.add(['dir1/file2'])
1789
source.commit('add3')
1790
target2 = self.make_branch_and_tree('target2')
1791
transform_result = build_tree(source.basis_tree(), target2)
1792
# children of non-root directories should not be renamed
1793
self.assertEqual(2, transform_result.rename_count)
1795
def create_ab_tree(self):
1796
"""Create a committed test tree with two files"""
1797
source = self.make_branch_and_tree('source')
1798
self.build_tree_contents([('source/file1', 'A')])
1799
self.build_tree_contents([('source/file2', 'B')])
1800
source.add(['file1', 'file2'], ['file1-id', 'file2-id'])
1801
source.commit('commit files')
1803
self.addCleanup(source.unlock)
1806
def test_build_tree_accelerator_tree(self):
1807
source = self.create_ab_tree()
1808
self.build_tree_contents([('source/file2', 'C')])
1810
real_source_get_file = source.get_file
1811
def get_file(file_id, path=None):
1812
calls.append(file_id)
1813
return real_source_get_file(file_id, path)
1814
source.get_file = get_file
1815
target = self.make_branch_and_tree('target')
1816
revision_tree = source.basis_tree()
1817
revision_tree.lock_read()
1818
self.addCleanup(revision_tree.unlock)
1819
build_tree(revision_tree, target, source)
1820
self.assertEqual(['file1-id'], calls)
1822
self.addCleanup(target.unlock)
1823
self.assertEqual([], list(target.iter_changes(revision_tree)))
1825
def test_build_tree_accelerator_tree_missing_file(self):
1826
source = self.create_ab_tree()
1827
os.unlink('source/file1')
1828
source.remove(['file2'])
1829
target = self.make_branch_and_tree('target')
1830
revision_tree = source.basis_tree()
1831
revision_tree.lock_read()
1832
self.addCleanup(revision_tree.unlock)
1833
build_tree(revision_tree, target, source)
1835
self.addCleanup(target.unlock)
1836
self.assertEqual([], list(target.iter_changes(revision_tree)))
1838
def test_build_tree_accelerator_wrong_kind(self):
1839
self.requireFeature(SymlinkFeature)
1840
source = self.make_branch_and_tree('source')
1841
self.build_tree_contents([('source/file1', '')])
1842
self.build_tree_contents([('source/file2', '')])
1843
source.add(['file1', 'file2'], ['file1-id', 'file2-id'])
1844
source.commit('commit files')
1845
os.unlink('source/file2')
1846
self.build_tree_contents([('source/file2/', 'C')])
1847
os.unlink('source/file1')
1848
os.symlink('file2', 'source/file1')
1850
real_source_get_file = source.get_file
1851
def get_file(file_id, path=None):
1852
calls.append(file_id)
1853
return real_source_get_file(file_id, path)
1854
source.get_file = get_file
1855
target = self.make_branch_and_tree('target')
1856
revision_tree = source.basis_tree()
1857
revision_tree.lock_read()
1858
self.addCleanup(revision_tree.unlock)
1859
build_tree(revision_tree, target, source)
1860
self.assertEqual([], calls)
1862
self.addCleanup(target.unlock)
1863
self.assertEqual([], list(target.iter_changes(revision_tree)))
1865
def test_build_tree_hardlink(self):
1866
self.requireFeature(HardlinkFeature)
1867
source = self.create_ab_tree()
1868
target = self.make_branch_and_tree('target')
1869
revision_tree = source.basis_tree()
1870
revision_tree.lock_read()
1871
self.addCleanup(revision_tree.unlock)
1872
build_tree(revision_tree, target, source, hardlink=True)
1874
self.addCleanup(target.unlock)
1875
self.assertEqual([], list(target.iter_changes(revision_tree)))
1876
source_stat = os.stat('source/file1')
1877
target_stat = os.stat('target/file1')
1878
self.assertEqual(source_stat, target_stat)
1880
# Explicitly disallowing hardlinks should prevent them.
1881
target2 = self.make_branch_and_tree('target2')
1882
build_tree(revision_tree, target2, source, hardlink=False)
1884
self.addCleanup(target2.unlock)
1885
self.assertEqual([], list(target2.iter_changes(revision_tree)))
1886
source_stat = os.stat('source/file1')
1887
target2_stat = os.stat('target2/file1')
1888
self.assertNotEqual(source_stat, target2_stat)
1890
def test_build_tree_accelerator_tree_moved(self):
1891
source = self.make_branch_and_tree('source')
1892
self.build_tree_contents([('source/file1', 'A')])
1893
source.add(['file1'], ['file1-id'])
1894
source.commit('commit files')
1895
source.rename_one('file1', 'file2')
1897
self.addCleanup(source.unlock)
1898
target = self.make_branch_and_tree('target')
1899
revision_tree = source.basis_tree()
1900
revision_tree.lock_read()
1901
self.addCleanup(revision_tree.unlock)
1902
build_tree(revision_tree, target, source)
1904
self.addCleanup(target.unlock)
1905
self.assertEqual([], list(target.iter_changes(revision_tree)))
1907
def test_build_tree_hardlinks_preserve_execute(self):
1908
self.requireFeature(HardlinkFeature)
1909
source = self.create_ab_tree()
1910
tt = TreeTransform(source)
1911
trans_id = tt.trans_id_tree_file_id('file1-id')
1912
tt.set_executability(True, trans_id)
1914
self.assertTrue(source.is_executable('file1-id'))
1915
target = self.make_branch_and_tree('target')
1916
revision_tree = source.basis_tree()
1917
revision_tree.lock_read()
1918
self.addCleanup(revision_tree.unlock)
1919
build_tree(revision_tree, target, source, hardlink=True)
1921
self.addCleanup(target.unlock)
1922
self.assertEqual([], list(target.iter_changes(revision_tree)))
1923
self.assertTrue(source.is_executable('file1-id'))
1925
def install_rot13_content_filter(self, pattern):
1926
original_registry = filters._reset_registry()
1927
def restore_registry():
1928
filters._reset_registry(original_registry)
1929
self.addCleanup(restore_registry)
1930
def rot13(chunks, context=None):
1931
return [''.join(chunks).encode('rot13')]
1932
rot13filter = filters.ContentFilter(rot13, rot13)
1933
filters.register_filter_stack_map('rot13', {'yes': [rot13filter]}.get)
1934
os.mkdir(self.test_home_dir + '/.bazaar')
1935
rules_filename = self.test_home_dir + '/.bazaar/rules'
1936
f = open(rules_filename, 'wb')
1937
f.write('[name %s]\nrot13=yes\n' % (pattern,))
1939
def uninstall_rules():
1940
os.remove(rules_filename)
1942
self.addCleanup(uninstall_rules)
1945
def test_build_tree_content_filtered_files_are_not_hardlinked(self):
1946
"""build_tree will not hardlink files that have content filtering rules
1947
applied to them (but will still hardlink other files from the same tree
1950
self.requireFeature(HardlinkFeature)
1951
self.install_rot13_content_filter('file1')
1952
source = self.create_ab_tree()
1953
target = self.make_branch_and_tree('target')
1954
revision_tree = source.basis_tree()
1955
revision_tree.lock_read()
1956
self.addCleanup(revision_tree.unlock)
1957
build_tree(revision_tree, target, source, hardlink=True)
1959
self.addCleanup(target.unlock)
1960
self.assertEqual([], list(target.iter_changes(revision_tree)))
1961
source_stat = os.stat('source/file1')
1962
target_stat = os.stat('target/file1')
1963
self.assertNotEqual(source_stat, target_stat)
1964
source_stat = os.stat('source/file2')
1965
target_stat = os.stat('target/file2')
1966
self.assertEqualStat(source_stat, target_stat)
1968
def test_case_insensitive_build_tree_inventory(self):
1969
if (tests.CaseInsensitiveFilesystemFeature.available()
1970
or tests.CaseInsCasePresFilenameFeature.available()):
1971
raise tests.UnavailableFeature('Fully case sensitive filesystem')
1972
source = self.make_branch_and_tree('source')
1973
self.build_tree(['source/file', 'source/FILE'])
1974
source.add(['file', 'FILE'], ['lower-id', 'upper-id'])
1975
source.commit('added files')
1976
# Don't try this at home, kids!
1977
# Force the tree to report that it is case insensitive
1978
target = self.make_branch_and_tree('target')
1979
target.case_sensitive = False
1980
build_tree(source.basis_tree(), target, source, delta_from_tree=True)
1981
self.assertEqual('file.moved', target.id2path('lower-id'))
1982
self.assertEqual('FILE', target.id2path('upper-id'))
1985
class TestCommitTransform(tests.TestCaseWithTransport):
1987
def get_branch(self):
1988
tree = self.make_branch_and_tree('tree')
1990
self.addCleanup(tree.unlock)
1991
tree.commit('empty commit')
1994
def get_branch_and_transform(self):
1995
branch = self.get_branch()
1996
tt = TransformPreview(branch.basis_tree())
1997
self.addCleanup(tt.finalize)
2000
def test_commit_wrong_basis(self):
2001
branch = self.get_branch()
2002
basis = branch.repository.revision_tree(
2003
_mod_revision.NULL_REVISION)
2004
tt = TransformPreview(basis)
2005
self.addCleanup(tt.finalize)
2006
e = self.assertRaises(ValueError, tt.commit, branch, '')
2007
self.assertEqual('TreeTransform not based on branch basis: null:',
2010
def test_empy_commit(self):
2011
branch, tt = self.get_branch_and_transform()
2012
rev = tt.commit(branch, 'my message')
2013
self.assertEqual(2, branch.revno())
2014
repo = branch.repository
2015
self.assertEqual('my message', repo.get_revision(rev).message)
2017
def test_merge_parents(self):
2018
branch, tt = self.get_branch_and_transform()
2019
rev = tt.commit(branch, 'my message', ['rev1b', 'rev1c'])
2020
self.assertEqual(['rev1b', 'rev1c'],
2021
branch.basis_tree().get_parent_ids()[1:])
2023
def test_first_commit(self):
2024
branch = self.make_branch('branch')
2026
self.addCleanup(branch.unlock)
2027
tt = TransformPreview(branch.basis_tree())
2028
self.addCleanup(tt.finalize)
2029
tt.new_directory('', ROOT_PARENT, 'TREE_ROOT')
2030
rev = tt.commit(branch, 'my message')
2031
self.assertEqual([], branch.basis_tree().get_parent_ids())
2032
self.assertNotEqual(_mod_revision.NULL_REVISION,
2033
branch.last_revision())
2035
def test_first_commit_with_merge_parents(self):
2036
branch = self.make_branch('branch')
2038
self.addCleanup(branch.unlock)
2039
tt = TransformPreview(branch.basis_tree())
2040
self.addCleanup(tt.finalize)
2041
e = self.assertRaises(ValueError, tt.commit, branch,
2042
'my message', ['rev1b-id'])
2043
self.assertEqual('Cannot supply merge parents for first commit.',
2045
self.assertEqual(_mod_revision.NULL_REVISION, branch.last_revision())
2047
def test_add_files(self):
2048
branch, tt = self.get_branch_and_transform()
2049
tt.new_file('file', tt.root, 'contents', 'file-id')
2050
trans_id = tt.new_directory('dir', tt.root, 'dir-id')
2051
if SymlinkFeature.available():
2052
tt.new_symlink('symlink', trans_id, 'target', 'symlink-id')
2053
rev = tt.commit(branch, 'message')
2054
tree = branch.basis_tree()
2055
self.assertEqual('file', tree.id2path('file-id'))
2056
self.assertEqual('contents', tree.get_file_text('file-id'))
2057
self.assertEqual('dir', tree.id2path('dir-id'))
2058
if SymlinkFeature.available():
2059
self.assertEqual('dir/symlink', tree.id2path('symlink-id'))
2060
self.assertEqual('target', tree.get_symlink_target('symlink-id'))
2062
def test_add_unversioned(self):
2063
branch, tt = self.get_branch_and_transform()
2064
tt.new_file('file', tt.root, 'contents')
2065
self.assertRaises(errors.StrictCommitFailed, tt.commit, branch,
2066
'message', strict=True)
2068
def test_modify_strict(self):
2069
branch, tt = self.get_branch_and_transform()
2070
tt.new_file('file', tt.root, 'contents', 'file-id')
2071
tt.commit(branch, 'message', strict=True)
2072
tt = TransformPreview(branch.basis_tree())
2073
self.addCleanup(tt.finalize)
2074
trans_id = tt.trans_id_file_id('file-id')
2075
tt.delete_contents(trans_id)
2076
tt.create_file('contents', trans_id)
2077
tt.commit(branch, 'message', strict=True)
2079
def test_commit_malformed(self):
2080
"""Committing a malformed transform should raise an exception.
2082
In this case, we are adding a file without adding its parent.
2084
branch, tt = self.get_branch_and_transform()
2085
parent_id = tt.trans_id_file_id('parent-id')
2086
tt.new_file('file', parent_id, 'contents', 'file-id')
2087
self.assertRaises(errors.MalformedTransform, tt.commit, branch,
2091
class MockTransform(object):
2093
def has_named_child(self, by_parent, parent_id, name):
2094
for child_id in by_parent[parent_id]:
2098
elif name == "name.~%s~" % child_id:
2103
class MockEntry(object):
2105
object.__init__(self)
2109
class TestGetBackupName(TestCase):
2110
def test_get_backup_name(self):
2111
tt = MockTransform()
2112
name = get_backup_name(MockEntry(), {'a':[]}, 'a', tt)
2113
self.assertEqual(name, 'name.~1~')
2114
name = get_backup_name(MockEntry(), {'a':['1']}, 'a', tt)
2115
self.assertEqual(name, 'name.~2~')
2116
name = get_backup_name(MockEntry(), {'a':['2']}, 'a', tt)
2117
self.assertEqual(name, 'name.~1~')
2118
name = get_backup_name(MockEntry(), {'a':['2'], 'b':[]}, 'b', tt)
2119
self.assertEqual(name, 'name.~1~')
2120
name = get_backup_name(MockEntry(), {'a':['1', '2', '3']}, 'a', tt)
2121
self.assertEqual(name, 'name.~4~')
2124
class TestFileMover(tests.TestCaseWithTransport):
2126
def test_file_mover(self):
2127
self.build_tree(['a/', 'a/b', 'c/', 'c/d'])
2128
mover = _FileMover()
2129
mover.rename('a', 'q')
2130
self.failUnlessExists('q')
2131
self.failIfExists('a')
2132
self.failUnlessExists('q/b')
2133
self.failUnlessExists('c')
2134
self.failUnlessExists('c/d')
2136
def test_pre_delete_rollback(self):
2137
self.build_tree(['a/'])
2138
mover = _FileMover()
2139
mover.pre_delete('a', 'q')
2140
self.failUnlessExists('q')
2141
self.failIfExists('a')
2143
self.failIfExists('q')
2144
self.failUnlessExists('a')
2146
def test_apply_deletions(self):
2147
self.build_tree(['a/', 'b/'])
2148
mover = _FileMover()
2149
mover.pre_delete('a', 'q')
2150
mover.pre_delete('b', 'r')
2151
self.failUnlessExists('q')
2152
self.failUnlessExists('r')
2153
self.failIfExists('a')
2154
self.failIfExists('b')
2155
mover.apply_deletions()
2156
self.failIfExists('q')
2157
self.failIfExists('r')
2158
self.failIfExists('a')
2159
self.failIfExists('b')
2161
def test_file_mover_rollback(self):
2162
self.build_tree(['a/', 'a/b', 'c/', 'c/d/', 'c/e/'])
2163
mover = _FileMover()
2164
mover.rename('c/d', 'c/f')
2165
mover.rename('c/e', 'c/d')
2167
mover.rename('a', 'c')
2168
except errors.FileExists, e:
2170
self.failUnlessExists('a')
2171
self.failUnlessExists('c/d')
2174
class Bogus(Exception):
2178
class TestTransformRollback(tests.TestCaseWithTransport):
2180
class ExceptionFileMover(_FileMover):
2182
def __init__(self, bad_source=None, bad_target=None):
2183
_FileMover.__init__(self)
2184
self.bad_source = bad_source
2185
self.bad_target = bad_target
2187
def rename(self, source, target):
2188
if (self.bad_source is not None and
2189
source.endswith(self.bad_source)):
2191
elif (self.bad_target is not None and
2192
target.endswith(self.bad_target)):
2195
_FileMover.rename(self, source, target)
2197
def test_rollback_rename(self):
2198
tree = self.make_branch_and_tree('.')
2199
self.build_tree(['a/', 'a/b'])
2200
tt = TreeTransform(tree)
2201
self.addCleanup(tt.finalize)
2202
a_id = tt.trans_id_tree_path('a')
2203
tt.adjust_path('c', tt.root, a_id)
2204
tt.adjust_path('d', a_id, tt.trans_id_tree_path('a/b'))
2205
self.assertRaises(Bogus, tt.apply,
2206
_mover=self.ExceptionFileMover(bad_source='a'))
2207
self.failUnlessExists('a')
2208
self.failUnlessExists('a/b')
2210
self.failUnlessExists('c')
2211
self.failUnlessExists('c/d')
2213
def test_rollback_rename_into_place(self):
2214
tree = self.make_branch_and_tree('.')
2215
self.build_tree(['a/', 'a/b'])
2216
tt = TreeTransform(tree)
2217
self.addCleanup(tt.finalize)
2218
a_id = tt.trans_id_tree_path('a')
2219
tt.adjust_path('c', tt.root, a_id)
2220
tt.adjust_path('d', a_id, tt.trans_id_tree_path('a/b'))
2221
self.assertRaises(Bogus, tt.apply,
2222
_mover=self.ExceptionFileMover(bad_target='c/d'))
2223
self.failUnlessExists('a')
2224
self.failUnlessExists('a/b')
2226
self.failUnlessExists('c')
2227
self.failUnlessExists('c/d')
2229
def test_rollback_deletion(self):
2230
tree = self.make_branch_and_tree('.')
2231
self.build_tree(['a/', 'a/b'])
2232
tt = TreeTransform(tree)
2233
self.addCleanup(tt.finalize)
2234
a_id = tt.trans_id_tree_path('a')
2235
tt.delete_contents(a_id)
2236
tt.adjust_path('d', tt.root, tt.trans_id_tree_path('a/b'))
2237
self.assertRaises(Bogus, tt.apply,
2238
_mover=self.ExceptionFileMover(bad_target='d'))
2239
self.failUnlessExists('a')
2240
self.failUnlessExists('a/b')
2242
def test_resolve_no_parent(self):
2243
wt = self.make_branch_and_tree('.')
2244
tt = TreeTransform(wt)
2245
self.addCleanup(tt.finalize)
2246
parent = tt.trans_id_file_id('parent-id')
2247
tt.new_file('file', parent, 'Contents')
2248
resolve_conflicts(tt)
2251
A_ENTRY = ('a-id', ('a', 'a'), True, (True, True),
2252
('TREE_ROOT', 'TREE_ROOT'), ('a', 'a'), ('file', 'file'),
2254
ROOT_ENTRY = ('TREE_ROOT', ('', ''), False, (True, True), (None, None),
2255
('', ''), ('directory', 'directory'), (False, None))
2258
class TestTransformPreview(tests.TestCaseWithTransport):
2260
def create_tree(self):
2261
tree = self.make_branch_and_tree('.')
2262
self.build_tree_contents([('a', 'content 1')])
2263
tree.set_root_id('TREE_ROOT')
2264
tree.add('a', 'a-id')
2265
tree.commit('rev1', rev_id='rev1')
2266
return tree.branch.repository.revision_tree('rev1')
2268
def get_empty_preview(self):
2269
repository = self.make_repository('repo')
2270
tree = repository.revision_tree(_mod_revision.NULL_REVISION)
2271
preview = TransformPreview(tree)
2272
self.addCleanup(preview.finalize)
2275
def test_transform_preview(self):
2276
revision_tree = self.create_tree()
2277
preview = TransformPreview(revision_tree)
2278
self.addCleanup(preview.finalize)
2280
def test_transform_preview_tree(self):
2281
revision_tree = self.create_tree()
2282
preview = TransformPreview(revision_tree)
2283
self.addCleanup(preview.finalize)
2284
preview.get_preview_tree()
2286
def test_transform_new_file(self):
2287
revision_tree = self.create_tree()
2288
preview = TransformPreview(revision_tree)
2289
self.addCleanup(preview.finalize)
2290
preview.new_file('file2', preview.root, 'content B\n', 'file2-id')
2291
preview_tree = preview.get_preview_tree()
2292
self.assertEqual(preview_tree.kind('file2-id'), 'file')
2294
preview_tree.get_file('file2-id').read(), 'content B\n')
2296
def test_diff_preview_tree(self):
2297
revision_tree = self.create_tree()
2298
preview = TransformPreview(revision_tree)
2299
self.addCleanup(preview.finalize)
2300
preview.new_file('file2', preview.root, 'content B\n', 'file2-id')
2301
preview_tree = preview.get_preview_tree()
2303
show_diff_trees(revision_tree, preview_tree, out)
2304
lines = out.getvalue().splitlines()
2305
self.assertEqual(lines[0], "=== added file 'file2'")
2306
# 3 lines of diff administrivia
2307
self.assertEqual(lines[4], "+content B")
2309
def test_transform_conflicts(self):
2310
revision_tree = self.create_tree()
2311
preview = TransformPreview(revision_tree)
2312
self.addCleanup(preview.finalize)
2313
preview.new_file('a', preview.root, 'content 2')
2314
resolve_conflicts(preview)
2315
trans_id = preview.trans_id_file_id('a-id')
2316
self.assertEqual('a.moved', preview.final_name(trans_id))
2318
def get_tree_and_preview_tree(self):
2319
revision_tree = self.create_tree()
2320
preview = TransformPreview(revision_tree)
2321
self.addCleanup(preview.finalize)
2322
a_trans_id = preview.trans_id_file_id('a-id')
2323
preview.delete_contents(a_trans_id)
2324
preview.create_file('b content', a_trans_id)
2325
preview_tree = preview.get_preview_tree()
2326
return revision_tree, preview_tree
2328
def test_iter_changes(self):
2329
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2330
root = revision_tree.inventory.root.file_id
2331
self.assertEqual([('a-id', ('a', 'a'), True, (True, True),
2332
(root, root), ('a', 'a'), ('file', 'file'),
2334
list(preview_tree.iter_changes(revision_tree)))
2336
def test_include_unchanged_succeeds(self):
2337
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2338
changes = preview_tree.iter_changes(revision_tree,
2339
include_unchanged=True)
2340
root = revision_tree.inventory.root.file_id
2342
self.assertEqual([ROOT_ENTRY, A_ENTRY], list(changes))
2344
def test_specific_files(self):
2345
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2346
changes = preview_tree.iter_changes(revision_tree,
2347
specific_files=[''])
2348
self.assertEqual([ROOT_ENTRY, A_ENTRY], list(changes))
2350
def test_want_unversioned(self):
2351
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2352
changes = preview_tree.iter_changes(revision_tree,
2353
want_unversioned=True)
2354
self.assertEqual([ROOT_ENTRY, A_ENTRY], list(changes))
2356
def test_ignore_extra_trees_no_specific_files(self):
2357
# extra_trees is harmless without specific_files, so we'll silently
2358
# accept it, even though we won't use it.
2359
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2360
preview_tree.iter_changes(revision_tree, extra_trees=[preview_tree])
2362
def test_ignore_require_versioned_no_specific_files(self):
2363
# require_versioned is meaningless without specific_files.
2364
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2365
preview_tree.iter_changes(revision_tree, require_versioned=False)
2367
def test_ignore_pb(self):
2368
# pb could be supported, but TT.iter_changes doesn't support it.
2369
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2370
preview_tree.iter_changes(revision_tree, pb=progress.DummyProgress())
2372
def test_kind(self):
2373
revision_tree = self.create_tree()
2374
preview = TransformPreview(revision_tree)
2375
self.addCleanup(preview.finalize)
2376
preview.new_file('file', preview.root, 'contents', 'file-id')
2377
preview.new_directory('directory', preview.root, 'dir-id')
2378
preview_tree = preview.get_preview_tree()
2379
self.assertEqual('file', preview_tree.kind('file-id'))
2380
self.assertEqual('directory', preview_tree.kind('dir-id'))
2382
def test_get_file_mtime(self):
2383
preview = self.get_empty_preview()
2384
file_trans_id = preview.new_file('file', preview.root, 'contents',
2386
limbo_path = preview._limbo_name(file_trans_id)
2387
preview_tree = preview.get_preview_tree()
2388
self.assertEqual(os.stat(limbo_path).st_mtime,
2389
preview_tree.get_file_mtime('file-id'))
2391
def test_get_file_mtime_renamed(self):
2392
work_tree = self.make_branch_and_tree('tree')
2393
self.build_tree(['tree/file'])
2394
work_tree.add('file', 'file-id')
2395
preview = TransformPreview(work_tree)
2396
self.addCleanup(preview.finalize)
2397
file_trans_id = preview.trans_id_tree_file_id('file-id')
2398
preview.adjust_path('renamed', preview.root, file_trans_id)
2399
preview_tree = preview.get_preview_tree()
2400
preview_mtime = preview_tree.get_file_mtime('file-id', 'renamed')
2401
work_mtime = work_tree.get_file_mtime('file-id', 'file')
2403
def test_get_file(self):
2404
preview = self.get_empty_preview()
2405
preview.new_file('file', preview.root, 'contents', 'file-id')
2406
preview_tree = preview.get_preview_tree()
2407
tree_file = preview_tree.get_file('file-id')
2409
self.assertEqual('contents', tree_file.read())
2413
def test_get_symlink_target(self):
2414
self.requireFeature(SymlinkFeature)
2415
preview = self.get_empty_preview()
2416
preview.new_symlink('symlink', preview.root, 'target', 'symlink-id')
2417
preview_tree = preview.get_preview_tree()
2418
self.assertEqual('target',
2419
preview_tree.get_symlink_target('symlink-id'))
2421
def test_all_file_ids(self):
2422
tree = self.make_branch_and_tree('tree')
2423
self.build_tree(['tree/a', 'tree/b', 'tree/c'])
2424
tree.add(['a', 'b', 'c'], ['a-id', 'b-id', 'c-id'])
2425
preview = TransformPreview(tree)
2426
self.addCleanup(preview.finalize)
2427
preview.unversion_file(preview.trans_id_file_id('b-id'))
2428
c_trans_id = preview.trans_id_file_id('c-id')
2429
preview.unversion_file(c_trans_id)
2430
preview.version_file('c-id', c_trans_id)
2431
preview_tree = preview.get_preview_tree()
2432
self.assertEqual(set(['a-id', 'c-id', tree.get_root_id()]),
2433
preview_tree.all_file_ids())
2435
def test_path2id_deleted_unchanged(self):
2436
tree = self.make_branch_and_tree('tree')
2437
self.build_tree(['tree/unchanged', 'tree/deleted'])
2438
tree.add(['unchanged', 'deleted'], ['unchanged-id', 'deleted-id'])
2439
preview = TransformPreview(tree)
2440
self.addCleanup(preview.finalize)
2441
preview.unversion_file(preview.trans_id_file_id('deleted-id'))
2442
preview_tree = preview.get_preview_tree()
2443
self.assertEqual('unchanged-id', preview_tree.path2id('unchanged'))
2444
self.assertIs(None, preview_tree.path2id('deleted'))
2446
def test_path2id_created(self):
2447
tree = self.make_branch_and_tree('tree')
2448
self.build_tree(['tree/unchanged'])
2449
tree.add(['unchanged'], ['unchanged-id'])
2450
preview = TransformPreview(tree)
2451
self.addCleanup(preview.finalize)
2452
preview.new_file('new', preview.trans_id_file_id('unchanged-id'),
2453
'contents', 'new-id')
2454
preview_tree = preview.get_preview_tree()
2455
self.assertEqual('new-id', preview_tree.path2id('unchanged/new'))
2457
def test_path2id_moved(self):
2458
tree = self.make_branch_and_tree('tree')
2459
self.build_tree(['tree/old_parent/', 'tree/old_parent/child'])
2460
tree.add(['old_parent', 'old_parent/child'],
2461
['old_parent-id', 'child-id'])
2462
preview = TransformPreview(tree)
2463
self.addCleanup(preview.finalize)
2464
new_parent = preview.new_directory('new_parent', preview.root,
2466
preview.adjust_path('child', new_parent,
2467
preview.trans_id_file_id('child-id'))
2468
preview_tree = preview.get_preview_tree()
2469
self.assertIs(None, preview_tree.path2id('old_parent/child'))
2470
self.assertEqual('child-id', preview_tree.path2id('new_parent/child'))
2472
def test_path2id_renamed_parent(self):
2473
tree = self.make_branch_and_tree('tree')
2474
self.build_tree(['tree/old_name/', 'tree/old_name/child'])
2475
tree.add(['old_name', 'old_name/child'],
2476
['parent-id', 'child-id'])
2477
preview = TransformPreview(tree)
2478
self.addCleanup(preview.finalize)
2479
preview.adjust_path('new_name', preview.root,
2480
preview.trans_id_file_id('parent-id'))
2481
preview_tree = preview.get_preview_tree()
2482
self.assertIs(None, preview_tree.path2id('old_name/child'))
2483
self.assertEqual('child-id', preview_tree.path2id('new_name/child'))
2485
def assertMatchingIterEntries(self, tt, specific_file_ids=None):
2486
preview_tree = tt.get_preview_tree()
2487
preview_result = list(preview_tree.iter_entries_by_dir(
2491
actual_result = list(tree.iter_entries_by_dir(specific_file_ids))
2492
self.assertEqual(actual_result, preview_result)
2494
def test_iter_entries_by_dir_new(self):
2495
tree = self.make_branch_and_tree('tree')
2496
tt = TreeTransform(tree)
2497
tt.new_file('new', tt.root, 'contents', 'new-id')
2498
self.assertMatchingIterEntries(tt)
2500
def test_iter_entries_by_dir_deleted(self):
2501
tree = self.make_branch_and_tree('tree')
2502
self.build_tree(['tree/deleted'])
2503
tree.add('deleted', 'deleted-id')
2504
tt = TreeTransform(tree)
2505
tt.delete_contents(tt.trans_id_file_id('deleted-id'))
2506
self.assertMatchingIterEntries(tt)
2508
def test_iter_entries_by_dir_unversioned(self):
2509
tree = self.make_branch_and_tree('tree')
2510
self.build_tree(['tree/removed'])
2511
tree.add('removed', 'removed-id')
2512
tt = TreeTransform(tree)
2513
tt.unversion_file(tt.trans_id_file_id('removed-id'))
2514
self.assertMatchingIterEntries(tt)
2516
def test_iter_entries_by_dir_moved(self):
2517
tree = self.make_branch_and_tree('tree')
2518
self.build_tree(['tree/moved', 'tree/new_parent/'])
2519
tree.add(['moved', 'new_parent'], ['moved-id', 'new_parent-id'])
2520
tt = TreeTransform(tree)
2521
tt.adjust_path('moved', tt.trans_id_file_id('new_parent-id'),
2522
tt.trans_id_file_id('moved-id'))
2523
self.assertMatchingIterEntries(tt)
2525
def test_iter_entries_by_dir_specific_file_ids(self):
2526
tree = self.make_branch_and_tree('tree')
2527
tree.set_root_id('tree-root-id')
2528
self.build_tree(['tree/parent/', 'tree/parent/child'])
2529
tree.add(['parent', 'parent/child'], ['parent-id', 'child-id'])
2530
tt = TreeTransform(tree)
2531
self.assertMatchingIterEntries(tt, ['tree-root-id', 'child-id'])
2533
def test_symlink_content_summary(self):
2534
self.requireFeature(SymlinkFeature)
2535
preview = self.get_empty_preview()
2536
preview.new_symlink('path', preview.root, 'target', 'path-id')
2537
summary = preview.get_preview_tree().path_content_summary('path')
2538
self.assertEqual(('symlink', None, None, 'target'), summary)
2540
def test_missing_content_summary(self):
2541
preview = self.get_empty_preview()
2542
summary = preview.get_preview_tree().path_content_summary('path')
2543
self.assertEqual(('missing', None, None, None), summary)
2545
def test_deleted_content_summary(self):
2546
tree = self.make_branch_and_tree('tree')
2547
self.build_tree(['tree/path/'])
2549
preview = TransformPreview(tree)
2550
self.addCleanup(preview.finalize)
2551
preview.delete_contents(preview.trans_id_tree_path('path'))
2552
summary = preview.get_preview_tree().path_content_summary('path')
2553
self.assertEqual(('missing', None, None, None), summary)
2555
def test_file_content_summary_executable(self):
2556
preview = self.get_empty_preview()
2557
path_id = preview.new_file('path', preview.root, 'contents', 'path-id')
2558
preview.set_executability(True, path_id)
2559
summary = preview.get_preview_tree().path_content_summary('path')
2560
self.assertEqual(4, len(summary))
2561
self.assertEqual('file', summary[0])
2562
# size must be known
2563
self.assertEqual(len('contents'), summary[1])
2565
self.assertEqual(True, summary[2])
2566
# will not have hash (not cheap to determine)
2567
self.assertIs(None, summary[3])
2569
def test_change_executability(self):
2570
tree = self.make_branch_and_tree('tree')
2571
self.build_tree(['tree/path'])
2573
preview = TransformPreview(tree)
2574
self.addCleanup(preview.finalize)
2575
path_id = preview.trans_id_tree_path('path')
2576
preview.set_executability(True, path_id)
2577
summary = preview.get_preview_tree().path_content_summary('path')
2578
self.assertEqual(True, summary[2])
2580
def test_file_content_summary_non_exec(self):
2581
preview = self.get_empty_preview()
2582
preview.new_file('path', preview.root, 'contents', 'path-id')
2583
summary = preview.get_preview_tree().path_content_summary('path')
2584
self.assertEqual(4, len(summary))
2585
self.assertEqual('file', summary[0])
2586
# size must be known
2587
self.assertEqual(len('contents'), summary[1])
2589
self.assertEqual(False, summary[2])
2590
# will not have hash (not cheap to determine)
2591
self.assertIs(None, summary[3])
2593
def test_dir_content_summary(self):
2594
preview = self.get_empty_preview()
2595
preview.new_directory('path', preview.root, 'path-id')
2596
summary = preview.get_preview_tree().path_content_summary('path')
2597
self.assertEqual(('directory', None, None, None), summary)
2599
def test_tree_content_summary(self):
2600
preview = self.get_empty_preview()
2601
path = preview.new_directory('path', preview.root, 'path-id')
2602
preview.set_tree_reference('rev-1', path)
2603
summary = preview.get_preview_tree().path_content_summary('path')
2604
self.assertEqual(4, len(summary))
2605
self.assertEqual('tree-reference', summary[0])
2607
def test_annotate(self):
2608
tree = self.make_branch_and_tree('tree')
2609
self.build_tree_contents([('tree/file', 'a\n')])
2610
tree.add('file', 'file-id')
2611
tree.commit('a', rev_id='one')
2612
self.build_tree_contents([('tree/file', 'a\nb\n')])
2613
preview = TransformPreview(tree)
2614
self.addCleanup(preview.finalize)
2615
file_trans_id = preview.trans_id_file_id('file-id')
2616
preview.delete_contents(file_trans_id)
2617
preview.create_file('a\nb\nc\n', file_trans_id)
2618
preview_tree = preview.get_preview_tree()
2624
annotation = preview_tree.annotate_iter('file-id', 'me:')
2625
self.assertEqual(expected, annotation)
2627
def test_annotate_missing(self):
2628
preview = self.get_empty_preview()
2629
preview.new_file('file', preview.root, 'a\nb\nc\n', 'file-id')
2630
preview_tree = preview.get_preview_tree()
2636
annotation = preview_tree.annotate_iter('file-id', 'me:')
2637
self.assertEqual(expected, annotation)
2639
def test_annotate_rename(self):
2640
tree = self.make_branch_and_tree('tree')
2641
self.build_tree_contents([('tree/file', 'a\n')])
2642
tree.add('file', 'file-id')
2643
tree.commit('a', rev_id='one')
2644
preview = TransformPreview(tree)
2645
self.addCleanup(preview.finalize)
2646
file_trans_id = preview.trans_id_file_id('file-id')
2647
preview.adjust_path('newname', preview.root, file_trans_id)
2648
preview_tree = preview.get_preview_tree()
2652
annotation = preview_tree.annotate_iter('file-id', 'me:')
2653
self.assertEqual(expected, annotation)
2655
def test_annotate_deleted(self):
2656
tree = self.make_branch_and_tree('tree')
2657
self.build_tree_contents([('tree/file', 'a\n')])
2658
tree.add('file', 'file-id')
2659
tree.commit('a', rev_id='one')
2660
self.build_tree_contents([('tree/file', 'a\nb\n')])
2661
preview = TransformPreview(tree)
2662
self.addCleanup(preview.finalize)
2663
file_trans_id = preview.trans_id_file_id('file-id')
2664
preview.delete_contents(file_trans_id)
2665
preview_tree = preview.get_preview_tree()
2666
annotation = preview_tree.annotate_iter('file-id', 'me:')
2667
self.assertIs(None, annotation)
2669
def test_stored_kind(self):
2670
preview = self.get_empty_preview()
2671
preview.new_file('file', preview.root, 'a\nb\nc\n', 'file-id')
2672
preview_tree = preview.get_preview_tree()
2673
self.assertEqual('file', preview_tree.stored_kind('file-id'))
2675
def test_is_executable(self):
2676
preview = self.get_empty_preview()
2677
preview.new_file('file', preview.root, 'a\nb\nc\n', 'file-id')
2678
preview.set_executability(True, preview.trans_id_file_id('file-id'))
2679
preview_tree = preview.get_preview_tree()
2680
self.assertEqual(True, preview_tree.is_executable('file-id'))
2682
def test_get_set_parent_ids(self):
2683
revision_tree, preview_tree = self.get_tree_and_preview_tree()
2684
self.assertEqual([], preview_tree.get_parent_ids())
2685
preview_tree.set_parent_ids(['rev-1'])
2686
self.assertEqual(['rev-1'], preview_tree.get_parent_ids())
2688
def test_plan_file_merge(self):
2689
work_a = self.make_branch_and_tree('wta')
2690
self.build_tree_contents([('wta/file', 'a\nb\nc\nd\n')])
2691
work_a.add('file', 'file-id')
2692
base_id = work_a.commit('base version')
2693
tree_b = work_a.bzrdir.sprout('wtb').open_workingtree()
2694
preview = TransformPreview(work_a)
2695
self.addCleanup(preview.finalize)
2696
trans_id = preview.trans_id_file_id('file-id')
2697
preview.delete_contents(trans_id)
2698
preview.create_file('b\nc\nd\ne\n', trans_id)
2699
self.build_tree_contents([('wtb/file', 'a\nc\nd\nf\n')])
2700
tree_a = preview.get_preview_tree()
2701
tree_a.set_parent_ids([base_id])
2703
('killed-a', 'a\n'),
2704
('killed-b', 'b\n'),
2705
('unchanged', 'c\n'),
2706
('unchanged', 'd\n'),
2709
], list(tree_a.plan_file_merge('file-id', tree_b)))
2711
def test_plan_file_merge_revision_tree(self):
2712
work_a = self.make_branch_and_tree('wta')
2713
self.build_tree_contents([('wta/file', 'a\nb\nc\nd\n')])
2714
work_a.add('file', 'file-id')
2715
base_id = work_a.commit('base version')
2716
tree_b = work_a.bzrdir.sprout('wtb').open_workingtree()
2717
preview = TransformPreview(work_a.basis_tree())
2718
self.addCleanup(preview.finalize)
2719
trans_id = preview.trans_id_file_id('file-id')
2720
preview.delete_contents(trans_id)
2721
preview.create_file('b\nc\nd\ne\n', trans_id)
2722
self.build_tree_contents([('wtb/file', 'a\nc\nd\nf\n')])
2723
tree_a = preview.get_preview_tree()
2724
tree_a.set_parent_ids([base_id])
2726
('killed-a', 'a\n'),
2727
('killed-b', 'b\n'),
2728
('unchanged', 'c\n'),
2729
('unchanged', 'd\n'),
2732
], list(tree_a.plan_file_merge('file-id', tree_b)))
2734
def test_walkdirs(self):
2735
preview = self.get_empty_preview()
2736
root = preview.new_directory('', ROOT_PARENT, 'tree-root')
2737
# FIXME: new_directory should mark root.
2738
preview.fixup_new_roots()
2739
preview_tree = preview.get_preview_tree()
2740
file_trans_id = preview.new_file('a', preview.root, 'contents',
2742
expected = [(('', 'tree-root'),
2743
[('a', 'a', 'file', None, 'a-id', 'file')])]
2744
self.assertEqual(expected, list(preview_tree.walkdirs()))
2746
def test_extras(self):
2747
work_tree = self.make_branch_and_tree('tree')
2748
self.build_tree(['tree/removed-file', 'tree/existing-file',
2749
'tree/not-removed-file'])
2750
work_tree.add(['removed-file', 'not-removed-file'])
2751
preview = TransformPreview(work_tree)
2752
self.addCleanup(preview.finalize)
2753
preview.new_file('new-file', preview.root, 'contents')
2754
preview.new_file('new-versioned-file', preview.root, 'contents',
2756
tree = preview.get_preview_tree()
2757
preview.unversion_file(preview.trans_id_tree_path('removed-file'))
2758
self.assertEqual(set(['new-file', 'removed-file', 'existing-file']),
2761
def test_merge_into_preview(self):
2762
work_tree = self.make_branch_and_tree('tree')
2763
self.build_tree_contents([('tree/file','b\n')])
2764
work_tree.add('file', 'file-id')
2765
work_tree.commit('first commit')
2766
child_tree = work_tree.bzrdir.sprout('child').open_workingtree()
2767
self.build_tree_contents([('child/file','b\nc\n')])
2768
child_tree.commit('child commit')
2769
child_tree.lock_write()
2770
self.addCleanup(child_tree.unlock)
2771
work_tree.lock_write()
2772
self.addCleanup(work_tree.unlock)
2773
preview = TransformPreview(work_tree)
2774
self.addCleanup(preview.finalize)
2775
file_trans_id = preview.trans_id_file_id('file-id')
2776
preview.delete_contents(file_trans_id)
2777
preview.create_file('a\nb\n', file_trans_id)
2778
pb = progress.DummyProgress()
2779
preview_tree = preview.get_preview_tree()
2780
merger = Merger.from_revision_ids(pb, preview_tree,
2781
child_tree.branch.last_revision(),
2782
other_branch=child_tree.branch,
2783
tree_branch=work_tree.branch)
2784
merger.merge_type = Merge3Merger
2785
tt = merger.make_merger().make_preview_transform()
2786
self.addCleanup(tt.finalize)
2787
final_tree = tt.get_preview_tree()
2788
self.assertEqual('a\nb\nc\n', final_tree.get_file_text('file-id'))
2790
def test_merge_preview_into_workingtree(self):
2791
tree = self.make_branch_and_tree('tree')
2792
tree.set_root_id('TREE_ROOT')
2793
tt = TransformPreview(tree)
2794
self.addCleanup(tt.finalize)
2795
tt.new_file('name', tt.root, 'content', 'file-id')
2796
tree2 = self.make_branch_and_tree('tree2')
2797
tree2.set_root_id('TREE_ROOT')
2798
pb = progress.DummyProgress()
2799
merger = Merger.from_uncommitted(tree2, tt.get_preview_tree(),
2800
pb, tree.basis_tree())
2801
merger.merge_type = Merge3Merger
2804
def test_merge_preview_into_workingtree_handles_conflicts(self):
2805
tree = self.make_branch_and_tree('tree')
2806
self.build_tree_contents([('tree/foo', 'bar')])
2807
tree.add('foo', 'foo-id')
2809
tt = TransformPreview(tree)
2810
self.addCleanup(tt.finalize)
2811
trans_id = tt.trans_id_file_id('foo-id')
2812
tt.delete_contents(trans_id)
2813
tt.create_file('baz', trans_id)
2814
tree2 = tree.bzrdir.sprout('tree2').open_workingtree()
2815
self.build_tree_contents([('tree2/foo', 'qux')])
2816
pb = progress.DummyProgress()
2817
merger = Merger.from_uncommitted(tree2, tt.get_preview_tree(),
2818
pb, tree.basis_tree())
2819
merger.merge_type = Merge3Merger
2822
def test_is_executable(self):
2823
tree = self.make_branch_and_tree('tree')
2824
preview = TransformPreview(tree)
2825
self.addCleanup(preview.finalize)
2826
preview.new_file('foo', preview.root, 'bar', 'baz-id')
2827
preview_tree = preview.get_preview_tree()
2828
self.assertEqual(False, preview_tree.is_executable('baz-id',
2830
self.assertEqual(False, preview_tree.is_executable('baz-id'))
2832
def test_commit_preview_tree(self):
2833
tree = self.make_branch_and_tree('tree')
2834
rev_id = tree.commit('rev1')
2835
tree.branch.lock_write()
2836
self.addCleanup(tree.branch.unlock)
2837
tt = TransformPreview(tree)
2838
tt.new_file('file', tt.root, 'contents', 'file_id')
2839
self.addCleanup(tt.finalize)
2840
preview = tt.get_preview_tree()
2841
preview.set_parent_ids([rev_id])
2842
builder = tree.branch.get_commit_builder([rev_id])
2843
list(builder.record_iter_changes(preview, rev_id, tt.iter_changes()))
2844
builder.finish_inventory()
2845
rev2_id = builder.commit('rev2')
2846
rev2_tree = tree.branch.repository.revision_tree(rev2_id)
2847
self.assertEqual('contents', rev2_tree.get_file_text('file_id'))
2849
def test_ascii_limbo_paths(self):
2850
self.requireFeature(tests.UnicodeFilenameFeature)
2851
branch = self.make_branch('any')
2852
tree = branch.repository.revision_tree(_mod_revision.NULL_REVISION)
2853
tt = TransformPreview(tree)
2854
self.addCleanup(tt.finalize)
2855
foo_id = tt.new_directory('', ROOT_PARENT)
2856
bar_id = tt.new_file(u'\u1234bar', foo_id, 'contents')
2857
limbo_path = tt._limbo_name(bar_id)
2858
self.assertEqual(limbo_path.encode('ascii', 'replace'), limbo_path)
2861
class FakeSerializer(object):
2862
"""Serializer implementation that simply returns the input.
2864
The input is returned in the order used by pack.ContainerPushParser.
2867
def bytes_record(bytes, names):
2871
class TestSerializeTransform(tests.TestCaseWithTransport):
2873
_test_needs_features = [tests.UnicodeFilenameFeature]
2875
def get_preview(self, tree=None):
2877
tree = self.make_branch_and_tree('tree')
2878
tt = TransformPreview(tree)
2879
self.addCleanup(tt.finalize)
2882
def assertSerializesTo(self, expected, tt):
2883
records = list(tt.serialize(FakeSerializer()))
2884
self.assertEqual(expected, records)
2887
def default_attribs():
2892
'_new_executability': {},
2894
'_tree_path_ids': {'': 'new-0'},
2896
'_removed_contents': [],
2897
'_non_present_ids': {},
2900
def make_records(self, attribs, contents):
2902
(((('attribs'),),), bencode.bencode(attribs))]
2903
records.extend([(((n, k),), c) for n, k, c in contents])
2906
def creation_records(self):
2907
attribs = self.default_attribs()
2908
attribs['_id_number'] = 3
2909
attribs['_new_name'] = {
2910
'new-1': u'foo\u1234'.encode('utf-8'), 'new-2': 'qux'}
2911
attribs['_new_id'] = {'new-1': 'baz', 'new-2': 'quxx'}
2912
attribs['_new_parent'] = {'new-1': 'new-0', 'new-2': 'new-0'}
2913
attribs['_new_executability'] = {'new-1': 1}
2915
('new-1', 'file', 'i 1\nbar\n'),
2916
('new-2', 'directory', ''),
2918
return self.make_records(attribs, contents)
2920
def test_serialize_creation(self):
2921
tt = self.get_preview()
2922
tt.new_file(u'foo\u1234', tt.root, 'bar', 'baz', True)
2923
tt.new_directory('qux', tt.root, 'quxx')
2924
self.assertSerializesTo(self.creation_records(), tt)
2926
def test_deserialize_creation(self):
2927
tt = self.get_preview()
2928
tt.deserialize(iter(self.creation_records()))
2929
self.assertEqual(3, tt._id_number)
2930
self.assertEqual({'new-1': u'foo\u1234',
2931
'new-2': 'qux'}, tt._new_name)
2932
self.assertEqual({'new-1': 'baz', 'new-2': 'quxx'}, tt._new_id)
2933
self.assertEqual({'new-1': tt.root, 'new-2': tt.root}, tt._new_parent)
2934
self.assertEqual({'baz': 'new-1', 'quxx': 'new-2'}, tt._r_new_id)
2935
self.assertEqual({'new-1': True}, tt._new_executability)
2936
self.assertEqual({'new-1': 'file',
2937
'new-2': 'directory'}, tt._new_contents)
2938
foo_limbo = open(tt._limbo_name('new-1'), 'rb')
2940
foo_content = foo_limbo.read()
2943
self.assertEqual('bar', foo_content)
2945
def symlink_creation_records(self):
2946
attribs = self.default_attribs()
2947
attribs['_id_number'] = 2
2948
attribs['_new_name'] = {'new-1': u'foo\u1234'.encode('utf-8')}
2949
attribs['_new_parent'] = {'new-1': 'new-0'}
2950
contents = [('new-1', 'symlink', u'bar\u1234'.encode('utf-8'))]
2951
return self.make_records(attribs, contents)
2953
def test_serialize_symlink_creation(self):
2954
self.requireFeature(tests.SymlinkFeature)
2955
tt = self.get_preview()
2956
tt.new_symlink(u'foo\u1234', tt.root, u'bar\u1234')
2957
self.assertSerializesTo(self.symlink_creation_records(), tt)
2959
def test_deserialize_symlink_creation(self):
2960
self.requireFeature(tests.SymlinkFeature)
2961
tt = self.get_preview()
2962
tt.deserialize(iter(self.symlink_creation_records()))
2963
abspath = tt._limbo_name('new-1')
2964
foo_content = osutils.readlink(abspath)
2965
self.assertEqual(u'bar\u1234', foo_content)
2967
def make_destruction_preview(self):
2968
tree = self.make_branch_and_tree('.')
2969
self.build_tree([u'foo\u1234', 'bar'])
2970
tree.add([u'foo\u1234', 'bar'], ['foo-id', 'bar-id'])
2971
return self.get_preview(tree)
2973
def destruction_records(self):
2974
attribs = self.default_attribs()
2975
attribs['_id_number'] = 3
2976
attribs['_removed_id'] = ['new-1']
2977
attribs['_removed_contents'] = ['new-2']
2978
attribs['_tree_path_ids'] = {
2980
u'foo\u1234'.encode('utf-8'): 'new-1',
2983
return self.make_records(attribs, [])
2985
def test_serialize_destruction(self):
2986
tt = self.make_destruction_preview()
2987
foo_trans_id = tt.trans_id_tree_file_id('foo-id')
2988
tt.unversion_file(foo_trans_id)
2989
bar_trans_id = tt.trans_id_tree_file_id('bar-id')
2990
tt.delete_contents(bar_trans_id)
2991
self.assertSerializesTo(self.destruction_records(), tt)
2993
def test_deserialize_destruction(self):
2994
tt = self.make_destruction_preview()
2995
tt.deserialize(iter(self.destruction_records()))
2996
self.assertEqual({u'foo\u1234': 'new-1',
2998
'': tt.root}, tt._tree_path_ids)
2999
self.assertEqual({'new-1': u'foo\u1234',
3001
tt.root: ''}, tt._tree_id_paths)
3002
self.assertEqual(set(['new-1']), tt._removed_id)
3003
self.assertEqual(set(['new-2']), tt._removed_contents)
3005
def missing_records(self):
3006
attribs = self.default_attribs()
3007
attribs['_id_number'] = 2
3008
attribs['_non_present_ids'] = {
3010
return self.make_records(attribs, [])
3012
def test_serialize_missing(self):
3013
tt = self.get_preview()
3014
boo_trans_id = tt.trans_id_file_id('boo')
3015
self.assertSerializesTo(self.missing_records(), tt)
3017
def test_deserialize_missing(self):
3018
tt = self.get_preview()
3019
tt.deserialize(iter(self.missing_records()))
3020
self.assertEqual({'boo': 'new-1'}, tt._non_present_ids)
3022
def make_modification_preview(self):
3023
LINES_ONE = 'aa\nbb\ncc\ndd\n'
3024
LINES_TWO = 'z\nbb\nx\ndd\n'
3025
tree = self.make_branch_and_tree('tree')
3026
self.build_tree_contents([('tree/file', LINES_ONE)])
3027
tree.add('file', 'file-id')
3028
return self.get_preview(tree), LINES_TWO
3030
def modification_records(self):
3031
attribs = self.default_attribs()
3032
attribs['_id_number'] = 2
3033
attribs['_tree_path_ids'] = {
3036
attribs['_removed_contents'] = ['new-1']
3037
contents = [('new-1', 'file',
3038
'i 1\nz\n\nc 0 1 1 1\ni 1\nx\n\nc 0 3 3 1\n')]
3039
return self.make_records(attribs, contents)
3041
def test_serialize_modification(self):
3042
tt, LINES = self.make_modification_preview()
3043
trans_id = tt.trans_id_file_id('file-id')
3044
tt.delete_contents(trans_id)
3045
tt.create_file(LINES, trans_id)
3046
self.assertSerializesTo(self.modification_records(), tt)
3048
def test_deserialize_modification(self):
3049
tt, LINES = self.make_modification_preview()
3050
tt.deserialize(iter(self.modification_records()))
3051
self.assertFileEqual(LINES, tt._limbo_name('new-1'))
3053
def make_kind_change_preview(self):
3054
LINES = 'a\nb\nc\nd\n'
3055
tree = self.make_branch_and_tree('tree')
3056
self.build_tree(['tree/foo/'])
3057
tree.add('foo', 'foo-id')
3058
return self.get_preview(tree), LINES
3060
def kind_change_records(self):
3061
attribs = self.default_attribs()
3062
attribs['_id_number'] = 2
3063
attribs['_tree_path_ids'] = {
3066
attribs['_removed_contents'] = ['new-1']
3067
contents = [('new-1', 'file',
3068
'i 4\na\nb\nc\nd\n\n')]
3069
return self.make_records(attribs, contents)
3071
def test_serialize_kind_change(self):
3072
tt, LINES = self.make_kind_change_preview()
3073
trans_id = tt.trans_id_file_id('foo-id')
3074
tt.delete_contents(trans_id)
3075
tt.create_file(LINES, trans_id)
3076
self.assertSerializesTo(self.kind_change_records(), tt)
3078
def test_deserialize_kind_change(self):
3079
tt, LINES = self.make_kind_change_preview()
3080
tt.deserialize(iter(self.kind_change_records()))
3081
self.assertFileEqual(LINES, tt._limbo_name('new-1'))
3083
def make_add_contents_preview(self):
3084
LINES = 'a\nb\nc\nd\n'
3085
tree = self.make_branch_and_tree('tree')
3086
self.build_tree(['tree/foo'])
3088
os.unlink('tree/foo')
3089
return self.get_preview(tree), LINES
3091
def add_contents_records(self):
3092
attribs = self.default_attribs()
3093
attribs['_id_number'] = 2
3094
attribs['_tree_path_ids'] = {
3097
contents = [('new-1', 'file',
3098
'i 4\na\nb\nc\nd\n\n')]
3099
return self.make_records(attribs, contents)
3101
def test_serialize_add_contents(self):
3102
tt, LINES = self.make_add_contents_preview()
3103
trans_id = tt.trans_id_tree_path('foo')
3104
tt.create_file(LINES, trans_id)
3105
self.assertSerializesTo(self.add_contents_records(), tt)
3107
def test_deserialize_add_contents(self):
3108
tt, LINES = self.make_add_contents_preview()
3109
tt.deserialize(iter(self.add_contents_records()))
3110
self.assertFileEqual(LINES, tt._limbo_name('new-1'))
3112
def test_get_parents_lines(self):
3113
LINES_ONE = 'aa\nbb\ncc\ndd\n'
3114
LINES_TWO = 'z\nbb\nx\ndd\n'
3115
tree = self.make_branch_and_tree('tree')
3116
self.build_tree_contents([('tree/file', LINES_ONE)])
3117
tree.add('file', 'file-id')
3118
tt = self.get_preview(tree)
3119
trans_id = tt.trans_id_tree_path('file')
3120
self.assertEqual((['aa\n', 'bb\n', 'cc\n', 'dd\n'],),
3121
tt._get_parents_lines(trans_id))
3123
def test_get_parents_texts(self):
3124
LINES_ONE = 'aa\nbb\ncc\ndd\n'
3125
LINES_TWO = 'z\nbb\nx\ndd\n'
3126
tree = self.make_branch_and_tree('tree')
3127
self.build_tree_contents([('tree/file', LINES_ONE)])
3128
tree.add('file', 'file-id')
3129
tt = self.get_preview(tree)
3130
trans_id = tt.trans_id_tree_path('file')
3131
self.assertEqual((LINES_ONE,),
3132
tt._get_parents_texts(trans_id))