271
def diff_cmd_helper(tree, specific_files, external_diff_options,
272
old_revision_spec=None, new_revision_spec=None,
274
old_label='a/', new_label='b/'):
275
"""Helper for cmd_diff.
280
:param specific_files:
281
The specific files to compare, or None
283
:param external_diff_options:
284
If non-None, run an external diff, and pass it these options
286
:param old_revision_spec:
287
If None, use basis tree as old revision, otherwise use the tree for
288
the specified revision.
290
:param new_revision_spec:
291
If None, use working tree as new revision, otherwise use the tree for
292
the specified revision.
294
:param revision_specs:
295
Zero, one or two RevisionSpecs from the command line, saying what revisions
296
to compare. This can be passed as an alternative to the old_revision_spec
297
and new_revision_spec parameters.
299
The more general form is show_diff_trees(), where the caller
300
supplies any two trees.
278
def _get_trees_to_diff(path_list, revision_specs, old_url, new_url):
279
"""Get the trees and specific files to diff given a list of paths.
281
This method works out the trees to be diff'ed and the files of
282
interest within those trees.
285
the list of arguments passed to the diff command
286
:param revision_specs:
287
Zero, one or two RevisionSpecs from the diff command line,
288
saying what revisions to compare.
290
The url of the old branch or tree. If None, the tree to use is
291
taken from the first path, if any, or the current working tree.
293
The url of the new branch or tree. If None, the tree to use is
294
taken from the first path, if any, or the current working tree.
296
a tuple of (old_tree, new_tree, specific_files, extra_trees) where
297
extra_trees is a sequence of additional trees to search in for
303
# TODO: perhaps remove the old parameters old_revision_spec and
304
# new_revision_spec, since this is only really for use from cmd_diff and
305
# it now always passes through a sequence of revision_specs -- mbp
310
revision = spec.in_store(tree.branch)
312
revision = spec.in_store(None)
313
revision_id = revision.rev_id
314
branch = revision.branch
315
return branch.repository.revision_tree(revision_id)
300
# Get the old and new revision specs
301
old_revision_spec = None
302
new_revision_spec = None
317
303
if revision_specs is not None:
318
assert (old_revision_spec is None
319
and new_revision_spec is None)
320
304
if len(revision_specs) > 0:
321
305
old_revision_spec = revision_specs[0]
307
old_url = old_revision_spec.get_branch()
322
308
if len(revision_specs) > 1:
323
309
new_revision_spec = revision_specs[1]
325
if old_revision_spec is None:
326
old_tree = tree.basis_tree()
328
old_tree = spec_tree(old_revision_spec)
330
if (new_revision_spec is None
331
or new_revision_spec.spec is None):
334
new_tree = spec_tree(new_revision_spec)
336
if new_tree is not tree:
337
extra_trees = (tree,)
341
return show_diff_trees(old_tree, new_tree, sys.stdout, specific_files,
342
external_diff_options,
343
old_label=old_label, new_label=new_label,
344
extra_trees=extra_trees)
311
new_url = new_revision_spec.get_branch()
314
make_paths_wt_relative = True
315
consider_relpath = True
316
if path_list is None or len(path_list) == 0:
317
# If no path is given, the current working tree is used
318
default_location = u'.'
319
consider_relpath = False
320
elif old_url is not None and new_url is not None:
321
other_paths = path_list
322
make_paths_wt_relative = False
324
default_location = path_list[0]
325
other_paths = path_list[1:]
327
# Get the old location
330
old_url = default_location
331
working_tree, branch, relpath = \
332
bzrdir.BzrDir.open_containing_tree_or_branch(old_url)
333
if consider_relpath and relpath != '':
334
specific_files.append(relpath)
335
old_tree = _get_tree_to_diff(old_revision_spec, working_tree, branch)
337
# Get the new location
339
new_url = default_location
340
if new_url != old_url:
341
working_tree, branch, relpath = \
342
bzrdir.BzrDir.open_containing_tree_or_branch(new_url)
343
if consider_relpath and relpath != '':
344
specific_files.append(relpath)
345
new_tree = _get_tree_to_diff(new_revision_spec, working_tree, branch,
346
basis_is_default=working_tree is None)
348
# Get the specific files (all files is None, no files is [])
349
if make_paths_wt_relative and working_tree is not None:
350
other_paths = _relative_paths_in_tree(working_tree, other_paths)
351
specific_files.extend(other_paths)
352
if len(specific_files) == 0:
353
specific_files = None
355
# Get extra trees that ought to be searched for file-ids
357
if working_tree is not None and working_tree not in (old_tree, new_tree):
358
extra_trees = (working_tree,)
359
return old_tree, new_tree, specific_files, extra_trees
362
def _get_tree_to_diff(spec, tree=None, branch=None, basis_is_default=True):
363
if branch is None and tree is not None:
365
if spec is None or spec.spec is None:
368
return tree.basis_tree()
370
return branch.basis_tree()
373
if not spec.needs_branch():
374
branch = _mod_branch.Branch.open(spec.get_branch())
375
revision_id = spec.as_revision_id(branch)
376
return branch.repository.revision_tree(revision_id)
379
def _relative_paths_in_tree(tree, paths):
380
"""Get the relative paths within a working tree.
382
Each path may be either an absolute path or a path relative to the
383
current working directory.
386
for filename in paths:
388
result.append(tree.relpath(osutils.dereference_path(filename)))
389
except errors.PathNotChild:
390
raise errors.BzrCommandError("Files are in different branches")
347
394
def show_diff_trees(old_tree, new_tree, to_file, specific_files=None,
348
395
external_diff_options=None,
349
396
old_label='a/', new_label='b/',
350
397
extra_trees=None,
351
path_encoding='utf8'):
398
path_encoding='utf8',
352
400
"""Show in text form the changes from one tree to another.
355
If set, include only changes to these files.
406
Include only changes to these files - None for all changes.
357
408
external_diff_options
358
409
If set, use an external GNU diff and pass these options.
385
436
old_tree.unlock()
388
def _show_diff_trees(old_tree, new_tree, to_file,
389
specific_files, external_diff_options, path_encoding,
390
old_label='a/', new_label='b/', extra_trees=None):
392
# GNU Patch uses the epoch date to detect files that are being added
393
# or removed in a diff.
394
EPOCH_DATE = '1970-01-01 00:00:00 +0000'
396
# TODO: Generation of pseudo-diffs for added/deleted files could
397
# be usefully made into a much faster special case.
399
if external_diff_options:
400
assert isinstance(external_diff_options, basestring)
401
opts = external_diff_options.split()
402
def diff_file(olab, olines, nlab, nlines, to_file):
403
external_diff(olab, olines, nlab, nlines, to_file, opts)
405
diff_file = internal_diff
407
delta = new_tree.changes_from(old_tree,
408
specific_files=specific_files,
409
extra_trees=extra_trees, require_versioned=True)
412
for path, file_id, kind in delta.removed:
414
path_encoded = path.encode(path_encoding, "replace")
415
print >>to_file, "=== removed %s '%s'" % (kind, path_encoded)
416
old_name = '%s%s\t%s' % (old_label, path,
417
_patch_header_date(old_tree, file_id, path))
418
new_name = '%s%s\t%s' % (new_label, path, EPOCH_DATE)
419
old_tree.inventory[file_id].diff(diff_file, old_name, old_tree,
420
new_name, None, None, to_file)
421
for path, file_id, kind in delta.added:
423
path_encoded = path.encode(path_encoding, "replace")
424
print >>to_file, "=== added %s '%s'" % (kind, path_encoded)
425
old_name = '%s%s\t%s' % (old_label, path, EPOCH_DATE)
426
new_name = '%s%s\t%s' % (new_label, path,
427
_patch_header_date(new_tree, file_id, path))
428
new_tree.inventory[file_id].diff(diff_file, new_name, new_tree,
429
old_name, None, None, to_file,
431
for (old_path, new_path, file_id, kind,
432
text_modified, meta_modified) in delta.renamed:
434
prop_str = get_prop_change(meta_modified)
435
oldpath_encoded = old_path.encode(path_encoding, "replace")
436
newpath_encoded = new_path.encode(path_encoding, "replace")
437
print >>to_file, "=== renamed %s '%s' => '%s'%s" % (kind,
438
oldpath_encoded, newpath_encoded, prop_str)
439
old_name = '%s%s\t%s' % (old_label, old_path,
440
_patch_header_date(old_tree, file_id,
442
new_name = '%s%s\t%s' % (new_label, new_path,
443
_patch_header_date(new_tree, file_id,
445
_maybe_diff_file_or_symlink(old_name, old_tree, file_id,
447
text_modified, kind, to_file, diff_file)
448
for path, file_id, kind, text_modified, meta_modified in delta.modified:
450
prop_str = get_prop_change(meta_modified)
451
path_encoded = path.encode(path_encoding, "replace")
452
print >>to_file, "=== modified %s '%s'%s" % (kind,
453
path_encoded, prop_str)
454
# The file may be in a different location in the old tree (because
455
# the containing dir was renamed, but the file itself was not)
456
old_path = old_tree.id2path(file_id)
457
old_name = '%s%s\t%s' % (old_label, old_path,
458
_patch_header_date(old_tree, file_id, old_path))
459
new_name = '%s%s\t%s' % (new_label, path,
460
_patch_header_date(new_tree, file_id, path))
462
_maybe_diff_file_or_symlink(old_name, old_tree, file_id,
464
True, kind, to_file, diff_file)
469
439
def _patch_header_date(tree, file_id, path):
470
440
"""Returns a timestamp suitable for use in a patch header."""
471
441
mtime = tree.get_file_mtime(file_id, path)
472
assert mtime is not None, \
473
"got an mtime of None for file-id %s, path %s in tree %s" % (
475
442
return timestamp.format_patch_date(mtime)
494
461
raise errors.PathsDoNotExist(sorted(s))
464
@deprecated_function(one_three)
497
465
def get_prop_change(meta_modified):
498
466
if meta_modified:
499
467
return " (properties changed)"
504
def _maybe_diff_file_or_symlink(old_path, old_tree, file_id,
505
new_path, new_tree, text_modified,
506
kind, to_file, diff_file):
508
new_entry = new_tree.inventory[file_id]
509
old_tree.inventory[file_id].diff(diff_file,
471
def get_executable_change(old_is_x, new_is_x):
472
descr = { True:"+x", False:"-x", None:"??" }
473
if old_is_x != new_is_x:
474
return ["%s to %s" % (descr[old_is_x], descr[new_is_x],)]
479
class DiffPath(object):
480
"""Base type for command object that compare files"""
482
# The type or contents of the file were unsuitable for diffing
483
CANNOT_DIFF = 'CANNOT_DIFF'
484
# The file has changed in a semantic way
486
# The file content may have changed, but there is no semantic change
487
UNCHANGED = 'UNCHANGED'
489
def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8'):
492
:param old_tree: The tree to show as the old tree in the comparison
493
:param new_tree: The tree to show as new in the comparison
494
:param to_file: The file to write comparison data to
495
:param path_encoding: The character encoding to write paths in
497
self.old_tree = old_tree
498
self.new_tree = new_tree
499
self.to_file = to_file
500
self.path_encoding = path_encoding
506
def from_diff_tree(klass, diff_tree):
507
return klass(diff_tree.old_tree, diff_tree.new_tree,
508
diff_tree.to_file, diff_tree.path_encoding)
511
def _diff_many(differs, file_id, old_path, new_path, old_kind, new_kind):
512
for file_differ in differs:
513
result = file_differ.diff(file_id, old_path, new_path, old_kind,
515
if result is not DiffPath.CANNOT_DIFF:
518
return DiffPath.CANNOT_DIFF
521
class DiffKindChange(object):
522
"""Special differ for file kind changes.
524
Represents kind change as deletion + creation. Uses the other differs
527
def __init__(self, differs):
528
self.differs = differs
534
def from_diff_tree(klass, diff_tree):
535
return klass(diff_tree.differs)
537
def diff(self, file_id, old_path, new_path, old_kind, new_kind):
538
"""Perform comparison
540
:param file_id: The file_id of the file to compare
541
:param old_path: Path of the file in the old tree
542
:param new_path: Path of the file in the new tree
543
:param old_kind: Old file-kind of the file
544
:param new_kind: New file-kind of the file
546
if None in (old_kind, new_kind):
547
return DiffPath.CANNOT_DIFF
548
result = DiffPath._diff_many(self.differs, file_id, old_path,
549
new_path, old_kind, None)
550
if result is DiffPath.CANNOT_DIFF:
552
return DiffPath._diff_many(self.differs, file_id, old_path, new_path,
556
class DiffDirectory(DiffPath):
558
def diff(self, file_id, old_path, new_path, old_kind, new_kind):
559
"""Perform comparison between two directories. (dummy)
562
if 'directory' not in (old_kind, new_kind):
563
return self.CANNOT_DIFF
564
if old_kind not in ('directory', None):
565
return self.CANNOT_DIFF
566
if new_kind not in ('directory', None):
567
return self.CANNOT_DIFF
571
class DiffSymlink(DiffPath):
573
def diff(self, file_id, old_path, new_path, old_kind, new_kind):
574
"""Perform comparison between two symlinks
576
:param file_id: The file_id of the file to compare
577
:param old_path: Path of the file in the old tree
578
:param new_path: Path of the file in the new tree
579
:param old_kind: Old file-kind of the file
580
:param new_kind: New file-kind of the file
582
if 'symlink' not in (old_kind, new_kind):
583
return self.CANNOT_DIFF
584
if old_kind == 'symlink':
585
old_target = self.old_tree.get_symlink_target(file_id)
586
elif old_kind is None:
589
return self.CANNOT_DIFF
590
if new_kind == 'symlink':
591
new_target = self.new_tree.get_symlink_target(file_id)
592
elif new_kind is None:
595
return self.CANNOT_DIFF
596
return self.diff_symlink(old_target, new_target)
598
def diff_symlink(self, old_target, new_target):
599
if old_target is None:
600
self.to_file.write('=== target is %r\n' % new_target)
601
elif new_target is None:
602
self.to_file.write('=== target was %r\n' % old_target)
604
self.to_file.write('=== target changed %r => %r\n' %
605
(old_target, new_target))
609
class DiffText(DiffPath):
611
# GNU Patch uses the epoch date to detect files that are being added
612
# or removed in a diff.
613
EPOCH_DATE = '1970-01-01 00:00:00 +0000'
615
def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8',
616
old_label='', new_label='', text_differ=internal_diff):
617
DiffPath.__init__(self, old_tree, new_tree, to_file, path_encoding)
618
self.text_differ = text_differ
619
self.old_label = old_label
620
self.new_label = new_label
621
self.path_encoding = path_encoding
623
def diff(self, file_id, old_path, new_path, old_kind, new_kind):
624
"""Compare two files in unified diff format
626
:param file_id: The file_id of the file to compare
627
:param old_path: Path of the file in the old tree
628
:param new_path: Path of the file in the new tree
629
:param old_kind: Old file-kind of the file
630
:param new_kind: New file-kind of the file
632
if 'file' not in (old_kind, new_kind):
633
return self.CANNOT_DIFF
634
from_file_id = to_file_id = file_id
635
if old_kind == 'file':
636
old_date = _patch_header_date(self.old_tree, file_id, old_path)
637
elif old_kind is None:
638
old_date = self.EPOCH_DATE
641
return self.CANNOT_DIFF
642
if new_kind == 'file':
643
new_date = _patch_header_date(self.new_tree, file_id, new_path)
644
elif new_kind is None:
645
new_date = self.EPOCH_DATE
648
return self.CANNOT_DIFF
649
from_label = '%s%s\t%s' % (self.old_label, old_path, old_date)
650
to_label = '%s%s\t%s' % (self.new_label, new_path, new_date)
651
return self.diff_text(from_file_id, to_file_id, from_label, to_label)
653
def diff_text(self, from_file_id, to_file_id, from_label, to_label):
654
"""Diff the content of given files in two trees
656
:param from_file_id: The id of the file in the from tree. If None,
657
the file is not present in the from tree.
658
:param to_file_id: The id of the file in the to tree. This may refer
659
to a different file from from_file_id. If None,
660
the file is not present in the to tree.
662
def _get_text(tree, file_id):
663
if file_id is not None:
664
return tree.get_file(file_id).readlines()
668
from_text = _get_text(self.old_tree, from_file_id)
669
to_text = _get_text(self.new_tree, to_file_id)
670
self.text_differ(from_label, from_text, to_label, to_text,
672
except errors.BinaryFile:
674
("Binary files %s and %s differ\n" %
675
(from_label, to_label)).encode(self.path_encoding))
679
class DiffFromTool(DiffPath):
681
def __init__(self, command_template, old_tree, new_tree, to_file,
682
path_encoding='utf-8'):
683
DiffPath.__init__(self, old_tree, new_tree, to_file, path_encoding)
684
self.command_template = command_template
685
self._root = tempfile.mkdtemp(prefix='bzr-diff-')
688
def from_string(klass, command_string, old_tree, new_tree, to_file,
689
path_encoding='utf-8'):
690
command_template = commands.shlex_split_unicode(command_string)
691
command_template.extend(['%(old_path)s', '%(new_path)s'])
692
return klass(command_template, old_tree, new_tree, to_file,
696
def make_from_diff_tree(klass, command_string):
697
def from_diff_tree(diff_tree):
698
return klass.from_string(command_string, diff_tree.old_tree,
699
diff_tree.new_tree, diff_tree.to_file)
700
return from_diff_tree
702
def _get_command(self, old_path, new_path):
703
my_map = {'old_path': old_path, 'new_path': new_path}
704
return [t % my_map for t in self.command_template]
706
def _execute(self, old_path, new_path):
707
command = self._get_command(old_path, new_path)
709
proc = subprocess.Popen(command, stdout=subprocess.PIPE,
712
if e.errno == errno.ENOENT:
713
raise errors.ExecutableMissing(command[0])
716
self.to_file.write(proc.stdout.read())
719
def _try_symlink_root(self, tree, prefix):
720
if (getattr(tree, 'abspath', None) is None
721
or not osutils.host_os_dereferences_symlinks()):
724
os.symlink(tree.abspath(''), osutils.pathjoin(self._root, prefix))
726
if e.errno != errno.EEXIST:
730
def _write_file(self, file_id, tree, prefix, relpath):
731
full_path = osutils.pathjoin(self._root, prefix, relpath)
732
if self._try_symlink_root(tree, prefix):
734
parent_dir = osutils.dirname(full_path)
736
os.makedirs(parent_dir)
738
if e.errno != errno.EEXIST:
740
source = tree.get_file(file_id, relpath)
742
target = open(full_path, 'wb')
744
osutils.pumpfile(source, target)
749
osutils.make_readonly(full_path)
750
mtime = tree.get_file_mtime(file_id)
751
os.utime(full_path, (mtime, mtime))
754
def _prepare_files(self, file_id, old_path, new_path):
755
old_disk_path = self._write_file(file_id, self.old_tree, 'old',
757
new_disk_path = self._write_file(file_id, self.new_tree, 'new',
759
return old_disk_path, new_disk_path
762
osutils.rmtree(self._root)
764
def diff(self, file_id, old_path, new_path, old_kind, new_kind):
765
if (old_kind, new_kind) != ('file', 'file'):
766
return DiffPath.CANNOT_DIFF
767
self._prepare_files(file_id, old_path, new_path)
768
self._execute(osutils.pathjoin('old', old_path),
769
osutils.pathjoin('new', new_path))
772
class DiffTree(object):
773
"""Provides textual representations of the difference between two trees.
775
A DiffTree examines two trees and where a file-id has altered
776
between them, generates a textual representation of the difference.
777
DiffTree uses a sequence of DiffPath objects which are each
778
given the opportunity to handle a given altered fileid. The list
779
of DiffPath objects can be extended globally by appending to
780
DiffTree.diff_factories, or for a specific diff operation by
781
supplying the extra_factories option to the appropriate method.
784
# list of factories that can provide instances of DiffPath objects
785
# may be extended by plugins.
786
diff_factories = [DiffSymlink.from_diff_tree,
787
DiffDirectory.from_diff_tree]
789
def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8',
790
diff_text=None, extra_factories=None):
793
:param old_tree: Tree to show as old in the comparison
794
:param new_tree: Tree to show as new in the comparison
795
:param to_file: File to write comparision to
796
:param path_encoding: Character encoding to write paths in
797
:param diff_text: DiffPath-type object to use as a last resort for
799
:param extra_factories: Factories of DiffPaths to try before any other
801
if diff_text is None:
802
diff_text = DiffText(old_tree, new_tree, to_file, path_encoding,
803
'', '', internal_diff)
804
self.old_tree = old_tree
805
self.new_tree = new_tree
806
self.to_file = to_file
807
self.path_encoding = path_encoding
809
if extra_factories is not None:
810
self.differs.extend(f(self) for f in extra_factories)
811
self.differs.extend(f(self) for f in self.diff_factories)
812
self.differs.extend([diff_text, DiffKindChange.from_diff_tree(self)])
815
def from_trees_options(klass, old_tree, new_tree, to_file,
816
path_encoding, external_diff_options, old_label,
818
"""Factory for producing a DiffTree.
820
Designed to accept options used by show_diff_trees.
821
:param old_tree: The tree to show as old in the comparison
822
:param new_tree: The tree to show as new in the comparison
823
:param to_file: File to write comparisons to
824
:param path_encoding: Character encoding to use for writing paths
825
:param external_diff_options: If supplied, use the installed diff
826
binary to perform file comparison, using supplied options.
827
:param old_label: Prefix to use for old file labels
828
:param new_label: Prefix to use for new file labels
829
:param using: Commandline to use to invoke an external diff tool
831
if using is not None:
832
extra_factories = [DiffFromTool.make_from_diff_tree(using)]
835
if external_diff_options:
836
opts = external_diff_options.split()
837
def diff_file(olab, olines, nlab, nlines, to_file):
838
external_diff(olab, olines, nlab, nlines, to_file, opts)
840
diff_file = internal_diff
841
diff_text = DiffText(old_tree, new_tree, to_file, path_encoding,
842
old_label, new_label, diff_file)
843
return klass(old_tree, new_tree, to_file, path_encoding, diff_text,
846
def show_diff(self, specific_files, extra_trees=None):
847
"""Write tree diff to self.to_file
849
:param sepecific_files: the specific files to compare (recursive)
850
:param extra_trees: extra trees to use for mapping paths to file_ids
853
return self._show_diff(specific_files, extra_trees)
855
for differ in self.differs:
858
def _show_diff(self, specific_files, extra_trees):
859
# TODO: Generation of pseudo-diffs for added/deleted files could
860
# be usefully made into a much faster special case.
861
iterator = self.new_tree.iter_changes(self.old_tree,
862
specific_files=specific_files,
863
extra_trees=extra_trees,
864
require_versioned=True)
866
def changes_key(change):
867
old_path, new_path = change[1]
872
def get_encoded_path(path):
874
return path.encode(self.path_encoding, "replace")
875
for (file_id, paths, changed_content, versioned, parent, name, kind,
876
executable) in sorted(iterator, key=changes_key):
877
# The root does not get diffed, and items with no known kind (that
878
# is, missing) in both trees are skipped as well.
879
if parent == (None, None) or kind == (None, None):
881
oldpath, newpath = paths
882
oldpath_encoded = get_encoded_path(paths[0])
883
newpath_encoded = get_encoded_path(paths[1])
884
old_present = (kind[0] is not None and versioned[0])
885
new_present = (kind[1] is not None and versioned[1])
886
renamed = (parent[0], name[0]) != (parent[1], name[1])
888
properties_changed = []
889
properties_changed.extend(get_executable_change(executable[0], executable[1]))
891
if properties_changed:
892
prop_str = " (properties changed: %s)" % (", ".join(properties_changed),)
896
if (old_present, new_present) == (True, False):
897
self.to_file.write("=== removed %s '%s'\n" %
898
(kind[0], oldpath_encoded))
900
elif (old_present, new_present) == (False, True):
901
self.to_file.write("=== added %s '%s'\n" %
902
(kind[1], newpath_encoded))
905
self.to_file.write("=== renamed %s '%s' => '%s'%s\n" %
906
(kind[0], oldpath_encoded, newpath_encoded, prop_str))
908
# if it was produced by iter_changes, it must be
909
# modified *somehow*, either content or execute bit.
910
self.to_file.write("=== modified %s '%s'%s\n" % (kind[0],
911
newpath_encoded, prop_str))
913
self.diff(file_id, oldpath, newpath)
919
def diff(self, file_id, old_path, new_path):
920
"""Perform a diff of a single file
922
:param file_id: file-id of the file
923
:param old_path: The path of the file in the old tree
924
:param new_path: The path of the file in the new tree
927
old_kind = self.old_tree.kind(file_id)
928
except (errors.NoSuchId, errors.NoSuchFile):
931
new_kind = self.new_tree.kind(file_id)
932
except (errors.NoSuchId, errors.NoSuchFile):
935
result = DiffPath._diff_many(self.differs, file_id, old_path,
936
new_path, old_kind, new_kind)
937
if result is DiffPath.CANNOT_DIFF:
938
error_path = new_path
939
if error_path is None:
940
error_path = old_path
941
raise errors.NoDiffFound(error_path)