233
290
tree1 = self.make_branch_and_tree('1')
234
291
tree2 = self.make_to_branch_and_tree('2')
235
292
tree1 = self.get_tree_no_parents_no_content(tree1)
236
tree2 = self.get_to_tree_no_parents_abc_content(tree2)
293
tree2 = self.get_tree_no_parents_abc_content(tree2)
294
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
237
295
self.assertRaises(errors.PathsNotVersionedError,
238
296
self.intertree_class(tree1, tree2).compare,
239
297
specific_files=['d'],
240
298
require_versioned=True)
300
def test_default_ignores_unversioned_files(self):
301
tree1 = self.make_branch_and_tree('tree1')
302
tree2 = self.make_to_branch_and_tree('tree2')
303
tree2.set_root_id(tree1.get_root_id())
304
self.build_tree(['tree1/a', 'tree1/c',
305
'tree2/a', 'tree2/b', 'tree2/c'])
306
tree1.add(['a', 'c'], ['a-id', 'c-id'])
307
tree2.add(['a', 'c'], ['a-id', 'c-id'])
309
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
310
d = self.intertree_class(tree1, tree2).compare()
311
self.assertEqual([], d.added)
312
self.assertEqual([(u'a', 'a-id', 'file', True, False),
313
(u'c', 'c-id', 'file', True, False)], d.modified)
314
self.assertEqual([], d.removed)
315
self.assertEqual([], d.renamed)
316
self.assertEqual([], d.unchanged)
317
self.assertEqual([], d.unversioned)
319
def test_unversioned_paths_in_tree(self):
320
tree1 = self.make_branch_and_tree('tree1')
321
tree2 = self.make_to_branch_and_tree('tree2')
322
tree2.set_root_id(tree1.get_root_id())
323
self.build_tree(['tree2/file', 'tree2/dir/'])
325
os.symlink('target', 'tree2/link')
326
links_supported = True
328
# links_supported = False
329
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
330
d = self.intertree_class(tree1, tree2).compare(want_unversioned=True)
331
self.assertEqual([], d.added)
332
self.assertEqual([], d.modified)
333
self.assertEqual([], d.removed)
334
self.assertEqual([], d.renamed)
335
self.assertEqual([], d.unchanged)
336
self.assertEqual([(u'dir', None, 'directory'), (u'file', None, 'file'),
337
(u'link', None, 'symlink')], d.unversioned)
243
340
class TestIterChanges(TestCaseWithTwoTrees):
244
341
"""Test the comparison iterator"""
343
def do_iter_changes(self, tree1, tree2, **extra_args):
344
"""Helper to run _iter_changes from tree1 to tree2.
346
:param tree1, tree2: The source and target trees. These will be locked
348
:param **extra_args: Extra args to pass to _iter_changes. This is not
349
inspected by this test helper.
354
# sort order of output is not strictly defined
355
return sorted(self.intertree_class(tree1, tree2)
356
._iter_changes(**extra_args))
361
def make_tree_with_special_names(self):
362
"""Create a tree with filenames chosen to exercise the walk order."""
363
tree1 = self.make_branch_and_tree('tree1')
364
tree2 = self.make_to_branch_and_tree('tree2')
365
tree2.set_root_id(tree1.get_root_id())
366
paths, path_ids = self._create_special_names(tree2, 'tree2')
367
tree2.commit('initial', rev_id='rev-1')
368
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
369
return (tree1, tree2, paths, path_ids)
371
def make_trees_with_special_names(self):
372
"""Both trees will use the special names.
374
But the contents will differ for each file.
376
tree1 = self.make_branch_and_tree('tree1')
377
tree2 = self.make_to_branch_and_tree('tree2')
378
tree2.set_root_id(tree1.get_root_id())
379
paths, path_ids = self._create_special_names(tree1, 'tree1')
380
paths, path_ids = self._create_special_names(tree2, 'tree2')
381
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
382
return (tree1, tree2, paths, path_ids)
384
def _create_special_names(self, tree, base_path):
385
"""Create a tree with paths that expose differences in sort orders."""
386
# Each directory will have a single file named 'f' inside
401
with_slashes.append(base_path + '/' + d + '/')
402
with_slashes.append(base_path + '/' + d + '/f')
405
path_ids.append(d.replace('/', '_') + '-id')
406
path_ids.append(d.replace('/', '_') + '_f-id')
407
self.build_tree(with_slashes)
408
tree.add(paths, path_ids)
409
return paths, path_ids
246
411
def test_compare_empty_trees(self):
247
412
tree1 = self.make_branch_and_tree('1')
248
413
tree2 = self.make_to_branch_and_tree('2')
249
414
tree1 = self.get_tree_no_parents_no_content(tree1)
250
tree2 = self.get_to_tree_no_parents_no_content(tree2)
251
self.assertEqual([], list(tree2._iter_changes(tree1)))
415
tree2 = self.get_tree_no_parents_no_content(tree2)
416
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
417
self.assertEqual([], self.do_iter_changes(tree1, tree2))
253
419
def added(self, tree, file_id):
254
420
entry = tree.inventory[file_id]
255
421
path = tree.id2path(file_id)
256
return (file_id, path, True, (False, True), (None, entry.parent_id),
257
(None, entry.name), (None, entry.kind),
422
return (file_id, (None, path), True, (False, True), (None, entry.parent_id),
423
(None, entry.name), (None, entry.kind),
258
424
(None, entry.executable))
426
def content_changed(self, tree, file_id):
427
entry = tree.inventory[file_id]
428
path = tree.id2path(file_id)
429
return (file_id, (path, path), True, (True, True), (entry.parent_id, entry.parent_id),
430
(entry.name, entry.name), (entry.kind, entry.kind),
431
(entry.executable, entry.executable))
433
def kind_changed(self, from_tree, to_tree, file_id):
434
old_entry = from_tree.inventory[file_id]
435
new_entry = to_tree.inventory[file_id]
436
path = to_tree.id2path(file_id)
437
from_path = from_tree.id2path(file_id)
438
return (file_id, (from_path, path), True, (True, True), (old_entry.parent_id, new_entry.parent_id),
439
(old_entry.name, new_entry.name), (old_entry.kind, new_entry.kind),
440
(old_entry.executable, new_entry.executable))
442
def missing(self, file_id, from_path, to_path, parent_id, kind):
443
_, from_basename = os.path.split(from_path)
444
_, to_basename = os.path.split(to_path)
445
# missing files have both paths, but no kind.
446
return (file_id, (from_path, to_path), True, (True, True),
447
(parent_id, parent_id),
448
(from_basename, to_basename), (kind, None), (False, False))
260
450
def deleted(self, tree, file_id):
261
451
entry = tree.inventory[file_id]
262
452
path = tree.id2path(file_id)
263
return (file_id, path, True, (True, False), (entry.parent_id, None),
264
(entry.name, None), (entry.kind, None),
453
return (file_id, (path, None), True, (True, False), (entry.parent_id, None),
454
(entry.name, None), (entry.kind, None),
265
455
(entry.executable, None))
457
def renamed(self, from_tree, to_tree, file_id, content_changed):
458
from_entry = from_tree.inventory[file_id]
459
to_entry = to_tree.inventory[file_id]
460
from_path = from_tree.id2path(file_id)
461
to_path = to_tree.id2path(file_id)
462
return (file_id, (from_path, to_path), content_changed, (True, True),
463
(from_entry.parent_id, to_entry.parent_id),
464
(from_entry.name, to_entry.name),
465
(from_entry.kind, to_entry.kind),
466
(from_entry.executable, to_entry.executable))
468
def unchanged(self, tree, file_id):
469
entry = tree.inventory[file_id]
470
parent = entry.parent_id
473
executable = entry.executable
474
path = tree.id2path(file_id)
475
return (file_id, (path, path), False, (True, True),
476
(parent, parent), (name, name), (kind, kind),
477
(executable, executable))
479
def unversioned(self, tree, path):
480
"""Create an unversioned result."""
481
_, basename = os.path.split(path)
482
kind = file_kind(tree.abspath(path))
483
return (None, (None, path), True, (False, False), (None, None),
484
(None, basename), (None, kind),
267
487
def test_empty_to_abc_content(self):
268
488
tree1 = self.make_branch_and_tree('1')
269
489
tree2 = self.make_to_branch_and_tree('2')
270
490
tree1 = self.get_tree_no_parents_no_content(tree1)
271
tree2 = self.get_to_tree_no_parents_abc_content(tree2)
273
self.assertEqual([self.added(tree2, 'root-id'),
274
self.added(tree2, 'a-id'),
275
self.added(tree2, 'b-id'),
276
self.added(tree2, 'c-id'),
277
self.deleted(tree1, 'empty-root-id')],
278
list(tree2._iter_changes(tree1)))
491
tree2 = self.get_tree_no_parents_abc_content(tree2)
492
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
495
expected_results = sorted([
496
self.added(tree2, 'root-id'),
497
self.added(tree2, 'a-id'),
498
self.added(tree2, 'b-id'),
499
self.added(tree2, 'c-id'),
500
self.deleted(tree1, 'empty-root-id')])
503
self.assertEqual(expected_results, self.do_iter_changes(tree1, tree2))
280
505
def test_empty_to_abc_content_a_only(self):
281
506
tree1 = self.make_branch_and_tree('1')
282
507
tree2 = self.make_to_branch_and_tree('2')
283
508
tree1 = self.get_tree_no_parents_no_content(tree1)
284
tree2 = self.get_to_tree_no_parents_abc_content(tree2)
285
self.assertEqual([self.added(tree2, 'a-id')],
286
list(tree2._iter_changes(tree1,
287
specific_file_ids=['a-id'])))
288
self.assertEqual([self.deleted(tree2, 'a-id')],
289
list(tree1._iter_changes(tree2,
290
specific_file_ids=['a-id'])))
509
tree2 = self.get_tree_no_parents_abc_content(tree2)
510
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
514
[self.added(tree2, 'a-id')],
515
self.do_iter_changes(tree1, tree2, specific_files=['a']))
519
def test_abc_content_to_empty_to_abc_content_a_only(self):
520
tree1 = self.make_branch_and_tree('1')
521
tree2 = self.make_to_branch_and_tree('2')
522
tree1 = self.get_tree_no_parents_abc_content(tree1)
523
tree2 = self.get_tree_no_parents_no_content(tree2)
524
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
528
[self.deleted(tree1, 'a-id')],
529
self.do_iter_changes(tree1, tree2, specific_files=['a']))
292
533
def test_empty_to_abc_content_a_and_c_only(self):
293
534
tree1 = self.make_branch_and_tree('1')
294
535
tree2 = self.make_to_branch_and_tree('2')
295
536
tree1 = self.get_tree_no_parents_no_content(tree1)
296
tree2 = self.get_to_tree_no_parents_abc_content(tree2)
297
self.assertEqual([self.added(tree2, 'a-id'),
298
self.added(tree2, 'c-id')],
299
list(tree2._iter_changes(tree1,
300
specific_file_ids=['a-id',
302
d = self.intertree_class(tree1, tree2).compare(
303
specific_files=['a', 'b/c'])
537
tree2 = self.get_tree_no_parents_abc_content(tree2)
538
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
541
expected_result = [self.added(tree2, 'a-id'), self.added(tree2, 'c-id')]
544
self.assertEqual(expected_result,
545
self.do_iter_changes(tree1, tree2, specific_files=['a', 'b/c']))
305
def test_abc_content(self):
547
def test_abc_content_to_empty(self):
306
548
tree1 = self.make_branch_and_tree('1')
307
549
tree2 = self.make_to_branch_and_tree('2')
308
tree1 = self.get_tree_no_parents_no_content(tree1)
309
tree2 = self.get_to_tree_no_parents_abc_content(tree2)
550
tree1 = self.get_tree_no_parents_abc_content(tree1)
551
tree2 = self.get_tree_no_parents_no_content(tree2)
552
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
310
555
def deleted(file_id):
311
entry = tree2.inventory[file_id]
312
path = tree2.id2path(file_id)
313
return (file_id, path, True, (True, False),
556
entry = tree1.inventory[file_id]
557
path = tree1.id2path(file_id)
558
return (file_id, (path, None), True, (True, False),
314
559
(entry.parent_id, None),
315
(entry.name, None), (entry.kind, None),
560
(entry.name, None), (entry.kind, None),
316
561
(entry.executable, None))
317
self.assertEqual([self.added(tree1, 'empty-root-id'),
318
deleted('root-id'), deleted('a-id'),
319
deleted('b-id'), deleted('c-id')],
320
list(tree1._iter_changes(tree2)))
562
expected_results = sorted([
563
self.added(tree2, 'empty-root-id'),
564
deleted('root-id'), deleted('a-id'),
565
deleted('b-id'), deleted('c-id')])
570
self.do_iter_changes(tree1, tree2))
322
572
def test_content_modification(self):
323
573
tree1 = self.make_branch_and_tree('1')
324
574
tree2 = self.make_to_branch_and_tree('2')
325
575
tree1 = self.get_tree_no_parents_abc_content(tree1)
326
tree2 = self.get_to_tree_no_parents_abc_content_2(tree2)
327
root_id = tree1.inventory.root.file_id
328
self.assertEqual([('a-id', 'a', True, (True, True),
329
(root_id, root_id), ('a', 'a'),
330
('file', 'file'), (False, False))],
331
list(tree2._iter_changes(tree1)))
576
tree2 = self.get_tree_no_parents_abc_content_2(tree2)
577
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
578
root_id = tree1.path2id('')
579
self.assertEqual([('a-id', ('a', 'a'), True, (True, True),
580
(root_id, root_id), ('a', 'a'),
581
('file', 'file'), (False, False))],
582
self.do_iter_changes(tree1, tree2))
333
584
def test_meta_modification(self):
334
585
tree1 = self.make_branch_and_tree('1')
335
586
tree2 = self.make_to_branch_and_tree('2')
336
587
tree1 = self.get_tree_no_parents_abc_content(tree1)
337
tree2 = self.get_to_tree_no_parents_abc_content_3(tree2)
338
self.assertEqual([('c-id', 'b/c', False, (True, True),
339
('b-id', 'b-id'), ('c', 'c'), ('file', 'file'),
340
(False, True))], list(tree2._iter_changes(tree1)))
588
tree2 = self.get_tree_no_parents_abc_content_3(tree2)
589
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
590
self.assertEqual([('c-id', ('b/c', 'b/c'), False, (True, True),
591
('b-id', 'b-id'), ('c', 'c'), ('file', 'file'),
593
self.do_iter_changes(tree1, tree2))
595
def test_empty_dir(self):
596
"""an empty dir should not cause glitches to surrounding files."""
597
tree1 = self.make_branch_and_tree('1')
598
tree2 = self.make_to_branch_and_tree('2')
599
tree1 = self.get_tree_no_parents_abc_content(tree1)
600
tree2 = self.get_tree_no_parents_abc_content(tree2)
601
# the pathname is chosen to fall between 'a' and 'b'.
602
self.build_tree(['1/a-empty/', '2/a-empty/'])
603
tree1.add(['a-empty'], ['a-empty'])
604
tree2.add(['a-empty'], ['a-empty'])
605
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
607
self.assertEqual(expected, self.do_iter_changes(tree1, tree2))
342
609
def test_file_rename(self):
343
610
tree1 = self.make_branch_and_tree('1')
344
611
tree2 = self.make_to_branch_and_tree('2')
345
612
tree1 = self.get_tree_no_parents_abc_content(tree1)
346
tree2 = self.get_to_tree_no_parents_abc_content_4(tree2)
347
root_id = tree1.inventory.root.file_id
348
self.assertEqual([('a-id', 'd', False, (True, True),
349
(root_id, root_id), ('a', 'd'), ('file', 'file'),
350
(False, False))], list(tree2._iter_changes(tree1)))
613
tree2 = self.get_tree_no_parents_abc_content_4(tree2)
614
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
615
root_id = tree1.path2id('')
616
self.assertEqual([('a-id', ('a', 'd'), False, (True, True),
617
(root_id, root_id), ('a', 'd'), ('file', 'file'),
619
self.do_iter_changes(tree1, tree2))
352
621
def test_file_rename_and_modification(self):
353
622
tree1 = self.make_branch_and_tree('1')
354
623
tree2 = self.make_to_branch_and_tree('2')
355
624
tree1 = self.get_tree_no_parents_abc_content(tree1)
356
tree2 = self.get_to_tree_no_parents_abc_content_5(tree2)
357
root_id = tree1.inventory.root.file_id
358
self.assertEqual([('a-id', 'd', True, (True, True),
359
(root_id, root_id), ('a', 'd'), ('file', 'file'),
360
(False, False))], list(tree2._iter_changes(tree1)))
625
tree2 = self.get_tree_no_parents_abc_content_5(tree2)
626
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
627
root_id = tree1.path2id('')
628
self.assertEqual([('a-id', ('a', 'd'), True, (True, True),
629
(root_id, root_id), ('a', 'd'), ('file', 'file'),
631
self.do_iter_changes(tree1, tree2))
362
633
def test_file_rename_and_meta_modification(self):
363
634
tree1 = self.make_branch_and_tree('1')
364
635
tree2 = self.make_to_branch_and_tree('2')
365
636
tree1 = self.get_tree_no_parents_abc_content(tree1)
366
tree2 = self.get_to_tree_no_parents_abc_content_6(tree2)
367
root_id = tree1.inventory.root.file_id
368
self.assertEqual([('c-id', 'e', False, (True, True),
369
('b-id', root_id), ('c', 'e'), ('file', 'file'),
370
(False, True))], list(tree2._iter_changes(tree1)))
637
tree2 = self.get_tree_no_parents_abc_content_6(tree2)
638
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
639
root_id = tree1.path2id('')
640
self.assertEqual([('c-id', ('b/c', 'e'), False, (True, True),
641
('b-id', root_id), ('c', 'e'), ('file', 'file'),
643
self.do_iter_changes(tree1, tree2))
645
def test_missing_in_target(self):
646
"""Test with the target files versioned but absent from disk."""
647
tree1 = self.make_branch_and_tree('1')
648
tree2 = self.make_to_branch_and_tree('2')
649
tree1 = self.get_tree_no_parents_abc_content(tree1)
650
tree2 = self.get_tree_no_parents_abc_content(tree2)
653
# TODO ? have a symlink here?
654
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
655
root_id = tree1.path2id('')
657
self.missing('a-id', 'a', 'a', root_id, 'file'),
658
self.missing('b-id', 'b', 'b', root_id, 'directory'),
659
self.missing('c-id', 'b/c', 'b/c', 'b-id', 'file'),
661
self.assertEqual(expected, self.do_iter_changes(tree1, tree2))
663
def test_missing_and_renamed(self):
664
tree1 = self.make_branch_and_tree('tree1')
665
tree2 = self.make_to_branch_and_tree('tree2')
666
tree2.set_root_id(tree1.get_root_id())
667
self.build_tree(['tree1/file'])
668
tree1.add(['file'], ['file-id'])
669
self.build_tree(['tree2/directory/'])
670
tree2.add(['directory'], ['file-id'])
671
os.rmdir('tree2/directory')
672
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
674
self.addCleanup(tree1.unlock)
676
self.addCleanup(tree2.unlock)
677
root_id = tree1.path2id('')
679
self.missing('file-id', 'file', 'directory', root_id, 'file'),
681
self.assertEqual(expected, self.do_iter_changes(tree1, tree2))
372
683
def test_unchanged_with_renames_and_modifications(self):
373
684
"""want_unchanged should generate a list of unchanged entries."""
374
685
tree1 = self.make_branch_and_tree('1')
375
686
tree2 = self.make_to_branch_and_tree('2')
376
687
tree1 = self.get_tree_no_parents_abc_content(tree1)
377
tree2 = self.get_to_tree_no_parents_abc_content_5(tree2)
378
root_id = tree1.inventory.root.file_id
379
def unchanged(file_id):
380
entry = tree1.inventory[file_id]
381
parent = entry.parent_id
384
executable = entry.executable
385
return (file_id, tree1.id2path(file_id), False, (True, True),
386
(parent, parent), (name, name), (kind, kind),
387
(executable, executable))
388
self.assertEqual([unchanged(root_id), unchanged('b-id'),
389
('a-id', 'd', True, (True, True),
390
(root_id, root_id), ('a', 'd'), ('file', 'file'),
391
(False, False)), unchanged('c-id')],
392
list(tree2._iter_changes(tree1,
393
include_unchanged=True)))
688
tree2 = self.get_tree_no_parents_abc_content_5(tree2)
689
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
690
root_id = tree1.path2id('')
692
self.addCleanup(tree1.unlock)
694
self.addCleanup(tree2.unlock)
695
self.assertEqual(sorted([self.unchanged(tree1, root_id),
696
self.unchanged(tree1, 'b-id'),
697
('a-id', ('a', 'd'), True, (True, True),
698
(root_id, root_id), ('a', 'd'), ('file', 'file'),
699
(False, False)), self.unchanged(tree1, 'c-id')]),
700
self.do_iter_changes(tree1, tree2, include_unchanged=True))
702
def test_compare_subtrees(self):
703
"""want_unchanged should generate a list of unchanged entries."""
704
tree1 = self.make_branch_and_tree('1')
705
if not tree1.supports_tree_reference():
706
raise tests.TestSkipped('Tree %s does not support references'
708
tree1.set_root_id('root-id')
709
subtree1 = self.make_branch_and_tree('1/sub')
710
subtree1.set_root_id('subtree-id')
711
tree1.add_reference(subtree1)
713
tree2 = self.make_to_branch_and_tree('2')
714
if not tree2.supports_tree_reference():
715
raise tests.TestSkipped('Tree %s does not support references'
717
tree2.set_root_id('root-id')
718
subtree2 = self.make_to_branch_and_tree('2/sub')
719
subtree2.set_root_id('subtree-id')
720
tree2.add_reference(subtree2)
721
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
723
self.addCleanup(tree1.unlock)
725
self.addCleanup(tree2.unlock)
726
self.assertEqual([], list(tree2._iter_changes(tree1)))
727
subtree1.commit('commit', rev_id='commit-a')
735
('directory', 'directory'),
741
('root-id', 'root-id'),
743
('tree-reference', 'tree-reference'),
745
list(tree2._iter_changes(tree1,
746
include_unchanged=True)))
748
def test_default_ignores_unversioned_files(self):
749
tree1 = self.make_branch_and_tree('tree1')
750
tree2 = self.make_to_branch_and_tree('tree2')
751
tree2.set_root_id(tree1.get_root_id())
752
self.build_tree(['tree1/a', 'tree1/c',
753
'tree2/a', 'tree2/b', 'tree2/c'])
754
tree1.add(['a', 'c'], ['a-id', 'c-id'])
755
tree2.add(['a', 'c'], ['a-id', 'c-id'])
757
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
759
self.addCleanup(tree1.unlock)
761
self.addCleanup(tree2.unlock)
763
# We should ignore the fact that 'b' exists in tree-2
764
# because the want_unversioned parameter was not given.
766
self.content_changed(tree2, 'a-id'),
767
self.content_changed(tree2, 'c-id'),
769
self.assertEqual(expected, self.do_iter_changes(tree1, tree2))
771
def test_unversioned_paths_in_tree(self):
772
tree1 = self.make_branch_and_tree('tree1')
773
tree2 = self.make_to_branch_and_tree('tree2')
774
tree2.set_root_id(tree1.get_root_id())
775
self.build_tree(['tree2/file', 'tree2/dir/'])
777
os.symlink('target', 'tree2/link')
778
links_supported = True
780
# links_supported = False
781
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
783
self.addCleanup(tree1.unlock)
785
self.addCleanup(tree2.unlock)
787
self.unversioned(tree2, 'file'),
788
self.unversioned(tree2, 'dir'),
791
expected.append(self.unversioned(tree2, 'link'))
792
expected = sorted(expected)
793
self.assertEqual(expected, self.do_iter_changes(tree1, tree2,
794
want_unversioned=True))
796
def test_unversioned_paths_in_tree_specific_files(self):
797
tree1 = self.make_branch_and_tree('tree1')
798
tree2 = self.make_to_branch_and_tree('tree2')
799
self.build_tree(['tree2/file', 'tree2/dir/'])
801
os.symlink('target', 'tree2/link')
802
links_supported = True
804
# links_supported = False
805
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
807
self.addCleanup(tree1.unlock)
809
self.addCleanup(tree2.unlock)
811
self.unversioned(tree2, 'file'),
812
self.unversioned(tree2, 'dir'),
814
specific_files=['file', 'dir']
816
expected.append(self.unversioned(tree2, 'link'))
817
specific_files.append('link')
818
expected = sorted(expected)
819
self.assertEqual(expected, self.do_iter_changes(tree1, tree2,
820
specific_files=specific_files, require_versioned=False,
821
want_unversioned=True))
823
def test_unversioned_paths_in_target_matching_source_old_names(self):
824
# its likely that naive implementations of unversioned file support
825
# will fail if the path was versioned, but is not any more,
826
# due to a rename, not due to unversioning it.
827
# That is, if the old tree has a versioned file 'foo', and
828
# the new tree has the same file but versioned as 'bar', and also
829
# has an unknown file 'foo', we should get back output for
831
tree1 = self.make_branch_and_tree('tree1')
832
tree2 = self.make_to_branch_and_tree('tree2')
833
tree2.set_root_id(tree1.get_root_id())
834
self.build_tree(['tree2/file', 'tree2/dir/',
835
'tree1/file', 'tree2/movedfile',
836
'tree1/dir/', 'tree2/moveddir/'])
838
os.symlink('target', 'tree1/link')
839
os.symlink('target', 'tree2/link')
840
os.symlink('target', 'tree2/movedlink')
841
links_supported = True
843
# links_supported = False
844
tree1.add(['file', 'dir', 'link'], ['file-id', 'dir-id', 'link-id'])
845
tree2.add(['movedfile', 'moveddir', 'movedlink'],
846
['file-id', 'dir-id', 'link-id'])
847
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
848
root_id = tree1.path2id('')
850
self.addCleanup(tree1.unlock)
852
self.addCleanup(tree2.unlock)
854
self.renamed(tree1, tree2, 'dir-id', False),
855
self.renamed(tree1, tree2, 'file-id', True),
856
self.unversioned(tree2, 'file'),
857
self.unversioned(tree2, 'dir'),
859
specific_files=['file', 'dir']
861
expected.append(self.renamed(tree1, tree2, 'link-id', False))
862
expected.append(self.unversioned(tree2, 'link'))
863
specific_files.append('link')
864
expected = sorted(expected)
865
# run once with, and once without specific files, to catch
866
# potentially different code paths.
867
self.assertEqual(expected, self.do_iter_changes(tree1, tree2,
868
require_versioned=False,
869
want_unversioned=True))
870
self.assertEqual(expected, self.do_iter_changes(tree1, tree2,
871
specific_files=specific_files, require_versioned=False,
872
want_unversioned=True))
874
def test_unversioned_subtree_only_emits_root(self):
875
tree1 = self.make_branch_and_tree('tree1')
876
tree2 = self.make_to_branch_and_tree('tree2')
877
tree2.set_root_id(tree1.get_root_id())
878
self.build_tree(['tree2/dir/', 'tree2/dir/file'])
879
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
881
self.unversioned(tree2, 'dir'),
883
self.assertEqual(expected, self.do_iter_changes(tree1, tree2,
884
want_unversioned=True))
886
def make_trees_with_symlinks(self):
887
tree1 = self.make_branch_and_tree('tree1')
888
tree2 = self.make_to_branch_and_tree('tree2')
889
tree2.set_root_id(tree1.get_root_id())
890
self.build_tree(['tree1/fromfile', 'tree1/fromdir/'])
891
self.build_tree(['tree2/tofile', 'tree2/todir/', 'tree2/unknown'])
893
os.symlink('original', 'tree1/changed')
894
os.symlink('original', 'tree1/removed')
895
os.symlink('original', 'tree1/tofile')
896
os.symlink('original', 'tree1/todir')
897
# we make the unchanged link point at unknown to catch incorrect
898
# symlink-following code in the specified_files test.
899
os.symlink('unknown', 'tree1/unchanged')
900
os.symlink('new', 'tree2/added')
901
os.symlink('new', 'tree2/changed')
902
os.symlink('new', 'tree2/fromfile')
903
os.symlink('new', 'tree2/fromdir')
904
os.symlink('unknown', 'tree2/unchanged')
905
from_paths_and_ids = [
923
tree1.add(from_paths_and_ids, from_paths_and_ids)
924
tree2.add(to_paths_and_ids, to_paths_and_ids)
926
# raise TestSkipped('OS does not support symlinks')
927
# links_supported = False
928
return self.mutable_trees_to_test_trees(tree1, tree2)
930
def test_versioned_symlinks(self):
931
tree1, tree2 = self.make_trees_with_symlinks()
932
root_id = tree1.path2id('')
934
self.addCleanup(tree1.unlock)
936
self.addCleanup(tree2.unlock)
938
self.unchanged(tree1, tree1.path2id('')),
939
self.added(tree2, 'added'),
940
self.content_changed(tree2, 'changed'),
941
self.kind_changed(tree1, tree2, 'fromdir'),
942
self.kind_changed(tree1, tree2, 'fromfile'),
943
self.deleted(tree1, 'removed'),
944
self.unchanged(tree2, 'unchanged'),
945
self.unversioned(tree2, 'unknown'),
946
self.kind_changed(tree1, tree2, 'todir'),
947
self.kind_changed(tree1, tree2, 'tofile'),
949
expected = sorted(expected)
950
self.assertEqual(expected,
951
self.do_iter_changes(tree1, tree2, include_unchanged=True,
952
want_unversioned=True))
954
def test_versioned_symlinks_specific_files(self):
955
tree1, tree2 = self.make_trees_with_symlinks()
956
root_id = tree1.path2id('')
958
self.addCleanup(tree1.unlock)
960
self.addCleanup(tree2.unlock)
962
self.added(tree2, 'added'),
963
self.content_changed(tree2, 'changed'),
964
self.kind_changed(tree1, tree2, 'fromdir'),
965
self.kind_changed(tree1, tree2, 'fromfile'),
966
self.deleted(tree1, 'removed'),
967
self.kind_changed(tree1, tree2, 'todir'),
968
self.kind_changed(tree1, tree2, 'tofile'),
970
expected = sorted(expected)
971
# we should get back just the changed links. We pass in 'unchanged' to
972
# make sure that it is correctly not returned - and neither is the
973
# unknown path 'unknown' which it points at.
974
self.assertEqual(expected, self.do_iter_changes(tree1, tree2,
975
specific_files=['added', 'changed', 'fromdir', 'fromfile',
976
'removed', 'unchanged', 'todir', 'tofile']))
978
def test_tree_with_special_names(self):
979
tree1, tree2, paths, path_ids = self.make_tree_with_special_names()
981
self.addCleanup(tree1.unlock)
983
self.addCleanup(tree2.unlock)
984
expected = sorted(self.added(tree2, f_id) for f_id in path_ids)
985
self.assertEqual(expected, self.do_iter_changes(tree1, tree2))
987
def test_trees_with_special_names(self):
988
tree1, tree2, paths, path_ids = self.make_trees_with_special_names()
990
self.addCleanup(tree1.unlock)
992
self.addCleanup(tree2.unlock)
993
expected = sorted(self.content_changed(tree2, f_id) for f_id in path_ids
994
if f_id.endswith('_f-id'))
995
self.assertEqual(expected, self.do_iter_changes(tree1, tree2))
997
def test_trees_with_deleted_dir(self):
998
tree1 = self.make_branch_and_tree('tree1')
999
tree2 = self.make_to_branch_and_tree('tree2')
1000
tree2.set_root_id(tree1.get_root_id())
1001
self.build_tree(['tree1/a', 'tree1/b/', 'tree1/b/c',
1002
'tree1/b/d/', 'tree1/b/d/e', 'tree1/f/', 'tree1/f/g',
1003
'tree2/a', 'tree2/f/', 'tree2/f/g'])
1004
tree1.add(['a', 'b', 'b/c', 'b/d/', 'b/d/e', 'f', 'f/g'],
1005
['a-id', 'b-id', 'c-id', 'd-id', 'e-id', 'f-id', 'g-id'])
1006
tree2.add(['a', 'f', 'f/g'], ['a-id', 'f-id', 'g-id'])
1008
tree1, tree2 = self.mutable_trees_to_test_trees(tree1, tree2)
1010
self.addCleanup(tree1.unlock)
1012
self.addCleanup(tree2.unlock)
1013
# We should notice that 'b' and all its children are deleted
1015
self.content_changed(tree2, 'a-id'),
1016
self.content_changed(tree2, 'g-id'),
1017
self.deleted(tree1, 'b-id'),
1018
self.deleted(tree1, 'c-id'),
1019
self.deleted(tree1, 'd-id'),
1020
self.deleted(tree1, 'e-id'),
1022
self.assertEqual(expected, self.do_iter_changes(tree1, tree2))