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:
456
mtime = tree.get_file_mtime(file_id, path)
482
457
return timestamp.format_patch_date(mtime)
626
601
# or removed in a diff.
627
602
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):
604
def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8',
605
old_label='', new_label='', text_differ=internal_diff):
632
606
DiffPath.__init__(self, old_tree, new_tree, to_file, path_encoding)
633
607
self.text_differ = text_differ
634
608
self.old_label = old_label
635
609
self.new_label = new_label
636
610
self.path_encoding = path_encoding
637
self.context_lines = context_lines
639
612
def diff(self, file_id, old_path, new_path, old_kind, new_kind):
640
613
"""Compare two files in unified diff format
664
637
return self.CANNOT_DIFF
665
638
from_label = '%s%s\t%s' % (self.old_label, old_path, old_date)
666
639
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)
640
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):
643
def diff_text(self, from_file_id, to_file_id, from_label, to_label,
644
from_path=None, to_path=None):
672
645
"""Diff the content of given files in two trees
674
:param from_path: The path in the from tree. If None,
647
:param from_file_id: The id of the file in the from tree. If None,
675
648
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,
649
:param to_file_id: The id of the file in the to tree. This may refer
650
to a different file from from_file_id. If None,
678
651
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
652
:param from_path: The path in the from tree or None if unknown.
653
:param to_path: The path in the to tree or None if unknown.
684
655
def _get_text(tree, file_id, path):
656
if file_id is not None:
657
return tree.get_file(file_id, path).readlines()
687
return tree.get_file_lines(path, file_id)
689
661
from_text = _get_text(self.old_tree, from_file_id, from_path)
690
662
to_text = _get_text(self.new_tree, to_file_id, to_path)
691
663
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
665
except errors.BinaryFile:
695
666
self.to_file.write(
696
667
("Binary files %s and %s differ\n" %
697
(from_label, to_label)).encode(self.path_encoding, 'replace'))
668
(from_label, to_label)).encode(self.path_encoding))
698
669
return self.CHANGED
704
675
path_encoding='utf-8'):
705
676
DiffPath.__init__(self, old_tree, new_tree, to_file, path_encoding)
706
677
self.command_template = command_template
707
self._root = osutils.mkdtemp(prefix='brz-diff-')
678
self._root = osutils.mkdtemp(prefix='bzr-diff-')
710
681
def from_string(klass, command_string, old_tree, new_tree, to_file,
711
682
path_encoding='utf-8'):
712
command_template = cmdline.split(command_string)
683
command_template = commands.shlex_split_unicode(command_string)
713
684
if '@' not in command_string:
714
685
command_template.extend(['@old_path', '@new_path'])
715
686
return klass(command_template, old_tree, new_tree, to_file,
719
def make_from_diff_tree(klass, command_string, external_diff_options=None):
690
def make_from_diff_tree(klass, command_string):
720
691
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,
692
return klass.from_string(command_string, diff_tree.old_tree,
725
693
diff_tree.new_tree, diff_tree.to_file)
726
694
return from_diff_tree
728
696
def _get_command(self, old_path, new_path):
729
697
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
698
return [AtTemplate(t).substitute(my_map) for t in
699
self.command_template]
743
701
def _execute(self, old_path, new_path):
744
702
command = self._get_command(old_path, new_path)
746
704
proc = subprocess.Popen(command, stdout=subprocess.PIPE,
749
707
if e.errno == errno.ENOENT:
750
708
raise errors.ExecutableMissing(command[0])
761
719
os.symlink(tree.abspath(''), osutils.pathjoin(self._root, prefix))
763
721
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):
725
def _write_file(self, file_id, tree, prefix, relpath, force_temp=False,
797
727
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)
728
return tree.abspath(tree.id2path(file_id))
730
full_path = osutils.pathjoin(self._root, prefix, relpath)
803
731
if not force_temp and self._try_symlink_root(tree, prefix):
805
733
parent_dir = osutils.dirname(full_path)
807
735
os.makedirs(parent_dir)
809
737
if e.errno != errno.EEXIST:
811
source = tree.get_file(relpath, file_id)
739
source = tree.get_file(file_id, relpath)
813
with open(full_path, 'wb') as target:
741
target = open(full_path, 'wb')
814
743
osutils.pumpfile(source, target)
818
mtime = tree.get_file_mtime(relpath, file_id)
819
except FileTimestampUnavailable:
822
os.utime(full_path, (mtime, mtime))
823
748
if not allow_write:
824
749
osutils.make_readonly(full_path)
750
mtime = tree.get_file_mtime(file_id)
751
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,
754
def _prepare_files(self, file_id, old_path, new_path, force_temp=False,
755
allow_write_new=False):
756
old_disk_path = self._write_file(file_id, self.old_tree, 'old',
757
old_path, force_temp)
758
new_disk_path = self._write_file(file_id, self.new_tree, 'new',
759
new_path, force_temp,
833
760
allow_write=allow_write_new)
834
761
return old_disk_path, new_disk_path
836
763
def finish(self):
838
765
osutils.rmtree(self._root)
840
767
if e.errno != errno.ENOENT:
841
768
mutter("The temporary directory \"%s\" was not "
842
769
"cleanly removed: %s." % (self._root, e))
845
772
if (old_kind, new_kind) != ('file', 'file'):
846
773
return DiffPath.CANNOT_DIFF
847
774
(old_disk_path, new_disk_path) = self._prepare_files(
848
old_path, new_path, file_id=file_id)
775
file_id, old_path, new_path)
849
776
self._execute(old_disk_path, new_disk_path)
851
def edit_file(self, old_path, new_path, file_id=None):
778
def edit_file(self, file_id):
852
779
"""Use this tool to edit a file.
854
781
A temporary copy will be edited, and the new contents will be
857
784
:param file_id: The id of the file to edit.
858
785
: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)
787
old_path = self.old_tree.id2path(file_id)
788
new_path = self.new_tree.id2path(file_id)
789
new_abs_path = self._prepare_files(file_id, old_path, new_path,
790
allow_write_new=True,
792
command = self._get_command(osutils.pathjoin('old', old_path),
793
osutils.pathjoin('new', new_path))
864
794
subprocess.call(command, cwd=self._root)
865
with open(new_abs_path, 'rb') as new_file:
795
new_file = open(new_abs_path, 'r')
866
797
return new_file.read()
869
802
class DiffTree(object):
927
859
:param using: Commandline to use to invoke an external diff tool
929
861
if using is not None:
930
extra_factories = [DiffFromTool.make_from_diff_tree(using, external_diff_options)]
862
extra_factories = [DiffFromTool.make_from_diff_tree(using)]
932
864
extra_factories = []
933
865
if external_diff_options:
934
866
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.
867
def diff_file(olab, olines, nlab, nlines, to_file):
939
868
external_diff(olab, olines, nlab, nlines, to_file, opts)
941
870
diff_file = internal_diff
942
871
diff_text = DiffText(old_tree, new_tree, to_file, path_encoding,
943
old_label, new_label, diff_file, context_lines=context_lines)
872
old_label, new_label, diff_file)
944
873
return klass(old_tree, new_tree, to_file, path_encoding, diff_text,
947
876
def show_diff(self, specific_files, extra_trees=None):
948
877
"""Write tree diff to self.to_file
950
:param specific_files: the specific files to compare (recursive)
879
:param sepecific_files: the specific files to compare (recursive)
951
880
:param extra_trees: extra trees to use for mapping paths to file_ids
990
919
properties_changed.extend(get_executable_change(executable[0], executable[1]))
992
921
if properties_changed:
993
prop_str = b" (properties changed: %s)" % (
994
b", ".join(properties_changed),)
922
prop_str = " (properties changed: %s)" % (", ".join(properties_changed),)
998
926
if (old_present, new_present) == (True, False):
999
self.to_file.write(b"=== removed %s '%s'\n" %
1000
(kind[0].encode('ascii'), oldpath_encoded))
927
self.to_file.write("=== removed %s '%s'\n" %
928
(kind[0], oldpath_encoded))
1001
929
newpath = oldpath
1002
930
elif (old_present, new_present) == (False, True):
1003
self.to_file.write(b"=== added %s '%s'\n" %
1004
(kind[1].encode('ascii'), newpath_encoded))
931
self.to_file.write("=== added %s '%s'\n" %
932
(kind[1], newpath_encoded))
1005
933
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))
935
self.to_file.write("=== renamed %s '%s' => '%s'%s\n" %
936
(kind[0], oldpath_encoded, newpath_encoded, prop_str))
1010
938
# if it was produced by iter_changes, it must be
1011
939
# modified *somehow*, either content or execute bit.
1012
self.to_file.write(b"=== modified %s '%s'%s\n" % (kind[0].encode('ascii'),
940
self.to_file.write("=== modified %s '%s'%s\n" % (kind[0],
1013
941
newpath_encoded, prop_str))
1014
942
if changed_content:
1015
self._diff(oldpath, newpath, kind[0], kind[1], file_id=file_id)
943
self._diff(file_id, oldpath, newpath, kind[0], kind[1])
1025
953
:param old_path: The path of the file in the old tree
1026
954
:param new_path: The path of the file in the new tree
1028
if old_path is None:
957
old_kind = self.old_tree.kind(file_id)
958
except (errors.NoSuchId, errors.NoSuchFile):
1031
old_kind = self.old_tree.kind(old_path, file_id)
1032
if new_path is None:
961
new_kind = self.new_tree.kind(file_id)
962
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):
964
self._diff(file_id, old_path, new_path, old_kind, new_kind)
967
def _diff(self, file_id, old_path, new_path, old_kind, new_kind):
1039
968
result = DiffPath._diff_many(self.differs, file_id, old_path,
1040
969
new_path, old_kind, new_kind)
1041
970
if result is DiffPath.CANNOT_DIFF: