93
95
if sequence_matcher is None:
94
96
sequence_matcher = patiencediff.PatienceSequenceMatcher
95
97
ud = patiencediff.unified_diff(oldlines, newlines,
96
fromfile=old_filename.encode(path_encoding, 'replace'),
97
tofile=new_filename.encode(path_encoding, 'replace'),
98
n=context_lines, sequencematcher=sequence_matcher)
98
fromfile=old_filename.encode(path_encoding),
99
tofile=new_filename.encode(path_encoding),
100
sequencematcher=sequence_matcher)
101
103
if len(ud) == 0: # Identical contents, nothing to do
143
145
stdout=subprocess.PIPE,
147
149
if e.errno == errno.ENOENT:
148
150
raise errors.NoDiff(str(e))
153
# diff style options as of GNU diff v3.2
154
style_option_list = ['-c', '-C', '--context',
156
'-f', '--forward-ed',
160
'-u', '-U', '--unified',
161
'-y', '--side-by-side',
164
def default_style_unified(diff_opts):
165
"""Default to unified diff style if alternative not specified in diff_opts.
167
diff only allows one style to be specified; they don't override.
168
Note that some of these take optargs, and the optargs can be
169
directly appended to the options.
170
This is only an approximate parser; it doesn't properly understand
173
:param diff_opts: List of options for external (GNU) diff.
174
:return: List of options with default style=='unified'.
176
for s in style_option_list:
184
diff_opts.append('-u')
188
156
def external_diff(old_filename, oldlines, new_filename, newlines, to_file,
228
diff_opts = default_style_unified(diff_opts)
196
# diff only allows one style to be specified; they don't override.
197
# note that some of these take optargs, and the optargs can be
198
# directly appended to the options.
199
# this is only an approximate parser; it doesn't properly understand
201
for s in ['-c', '-u', '-C', '-U',
206
'-y', '--side-by-side',
231
218
diffcmd.extend(diff_opts)
233
220
pipe = _spawn_external_diff(diffcmd, capture_errors=True)
234
out, err = pipe.communicate()
221
out,err = pipe.communicate()
235
222
rc = pipe.returncode
237
224
# internal_diff() adds a trailing newline, add one here for consistency
276
263
msg = 'exit code %d' % rc
278
265
raise errors.BzrError('external diff failed with %s; command: %r'
283
270
oldtmpf.close() # and delete
287
# Warn in case the file couldn't be deleted (in case windows still
288
# holds the file open, but not if the files have already been
293
if e.errno not in (errno.ENOENT,):
294
warning('Failed to delete temporary file: %s %s', path, e)
300
def get_trees_and_branches_to_diff_locked(
301
path_list, revision_specs, old_url, new_url, add_cleanup, apply_view=True):
272
# Clean up. Warn in case the files couldn't be deleted
273
# (in case windows still holds the file open, but not
274
# if the files have already been deleted)
276
os.remove(old_abspath)
278
if e.errno not in (errno.ENOENT,):
279
warning('Failed to delete temporary file: %s %s',
282
os.remove(new_abspath)
284
if e.errno not in (errno.ENOENT,):
285
warning('Failed to delete temporary file: %s %s',
289
def get_trees_and_branches_to_diff(path_list, revision_specs, old_url, new_url,
302
291
"""Get the trees and specific files to diff given a list of paths.
304
293
This method works out the trees to be diff'ed and the files of
316
305
The url of the new branch or tree. If None, the tree to use is
317
306
taken from the first path, if any, or the current working tree.
319
a callable like Command.add_cleanup. get_trees_and_branches_to_diff
320
will register cleanups that must be run to unlock the trees, etc.
321
307
:param apply_view:
322
308
if True and a view is set, apply the view or check that the paths
325
311
a tuple of (old_tree, new_tree, old_branch, new_branch,
326
312
specific_files, extra_trees) where extra_trees is a sequence of
327
additional trees to search in for file-ids. The trees and branches
328
will be read-locked until the cleanups registered via the add_cleanup
313
additional trees to search in for file-ids.
331
315
# Get the old and new revision specs
332
316
old_revision_spec = None
407
384
specific_files = view_files
408
385
view_str = views.view_display_str(view_files)
409
note(gettext("*** Ignoring files outside view. View is %s") % view_str)
386
note("*** Ignoring files outside view. View is %s" % view_str)
411
388
# Get extra trees that ought to be searched for file-ids
412
389
extra_trees = None
413
390
if working_tree is not None and working_tree not in (old_tree, new_tree):
414
391
extra_trees = (working_tree,)
415
return (old_tree, new_tree, old_branch, new_branch,
416
specific_files, extra_trees)
392
return old_tree, new_tree, old_branch, new_branch, specific_files, extra_trees
419
395
def _get_tree_to_diff(spec, tree=None, branch=None, basis_is_default=True):
435
411
old_label='a/', new_label='b/',
436
412
extra_trees=None,
437
413
path_encoding='utf8',
440
context=DEFAULT_CONTEXT_AMOUNT):
441
415
"""Show in text form the changes from one tree to another.
443
:param to_file: The output stream.
444
:param specific_files: Include only changes to these files - None for all
446
:param external_diff_options: If set, use an external GNU diff and pass
448
:param extra_trees: If set, more Trees to use for looking up file ids
449
:param path_encoding: If set, the path will be encoded as specified,
450
otherwise is supposed to be utf8
451
:param format_cls: Formatter class (DiffTree subclass)
421
Include only changes to these files - None for all changes.
423
external_diff_options
424
If set, use an external GNU diff and pass these options.
427
If set, more Trees to use for looking up file ids
430
If set, the path will be encoded as specified, otherwise is supposed
454
context = DEFAULT_CONTEXT_AMOUNT
455
if format_cls is None:
456
format_cls = DiffTree
457
with old_tree.lock_read():
458
435
if extra_trees is not None:
459
436
for tree in extra_trees:
461
438
new_tree.lock_read()
463
differ = format_cls.from_trees_options(old_tree, new_tree, to_file,
465
external_diff_options,
466
old_label, new_label, using,
467
context_lines=context)
440
differ = DiffTree.from_trees_options(old_tree, new_tree, to_file,
442
external_diff_options,
443
old_label, new_label, using)
468
444
return differ.show_diff(specific_files, extra_trees)
470
446
new_tree.unlock()
471
447
if extra_trees is not None:
472
448
for tree in extra_trees:
476
454
def _patch_header_date(tree, file_id, path):
477
455
"""Returns a timestamp suitable for use in a patch header."""
479
mtime = tree.get_file_mtime(path, file_id)
480
except FileTimestampUnavailable:
457
mtime = tree.get_file_mtime(file_id, path)
458
except errors.FileTimestampUnavailable:
482
460
return timestamp.format_patch_date(mtime)
626
604
# or removed in a diff.
627
605
EPOCH_DATE = '1970-01-01 00:00:00 +0000'
629
def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8',
630
old_label='', new_label='', text_differ=internal_diff,
631
context_lines=DEFAULT_CONTEXT_AMOUNT):
607
def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8',
608
old_label='', new_label='', text_differ=internal_diff):
632
609
DiffPath.__init__(self, old_tree, new_tree, to_file, path_encoding)
633
610
self.text_differ = text_differ
634
611
self.old_label = old_label
635
612
self.new_label = new_label
636
613
self.path_encoding = path_encoding
637
self.context_lines = context_lines
639
615
def diff(self, file_id, old_path, new_path, old_kind, new_kind):
640
616
"""Compare two files in unified diff format
664
640
return self.CANNOT_DIFF
665
641
from_label = '%s%s\t%s' % (self.old_label, old_path, old_date)
666
642
to_label = '%s%s\t%s' % (self.new_label, new_path, new_date)
667
return self.diff_text(old_path, new_path, from_label, to_label,
668
from_file_id, to_file_id)
643
return self.diff_text(from_file_id, to_file_id, from_label, to_label,
670
def diff_text(self, from_path, to_path, from_label, to_label,
671
from_file_id=None, to_file_id=None):
646
def diff_text(self, from_file_id, to_file_id, from_label, to_label,
647
from_path=None, to_path=None):
672
648
"""Diff the content of given files in two trees
674
:param from_path: The path in the from tree. If None,
650
:param from_file_id: The id of the file in the from tree. If None,
675
651
the file is not present in the from tree.
676
:param to_path: The path in the to tree. This may refer
677
to a different file from from_path. If None,
652
:param to_file_id: The id of the file in the to tree. This may refer
653
to a different file from from_file_id. If None,
678
654
the file is not present in the to tree.
679
:param from_file_id: The id of the file in the from tree or None if
681
:param to_file_id: The id of the file in the to tree or None if
655
:param from_path: The path in the from tree or None if unknown.
656
:param to_path: The path in the to tree or None if unknown.
684
658
def _get_text(tree, file_id, path):
659
if file_id is not None:
660
return tree.get_file(file_id, path).readlines()
687
return tree.get_file_lines(path, file_id)
689
664
from_text = _get_text(self.old_tree, from_file_id, from_path)
690
665
to_text = _get_text(self.new_tree, to_file_id, to_path)
691
666
self.text_differ(from_label, from_text, to_label, to_text,
692
self.to_file, path_encoding=self.path_encoding,
693
context_lines=self.context_lines)
694
668
except errors.BinaryFile:
695
669
self.to_file.write(
696
670
("Binary files %s and %s differ\n" %
697
(from_label, to_label)).encode(self.path_encoding, 'replace'))
671
(from_label, to_label)).encode(self.path_encoding))
698
672
return self.CHANGED
719
def make_from_diff_tree(klass, command_string, external_diff_options=None):
693
def make_from_diff_tree(klass, command_string):
720
694
def from_diff_tree(diff_tree):
721
full_command_string = [command_string]
722
if external_diff_options is not None:
723
full_command_string += ' ' + external_diff_options
724
return klass.from_string(full_command_string, diff_tree.old_tree,
695
return klass.from_string(command_string, diff_tree.old_tree,
725
696
diff_tree.new_tree, diff_tree.to_file)
726
697
return from_diff_tree
728
699
def _get_command(self, old_path, new_path):
729
700
my_map = {'old_path': old_path, 'new_path': new_path}
730
command = [AtTemplate(t).substitute(my_map) for t in
731
self.command_template]
732
if sys.platform == 'win32': # Popen doesn't accept unicode on win32
735
if isinstance(c, text_type):
736
command_encoded.append(c.encode('mbcs'))
738
command_encoded.append(c)
739
return command_encoded
701
return [AtTemplate(t).substitute(my_map) for t in
702
self.command_template]
743
704
def _execute(self, old_path, new_path):
744
705
command = self._get_command(old_path, new_path)
746
707
proc = subprocess.Popen(command, stdout=subprocess.PIPE,
749
710
if e.errno == errno.ENOENT:
750
711
raise errors.ExecutableMissing(command[0])
761
722
os.symlink(tree.abspath(''), osutils.pathjoin(self._root, prefix))
763
724
if e.errno != errno.EEXIST:
769
"""Returns safe encoding for passing file path to diff tool"""
770
if sys.platform == 'win32':
773
# Don't fallback to 'utf-8' because subprocess may not be able to
774
# handle utf-8 correctly when locale is not utf-8.
775
return sys.getfilesystemencoding() or 'ascii'
777
def _is_safepath(self, path):
778
"""Return true if `path` may be able to pass to subprocess."""
781
return path == path.encode(fenc).decode(fenc)
785
def _safe_filename(self, prefix, relpath):
786
"""Replace unsafe character in `relpath` then join `self._root`,
787
`prefix` and `relpath`."""
789
# encoded_str.replace('?', '_') may break multibyte char.
790
# So we should encode, decode, then replace(u'?', u'_')
791
relpath_tmp = relpath.encode(fenc, 'replace').decode(fenc, 'replace')
792
relpath_tmp = relpath_tmp.replace(u'?', u'_')
793
return osutils.pathjoin(self._root, prefix, relpath_tmp)
795
def _write_file(self, relpath, tree, prefix, force_temp=False,
796
allow_write=False, file_id=None):
728
def _write_file(self, file_id, tree, prefix, relpath, force_temp=False,
797
730
if not force_temp and isinstance(tree, WorkingTree):
798
full_path = tree.abspath(relpath)
799
if self._is_safepath(full_path):
802
full_path = self._safe_filename(prefix, relpath)
731
return tree.abspath(tree.id2path(file_id))
733
full_path = osutils.pathjoin(self._root, prefix, relpath)
803
734
if not force_temp and self._try_symlink_root(tree, prefix):
805
736
parent_dir = osutils.dirname(full_path)
807
738
os.makedirs(parent_dir)
809
740
if e.errno != errno.EEXIST:
811
source = tree.get_file(relpath, file_id)
742
source = tree.get_file(file_id, relpath)
813
with open(full_path, 'wb') as target:
744
target = open(full_path, 'wb')
814
746
osutils.pumpfile(source, target)
818
mtime = tree.get_file_mtime(relpath, file_id)
819
except FileTimestampUnavailable:
822
os.utime(full_path, (mtime, mtime))
823
751
if not allow_write:
824
752
osutils.make_readonly(full_path)
754
mtime = tree.get_file_mtime(file_id)
755
except errors.FileTimestampUnavailable:
757
os.utime(full_path, (mtime, mtime))
827
def _prepare_files(self, old_path, new_path, force_temp=False,
828
allow_write_new=False, file_id=None):
829
old_disk_path = self._write_file(old_path, self.old_tree, 'old',
830
force_temp, file_id=file_id)
831
new_disk_path = self._write_file(new_path, self.new_tree, 'new',
832
force_temp, file_id=file_id,
760
def _prepare_files(self, file_id, old_path, new_path, force_temp=False,
761
allow_write_new=False):
762
old_disk_path = self._write_file(file_id, self.old_tree, 'old',
763
old_path, force_temp)
764
new_disk_path = self._write_file(file_id, self.new_tree, 'new',
765
new_path, force_temp,
833
766
allow_write=allow_write_new)
834
767
return old_disk_path, new_disk_path
836
769
def finish(self):
838
771
osutils.rmtree(self._root)
840
773
if e.errno != errno.ENOENT:
841
774
mutter("The temporary directory \"%s\" was not "
842
775
"cleanly removed: %s." % (self._root, e))
845
778
if (old_kind, new_kind) != ('file', 'file'):
846
779
return DiffPath.CANNOT_DIFF
847
780
(old_disk_path, new_disk_path) = self._prepare_files(
848
old_path, new_path, file_id=file_id)
781
file_id, old_path, new_path)
849
782
self._execute(old_disk_path, new_disk_path)
851
def edit_file(self, old_path, new_path, file_id=None):
784
def edit_file(self, file_id):
852
785
"""Use this tool to edit a file.
854
787
A temporary copy will be edited, and the new contents will be
857
790
:param file_id: The id of the file to edit.
858
791
:return: The new contents of the file.
860
old_abs_path, new_abs_path = self._prepare_files(
861
old_path, new_path, allow_write_new=True, force_temp=True,
863
command = self._get_command(old_abs_path, new_abs_path)
793
old_path = self.old_tree.id2path(file_id)
794
new_path = self.new_tree.id2path(file_id)
795
new_abs_path = self._prepare_files(file_id, old_path, new_path,
796
allow_write_new=True,
798
command = self._get_command(osutils.pathjoin('old', old_path),
799
osutils.pathjoin('new', new_path))
864
800
subprocess.call(command, cwd=self._root)
865
with open(new_abs_path, 'rb') as new_file:
801
new_file = open(new_abs_path, 'r')
866
803
return new_file.read()
869
808
class DiffTree(object):
927
865
:param using: Commandline to use to invoke an external diff tool
929
867
if using is not None:
930
extra_factories = [DiffFromTool.make_from_diff_tree(using, external_diff_options)]
868
extra_factories = [DiffFromTool.make_from_diff_tree(using)]
932
870
extra_factories = []
933
871
if external_diff_options:
934
872
opts = external_diff_options.split()
935
def diff_file(olab, olines, nlab, nlines, to_file, path_encoding=None, context_lines=None):
936
""":param path_encoding: not used but required
937
to match the signature of internal_diff.
873
def diff_file(olab, olines, nlab, nlines, to_file):
939
874
external_diff(olab, olines, nlab, nlines, to_file, opts)
941
876
diff_file = internal_diff
942
877
diff_text = DiffText(old_tree, new_tree, to_file, path_encoding,
943
old_label, new_label, diff_file, context_lines=context_lines)
878
old_label, new_label, diff_file)
944
879
return klass(old_tree, new_tree, to_file, path_encoding, diff_text,
947
882
def show_diff(self, specific_files, extra_trees=None):
948
883
"""Write tree diff to self.to_file
950
:param specific_files: the specific files to compare (recursive)
885
:param sepecific_files: the specific files to compare (recursive)
951
886
:param extra_trees: extra trees to use for mapping paths to file_ids
990
925
properties_changed.extend(get_executable_change(executable[0], executable[1]))
992
927
if properties_changed:
993
prop_str = b" (properties changed: %s)" % (
994
b", ".join(properties_changed),)
928
prop_str = " (properties changed: %s)" % (", ".join(properties_changed),)
998
932
if (old_present, new_present) == (True, False):
999
self.to_file.write(b"=== removed %s '%s'\n" %
1000
(kind[0].encode('ascii'), oldpath_encoded))
933
self.to_file.write("=== removed %s '%s'\n" %
934
(kind[0], oldpath_encoded))
1001
935
newpath = oldpath
1002
936
elif (old_present, new_present) == (False, True):
1003
self.to_file.write(b"=== added %s '%s'\n" %
1004
(kind[1].encode('ascii'), newpath_encoded))
937
self.to_file.write("=== added %s '%s'\n" %
938
(kind[1], newpath_encoded))
1005
939
oldpath = newpath
1007
self.to_file.write(b"=== renamed %s '%s' => '%s'%s\n" %
1008
(kind[0].encode('ascii'), oldpath_encoded, newpath_encoded, prop_str))
941
self.to_file.write("=== renamed %s '%s' => '%s'%s\n" %
942
(kind[0], oldpath_encoded, newpath_encoded, prop_str))
1010
944
# if it was produced by iter_changes, it must be
1011
945
# modified *somehow*, either content or execute bit.
1012
self.to_file.write(b"=== modified %s '%s'%s\n" % (kind[0].encode('ascii'),
946
self.to_file.write("=== modified %s '%s'%s\n" % (kind[0],
1013
947
newpath_encoded, prop_str))
1014
948
if changed_content:
1015
self._diff(oldpath, newpath, kind[0], kind[1], file_id=file_id)
949
self._diff(file_id, oldpath, newpath, kind[0], kind[1])
1025
959
:param old_path: The path of the file in the old tree
1026
960
:param new_path: The path of the file in the new tree
1028
if old_path is None:
963
old_kind = self.old_tree.kind(file_id)
964
except (errors.NoSuchId, errors.NoSuchFile):
1031
old_kind = self.old_tree.kind(old_path, file_id)
1032
if new_path is None:
967
new_kind = self.new_tree.kind(file_id)
968
except (errors.NoSuchId, errors.NoSuchFile):
1035
new_kind = self.new_tree.kind(new_path, file_id)
1036
self._diff(old_path, new_path, old_kind, new_kind, file_id=file_id)
1038
def _diff(self, old_path, new_path, old_kind, new_kind, file_id):
970
self._diff(file_id, old_path, new_path, old_kind, new_kind)
973
def _diff(self, file_id, old_path, new_path, old_kind, new_kind):
1039
974
result = DiffPath._diff_many(self.differs, file_id, old_path,
1040
975
new_path, old_kind, new_kind)
1041
976
if result is DiffPath.CANNOT_DIFF: