215
215
tree_z.commit('removed b')
216
216
merge_inner(tree_z.branch, tree_a, base_tree, this_tree=tree_z)
217
217
self.assertEqual([
218
conflicts.MissingParent('Created directory', 'b', b'b-id'),
219
conflicts.UnversionedParent('Versioned directory', 'b', b'b-id')],
218
conflicts.MissingParent('Created directory', 'b', 'b-id'),
219
conflicts.UnversionedParent('Versioned directory', 'b', 'b-id')],
220
220
tree_z.conflicts())
221
221
merge_inner(tree_a.branch, tree_z.basis_tree(), base_tree,
222
222
this_tree=tree_a)
223
223
self.assertEqual([
224
conflicts.DeletingParent('Not deleting', 'b', b'b-id'),
225
conflicts.UnversionedParent('Versioned directory', 'b', b'b-id')],
224
conflicts.DeletingParent('Not deleting', 'b', 'b-id'),
225
conflicts.UnversionedParent('Versioned directory', 'b', 'b-id')],
226
226
tree_a.conflicts())
228
228
def test_nested_merge(self):
230
'iter_changes doesn\'t work with changes in nested trees')
231
229
tree = self.make_branch_and_tree('tree',
232
230
format='development-subtree')
233
231
sub_tree = self.make_branch_and_tree('tree/sub-tree',
234
232
format='development-subtree')
235
sub_tree.set_root_id(b'sub-tree-root')
236
self.build_tree_contents([('tree/sub-tree/file', b'text1')])
233
sub_tree.set_root_id('sub-tree-root')
234
self.build_tree_contents([('tree/sub-tree/file', 'text1')])
237
235
sub_tree.add('file')
238
236
sub_tree.commit('foo')
239
237
tree.add_reference(sub_tree)
240
238
tree.commit('set text to 1')
241
tree2 = tree.controldir.sprout('tree2').open_workingtree()
239
tree2 = tree.bzrdir.sprout('tree2').open_workingtree()
242
240
# modify the file in the subtree
243
self.build_tree_contents([('tree2/sub-tree/file', b'text2')])
241
self.build_tree_contents([('tree2/sub-tree/file', 'text2')])
244
242
# and merge the changes from the diverged subtree into the containing
246
244
tree2.commit('changed file text')
382
380
def test_weave_cannot_reverse_cherrypick(self):
383
381
this_tree, other_tree = self.prepare_cherrypick()
384
merger = _mod_merge.Merger.from_revision_ids(
382
merger = _mod_merge.Merger.from_revision_ids(None,
385
383
this_tree, 'rev2b', 'rev3b', other_tree.branch)
386
384
merger.merge_type = _mod_merge.WeaveMerger
387
385
self.assertRaises(errors.CannotReverseCherrypick, merger.do_merge)
389
387
def test_merge3_can_reverse_cherrypick(self):
390
388
this_tree, other_tree = self.prepare_cherrypick()
391
merger = _mod_merge.Merger.from_revision_ids(
389
merger = _mod_merge.Merger.from_revision_ids(None,
392
390
this_tree, 'rev2b', 'rev3b', other_tree.branch)
393
391
merger.merge_type = _mod_merge.Merge3Merger
394
392
merger.do_merge()
396
394
def test_merge3_will_detect_cherrypick(self):
397
395
this_tree = self.make_branch_and_tree('this')
398
self.build_tree_contents([('this/file', b"a\n")])
396
self.build_tree_contents([('this/file', "a\n")])
399
397
this_tree.add('file')
400
398
this_tree.commit('rev1')
401
other_tree = this_tree.controldir.sprout('other').open_workingtree()
402
self.build_tree_contents([('other/file', b"a\nb\n")])
403
other_tree.commit('rev2b', rev_id=b'rev2b')
404
self.build_tree_contents([('other/file', b"a\nb\nc\n")])
405
other_tree.commit('rev3b', rev_id=b'rev3b')
399
other_tree = this_tree.bzrdir.sprout('other').open_workingtree()
400
self.build_tree_contents([('other/file', "a\nb\n")])
401
other_tree.commit('rev2b', rev_id='rev2b')
402
self.build_tree_contents([('other/file', "a\nb\nc\n")])
403
other_tree.commit('rev3b', rev_id='rev3b')
406
404
this_tree.lock_write()
407
405
self.addCleanup(this_tree.unlock)
409
merger = _mod_merge.Merger.from_revision_ids(
407
merger = _mod_merge.Merger.from_revision_ids(None,
410
408
this_tree, 'rev3b', 'rev2b', other_tree.branch)
411
409
merger.merge_type = _mod_merge.Merge3Merger
412
410
merger.do_merge()
458
456
def test_make_preview_transform(self):
459
457
this_tree = self.make_branch_and_tree('this')
460
self.build_tree_contents([('this/file', b'1\n')])
461
this_tree.add('file', b'file-id')
462
this_tree.commit('rev1', rev_id=b'rev1')
463
other_tree = this_tree.controldir.sprout('other').open_workingtree()
464
self.build_tree_contents([('this/file', b'1\n2a\n')])
465
this_tree.commit('rev2', rev_id=b'rev2a')
466
self.build_tree_contents([('other/file', b'2b\n1\n')])
467
other_tree.commit('rev2', rev_id=b'rev2b')
458
self.build_tree_contents([('this/file', '1\n')])
459
this_tree.add('file', 'file-id')
460
this_tree.commit('rev1', rev_id='rev1')
461
other_tree = this_tree.bzrdir.sprout('other').open_workingtree()
462
self.build_tree_contents([('this/file', '1\n2a\n')])
463
this_tree.commit('rev2', rev_id='rev2a')
464
self.build_tree_contents([('other/file', '2b\n1\n')])
465
other_tree.commit('rev2', rev_id='rev2b')
468
466
this_tree.lock_write()
469
467
self.addCleanup(this_tree.unlock)
470
merger = _mod_merge.Merger.from_revision_ids(
471
this_tree, b'rev2b', other_branch=other_tree.branch)
468
merger = _mod_merge.Merger.from_revision_ids(None,
469
this_tree, 'rev2b', other_branch=other_tree.branch)
472
470
merger.merge_type = _mod_merge.Merge3Merger
473
471
tree_merger = merger.make_merger()
474
472
tt = tree_merger.make_preview_transform()
475
473
self.addCleanup(tt.finalize)
476
474
preview_tree = tt.get_preview_tree()
477
tree_file = this_tree.get_file('file')
475
tree_file = this_tree.get_file('file-id')
479
477
self.assertEqual('1\n2a\n', tree_file.read())
481
479
tree_file.close()
482
preview_file = preview_tree.get_file('file')
480
preview_file = preview_tree.get_file('file-id')
484
482
self.assertEqual('2b\n1\n2a\n', preview_file.read())
488
486
def test_do_merge(self):
489
487
this_tree = self.make_branch_and_tree('this')
490
self.build_tree_contents([('this/file', b'1\n')])
491
this_tree.add('file', b'file-id')
492
this_tree.commit('rev1', rev_id=b'rev1')
493
other_tree = this_tree.controldir.sprout('other').open_workingtree()
494
self.build_tree_contents([('this/file', b'1\n2a\n')])
495
this_tree.commit('rev2', rev_id=b'rev2a')
496
self.build_tree_contents([('other/file', b'2b\n1\n')])
497
other_tree.commit('rev2', rev_id=b'rev2b')
488
self.build_tree_contents([('this/file', '1\n')])
489
this_tree.add('file', 'file-id')
490
this_tree.commit('rev1', rev_id='rev1')
491
other_tree = this_tree.bzrdir.sprout('other').open_workingtree()
492
self.build_tree_contents([('this/file', '1\n2a\n')])
493
this_tree.commit('rev2', rev_id='rev2a')
494
self.build_tree_contents([('other/file', '2b\n1\n')])
495
other_tree.commit('rev2', rev_id='rev2b')
498
496
this_tree.lock_write()
499
497
self.addCleanup(this_tree.unlock)
500
merger = _mod_merge.Merger.from_revision_ids(
501
this_tree, b'rev2b', other_branch=other_tree.branch)
498
merger = _mod_merge.Merger.from_revision_ids(None,
499
this_tree, 'rev2b', other_branch=other_tree.branch)
502
500
merger.merge_type = _mod_merge.Merge3Merger
503
501
tree_merger = merger.make_merger()
504
502
tt = tree_merger.do_merge()
505
tree_file = this_tree.get_file('file')
503
tree_file = this_tree.get_file('file-id')
507
505
self.assertEqual('2b\n1\n2a\n', tree_file.read())
1397
1396
class TestMergerEntriesLCA(TestMergerBase):
1399
1398
def make_merge_obj(self, builder, other_revision_id,
1400
interesting_files=None):
1399
interesting_files=None, interesting_ids=None):
1401
1400
merger = self.make_Merger(builder, other_revision_id,
1402
interesting_files=interesting_files)
1401
interesting_files=interesting_files,
1402
interesting_ids=interesting_ids)
1403
1403
return merger.make_merger()
1405
1405
def test_simple(self):
1406
1406
builder = self.get_builder()
1407
builder.build_snapshot(None,
1407
builder.build_snapshot('A-id', None,
1408
1408
[('add', (u'', 'a-root-id', 'directory', None)),
1409
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
1411
builder.build_snapshot(['A-id'],
1412
[('modify', ('a', 'a\nb\nC\nc\n'))],
1414
builder.build_snapshot(['A-id'],
1415
[('modify', ('a', 'a\nB\nb\nc\n'))],
1417
builder.build_snapshot(['C-id', 'B-id'],
1418
[('modify', ('a', 'a\nB\nb\nC\nc\nE\n'))],
1420
builder.build_snapshot(['B-id', 'C-id'],
1421
[('modify', ('a', 'a\nB\nb\nC\nc\n'))],
1422
revision_id='D-id', )
1409
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
1410
builder.build_snapshot('C-id', ['A-id'],
1411
[('modify', ('a-id', 'a\nb\nC\nc\n'))])
1412
builder.build_snapshot('B-id', ['A-id'],
1413
[('modify', ('a-id', 'a\nB\nb\nc\n'))])
1414
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1415
[('modify', ('a-id', 'a\nB\nb\nC\nc\nE\n'))])
1416
builder.build_snapshot('D-id', ['B-id', 'C-id'],
1417
[('modify', ('a-id', 'a\nB\nb\nC\nc\n'))])
1423
1418
merge_obj = self.make_merge_obj(builder, 'E-id')
1425
1420
self.assertEqual(['B-id', 'C-id'], [t.get_revision_id()
1450
1444
# G modifies 'bar'
1452
1446
builder = self.get_builder()
1453
builder.build_snapshot(None,
1454
[('add', (u'', 'a-root-id', 'directory', None))],
1456
builder.build_snapshot(['A-id'],
1457
[('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))],
1459
builder.build_snapshot(['A-id'],
1460
[('add', (u'bar', 'bar-id', 'file', 'd\ne\nf\n'))],
1462
builder.build_snapshot(['B-id', 'C-id'],
1463
[('add', (u'bar', 'bar-id', 'file', 'd\ne\nf\n'))],
1465
builder.build_snapshot(['C-id', 'B-id'],
1466
[('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))],
1468
builder.build_snapshot(['E-id', 'D-id'],
1469
[('modify', (u'bar', 'd\ne\nf\nG\n'))],
1471
builder.build_snapshot(['D-id', 'E-id'], [], revision_id='F-id')
1447
builder.build_snapshot('A-id', None,
1448
[('add', (u'', 'a-root-id', 'directory', None))])
1449
builder.build_snapshot('B-id', ['A-id'],
1450
[('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))])
1451
builder.build_snapshot('C-id', ['A-id'],
1452
[('add', (u'bar', 'bar-id', 'file', 'd\ne\nf\n'))])
1453
builder.build_snapshot('D-id', ['B-id', 'C-id'],
1454
[('add', (u'bar', 'bar-id', 'file', 'd\ne\nf\n'))])
1455
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1456
[('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))])
1457
builder.build_snapshot('G-id', ['E-id', 'D-id'],
1458
[('modify', (u'bar-id', 'd\ne\nf\nG\n'))])
1459
builder.build_snapshot('F-id', ['D-id', 'E-id'], [])
1472
1460
merge_obj = self.make_merge_obj(builder, 'G-id')
1474
1462
self.assertEqual(['D-id', 'E-id'], [t.get_revision_id()
1486
1473
def test_not_in_this(self):
1487
1474
builder = self.get_builder()
1488
builder.build_snapshot(None,
1475
builder.build_snapshot('A-id', None,
1489
1476
[('add', (u'', 'a-root-id', 'directory', None)),
1490
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
1492
builder.build_snapshot(['A-id'],
1493
[('modify', ('a', 'a\nB\nb\nc\n'))],
1495
builder.build_snapshot(['A-id'],
1496
[('modify', ('a', 'a\nb\nC\nc\n'))],
1498
builder.build_snapshot(['C-id', 'B-id'],
1499
[('modify', ('a', 'a\nB\nb\nC\nc\nE\n'))],
1501
builder.build_snapshot(['B-id', 'C-id'],
1502
[('unversion', 'a')],
1477
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
1478
builder.build_snapshot('B-id', ['A-id'],
1479
[('modify', ('a-id', 'a\nB\nb\nc\n'))])
1480
builder.build_snapshot('C-id', ['A-id'],
1481
[('modify', ('a-id', 'a\nb\nC\nc\n'))])
1482
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1483
[('modify', ('a-id', 'a\nB\nb\nC\nc\nE\n'))])
1484
builder.build_snapshot('D-id', ['B-id', 'C-id'],
1485
[('unversion', 'a-id')])
1504
1486
merge_obj = self.make_merge_obj(builder, 'E-id')
1506
1488
self.assertEqual(['B-id', 'C-id'], [t.get_revision_id()
1524
1505
# D E # D and E both have the file, unchanged from C
1525
1506
builder = self.get_builder()
1526
builder.build_snapshot(None,
1527
[('add', (u'', 'a-root-id', 'directory', None))],
1529
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1530
builder.build_snapshot(['A-id'],
1531
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
1533
builder.build_snapshot(['C-id', 'B-id'],
1534
[], revision_id='E-id') # Inherited from C
1535
builder.build_snapshot(['B-id', 'C-id'], # Merged from C
1536
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
1507
builder.build_snapshot('A-id', None,
1508
[('add', (u'', 'a-root-id', 'directory', None))])
1509
builder.build_snapshot('B-id', ['A-id'], [])
1510
builder.build_snapshot('C-id', ['A-id'],
1511
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
1512
builder.build_snapshot('E-id', ['C-id', 'B-id'], []) # Inherited from C
1513
builder.build_snapshot('D-id', ['B-id', 'C-id'], # Merged from C
1514
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
1538
1515
merge_obj = self.make_merge_obj(builder, 'E-id')
1540
1517
self.assertEqual(['B-id', 'C-id'], [t.get_revision_id()
1547
1524
def test_not_in_other(self):
1548
1525
builder = self.get_builder()
1549
builder.build_snapshot(None,
1526
builder.build_snapshot('A-id', None,
1550
1527
[('add', (u'', 'a-root-id', 'directory', None)),
1551
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
1553
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1554
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1555
builder.build_snapshot(
1557
[('unversion', 'a')], revision_id='E-id')
1558
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1528
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
1529
builder.build_snapshot('B-id', ['A-id'], [])
1530
builder.build_snapshot('C-id', ['A-id'], [])
1531
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1532
[('unversion', 'a-id')])
1533
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1559
1534
merge_obj = self.make_merge_obj(builder, 'E-id')
1561
1536
entries = list(merge_obj._entries_lca())
1562
1537
root_id = 'a-root-id'
1563
1538
self.assertEqual([('a-id', True,
1564
((u'a', [u'a', u'a']), None, u'a'),
1565
1539
((root_id, [root_id, root_id]), None, root_id),
1566
1540
((u'a', [u'a', u'a']), None, u'a'),
1567
1541
((False, [False, False]), None, False)),
1580
1554
# C => E, no changes
1581
1555
# D would then win 'cleanly' and no record would be given
1582
1556
builder = self.get_builder()
1583
builder.build_snapshot(None,
1557
builder.build_snapshot('A-id', None,
1584
1558
[('add', (u'', 'a-root-id', 'directory', None)),
1585
('add', (u'foo', 'foo-id', 'file', 'content\n'))],
1587
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1588
builder.build_snapshot(['A-id'],
1589
[('unversion', 'foo')], revision_id='C-id')
1590
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
1591
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1559
('add', (u'foo', 'foo-id', 'file', 'content\n'))])
1560
builder.build_snapshot('B-id', ['A-id'], [])
1561
builder.build_snapshot('C-id', ['A-id'],
1562
[('unversion', 'foo-id')])
1563
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
1564
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1592
1565
merge_obj = self.make_merge_obj(builder, 'E-id')
1594
1567
entries = list(merge_obj._entries_lca())
1610
1583
# In this case, we have a conflict of how the changes were resolved. E
1611
1584
# picked C and D picked B, so we should issue a conflict
1612
1585
builder = self.get_builder()
1613
builder.build_snapshot(None,
1586
builder.build_snapshot('A-id', None,
1614
1587
[('add', (u'', 'a-root-id', 'directory', None)),
1615
('add', (u'foo', 'foo-id', 'file', 'content\n'))],
1617
builder.build_snapshot(['A-id'], [
1618
('modify', ('foo', 'new-content\n'))],
1620
builder.build_snapshot(['A-id'],
1621
[('unversion', 'foo')],
1623
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
1624
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1588
('add', (u'foo', 'foo-id', 'file', 'content\n'))])
1589
builder.build_snapshot('B-id', ['A-id'], [
1590
('modify', ('foo-id', 'new-content\n'))])
1591
builder.build_snapshot('C-id', ['A-id'],
1592
[('unversion', 'foo-id')])
1593
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
1594
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1625
1595
merge_obj = self.make_merge_obj(builder, 'E-id')
1627
1597
entries = list(merge_obj._entries_lca())
1628
1598
root_id = 'a-root-id'
1629
1599
self.assertEqual([('foo-id', True,
1630
((u'foo', [u'foo', None]), None, u'foo'),
1631
1600
((root_id, [root_id, None]), None, root_id),
1632
1601
((u'foo', [u'foo', None]), None, 'foo'),
1633
1602
((False, [False, None]), None, False)),
1650
1619
# A => C, add file, thus C supersedes B
1651
1620
# w/ C=BASE, D=THIS, E=OTHER we have 'happy convergence'
1652
1621
builder = self.get_builder()
1653
builder.build_snapshot(None,
1654
[('add', (u'', 'a-root-id', 'directory', None))],
1656
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1657
builder.build_snapshot(['A-id'],
1658
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
1660
builder.build_snapshot(['C-id', 'B-id'],
1661
[('unversion', 'a')],
1663
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1622
builder.build_snapshot('A-id', None,
1623
[('add', (u'', 'a-root-id', 'directory', None))])
1624
builder.build_snapshot('B-id', ['A-id'], [])
1625
builder.build_snapshot('C-id', ['A-id'],
1626
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
1627
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1628
[('unversion', 'a-id')])
1629
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1664
1630
merge_obj = self.make_merge_obj(builder, 'E-id')
1666
1632
entries = list(merge_obj._entries_lca())
1669
1635
def test_only_in_other(self):
1670
1636
builder = self.get_builder()
1671
builder.build_snapshot(None,
1672
[('add', (u'', 'a-root-id', 'directory', None))],
1674
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1675
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1676
builder.build_snapshot(['C-id', 'B-id'],
1677
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
1679
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1637
builder.build_snapshot('A-id', None,
1638
[('add', (u'', 'a-root-id', 'directory', None))])
1639
builder.build_snapshot('B-id', ['A-id'], [])
1640
builder.build_snapshot('C-id', ['A-id'], [])
1641
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1642
[('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
1643
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1680
1644
merge_obj = self.make_merge_obj(builder, 'E-id')
1682
1646
entries = list(merge_obj._entries_lca())
1683
1647
root_id = 'a-root-id'
1684
1648
self.assertEqual([('a-id', True,
1685
((None, [None, None]), u'a', None),
1686
1649
((None, [None, None]), root_id, None),
1687
1650
((None, [None, None]), u'a', None),
1688
1651
((None, [None, None]), False, None)),
1703
1666
# though its LCAs disagree. This is because the modification in E
1704
1667
# completely supersedes the value in D.
1705
1668
builder = self.get_builder()
1706
builder.build_snapshot(None,
1669
builder.build_snapshot('A-id', None,
1707
1670
[('add', (u'', 'a-root-id', 'directory', None)),
1708
('add', (u'foo', 'foo-id', 'file', 'A content\n'))],
1710
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1711
builder.build_snapshot(['A-id'],
1712
[('modify', ('foo', 'B content\n'))],
1714
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1715
builder.build_snapshot(['C-id', 'B-id'],
1716
[('modify', ('foo', 'E content\n'))],
1718
builder.build_snapshot(['E-id', 'D-id'], [], revision_id='G-id')
1719
builder.build_snapshot(['D-id', 'E-id'],
1720
[('modify', ('foo', 'F content\n'))],
1671
('add', (u'foo', 'foo-id', 'file', 'A content\n'))])
1672
builder.build_snapshot('C-id', ['A-id'], [])
1673
builder.build_snapshot('B-id', ['A-id'],
1674
[('modify', ('foo-id', 'B content\n'))])
1675
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1676
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1677
[('modify', ('foo-id', 'E content\n'))])
1678
builder.build_snapshot('G-id', ['E-id', 'D-id'], [])
1679
builder.build_snapshot('F-id', ['D-id', 'E-id'],
1680
[('modify', ('foo-id', 'F content\n'))])
1722
1681
merge_obj = self.make_merge_obj(builder, 'G-id')
1724
1683
self.assertEqual([], list(merge_obj._entries_lca()))
1753
1712
# aren't supporting it yet.
1755
1714
builder = self.get_builder()
1756
builder.build_snapshot(None,
1715
builder.build_snapshot('A-id', None,
1757
1716
[('add', (u'', 'a-root-id', 'directory', None)),
1758
('add', (u'foo', 'foo-id', 'file', 'A content\n'))],
1760
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1761
builder.build_snapshot(['A-id'],
1762
[('rename', ('foo', 'bar'))],
1764
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1765
builder.build_snapshot(['C-id', 'B-id'],
1766
[('rename', ('foo', 'bing'))],
1767
revision_id='E-id') # override to bing
1768
builder.build_snapshot(['E-id', 'D-id'],
1769
[('rename', ('bing', 'barry'))],
1770
revision_id='G-id') # override to barry
1771
builder.build_snapshot(['D-id', 'E-id'],
1772
[('rename', ('bar', 'bing'))],
1773
revision_id='F-id') # Merge in E's change
1717
('add', (u'foo', 'foo-id', 'file', 'A content\n'))])
1718
builder.build_snapshot('C-id', ['A-id'], [])
1719
builder.build_snapshot('B-id', ['A-id'],
1720
[('rename', ('foo', 'bar'))])
1721
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1722
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1723
[('rename', ('foo', 'bing'))]) # override to bing
1724
builder.build_snapshot('G-id', ['E-id', 'D-id'],
1725
[('rename', ('bing', 'barry'))]) # override to barry
1726
builder.build_snapshot('F-id', ['D-id', 'E-id'],
1727
[('rename', ('bar', 'bing'))]) # Merge in E's change
1774
1728
merge_obj = self.make_merge_obj(builder, 'G-id')
1776
1730
self.expectFailure("We don't do an actual heads() check on lca values,"
1791
1745
# be pruned from the LCAs, even though it was newly introduced by E
1792
1746
# (superseding B).
1793
1747
builder = self.get_builder()
1794
builder.build_snapshot(None,
1748
builder.build_snapshot('A-id', None,
1795
1749
[('add', (u'', 'a-root-id', 'directory', None)),
1796
('add', (u'foo', 'foo-id', 'file', 'A content\n'))],
1798
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1799
builder.build_snapshot(['A-id'],
1800
[('rename', ('foo', 'bar'))],
1802
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1803
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
1804
builder.build_snapshot(['E-id', 'D-id'],
1805
[('rename', ('foo', 'bar'))],
1807
builder.build_snapshot(['D-id', 'E-id'],
1808
[('rename', ('bar', 'bing'))],
1809
revision_id='F-id') # should end up conflicting
1750
('add', (u'foo', 'foo-id', 'file', 'A content\n'))])
1751
builder.build_snapshot('C-id', ['A-id'], [])
1752
builder.build_snapshot('B-id', ['A-id'],
1753
[('rename', ('foo', 'bar'))])
1754
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1755
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
1756
builder.build_snapshot('G-id', ['E-id', 'D-id'],
1757
[('rename', ('foo', 'bar'))])
1758
builder.build_snapshot('F-id', ['D-id', 'E-id'],
1759
[('rename', ('bar', 'bing'))]) # should end up conflicting
1810
1760
merge_obj = self.make_merge_obj(builder, 'G-id')
1812
1762
entries = list(merge_obj._entries_lca())
1828
1778
# D E D reverts to B, E reverts to C
1829
1779
# This should conflict
1830
1780
builder = self.get_builder()
1831
builder.build_snapshot(None,
1781
builder.build_snapshot('A-id', None,
1832
1782
[('add', (u'', 'a-root-id', 'directory', None)),
1833
('add', (u'foo', 'foo-id', 'file', 'A content\n'))],
1835
builder.build_snapshot(['A-id'],
1836
[('modify', ('foo', 'B content\n'))],
1838
builder.build_snapshot(['A-id'],
1839
[('modify', ('foo', 'C content\n'))],
1841
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
1842
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1783
('add', (u'foo', 'foo-id', 'file', 'A content\n'))])
1784
builder.build_snapshot('B-id', ['A-id'],
1785
[('modify', ('foo-id', 'B content\n'))])
1786
builder.build_snapshot('C-id', ['A-id'],
1787
[('modify', ('foo-id', 'C content\n'))])
1788
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
1789
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1843
1790
merge_obj = self.make_merge_obj(builder, 'E-id')
1845
1792
entries = list(merge_obj._entries_lca())
1846
1793
root_id = 'a-root-id'
1847
1794
self.assertEqual([('foo-id', True,
1848
((u'foo', [u'foo', u'foo']), u'foo', u'foo'),
1849
1795
((root_id, [root_id, root_id]), root_id, root_id),
1850
1796
((u'foo', [u'foo', u'foo']), u'foo', u'foo'),
1851
1797
((False, [False, False]), False, False)),
1863
1809
# We need to emit an entry for 'foo', because D & E differed on the
1864
1810
# merge resolution
1865
1811
builder = self.get_builder()
1866
builder.build_snapshot(None,
1812
builder.build_snapshot('A-id', None,
1867
1813
[('add', (u'', 'a-root-id', 'directory', None)),
1868
('add', (u'foo', 'foo-id', 'file', 'A content\n'))],
1870
builder.build_snapshot(['A-id'],
1871
[('modify', ('foo', 'B content\n'))],
1873
builder.build_snapshot(['A-id'],
1874
[('modify', ('foo', 'C content\n'))],
1875
revision_id='C-id', )
1876
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
1877
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1878
builder.build_snapshot(['D-id'],
1879
[('modify', ('foo', 'F content\n'))],
1814
('add', (u'foo', 'foo-id', 'file', 'A content\n'))])
1815
builder.build_snapshot('B-id', ['A-id'],
1816
[('modify', ('foo-id', 'B content\n'))])
1817
builder.build_snapshot('C-id', ['A-id'],
1818
[('modify', ('foo-id', 'C content\n'))])
1819
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
1820
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1821
builder.build_snapshot('F-id', ['D-id'],
1822
[('modify', ('foo-id', 'F content\n'))])
1881
1823
merge_obj = self.make_merge_obj(builder, 'E-id')
1883
1825
entries = list(merge_obj._entries_lca())
1884
1826
root_id = 'a-root-id'
1885
1827
self.assertEqual([('foo-id', True,
1886
((u'foo', [u'foo', u'foo']), u'foo', u'foo'),
1887
1828
((root_id, [root_id, root_id]), root_id, root_id),
1888
1829
((u'foo', [u'foo', u'foo']), u'foo', u'foo'),
1889
1830
((False, [False, False]), False, False)),
1904
1845
# We need to conflict.
1906
1847
builder = self.get_builder()
1907
builder.build_snapshot(None,
1848
builder.build_snapshot('A-id', None,
1908
1849
[('add', (u'', 'a-root-id', 'directory', None)),
1909
('add', (u'foo', 'foo-id', 'file', 'A content\n'))],
1911
builder.build_snapshot(['A-id'],
1912
[('modify', ('foo', 'B content\n'))],
1914
builder.build_snapshot(['A-id'],
1915
[('modify', ('foo', 'C content\n'))],
1917
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
1918
builder.build_snapshot(['B-id', 'C-id'],
1919
[('modify', ('foo', 'C content\n'))],
1920
revision_id='D-id') # Same as E
1921
builder.build_snapshot(['D-id'],
1922
[('modify', ('foo', 'F content\n'))],
1850
('add', (u'foo', 'foo-id', 'file', 'A content\n'))])
1851
builder.build_snapshot('B-id', ['A-id'],
1852
[('modify', ('foo-id', 'B content\n'))])
1853
builder.build_snapshot('C-id', ['A-id'],
1854
[('modify', ('foo-id', 'C content\n'))])
1855
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
1856
builder.build_snapshot('D-id', ['B-id', 'C-id'],
1857
[('modify', ('foo-id', 'C content\n'))]) # Same as E
1858
builder.build_snapshot('F-id', ['D-id'],
1859
[('modify', ('foo-id', 'F content\n'))])
1924
1860
merge_obj = self.make_merge_obj(builder, 'E-id')
1926
1862
entries = list(merge_obj._entries_lca())
1931
1867
def test_only_path_changed(self):
1932
1868
builder = self.get_builder()
1933
builder.build_snapshot(None,
1869
builder.build_snapshot('A-id', None,
1934
1870
[('add', (u'', 'a-root-id', 'directory', None)),
1935
('add', (u'a', 'a-id', 'file', 'content\n'))],
1937
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1938
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1939
builder.build_snapshot(['C-id', 'B-id'],
1940
[('rename', (u'a', u'b'))],
1942
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1871
('add', (u'a', 'a-id', 'file', 'content\n'))])
1872
builder.build_snapshot('B-id', ['A-id'], [])
1873
builder.build_snapshot('C-id', ['A-id'], [])
1874
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1875
[('rename', (u'a', u'b'))])
1876
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1943
1877
merge_obj = self.make_merge_obj(builder, 'E-id')
1944
1878
entries = list(merge_obj._entries_lca())
1945
1879
root_id = 'a-root-id'
1946
1880
# The content was not changed, only the path
1947
1881
self.assertEqual([('a-id', False,
1948
((u'a', [u'a', u'a']), u'b', u'a'),
1949
1882
((root_id, [root_id, root_id]), root_id, root_id),
1950
1883
((u'a', [u'a', u'a']), u'b', u'a'),
1951
1884
((False, [False, False]), False, False)),
1954
1887
def test_kind_changed(self):
1955
1888
# Identical content, except 'D' changes a-id into a directory
1956
1889
builder = self.get_builder()
1957
builder.build_snapshot(None,
1890
builder.build_snapshot('A-id', None,
1958
1891
[('add', (u'', 'a-root-id', 'directory', None)),
1959
('add', (u'a', 'a-id', 'file', 'content\n'))],
1961
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1962
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1963
builder.build_snapshot(['C-id', 'B-id'],
1964
[('unversion', 'a'),
1892
('add', (u'a', 'a-id', 'file', 'content\n'))])
1893
builder.build_snapshot('B-id', ['A-id'], [])
1894
builder.build_snapshot('C-id', ['A-id'], [])
1895
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1896
[('unversion', 'a-id'),
1965
1897
('flush', None),
1966
('add', (u'a', 'a-id', 'directory', None))],
1968
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1898
('add', (u'a', 'a-id', 'directory', None))])
1899
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
1969
1900
merge_obj = self.make_merge_obj(builder, 'E-id')
1970
1901
entries = list(merge_obj._entries_lca())
1971
1902
root_id = 'a-root-id'
1972
1903
# Only the kind was changed (content)
1973
1904
self.assertEqual([('a-id', True,
1974
((u'a', [u'a', u'a']), u'a', u'a'),
1975
1905
((root_id, [root_id, root_id]), root_id, root_id),
1976
1906
((u'a', [u'a', u'a']), u'a', u'a'),
1977
1907
((False, [False, False]), False, False)),
1980
1910
def test_this_changed_kind(self):
1981
1911
# Identical content, but THIS changes a file to a directory
1982
1912
builder = self.get_builder()
1983
builder.build_snapshot(None,
1913
builder.build_snapshot('A-id', None,
1984
1914
[('add', (u'', 'a-root-id', 'directory', None)),
1985
('add', (u'a', 'a-id', 'file', 'content\n'))],
1987
builder.build_snapshot(['A-id'], [], revision_id='B-id')
1988
builder.build_snapshot(['A-id'], [], revision_id='C-id')
1989
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
1990
builder.build_snapshot(['B-id', 'C-id'],
1991
[('unversion', 'a'),
1915
('add', (u'a', 'a-id', 'file', 'content\n'))])
1916
builder.build_snapshot('B-id', ['A-id'], [])
1917
builder.build_snapshot('C-id', ['A-id'], [])
1918
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
1919
builder.build_snapshot('D-id', ['B-id', 'C-id'],
1920
[('unversion', 'a-id'),
1992
1921
('flush', None),
1993
('add', (u'a', 'a-id', 'directory', None))],
1922
('add', (u'a', 'a-id', 'directory', None))])
1995
1923
merge_obj = self.make_merge_obj(builder, 'E-id')
1996
1924
entries = list(merge_obj._entries_lca())
1997
1925
# Only the kind was changed (content)
2000
1928
def test_interesting_files(self):
2001
1929
# Two files modified, but we should filter one of them
2002
1930
builder = self.get_builder()
2003
builder.build_snapshot(None,
1931
builder.build_snapshot('A-id', None,
2004
1932
[('add', (u'', 'a-root-id', 'directory', None)),
2005
1933
('add', (u'a', 'a-id', 'file', 'content\n')),
2006
('add', (u'b', 'b-id', 'file', 'content\n'))],
2008
builder.build_snapshot(['A-id'], [], revision_id='B-id')
2009
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2010
builder.build_snapshot(['C-id', 'B-id'],
2011
[('modify', ('a', 'new-content\n')),
2012
('modify', ('b', 'new-content\n'))],
2014
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1934
('add', (u'b', 'b-id', 'file', 'content\n'))])
1935
builder.build_snapshot('B-id', ['A-id'], [])
1936
builder.build_snapshot('C-id', ['A-id'], [])
1937
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1938
[('modify', ('a-id', 'new-content\n')),
1939
('modify', ('b-id', 'new-content\n'))])
1940
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2015
1941
merge_obj = self.make_merge_obj(builder, 'E-id',
2016
1942
interesting_files=['b'])
2017
1943
entries = list(merge_obj._entries_lca())
2018
1944
root_id = 'a-root-id'
2019
1945
self.assertEqual([('b-id', True,
2020
((u'b', [u'b', u'b']), u'b', u'b'),
2021
1946
((root_id, [root_id, root_id]), root_id, root_id),
2022
1947
((u'b', [u'b', u'b']), u'b', u'b'),
2023
1948
((False, [False, False]), False, False)),
2026
1951
def test_interesting_file_in_this(self):
2027
1952
# This renamed the file, but it should still match the entry in other
2028
1953
builder = self.get_builder()
2029
builder.build_snapshot(None,
1954
builder.build_snapshot('A-id', None,
2030
1955
[('add', (u'', 'a-root-id', 'directory', None)),
2031
1956
('add', (u'a', 'a-id', 'file', 'content\n')),
2032
('add', (u'b', 'b-id', 'file', 'content\n'))],
2034
builder.build_snapshot(['A-id'], [], revision_id='B-id')
2035
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2036
builder.build_snapshot(['C-id', 'B-id'],
2037
[('modify', ('a', 'new-content\n')),
2038
('modify', ('b', 'new-content\n'))],
2040
builder.build_snapshot(['B-id', 'C-id'],
2041
[('rename', ('b', 'c'))],
1957
('add', (u'b', 'b-id', 'file', 'content\n'))])
1958
builder.build_snapshot('B-id', ['A-id'], [])
1959
builder.build_snapshot('C-id', ['A-id'], [])
1960
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1961
[('modify', ('a-id', 'new-content\n')),
1962
('modify', ('b-id', 'new-content\n'))])
1963
builder.build_snapshot('D-id', ['B-id', 'C-id'],
1964
[('rename', ('b', 'c'))])
2043
1965
merge_obj = self.make_merge_obj(builder, 'E-id',
2044
1966
interesting_files=['c'])
2045
1967
entries = list(merge_obj._entries_lca())
2046
1968
root_id = 'a-root-id'
2047
1969
self.assertEqual([('b-id', True,
2048
((u'b', [u'b', u'b']), u'b', u'c'),
2049
1970
((root_id, [root_id, root_id]), root_id, root_id),
2050
1971
((u'b', [u'b', u'b']), u'b', u'c'),
2051
1972
((False, [False, False]), False, False)),
2054
1975
def test_interesting_file_in_base(self):
2055
1976
# This renamed the file, but it should still match the entry in BASE
2056
1977
builder = self.get_builder()
2057
builder.build_snapshot(None,
1978
builder.build_snapshot('A-id', None,
2058
1979
[('add', (u'', 'a-root-id', 'directory', None)),
2059
1980
('add', (u'a', 'a-id', 'file', 'content\n')),
2060
('add', (u'c', 'c-id', 'file', 'content\n'))],
2062
builder.build_snapshot(['A-id'],
2063
[('rename', ('c', 'b'))],
2065
builder.build_snapshot(['A-id'],
2066
[('rename', ('c', 'b'))],
2068
builder.build_snapshot(['C-id', 'B-id'],
2069
[('modify', ('a', 'new-content\n')),
2070
('modify', ('b', 'new-content\n'))],
2072
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
1981
('add', (u'c', 'c-id', 'file', 'content\n'))])
1982
builder.build_snapshot('B-id', ['A-id'],
1983
[('rename', ('c', 'b'))])
1984
builder.build_snapshot('C-id', ['A-id'],
1985
[('rename', ('c', 'b'))])
1986
builder.build_snapshot('E-id', ['C-id', 'B-id'],
1987
[('modify', ('a-id', 'new-content\n')),
1988
('modify', ('c-id', 'new-content\n'))])
1989
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2073
1990
merge_obj = self.make_merge_obj(builder, 'E-id',
2074
1991
interesting_files=['c'])
2075
1992
entries = list(merge_obj._entries_lca())
2076
1993
root_id = 'a-root-id'
2077
1994
self.assertEqual([('c-id', True,
2078
((u'c', [u'b', u'b']), u'b', u'b'),
2079
1995
((root_id, [root_id, root_id]), root_id, root_id),
2080
1996
((u'c', [u'b', u'b']), u'b', u'b'),
2081
1997
((False, [False, False]), False, False)),
2084
2000
def test_interesting_file_in_lca(self):
2085
2001
# This renamed the file, but it should still match the entry in LCA
2086
2002
builder = self.get_builder()
2087
builder.build_snapshot(None,
2003
builder.build_snapshot('A-id', None,
2088
2004
[('add', (u'', 'a-root-id', 'directory', None)),
2089
2005
('add', (u'a', 'a-id', 'file', 'content\n')),
2090
('add', (u'b', 'b-id', 'file', 'content\n'))],
2092
builder.build_snapshot(['A-id'],
2093
[('rename', ('b', 'c'))], revision_id='B-id')
2094
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2095
builder.build_snapshot(['C-id', 'B-id'],
2096
[('modify', ('a', 'new-content\n')),
2097
('modify', ('b', 'new-content\n'))],
2099
builder.build_snapshot(['B-id', 'C-id'],
2100
[('rename', ('c', 'b'))], revision_id='D-id')
2006
('add', (u'b', 'b-id', 'file', 'content\n'))])
2007
builder.build_snapshot('B-id', ['A-id'],
2008
[('rename', ('b', 'c'))])
2009
builder.build_snapshot('C-id', ['A-id'], [])
2010
builder.build_snapshot('E-id', ['C-id', 'B-id'],
2011
[('modify', ('a-id', 'new-content\n')),
2012
('modify', ('b-id', 'new-content\n'))])
2013
builder.build_snapshot('D-id', ['B-id', 'C-id'],
2014
[('rename', ('c', 'b'))])
2101
2015
merge_obj = self.make_merge_obj(builder, 'E-id',
2102
2016
interesting_files=['c'])
2103
2017
entries = list(merge_obj._entries_lca())
2104
2018
root_id = 'a-root-id'
2105
2019
self.assertEqual([('b-id', True,
2106
((u'b', [u'c', u'b']), u'b', u'b'),
2107
2020
((root_id, [root_id, root_id]), root_id, root_id),
2108
2021
((u'b', [u'c', u'b']), u'b', u'b'),
2109
2022
((False, [False, False]), False, False)),
2112
def test_interesting_files(self):
2025
def test_interesting_ids(self):
2113
2026
# Two files modified, but we should filter one of them
2114
2027
builder = self.get_builder()
2115
builder.build_snapshot(None,
2028
builder.build_snapshot('A-id', None,
2116
2029
[('add', (u'', 'a-root-id', 'directory', None)),
2117
2030
('add', (u'a', 'a-id', 'file', 'content\n')),
2118
('add', (u'b', 'b-id', 'file', 'content\n'))],
2120
builder.build_snapshot(['A-id'], [], revision_id='B-id')
2121
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2122
builder.build_snapshot(['C-id', 'B-id'],
2123
[('modify', ('a', 'new-content\n')),
2124
('modify', ('b', 'new-content\n'))], revision_id='E-id')
2125
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
2031
('add', (u'b', 'b-id', 'file', 'content\n'))])
2032
builder.build_snapshot('B-id', ['A-id'], [])
2033
builder.build_snapshot('C-id', ['A-id'], [])
2034
builder.build_snapshot('E-id', ['C-id', 'B-id'],
2035
[('modify', ('a-id', 'new-content\n')),
2036
('modify', ('b-id', 'new-content\n'))])
2037
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2126
2038
merge_obj = self.make_merge_obj(builder, 'E-id',
2127
interesting_files=['b'])
2039
interesting_ids=['b-id'])
2128
2040
entries = list(merge_obj._entries_lca())
2129
2041
root_id = 'a-root-id'
2130
2042
self.assertEqual([('b-id', True,
2131
((u'b', [u'b', u'b']), u'b', u'b'),
2132
2043
((root_id, [root_id, root_id]), root_id, root_id),
2133
2044
((u'b', [u'b', u'b']), u'b', u'b'),
2134
2045
((False, [False, False]), False, False)),
2160
2071
def do_merge(self, builder, other_revision_id):
2161
2072
wt = self.get_wt_from_builder(builder)
2162
merger = _mod_merge.Merger.from_revision_ids(
2073
merger = _mod_merge.Merger.from_revision_ids(None,
2163
2074
wt, other_revision_id)
2164
2075
merger.merge_type = _mod_merge.Merge3Merger
2165
2076
return wt, merger.do_merge()
2167
2078
def test_simple_lca(self):
2168
2079
builder = self.get_builder()
2169
builder.build_snapshot(None,
2080
builder.build_snapshot('A-id', None,
2170
2081
[('add', (u'', 'a-root-id', 'directory', None)),
2171
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))],
2173
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2174
builder.build_snapshot(['A-id'], [], revision_id='B-id')
2175
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
2176
builder.build_snapshot(['B-id', 'C-id'],
2177
[('modify', ('a', 'a\nb\nc\nd\ne\nf\n'))],
2082
('add', (u'a', 'a-id', 'file', 'a\nb\nc\n'))])
2083
builder.build_snapshot('C-id', ['A-id'], [])
2084
builder.build_snapshot('B-id', ['A-id'], [])
2085
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
2086
builder.build_snapshot('D-id', ['B-id', 'C-id'],
2087
[('modify', ('a-id', 'a\nb\nc\nd\ne\nf\n'))])
2179
2088
wt, conflicts = self.do_merge(builder, 'E-id')
2180
2089
self.assertEqual(0, conflicts)
2181
2090
# The merge should have simply update the contents of 'a'
2182
self.assertEqual('a\nb\nc\nd\ne\nf\n', wt.get_file_text('a'))
2091
self.assertEqual('a\nb\nc\nd\ne\nf\n', wt.get_file_text('a-id'))
2184
2093
def test_conflict_without_lca(self):
2185
2094
# This test would cause a merge conflict, unless we use the lca trees
2195
2104
# F Path at 'baz' in F, which supersedes 'bar' and 'foo'
2196
2105
builder = self.get_builder()
2197
builder.build_snapshot(None,
2106
builder.build_snapshot('A-id', None,
2198
2107
[('add', (u'', 'a-root-id', 'directory', None)),
2199
('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))],
2201
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2202
builder.build_snapshot(['A-id'],
2203
[('rename', ('foo', 'bar'))], revision_id='B-id', )
2204
builder.build_snapshot(['C-id', 'B-id'], # merge the rename
2205
[('rename', ('foo', 'bar'))], revision_id='E-id')
2206
builder.build_snapshot(['E-id'],
2207
[('rename', ('bar', 'baz'))], revision_id='F-id')
2208
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
2108
('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))])
2109
builder.build_snapshot('C-id', ['A-id'], [])
2110
builder.build_snapshot('B-id', ['A-id'],
2111
[('rename', ('foo', 'bar'))])
2112
builder.build_snapshot('E-id', ['C-id', 'B-id'], # merge the rename
2113
[('rename', ('foo', 'bar'))])
2114
builder.build_snapshot('F-id', ['E-id'],
2115
[('rename', ('bar', 'baz'))])
2116
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2209
2117
wt, conflicts = self.do_merge(builder, 'F-id')
2210
2118
self.assertEqual(0, conflicts)
2211
2119
# The merge should simply recognize that the final rename takes
2213
self.assertEqual('baz', wt.id2path(b'foo-id'))
2121
self.assertEqual('baz', wt.id2path('foo-id'))
2215
2123
def test_other_deletes_lca_renames(self):
2216
2124
# This test would cause a merge conflict, unless we use the lca trees
2226
2134
# F F deletes 'bar'
2227
2135
builder = self.get_builder()
2228
builder.build_snapshot(None,
2136
builder.build_snapshot('A-id', None,
2229
2137
[('add', (u'', 'a-root-id', 'directory', None)),
2230
('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))],
2232
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2233
builder.build_snapshot(['A-id'],
2234
[('rename', ('foo', 'bar'))], revision_id='B-id')
2235
builder.build_snapshot(['C-id', 'B-id'], # merge the rename
2236
[('rename', ('foo', 'bar'))], revision_id='E-id')
2237
builder.build_snapshot(['E-id'],
2238
[('unversion', 'bar')], revision_id='F-id')
2239
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
2138
('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))])
2139
builder.build_snapshot('C-id', ['A-id'], [])
2140
builder.build_snapshot('B-id', ['A-id'],
2141
[('rename', ('foo', 'bar'))])
2142
builder.build_snapshot('E-id', ['C-id', 'B-id'], # merge the rename
2143
[('rename', ('foo', 'bar'))])
2144
builder.build_snapshot('F-id', ['E-id'],
2145
[('unversion', 'foo-id')])
2146
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2240
2147
wt, conflicts = self.do_merge(builder, 'F-id')
2241
2148
self.assertEqual(0, conflicts)
2242
2149
self.assertRaises(errors.NoSuchId, wt.id2path, 'foo-id')
2253
2160
# F Executable bit changed
2254
2161
builder = self.get_builder()
2255
builder.build_snapshot(None,
2162
builder.build_snapshot('A-id', None,
2256
2163
[('add', (u'', 'a-root-id', 'directory', None)),
2257
('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))],
2259
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2260
builder.build_snapshot(['A-id'], [], revision_id='B-id')
2261
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
2262
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
2164
('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))])
2165
builder.build_snapshot('C-id', ['A-id'], [])
2166
builder.build_snapshot('B-id', ['A-id'], [])
2167
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2168
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
2263
2169
# Have to use a real WT, because BranchBuilder doesn't support exec bit
2264
2170
wt = self.get_wt_from_builder(builder)
2265
2171
tt = transform.TreeTransform(wt)
2267
tt.set_executability(True, tt.trans_id_tree_path('foo'))
2173
tt.set_executability(True, tt.trans_id_tree_file_id('foo-id'))
2272
self.assertTrue(wt.is_executable('foo'))
2273
wt.commit('F-id', rev_id=b'F-id')
2178
self.assertTrue(wt.is_executable('foo-id'))
2179
wt.commit('F-id', rev_id='F-id')
2274
2180
# Reset to D, so that we can merge F
2275
2181
wt.set_parent_ids(['D-id'])
2276
2182
wt.branch.set_last_revision_info(3, 'D-id')
2278
self.assertFalse(wt.is_executable('foo'))
2184
self.assertFalse(wt.is_executable('foo-id'))
2279
2185
conflicts = wt.merge_from_branch(wt.branch, to_revision='F-id')
2280
2186
self.assertEqual(0, conflicts)
2281
self.assertTrue(wt.is_executable('foo'))
2187
self.assertTrue(wt.is_executable('foo-id'))
2283
2189
def test_create_symlink(self):
2284
2190
self.requireFeature(features.SymlinkFeature)
2294
2200
# Have to use a real WT, because BranchBuilder and MemoryTree don't
2295
2201
# have symlink support
2296
2202
builder = self.get_builder()
2297
builder.build_snapshot(None,
2298
[('add', (u'', 'a-root-id', 'directory', None))],
2300
builder.build_snapshot(['A-id'], [], revision_id='C-id')
2301
builder.build_snapshot(['A-id'], [], revision_id='B-id')
2302
builder.build_snapshot(['B-id', 'C-id'], [], revision_id='D-id')
2303
builder.build_snapshot(['C-id', 'B-id'], [], revision_id='E-id')
2203
builder.build_snapshot('A-id', None,
2204
[('add', (u'', 'a-root-id', 'directory', None))])
2205
builder.build_snapshot('C-id', ['A-id'], [])
2206
builder.build_snapshot('B-id', ['A-id'], [])
2207
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2208
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
2304
2209
# Have to use a real WT, because BranchBuilder doesn't support exec bit
2305
2210
wt = self.get_wt_from_builder(builder)
2306
2211
os.symlink('bar', 'path/foo')
2307
wt.add(['foo'], [b'foo-id'])
2308
self.assertEqual('bar', wt.get_symlink_target('foo'))
2309
wt.commit('add symlink', rev_id=b'F-id')
2212
wt.add(['foo'], ['foo-id'])
2213
self.assertEqual('bar', wt.get_symlink_target('foo-id'))
2214
wt.commit('add symlink', rev_id='F-id')
2310
2215
# Reset to D, so that we can merge F
2311
wt.set_parent_ids([b'D-id'])
2312
wt.branch.set_last_revision_info(3, b'D-id')
2216
wt.set_parent_ids(['D-id'])
2217
wt.branch.set_last_revision_info(3, 'D-id')
2314
self.assertFalse(wt.is_versioned('foo'))
2315
conflicts = wt.merge_from_branch(wt.branch, to_revision=b'F-id')
2219
self.assertIs(None, wt.path2id('foo'))
2220
conflicts = wt.merge_from_branch(wt.branch, to_revision='F-id')
2316
2221
self.assertEqual(0, conflicts)
2317
self.assertEqual(b'foo-id', wt.path2id('foo'))
2318
self.assertEqual('bar', wt.get_symlink_target('foo'))
2222
self.assertEqual('foo-id', wt.path2id('foo'))
2223
self.assertEqual('bar', wt.get_symlink_target('foo-id'))
2320
2225
def test_both_sides_revert(self):
2321
2226
# Both sides of a criss-cross revert the text to the lca
2328
2233
# This must be done with a real WorkingTree, because normally their
2329
2234
# inventory contains "None" rather than a real sha1
2330
2235
builder = self.get_builder()
2331
builder.build_snapshot(None,
2236
builder.build_snapshot('A-id', None,
2332
2237
[('add', (u'', 'a-root-id', 'directory', None)),
2333
('add', (u'foo', 'foo-id', 'file', 'A content\n'))],
2335
builder.build_snapshot(['A-id'],
2336
[('modify', ('foo', 'B content\n'))],
2338
builder.build_snapshot(['A-id'],
2339
[('modify', ('foo', 'C content\n'))],
2341
builder.build_snapshot(['C-id', 'B-id'], [],
2343
builder.build_snapshot(['B-id', 'C-id'], [],
2238
('add', (u'foo', 'foo-id', 'file', 'A content\n'))])
2239
builder.build_snapshot('B-id', ['A-id'],
2240
[('modify', ('foo-id', 'B content\n'))])
2241
builder.build_snapshot('C-id', ['A-id'],
2242
[('modify', ('foo-id', 'C content\n'))])
2243
builder.build_snapshot('E-id', ['C-id', 'B-id'], [])
2244
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2345
2245
wt, conflicts = self.do_merge(builder, 'E-id')
2346
2246
self.assertEqual(1, conflicts)
2347
2247
self.assertEqualDiff('<<<<<<< TREE\n'
2372
2272
wt.lock_write()
2373
2273
self.addCleanup(wt.unlock)
2374
2274
os.symlink('bar', 'path/foo')
2375
wt.add(['foo'], [b'foo-id'])
2376
wt.commit('add symlink', rev_id=b'A-id')
2275
wt.add(['foo'], ['foo-id'])
2276
wt.commit('add symlink', rev_id='A-id')
2377
2277
os.remove('path/foo')
2378
2278
os.symlink('baz', 'path/foo')
2379
wt.commit('foo => baz', rev_id=b'B-id')
2380
wt.set_last_revision(b'A-id')
2279
wt.commit('foo => baz', rev_id='B-id')
2280
wt.set_last_revision('A-id')
2381
2281
wt.branch.set_last_revision_info(1, 'A-id')
2383
wt.commit('C', rev_id=b'C-id')
2384
wt.merge_from_branch(wt.branch, b'B-id')
2385
self.assertEqual('baz', wt.get_symlink_target('foo'))
2386
wt.commit('E merges C & B', rev_id=b'E-id')
2283
wt.commit('C', rev_id='C-id')
2284
wt.merge_from_branch(wt.branch, 'B-id')
2285
self.assertEqual('baz', wt.get_symlink_target('foo-id'))
2286
wt.commit('E merges C & B', rev_id='E-id')
2387
2287
os.remove('path/foo')
2388
2288
os.symlink('bing', 'path/foo')
2389
wt.commit('F foo => bing', rev_id=b'F-id')
2289
wt.commit('F foo => bing', rev_id='F-id')
2390
2290
wt.set_last_revision('B-id')
2391
wt.branch.set_last_revision_info(2, b'B-id')
2291
wt.branch.set_last_revision_info(2, 'B-id')
2393
wt.merge_from_branch(wt.branch, b'C-id')
2394
wt.commit('D merges B & C', rev_id=b'D-id')
2395
conflicts = wt.merge_from_branch(wt.branch, to_revision=b'F-id')
2293
wt.merge_from_branch(wt.branch, 'C-id')
2294
wt.commit('D merges B & C', rev_id='D-id')
2295
conflicts = wt.merge_from_branch(wt.branch, to_revision='F-id')
2396
2296
self.assertEqual(0, conflicts)
2397
self.assertEqual('bing', wt.get_symlink_target('foo'))
2297
self.assertEqual('bing', wt.get_symlink_target('foo-id'))
2399
2299
def test_renamed_symlink(self):
2400
2300
self.requireFeature(features.SymlinkFeature)
2415
2315
wt.lock_write()
2416
2316
self.addCleanup(wt.unlock)
2417
2317
os.symlink('bar', 'path/foo')
2418
wt.add(['foo'], [b'foo-id'])
2419
wt.commit('A add symlink', rev_id=b'A-id')
2318
wt.add(['foo'], ['foo-id'])
2319
wt.commit('A add symlink', rev_id='A-id')
2420
2320
wt.rename_one('foo', 'barry')
2421
wt.commit('B foo => barry', rev_id=b'B-id')
2321
wt.commit('B foo => barry', rev_id='B-id')
2422
2322
wt.set_last_revision('A-id')
2423
wt.branch.set_last_revision_info(1, b'A-id')
2323
wt.branch.set_last_revision_info(1, 'A-id')
2425
wt.commit('C', rev_id=b'C-id')
2426
wt.merge_from_branch(wt.branch, b'B-id')
2427
self.assertEqual('barry', wt.id2path(b'foo-id'))
2428
self.assertEqual('bar', wt.get_symlink_target('barry'))
2429
wt.commit('E merges C & B', rev_id=b'E-id')
2325
wt.commit('C', rev_id='C-id')
2326
wt.merge_from_branch(wt.branch, 'B-id')
2327
self.assertEqual('barry', wt.id2path('foo-id'))
2328
self.assertEqual('bar', wt.get_symlink_target('foo-id'))
2329
wt.commit('E merges C & B', rev_id='E-id')
2430
2330
wt.rename_one('barry', 'blah')
2431
wt.commit('F barry => blah', rev_id=b'F-id')
2432
wt.set_last_revision(b'B-id')
2433
wt.branch.set_last_revision_info(2, b'B-id')
2331
wt.commit('F barry => blah', rev_id='F-id')
2332
wt.set_last_revision('B-id')
2333
wt.branch.set_last_revision_info(2, 'B-id')
2435
wt.merge_from_branch(wt.branch, b'C-id')
2436
wt.commit('D merges B & C', rev_id=b'D-id')
2437
self.assertEqual('barry', wt.id2path(b'foo-id'))
2335
wt.merge_from_branch(wt.branch, 'C-id')
2336
wt.commit('D merges B & C', rev_id='D-id')
2337
self.assertEqual('barry', wt.id2path('foo-id'))
2438
2338
# Check the output of the Merger object directly
2439
merger = _mod_merge.Merger.from_revision_ids(wt, b'F-id')
2339
merger = _mod_merge.Merger.from_revision_ids(None,
2440
2341
merger.merge_type = _mod_merge.Merge3Merger
2441
2342
merge_obj = merger.make_merger()
2442
2343
root_id = wt.path2id('')
2443
2344
entries = list(merge_obj._entries_lca())
2444
2345
# No content change, just a path change
2445
self.assertEqual([(b'foo-id', False,
2446
((u'foo', [u'barry', u'foo']), u'blah', u'barry'),
2346
self.assertEqual([('foo-id', False,
2447
2347
((root_id, [root_id, root_id]), root_id, root_id),
2448
2348
((u'foo', [u'barry', u'foo']), u'blah', u'barry'),
2449
2349
((False, [False, False]), False, False)),
2451
conflicts = wt.merge_from_branch(wt.branch, to_revision=b'F-id')
2351
conflicts = wt.merge_from_branch(wt.branch, to_revision='F-id')
2452
2352
self.assertEqual(0, conflicts)
2453
self.assertEqual('blah', wt.id2path(b'foo-id'))
2353
self.assertEqual('blah', wt.id2path('foo-id'))
2455
2355
def test_symlink_no_content_change(self):
2456
2356
self.requireFeature(features.SymlinkFeature)
2470
2370
wt.lock_write()
2471
2371
self.addCleanup(wt.unlock)
2472
2372
os.symlink('bar', 'path/foo')
2473
wt.add(['foo'], [b'foo-id'])
2474
wt.commit('add symlink', rev_id=b'A-id')
2373
wt.add(['foo'], ['foo-id'])
2374
wt.commit('add symlink', rev_id='A-id')
2475
2375
os.remove('path/foo')
2476
2376
os.symlink('baz', 'path/foo')
2477
wt.commit('foo => baz', rev_id=b'B-id')
2478
wt.set_last_revision(b'A-id')
2479
wt.branch.set_last_revision_info(1, b'A-id')
2481
wt.commit('C', rev_id=b'C-id')
2482
wt.merge_from_branch(wt.branch, b'B-id')
2483
self.assertEqual('baz', wt.get_symlink_target('foo'))
2484
wt.commit('E merges C & B', rev_id=b'E-id')
2485
wt.set_last_revision(b'B-id')
2486
wt.branch.set_last_revision_info(2, b'B-id')
2488
wt.merge_from_branch(wt.branch, b'C-id')
2489
wt.commit('D merges B & C', rev_id=b'D-id')
2377
wt.commit('foo => baz', rev_id='B-id')
2378
wt.set_last_revision('A-id')
2379
wt.branch.set_last_revision_info(1, 'A-id')
2381
wt.commit('C', rev_id='C-id')
2382
wt.merge_from_branch(wt.branch, 'B-id')
2383
self.assertEqual('baz', wt.get_symlink_target('foo-id'))
2384
wt.commit('E merges C & B', rev_id='E-id')
2385
wt.set_last_revision('B-id')
2386
wt.branch.set_last_revision_info(2, 'B-id')
2388
wt.merge_from_branch(wt.branch, 'C-id')
2389
wt.commit('D merges B & C', rev_id='D-id')
2490
2390
os.remove('path/foo')
2491
2391
os.symlink('bing', 'path/foo')
2492
wt.commit('F foo => bing', rev_id=b'F-id')
2392
wt.commit('F foo => bing', rev_id='F-id')
2494
2394
# Check the output of the Merger object directly
2495
merger = _mod_merge.Merger.from_revision_ids(wt, b'E-id')
2395
merger = _mod_merge.Merger.from_revision_ids(None,
2496
2397
merger.merge_type = _mod_merge.Merge3Merger
2497
2398
merge_obj = merger.make_merger()
2498
2399
# Nothing interesting happened in OTHER relative to BASE
2499
2400
self.assertEqual([], list(merge_obj._entries_lca()))
2500
2401
# Now do a real merge, just to test the rest of the stack
2501
conflicts = wt.merge_from_branch(wt.branch, to_revision=b'E-id')
2402
conflicts = wt.merge_from_branch(wt.branch, to_revision='E-id')
2502
2403
self.assertEqual(0, conflicts)
2503
self.assertEqual('bing', wt.get_symlink_target('foo'))
2404
self.assertEqual('bing', wt.get_symlink_target('foo-id'))
2505
2406
def test_symlink_this_changed_kind(self):
2506
2407
self.requireFeature(features.SymlinkFeature)
2517
2418
wt = self.make_branch_and_tree('path')
2518
2419
wt.lock_write()
2519
2420
self.addCleanup(wt.unlock)
2520
wt.commit('base', rev_id=b'A-id')
2421
wt.commit('base', rev_id='A-id')
2521
2422
os.symlink('bar', 'path/foo')
2522
wt.add(['foo'], [b'foo-id'])
2523
wt.commit('add symlink foo => bar', rev_id=b'B-id')
2524
wt.set_last_revision(b'A-id')
2525
wt.branch.set_last_revision_info(1, b'A-id')
2423
wt.add(['foo'], ['foo-id'])
2424
wt.commit('add symlink foo => bar', rev_id='B-id')
2425
wt.set_last_revision('A-id')
2426
wt.branch.set_last_revision_info(1, 'A-id')
2527
wt.commit('C', rev_id=b'C-id')
2528
wt.merge_from_branch(wt.branch, b'B-id')
2529
self.assertEqual('bar', wt.get_symlink_target('foo'))
2428
wt.commit('C', rev_id='C-id')
2429
wt.merge_from_branch(wt.branch, 'B-id')
2430
self.assertEqual('bar', wt.get_symlink_target('foo-id'))
2530
2431
os.remove('path/foo')
2531
2432
# We have to change the link in E, or it won't try to do a comparison
2532
2433
os.symlink('bing', 'path/foo')
2533
wt.commit('E merges C & B, overrides to bing', rev_id=b'E-id')
2434
wt.commit('E merges C & B, overrides to bing', rev_id='E-id')
2534
2435
wt.set_last_revision('B-id')
2535
wt.branch.set_last_revision_info(2, b'B-id')
2436
wt.branch.set_last_revision_info(2, 'B-id')
2537
wt.merge_from_branch(wt.branch, b'C-id')
2438
wt.merge_from_branch(wt.branch, 'C-id')
2538
2439
os.remove('path/foo')
2539
self.build_tree_contents([('path/foo', b'file content\n')])
2440
self.build_tree_contents([('path/foo', 'file content\n')])
2540
2441
# XXX: workaround, WT doesn't detect kind changes unless you do
2541
2442
# iter_changes()
2542
2443
list(wt.iter_changes(wt.basis_tree()))
2543
wt.commit('D merges B & C, makes it a file', rev_id=b'D-id')
2444
wt.commit('D merges B & C, makes it a file', rev_id='D-id')
2545
merger = _mod_merge.Merger.from_revision_ids(wt, b'E-id')
2446
merger = _mod_merge.Merger.from_revision_ids(None,
2546
2448
merger.merge_type = _mod_merge.Merge3Merger
2547
2449
merge_obj = merger.make_merger()
2548
2450
entries = list(merge_obj._entries_lca())
2549
2451
root_id = wt.path2id('')
2550
self.assertEqual([(b'foo-id', True,
2551
((None, [u'foo', None]), u'foo', u'foo'),
2452
self.assertEqual([('foo-id', True,
2552
2453
((None, [root_id, None]), root_id, root_id),
2553
2454
((None, [u'foo', None]), u'foo', u'foo'),
2554
2455
((None, [False, None]), False, False)),
2573
2474
wt.lock_write()
2574
2475
self.addCleanup(wt.unlock)
2575
2476
os.symlink('bar', 'path/foo')
2576
wt.add(['foo'], [b'foo-id'])
2577
wt.commit('add symlink', rev_id=b'A-id')
2477
wt.add(['foo'], ['foo-id'])
2478
wt.commit('add symlink', rev_id='A-id')
2578
2479
os.remove('path/foo')
2579
2480
os.symlink('baz', 'path/foo')
2580
wt.commit('foo => baz', rev_id=b'B-id')
2581
wt.set_last_revision(b'A-id')
2582
wt.branch.set_last_revision_info(1, b'A-id')
2481
wt.commit('foo => baz', rev_id='B-id')
2482
wt.set_last_revision('A-id')
2483
wt.branch.set_last_revision_info(1, 'A-id')
2584
wt.commit('C', rev_id=b'C-id')
2585
wt.merge_from_branch(wt.branch, b'B-id')
2586
self.assertEqual('baz', wt.get_symlink_target('foo'))
2587
wt.commit('E merges C & B', rev_id=b'E-id')
2485
wt.commit('C', rev_id='C-id')
2486
wt.merge_from_branch(wt.branch, 'B-id')
2487
self.assertEqual('baz', wt.get_symlink_target('foo-id'))
2488
wt.commit('E merges C & B', rev_id='E-id')
2588
2489
os.remove('path/foo')
2589
2490
os.symlink('bing', 'path/foo')
2590
wt.commit('F foo => bing', rev_id=b'F-id')
2491
wt.commit('F foo => bing', rev_id='F-id')
2591
2492
wt.set_last_revision('B-id')
2592
wt.branch.set_last_revision_info(2, b'B-id')
2493
wt.branch.set_last_revision_info(2, 'B-id')
2594
wt.merge_from_branch(wt.branch, b'C-id')
2595
wt.commit('D merges B & C', rev_id=b'D-id')
2596
wt_base = wt.controldir.sprout('base', b'A-id').open_workingtree()
2495
wt.merge_from_branch(wt.branch, 'C-id')
2496
wt.commit('D merges B & C', rev_id='D-id')
2497
wt_base = wt.bzrdir.sprout('base', 'A-id').open_workingtree()
2597
2498
wt_base.lock_read()
2598
2499
self.addCleanup(wt_base.unlock)
2599
wt_lca1 = wt.controldir.sprout('b-tree', b'B-id').open_workingtree()
2500
wt_lca1 = wt.bzrdir.sprout('b-tree', 'B-id').open_workingtree()
2600
2501
wt_lca1.lock_read()
2601
2502
self.addCleanup(wt_lca1.unlock)
2602
wt_lca2 = wt.controldir.sprout('c-tree', b'C-id').open_workingtree()
2503
wt_lca2 = wt.bzrdir.sprout('c-tree', 'C-id').open_workingtree()
2603
2504
wt_lca2.lock_read()
2604
2505
self.addCleanup(wt_lca2.unlock)
2605
wt_other = wt.controldir.sprout('other', b'F-id').open_workingtree()
2506
wt_other = wt.bzrdir.sprout('other', 'F-id').open_workingtree()
2606
2507
wt_other.lock_read()
2607
2508
self.addCleanup(wt_other.unlock)
2608
2509
merge_obj = _mod_merge.Merge3Merger(wt, wt, wt_base,
2609
2510
wt_other, lca_trees=[wt_lca1, wt_lca2], do_merge=False)
2610
2511
entries = list(merge_obj._entries_lca())
2611
2512
root_id = wt.path2id('')
2612
self.assertEqual([(b'foo-id', True,
2613
((u'foo', [u'foo', u'foo']), u'foo', u'foo'),
2513
self.assertEqual([('foo-id', True,
2614
2514
((root_id, [root_id, root_id]), root_id, root_id),
2615
2515
((u'foo', [u'foo', u'foo']), u'foo', u'foo'),
2616
2516
((False, [False, False]), False, False)),
2628
2528
# F Path at 'foo'
2629
2529
builder = self.get_builder()
2630
builder.build_snapshot(None,
2631
[('add', (u'', b'a-root-id', 'directory', None)),
2632
('add', (u'foo', b'foo-id', 'file', 'a\nb\nc\n'))],
2633
revision_id=b'A-id')
2634
builder.build_snapshot([b'A-id'], [], revision_id=b'C-id')
2635
builder.build_snapshot([b'A-id'],
2636
[('rename', ('foo', 'bar'))], revision_id=b'B-id')
2637
builder.build_snapshot([b'C-id', b'B-id'],
2638
[('rename', ('foo', 'bar'))], revision_id=b'E-id') # merge the rename
2639
builder.build_snapshot([b'E-id'],
2640
[('rename', ('bar', 'foo'))], revision_id=b'F-id') # Rename back to BASE
2641
builder.build_snapshot([b'B-id', b'C-id'], [], revision_id=b'D-id')
2642
wt, conflicts = self.do_merge(builder, b'F-id')
2530
builder.build_snapshot('A-id', None,
2531
[('add', (u'', 'a-root-id', 'directory', None)),
2532
('add', (u'foo', 'foo-id', 'file', 'a\nb\nc\n'))])
2533
builder.build_snapshot('C-id', ['A-id'], [])
2534
builder.build_snapshot('B-id', ['A-id'],
2535
[('rename', ('foo', 'bar'))])
2536
builder.build_snapshot('E-id', ['C-id', 'B-id'],
2537
[('rename', ('foo', 'bar'))]) # merge the rename
2538
builder.build_snapshot('F-id', ['E-id'],
2539
[('rename', ('bar', 'foo'))]) # Rename back to BASE
2540
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2541
wt, conflicts = self.do_merge(builder, 'F-id')
2643
2542
self.assertEqual(0, conflicts)
2644
self.assertEqual('foo', wt.id2path(b'foo-id'))
2543
self.assertEqual('foo', wt.id2path('foo-id'))
2646
2545
def test_other_reverted_content_to_base(self):
2647
2546
builder = self.get_builder()
2648
builder.build_snapshot(None,
2649
[('add', (u'', b'a-root-id', 'directory', None)),
2650
('add', (u'foo', b'foo-id', 'file', b'base content\n'))],
2651
revision_id=b'A-id')
2652
builder.build_snapshot([b'A-id'], [], revision_id=b'C-id')
2653
builder.build_snapshot([b'A-id'],
2654
[('modify', ('foo', b'B content\n'))],
2655
revision_id=b'B-id')
2656
builder.build_snapshot([b'C-id', b'B-id'],
2657
[('modify', ('foo', b'B content\n'))],
2658
revision_id=b'E-id') # merge the content
2659
builder.build_snapshot([b'E-id'],
2660
[('modify', ('foo', b'base content\n'))],
2661
revision_id=b'F-id') # Revert back to BASE
2662
builder.build_snapshot([b'B-id', b'C-id'], [], revision_id=b'D-id')
2663
wt, conflicts = self.do_merge(builder, b'F-id')
2547
builder.build_snapshot('A-id', None,
2548
[('add', (u'', 'a-root-id', 'directory', None)),
2549
('add', (u'foo', 'foo-id', 'file', 'base content\n'))])
2550
builder.build_snapshot('C-id', ['A-id'], [])
2551
builder.build_snapshot('B-id', ['A-id'],
2552
[('modify', ('foo-id', 'B content\n'))])
2553
builder.build_snapshot('E-id', ['C-id', 'B-id'],
2554
[('modify', ('foo-id', 'B content\n'))]) # merge the content
2555
builder.build_snapshot('F-id', ['E-id'],
2556
[('modify', ('foo-id', 'base content\n'))]) # Revert back to BASE
2557
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2558
wt, conflicts = self.do_merge(builder, 'F-id')
2664
2559
self.assertEqual(0, conflicts)
2665
2560
# TODO: We need to use the per-file graph to properly select a BASE
2666
2561
# before this will work. Or at least use the LCA trees to find
2667
2562
# the appropriate content base. (which is B, not A).
2668
self.assertEqual('base content\n', wt.get_file_text('foo'))
2563
self.assertEqual('base content\n', wt.get_file_text('foo-id'))
2670
2565
def test_other_modified_content(self):
2671
2566
builder = self.get_builder()
2672
builder.build_snapshot(None,
2673
[('add', (u'', b'a-root-id', 'directory', None)),
2674
('add', (u'foo', b'foo-id', 'file', b'base content\n'))],
2675
revision_id=b'A-id')
2676
builder.build_snapshot([b'A-id'], [], revision_id=b'C-id')
2677
builder.build_snapshot([b'A-id'],
2678
[('modify', ('foo', b'B content\n'))],
2679
revision_id=b'B-id')
2680
builder.build_snapshot([b'C-id', b'B-id'],
2681
[('modify', ('foo', b'B content\n'))],
2682
revision_id=b'E-id') # merge the content
2683
builder.build_snapshot([b'E-id'],
2684
[('modify', ('foo', b'F content\n'))],
2685
revision_id=b'F-id') # Override B content
2686
builder.build_snapshot([b'B-id', b'C-id'], [], revision_id=b'D-id')
2687
wt, conflicts = self.do_merge(builder, b'F-id')
2567
builder.build_snapshot('A-id', None,
2568
[('add', (u'', 'a-root-id', 'directory', None)),
2569
('add', (u'foo', 'foo-id', 'file', 'base content\n'))])
2570
builder.build_snapshot('C-id', ['A-id'], [])
2571
builder.build_snapshot('B-id', ['A-id'],
2572
[('modify', ('foo-id', 'B content\n'))])
2573
builder.build_snapshot('E-id', ['C-id', 'B-id'],
2574
[('modify', ('foo-id', 'B content\n'))]) # merge the content
2575
builder.build_snapshot('F-id', ['E-id'],
2576
[('modify', ('foo-id', 'F content\n'))]) # Override B content
2577
builder.build_snapshot('D-id', ['B-id', 'C-id'], [])
2578
wt, conflicts = self.do_merge(builder, 'F-id')
2688
2579
self.assertEqual(0, conflicts)
2689
self.assertEqual(b'F content\n', wt.get_file_text('foo'))
2580
self.assertEqual('F content\n', wt.get_file_text('foo-id'))
2691
2582
def test_all_wt(self):
2692
2583
"""Check behavior if all trees are Working Trees."""
2700
2591
# D E E updates content, renames 'b' => 'c'
2701
2592
builder = self.get_builder()
2702
builder.build_snapshot(None,
2593
builder.build_snapshot('A-id', None,
2703
2594
[('add', (u'', 'a-root-id', 'directory', None)),
2704
2595
('add', (u'a', 'a-id', 'file', 'base content\n')),
2705
('add', (u'foo', 'foo-id', 'file', 'base content\n'))],
2707
builder.build_snapshot(['A-id'],
2708
[('modify', ('foo', 'B content\n'))],
2710
builder.build_snapshot(['A-id'],
2711
[('rename', ('a', 'b'))],
2713
builder.build_snapshot(['C-id', 'B-id'],
2596
('add', (u'foo', 'foo-id', 'file', 'base content\n'))])
2597
builder.build_snapshot('B-id', ['A-id'],
2598
[('modify', ('foo-id', 'B content\n'))])
2599
builder.build_snapshot('C-id', ['A-id'],
2600
[('rename', ('a', 'b'))])
2601
builder.build_snapshot('E-id', ['C-id', 'B-id'],
2714
2602
[('rename', ('b', 'c')),
2715
('modify', ('foo', 'E content\n'))],
2717
builder.build_snapshot(['B-id', 'C-id'],
2718
[('rename', ('a', 'b'))], revision_id='D-id') # merged change
2603
('modify', ('foo-id', 'E content\n'))])
2604
builder.build_snapshot('D-id', ['B-id', 'C-id'],
2605
[('rename', ('a', 'b'))]) # merged change
2719
2606
wt_this = self.get_wt_from_builder(builder)
2720
wt_base = wt_this.controldir.sprout('base', 'A-id').open_workingtree()
2607
wt_base = wt_this.bzrdir.sprout('base', 'A-id').open_workingtree()
2721
2608
wt_base.lock_read()
2722
2609
self.addCleanup(wt_base.unlock)
2723
wt_lca1 = wt_this.controldir.sprout('b-tree', 'B-id').open_workingtree()
2610
wt_lca1 = wt_this.bzrdir.sprout('b-tree', 'B-id').open_workingtree()
2724
2611
wt_lca1.lock_read()
2725
2612
self.addCleanup(wt_lca1.unlock)
2726
wt_lca2 = wt_this.controldir.sprout('c-tree', 'C-id').open_workingtree()
2613
wt_lca2 = wt_this.bzrdir.sprout('c-tree', 'C-id').open_workingtree()
2727
2614
wt_lca2.lock_read()
2728
2615
self.addCleanup(wt_lca2.unlock)
2729
wt_other = wt_this.controldir.sprout('other', 'E-id').open_workingtree()
2616
wt_other = wt_this.bzrdir.sprout('other', 'E-id').open_workingtree()
2730
2617
wt_other.lock_read()
2731
2618
self.addCleanup(wt_other.unlock)
2732
2619
merge_obj = _mod_merge.Merge3Merger(wt_this, wt_this, wt_base,
2754
2639
self.addCleanup(wt.unlock)
2755
2640
sub_tree = self.make_branch_and_tree('tree/sub-tree',
2756
2641
format='development-subtree')
2757
wt.set_root_id(b'a-root-id')
2758
sub_tree.set_root_id(b'sub-tree-root')
2759
self.build_tree_contents([('tree/sub-tree/file', b'text1')])
2642
wt.set_root_id('a-root-id')
2643
sub_tree.set_root_id('sub-tree-root')
2644
self.build_tree_contents([('tree/sub-tree/file', 'text1')])
2760
2645
sub_tree.add('file')
2761
sub_tree.commit('foo', rev_id=b'sub-A-id')
2646
sub_tree.commit('foo', rev_id='sub-A-id')
2762
2647
wt.add_reference(sub_tree)
2763
wt.commit('set text to 1', rev_id=b'A-id', recursive=None)
2648
wt.commit('set text to 1', rev_id='A-id', recursive=None)
2764
2649
# Now create a criss-cross merge in the parent, without modifying the
2766
wt.commit('B', rev_id=b'B-id', recursive=None)
2651
wt.commit('B', rev_id='B-id', recursive=None)
2767
2652
wt.set_last_revision('A-id')
2768
2653
wt.branch.set_last_revision_info(1, 'A-id')
2769
wt.commit('C', rev_id=b'C-id', recursive=None)
2654
wt.commit('C', rev_id='C-id', recursive=None)
2770
2655
wt.merge_from_branch(wt.branch, to_revision='B-id')
2771
wt.commit('E', rev_id=b'E-id', recursive=None)
2656
wt.commit('E', rev_id='E-id', recursive=None)
2772
2657
wt.set_parent_ids(['B-id', 'C-id'])
2773
2658
wt.branch.set_last_revision_info(2, 'B-id')
2774
wt.commit('D', rev_id=b'D-id', recursive=None)
2659
wt.commit('D', rev_id='D-id', recursive=None)
2776
merger = _mod_merge.Merger.from_revision_ids(wt, 'E-id')
2661
merger = _mod_merge.Merger.from_revision_ids(None,
2777
2663
merger.merge_type = _mod_merge.Merge3Merger
2778
2664
merge_obj = merger.make_merger()
2779
2665
entries = list(merge_obj._entries_lca())
2788
2674
self.addCleanup(wt.unlock)
2789
2675
sub_tree = self.make_branch_and_tree('tree/sub',
2790
2676
format='development-subtree')
2791
wt.set_root_id(b'a-root-id')
2792
sub_tree.set_root_id(b'sub-tree-root')
2793
self.build_tree_contents([('tree/sub/file', b'text1')])
2677
wt.set_root_id('a-root-id')
2678
sub_tree.set_root_id('sub-tree-root')
2679
self.build_tree_contents([('tree/sub/file', 'text1')])
2794
2680
sub_tree.add('file')
2795
sub_tree.commit('foo', rev_id=b'sub-A-id')
2681
sub_tree.commit('foo', rev_id='sub-A-id')
2796
2682
wt.add_reference(sub_tree)
2797
wt.commit('set text to 1', rev_id=b'A-id', recursive=None)
2683
wt.commit('set text to 1', rev_id='A-id', recursive=None)
2798
2684
# Now create a criss-cross merge in the parent, without modifying the
2800
wt.commit('B', rev_id=b'B-id', recursive=None)
2686
wt.commit('B', rev_id='B-id', recursive=None)
2801
2687
wt.set_last_revision('A-id')
2802
2688
wt.branch.set_last_revision_info(1, 'A-id')
2803
wt.commit('C', rev_id=b'C-id', recursive=None)
2689
wt.commit('C', rev_id='C-id', recursive=None)
2804
2690
wt.merge_from_branch(wt.branch, to_revision='B-id')
2805
self.build_tree_contents([('tree/sub/file', b'text2')])
2806
sub_tree.commit('modify contents', rev_id=b'sub-B-id')
2807
wt.commit('E', rev_id=b'E-id', recursive=None)
2691
self.build_tree_contents([('tree/sub/file', 'text2')])
2692
sub_tree.commit('modify contents', rev_id='sub-B-id')
2693
wt.commit('E', rev_id='E-id', recursive=None)
2808
2694
wt.set_parent_ids(['B-id', 'C-id'])
2809
2695
wt.branch.set_last_revision_info(2, 'B-id')
2810
wt.commit('D', rev_id=b'D-id', recursive=None)
2696
wt.commit('D', rev_id='D-id', recursive=None)
2812
merger = _mod_merge.Merger.from_revision_ids(wt, 'E-id')
2698
merger = _mod_merge.Merger.from_revision_ids(None,
2813
2700
merger.merge_type = _mod_merge.Merge3Merger
2814
2701
merge_obj = merger.make_merger()
2815
2702
entries = list(merge_obj._entries_lca())
2826
2713
self.addCleanup(wt.unlock)
2827
2714
sub_tree = self.make_branch_and_tree('tree/sub',
2828
2715
format='development-subtree')
2829
wt.set_root_id(b'a-root-id')
2830
sub_tree.set_root_id(b'sub-tree-root')
2831
self.build_tree_contents([('tree/sub/file', b'text1')])
2716
wt.set_root_id('a-root-id')
2717
sub_tree.set_root_id('sub-tree-root')
2718
self.build_tree_contents([('tree/sub/file', 'text1')])
2832
2719
sub_tree.add('file')
2833
sub_tree.commit('foo', rev_id=b'sub-A-id')
2720
sub_tree.commit('foo', rev_id='sub-A-id')
2834
2721
wt.add_reference(sub_tree)
2835
wt.commit('set text to 1', rev_id=b'A-id', recursive=None)
2722
wt.commit('set text to 1', rev_id='A-id', recursive=None)
2836
2723
# Now create a criss-cross merge in the parent, without modifying the
2838
wt.commit('B', rev_id=b'B-id', recursive=None)
2725
wt.commit('B', rev_id='B-id', recursive=None)
2839
2726
wt.set_last_revision('A-id')
2840
2727
wt.branch.set_last_revision_info(1, 'A-id')
2841
wt.commit('C', rev_id=b'C-id', recursive=None)
2728
wt.commit('C', rev_id='C-id', recursive=None)
2842
2729
wt.merge_from_branch(wt.branch, to_revision='B-id')
2843
2730
wt.rename_one('sub', 'alt_sub')
2844
wt.commit('E', rev_id=b'E-id', recursive=None)
2731
wt.commit('E', rev_id='E-id', recursive=None)
2845
2732
wt.set_last_revision('B-id')
2847
2734
wt.set_parent_ids(['B-id', 'C-id'])
2848
2735
wt.branch.set_last_revision_info(2, 'B-id')
2849
wt.commit('D', rev_id=b'D-id', recursive=None)
2736
wt.commit('D', rev_id='D-id', recursive=None)
2851
merger = _mod_merge.Merger.from_revision_ids(wt, 'E-id')
2738
merger = _mod_merge.Merger.from_revision_ids(None,
2852
2740
merger.merge_type = _mod_merge.Merge3Merger
2853
2741
merge_obj = merger.make_merger()
2854
2742
entries = list(merge_obj._entries_lca())
2855
2743
root_id = 'a-root-id'
2856
2744
self.assertEqual([('sub-tree-root', False,
2857
((u'sub', [u'sub', u'sub']), u'alt_sub', u'sub'),
2858
2745
((root_id, [root_id, root_id]), root_id, root_id),
2859
2746
((u'sub', [u'sub', u'sub']), u'alt_sub', u'sub'),
2860
2747
((False, [False, False]), False, False)),
2869
2756
self.addCleanup(wt.unlock)
2870
2757
sub_tree = self.make_branch_and_tree('tree/sub',
2871
2758
format='development-subtree')
2872
wt.set_root_id(b'a-root-id')
2873
sub_tree.set_root_id(b'sub-tree-root')
2874
self.build_tree_contents([('tree/sub/file', b'text1')])
2759
wt.set_root_id('a-root-id')
2760
sub_tree.set_root_id('sub-tree-root')
2761
self.build_tree_contents([('tree/sub/file', 'text1')])
2875
2762
sub_tree.add('file')
2876
sub_tree.commit('foo', rev_id=b'sub-A-id')
2763
sub_tree.commit('foo', rev_id='sub-A-id')
2877
2764
wt.add_reference(sub_tree)
2878
wt.commit('set text to 1', rev_id=b'A-id', recursive=None)
2765
wt.commit('set text to 1', rev_id='A-id', recursive=None)
2879
2766
# Now create a criss-cross merge in the parent, without modifying the
2881
wt.commit('B', rev_id=b'B-id', recursive=None)
2768
wt.commit('B', rev_id='B-id', recursive=None)
2882
2769
wt.set_last_revision('A-id')
2883
2770
wt.branch.set_last_revision_info(1, 'A-id')
2884
wt.commit('C', rev_id=b'C-id', recursive=None)
2771
wt.commit('C', rev_id='C-id', recursive=None)
2885
2772
wt.merge_from_branch(wt.branch, to_revision='B-id')
2886
self.build_tree_contents([('tree/sub/file', b'text2')])
2887
sub_tree.commit('modify contents', rev_id=b'sub-B-id')
2773
self.build_tree_contents([('tree/sub/file', 'text2')])
2774
sub_tree.commit('modify contents', rev_id='sub-B-id')
2888
2775
wt.rename_one('sub', 'alt_sub')
2889
wt.commit('E', rev_id=b'E-id', recursive=None)
2776
wt.commit('E', rev_id='E-id', recursive=None)
2890
2777
wt.set_last_revision('B-id')
2892
2779
wt.set_parent_ids(['B-id', 'C-id'])
2893
2780
wt.branch.set_last_revision_info(2, 'B-id')
2894
wt.commit('D', rev_id=b'D-id', recursive=None)
2781
wt.commit('D', rev_id='D-id', recursive=None)
2896
merger = _mod_merge.Merger.from_revision_ids(wt, 'E-id')
2783
merger = _mod_merge.Merger.from_revision_ids(None,
2897
2785
merger.merge_type = _mod_merge.Merge3Merger
2898
2786
merge_obj = merger.make_merger()
2899
2787
entries = list(merge_obj._entries_lca())
2900
2788
root_id = 'a-root-id'
2901
2789
self.assertEqual([('sub-tree-root', False,
2902
((u'sub', [u'sub', u'sub']), u'alt_sub', u'sub'),
2903
2790
((root_id, [root_id, root_id]), root_id, root_id),
2904
2791
((u'sub', [u'sub', u'sub']), u'alt_sub', u'sub'),
2905
2792
((False, [False, False]), False, False)),