bzr branch
http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
| 1658.1.9
by Martin Pool Give an error for bzr diff on an nonexistent file (Malone #3619) | 1 | # Copyright (C) 2004, 2005, 2006 Canonical Ltd.
 | 
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 2 | #
 | 
| 1
by mbp at sourcefrog import from baz patch-364 | 3 | # This program is free software; you can redistribute it and/or modify
 | 
| 4 | # it under the terms of the GNU General Public License as published by
 | |
| 5 | # the Free Software Foundation; either version 2 of the License, or
 | |
| 6 | # (at your option) any later version.
 | |
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 7 | #
 | 
| 1
by mbp at sourcefrog import from baz patch-364 | 8 | # This program is distributed in the hope that it will be useful,
 | 
| 9 | # but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| 10 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| 11 | # GNU General Public License for more details.
 | |
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 12 | #
 | 
| 1
by mbp at sourcefrog import from baz patch-364 | 13 | # You should have received a copy of the GNU General Public License
 | 
| 14 | # along with this program; if not, write to the Free Software
 | |
| 15 | # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 | |
| 16 | ||
| 2520.4.140
by Aaron Bentley Use matching blocks from mpdiff for knit delta creation | 17 | import difflib | 
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 18 | import os | 
| 1899.1.5
by John Arbash Meinel Always buffer the output of diff, so we can check if retcode==2 is because of Binary files | 19 | import re | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 20 | import shutil | 
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 21 | import sys | 
| 22 | ||
| 23 | from bzrlib.lazy_import import lazy_import | |
| 24 | lazy_import(globals(), """ | |
| 25 | import errno
 | |
| 1692.8.7
by James Henstridge changes suggested by John Meinel | 26 | import subprocess
 | 
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 27 | import tempfile
 | 
| 1740.2.5
by Aaron Bentley Merge from bzr.dev | 28 | import time
 | 
| 29 | ||
| 1955.2.10
by John Arbash Meinel Unset a few other LANG type variables when spawning diff | 30 | from bzrlib import (
 | 
| 1551.19.33
by Aaron Bentley Use as_revision_id for diff | 31 |     branch as _mod_branch,
 | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 32 |     bzrdir,
 | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 33 |     commands,
 | 
| 1955.2.10
by John Arbash Meinel Unset a few other LANG type variables when spawning diff | 34 |     errors,
 | 
| 35 |     osutils,
 | |
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 36 |     patiencediff,
 | 
| 37 |     textfile,
 | |
| 1551.12.29
by Aaron Bentley Copy and extend patch date formatting code, add patch-date parsing | 38 |     timestamp,
 | 
| 1955.2.10
by John Arbash Meinel Unset a few other LANG type variables when spawning diff | 39 |     )
 | 
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 40 | """) | 
| 41 | ||
| 42 | from bzrlib.symbol_versioning import ( | |
| 43 | deprecated_function, | |
| 3268.1.1
by C Miller Describe the property changes in diffs. Currently, this is the executable-bit | 44 |         one_three
 | 
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 45 |         )
 | 
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 46 | from bzrlib.trace import mutter, warning | 
| 1
by mbp at sourcefrog import from baz patch-364 | 47 | |
| 1711.2.24
by John Arbash Meinel Late bind to PatienceSequenceMatcher to allow plugin to override. | 48 | |
| 767
by Martin Pool - files are only reported as modified if their name or parent has changed, | 49 | # TODO: Rather than building a changeset object, we should probably
 | 
| 50 | # invoke callbacks on an object.  That object can either accumulate a
 | |
| 51 | # list, write them out directly, etc etc.
 | |
| 52 | ||
| 2520.4.140
by Aaron Bentley Use matching blocks from mpdiff for knit delta creation | 53 | |
| 54 | class _PrematchedMatcher(difflib.SequenceMatcher): | |
| 55 | """Allow SequenceMatcher operations to use predetermined blocks""" | |
| 56 | ||
| 57 | def __init__(self, matching_blocks): | |
| 58 | difflib.SequenceMatcher(self, None, None) | |
| 59 | self.matching_blocks = matching_blocks | |
| 60 | self.opcodes = None | |
| 61 | ||
| 62 | ||
| 1558.15.11
by Aaron Bentley Apply merge review suggestions | 63 | def internal_diff(old_filename, oldlines, new_filename, newlines, to_file, | 
| 1711.2.30
by John Arbash Meinel Fix bug in internal_diff handling of unicode paths | 64 | allow_binary=False, sequence_matcher=None, | 
| 65 | path_encoding='utf8'): | |
| 475
by Martin Pool - rewrite diff using compare_trees() | 66 |     # FIXME: difflib is wrong if there is no trailing newline.
 | 
| 67 |     # The syntax used by patch seems to be "\ No newline at
 | |
| 68 |     # end of file" following the last diff line from that
 | |
| 69 |     # file.  This is not trivial to insert into the
 | |
| 70 |     # unified_diff output and it might be better to just fix
 | |
| 71 |     # or replace that function.
 | |
| 72 | ||
| 73 |     # In the meantime we at least make sure the patch isn't
 | |
| 74 |     # mangled.
 | |
| 75 | ||
| 76 | ||
| 77 |     # Special workaround for Python2.3, where difflib fails if
 | |
| 78 |     # both sequences are empty.
 | |
| 79 | if not oldlines and not newlines: | |
| 80 |         return
 | |
| 1558.15.2
by Aaron Bentley Implemented binary file handling for diff | 81 | |
| 1558.15.11
by Aaron Bentley Apply merge review suggestions | 82 | if allow_binary is False: | 
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 83 | textfile.check_text_lines(oldlines) | 
| 84 | textfile.check_text_lines(newlines) | |
| 475
by Martin Pool - rewrite diff using compare_trees() | 85 | |
| 1185.81.8
by John Arbash Meinel Updating unified_diff to take a factory, using the new diff algorithm in the code. | 86 | if sequence_matcher is None: | 
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 87 | sequence_matcher = patiencediff.PatienceSequenceMatcher | 
| 88 | ud = patiencediff.unified_diff(oldlines, newlines, | |
| 1740.2.5
by Aaron Bentley Merge from bzr.dev | 89 | fromfile=old_filename.encode(path_encoding), | 
| 90 | tofile=new_filename.encode(path_encoding), | |
| 1185.81.8
by John Arbash Meinel Updating unified_diff to take a factory, using the new diff algorithm in the code. | 91 | sequencematcher=sequence_matcher) | 
| 475
by Martin Pool - rewrite diff using compare_trees() | 92 | |
| 1092.1.50
by Robert Collins make diff lsdiff/filterdiff friendly | 93 | ud = list(ud) | 
| 3085.1.1
by John Arbash Meinel Fix internal_diff to not fail when the texts are identical. | 94 | if len(ud) == 0: # Identical contents, nothing to do | 
| 95 |         return
 | |
| 475
by Martin Pool - rewrite diff using compare_trees() | 96 |     # work-around for difflib being too smart for its own good
 | 
| 97 |     # if /dev/null is "1,0", patch won't recognize it as /dev/null
 | |
| 98 | if not oldlines: | |
| 99 | ud[2] = ud[2].replace('-1,0', '-0,0') | |
| 100 | elif not newlines: | |
| 101 | ud[2] = ud[2].replace('+1,0', '+0,0') | |
| 1092.1.50
by Robert Collins make diff lsdiff/filterdiff friendly | 102 |     # work around for difflib emitting random spaces after the label
 | 
| 103 | ud[0] = ud[0][:-2] + '\n' | |
| 104 | ud[1] = ud[1][:-2] + '\n' | |
| 475
by Martin Pool - rewrite diff using compare_trees() | 105 | |
| 804
by Martin Pool Patch from John: | 106 | for line in ud: | 
| 107 | to_file.write(line) | |
| 974.1.5
by Aaron Bentley Fixed handling of missing newlines in udiffs | 108 | if not line.endswith('\n'): | 
| 109 | to_file.write("\n\\ No newline at end of file\n") | |
| 2911.6.1
by Blake Winton Change 'print >> f,'s to 'f.write('s. | 110 | to_file.write('\n') | 
| 475
by Martin Pool - rewrite diff using compare_trees() | 111 | |
| 112 | ||
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 113 | def _spawn_external_diff(diffcmd, capture_errors=True): | 
| 114 | """Spawn the externall diff process, and return the child handle. | |
| 115 | ||
| 116 |     :param diffcmd: The command list to spawn
 | |
| 2138.1.1
by Wouter van Heyst Robuster external diff output handling. | 117 |     :param capture_errors: Capture stderr as well as setting LANG=C
 | 
| 118 |         and LC_ALL=C. This lets us read and understand the output of diff,
 | |
| 119 |         and respond to any errors.
 | |
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 120 |     :return: A Popen object.
 | 
| 121 |     """
 | |
| 122 | if capture_errors: | |
| 2321.2.2
by Alexander Belchenko win32 fixes for test_external_diff_binary (gettext on win32 rely on $LANGUAGE) | 123 |         # construct minimal environment
 | 
| 124 | env = {} | |
| 125 | path = os.environ.get('PATH') | |
| 126 | if path is not None: | |
| 127 | env['PATH'] = path | |
| 2321.2.5
by Alexander Belchenko external diff: no need for special code path for win32 (suggested by John Meinel) | 128 | env['LANGUAGE'] = 'C' # on win32 only LANGUAGE has effect | 
| 129 | env['LANG'] = 'C' | |
| 130 | env['LC_ALL'] = 'C' | |
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 131 | stderr = subprocess.PIPE | 
| 132 | else: | |
| 2321.2.2
by Alexander Belchenko win32 fixes for test_external_diff_binary (gettext on win32 rely on $LANGUAGE) | 133 | env = None | 
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 134 | stderr = None | 
| 135 | ||
| 136 | try: | |
| 137 | pipe = subprocess.Popen(diffcmd, | |
| 138 | stdin=subprocess.PIPE, | |
| 139 | stdout=subprocess.PIPE, | |
| 140 | stderr=stderr, | |
| 2321.2.2
by Alexander Belchenko win32 fixes for test_external_diff_binary (gettext on win32 rely on $LANGUAGE) | 141 | env=env) | 
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 142 | except OSError, e: | 
| 143 | if e.errno == errno.ENOENT: | |
| 144 | raise errors.NoDiff(str(e)) | |
| 145 |         raise
 | |
| 146 | ||
| 147 | return pipe | |
| 148 | ||
| 149 | ||
| 1185.35.29
by Aaron Bentley Support whitespace in diff filenames | 150 | def external_diff(old_filename, oldlines, new_filename, newlines, to_file, | 
| 571
by Martin Pool - new --diff-options to pass options through to external | 151 | diff_opts): | 
| 568
by Martin Pool - start adding support for showing diffs by calling out to | 152 | """Display a diff by calling out to the external diff program.""" | 
| 581
by Martin Pool - make sure any bzr output is flushed before | 153 |     # make sure our own output is properly ordered before the diff
 | 
| 154 | to_file.flush() | |
| 155 | ||
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 156 | oldtmp_fd, old_abspath = tempfile.mkstemp(prefix='bzr-diff-old-') | 
| 157 | newtmp_fd, new_abspath = tempfile.mkstemp(prefix='bzr-diff-new-') | |
| 158 | oldtmpf = os.fdopen(oldtmp_fd, 'wb') | |
| 159 | newtmpf = os.fdopen(newtmp_fd, 'wb') | |
| 568
by Martin Pool - start adding support for showing diffs by calling out to | 160 | |
| 161 | try: | |
| 162 |         # TODO: perhaps a special case for comparing to or from the empty
 | |
| 163 |         # sequence; can just use /dev/null on Unix
 | |
| 164 | ||
| 165 |         # TODO: if either of the files being compared already exists as a
 | |
| 166 |         # regular named file (e.g. in the working directory) then we can
 | |
| 167 |         # compare directly to that, rather than copying it.
 | |
| 168 | ||
| 169 | oldtmpf.writelines(oldlines) | |
| 170 | newtmpf.writelines(newlines) | |
| 171 | ||
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 172 | oldtmpf.close() | 
| 173 | newtmpf.close() | |
| 568
by Martin Pool - start adding support for showing diffs by calling out to | 174 | |
| 571
by Martin Pool - new --diff-options to pass options through to external | 175 | if not diff_opts: | 
| 176 | diff_opts = [] | |
| 177 | diffcmd = ['diff', | |
| 1740.2.5
by Aaron Bentley Merge from bzr.dev | 178 | '--label', old_filename, | 
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 179 | old_abspath, | 
| 1740.2.5
by Aaron Bentley Merge from bzr.dev | 180 | '--label', new_filename, | 
| 1711.2.56
by John Arbash Meinel Raise NoDiff if 'diff' not present. | 181 | new_abspath, | 
| 182 | '--binary', | |
| 183 |                   ]
 | |
| 571
by Martin Pool - new --diff-options to pass options through to external | 184 | |
| 185 |         # diff only allows one style to be specified; they don't override.
 | |
| 186 |         # note that some of these take optargs, and the optargs can be
 | |
| 187 |         # directly appended to the options.
 | |
| 188 |         # this is only an approximate parser; it doesn't properly understand
 | |
| 189 |         # the grammar.
 | |
| 190 | for s in ['-c', '-u', '-C', '-U', | |
| 191 | '-e', '--ed', | |
| 192 | '-q', '--brief', | |
| 193 | '--normal', | |
| 194 | '-n', '--rcs', | |
| 195 | '-y', '--side-by-side', | |
| 196 | '-D', '--ifdef']: | |
| 197 | for j in diff_opts: | |
| 198 | if j.startswith(s): | |
| 199 |                     break
 | |
| 200 | else: | |
| 201 |                 continue
 | |
| 202 |             break
 | |
| 203 | else: | |
| 204 | diffcmd.append('-u') | |
| 205 | ||
| 206 | if diff_opts: | |
| 207 | diffcmd.extend(diff_opts) | |
| 208 | ||
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 209 | pipe = _spawn_external_diff(diffcmd, capture_errors=True) | 
| 210 | out,err = pipe.communicate() | |
| 211 | rc = pipe.returncode | |
| 571
by Martin Pool - new --diff-options to pass options through to external | 212 | |
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 213 |         # internal_diff() adds a trailing newline, add one here for consistency
 | 
| 214 | out += '\n' | |
| 1899.1.5
by John Arbash Meinel Always buffer the output of diff, so we can check if retcode==2 is because of Binary files | 215 | if rc == 2: | 
| 216 |             # 'diff' gives retcode == 2 for all sorts of errors
 | |
| 217 |             # one of those is 'Binary files differ'.
 | |
| 218 |             # Bad options could also be the problem.
 | |
| 1904.1.4
by Marien Zwart Make external diff in binary mode work with recent versions of diffutils. | 219 |             # 'Binary files' is not a real error, so we suppress that error.
 | 
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 220 | lang_c_out = out | 
| 221 | ||
| 222 |             # Since we got here, we want to make sure to give an i18n error
 | |
| 223 | pipe = _spawn_external_diff(diffcmd, capture_errors=False) | |
| 224 | out, err = pipe.communicate() | |
| 225 | ||
| 226 |             # Write out the new i18n diff response
 | |
| 227 | to_file.write(out+'\n') | |
| 228 | if pipe.returncode != 2: | |
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 229 | raise errors.BzrError( | 
| 230 |                                'external diff failed with exit code 2'
 | |
| 2138.1.1
by Wouter van Heyst Robuster external diff output handling. | 231 |                                ' when run with LANG=C and LC_ALL=C,'
 | 
| 232 | ' but not when run natively: %r' % (diffcmd,)) | |
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 233 | |
| 234 | first_line = lang_c_out.split('\n', 1)[0] | |
| 1904.1.4
by Marien Zwart Make external diff in binary mode work with recent versions of diffutils. | 235 |             # Starting with diffutils 2.8.4 the word "binary" was dropped.
 | 
| 236 | m = re.match('^(binary )?files.*differ$', first_line, re.I) | |
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 237 | if m is None: | 
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 238 | raise errors.BzrError('external diff failed with exit code 2;' | 
| 239 | ' command: %r' % (diffcmd,)) | |
| 1920.1.1
by John Arbash Meinel fix bug #56307, handle binary files even when LANG is not english | 240 | else: | 
| 241 |                 # Binary files differ, just return
 | |
| 242 |                 return
 | |
| 243 | ||
| 244 |         # If we got to here, we haven't written out the output of diff
 | |
| 245 |         # do so now
 | |
| 246 | to_file.write(out) | |
| 247 | if rc not in (0, 1): | |
| 571
by Martin Pool - new --diff-options to pass options through to external | 248 |             # returns 1 if files differ; that's OK
 | 
| 249 | if rc < 0: | |
| 250 | msg = 'signal %d' % (-rc) | |
| 251 | else: | |
| 252 | msg = 'exit code %d' % rc | |
| 253 | ||
| 1996.3.9
by John Arbash Meinel lazy_import diff.py | 254 | raise errors.BzrError('external diff failed with %s; command: %r' | 
| 255 | % (rc, diffcmd)) | |
| 1899.1.6
by John Arbash Meinel internal_diff always adds a trailing \n, make sure external_diff does too | 256 | |
| 257 | ||
| 568
by Martin Pool - start adding support for showing diffs by calling out to | 258 | finally: | 
| 259 | oldtmpf.close() # and delete | |
| 260 | newtmpf.close() | |
| 1711.2.54
by John Arbash Meinel Use mkstemp instead of NamedTemporary file for external diff. | 261 |         # Clean up. Warn in case the files couldn't be deleted
 | 
| 262 |         # (in case windows still holds the file open, but not
 | |
| 263 |         # if the files have already been deleted)
 | |
| 264 | try: | |
| 265 | os.remove(old_abspath) | |
| 266 | except OSError, e: | |
| 267 | if e.errno not in (errno.ENOENT,): | |
| 268 | warning('Failed to delete temporary file: %s %s', | |
| 269 | old_abspath, e) | |
| 270 | try: | |
| 271 | os.remove(new_abspath) | |
| 272 | except OSError: | |
| 273 | if e.errno not in (errno.ENOENT,): | |
| 274 | warning('Failed to delete temporary file: %s %s', | |
| 275 | new_abspath, e) | |
| 568
by Martin Pool - start adding support for showing diffs by calling out to | 276 | |
| 1551.2.13
by Aaron Bentley Got diff working properly with checkouts | 277 | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 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. | |
| 280 | ||
| 281 |     This method works out the trees to be diff'ed and the files of
 | |
| 282 |     interest within those trees.
 | |
| 283 | ||
| 284 |     :param path_list:
 | |
| 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.
 | |
| 289 |     :param old_url:
 | |
| 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.
 | |
| 292 |     :param new_url:
 | |
| 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.
 | |
| 295 |     :returns:
 | |
| 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
 | |
| 298 |         file-ids.
 | |
| 299 |     """
 | |
| 300 |     # Get the old and new revision specs
 | |
| 301 | old_revision_spec = None | |
| 302 | new_revision_spec = None | |
| 303 | if revision_specs is not None: | |
| 304 | if len(revision_specs) > 0: | |
| 305 | old_revision_spec = revision_specs[0] | |
| 3072.1.5
by Ian Clatworthy more good ideas from abentley | 306 | if old_url is None: | 
| 307 | old_url = old_revision_spec.get_branch() | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 308 | if len(revision_specs) > 1: | 
| 309 | new_revision_spec = revision_specs[1] | |
| 3072.1.5
by Ian Clatworthy more good ideas from abentley | 310 | if new_url is None: | 
| 311 | new_url = new_revision_spec.get_branch() | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 312 | |
| 3072.1.5
by Ian Clatworthy more good ideas from abentley | 313 | other_paths = [] | 
| 314 | make_paths_wt_relative = True | |
| 3164.1.1
by Ian Clatworthy diff without arguments means the current tree, not the current directory | 315 | consider_relpath = True | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 316 | if path_list is None or len(path_list) == 0: | 
| 3164.1.1
by Ian Clatworthy diff without arguments means the current tree, not the current directory | 317 |         # If no path is given, the current working tree is used
 | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 318 | default_location = u'.' | 
| 3164.1.1
by Ian Clatworthy diff without arguments means the current tree, not the current directory | 319 | consider_relpath = False | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 320 | elif old_url is not None and new_url is not None: | 
| 321 | other_paths = path_list | |
| 3072.1.5
by Ian Clatworthy more good ideas from abentley | 322 | make_paths_wt_relative = False | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 323 | else: | 
| 324 | default_location = path_list[0] | |
| 325 | other_paths = path_list[1:] | |
| 326 | ||
| 327 |     # Get the old location
 | |
| 3072.1.2
by Ian Clatworthy Test various --old and --new combinations | 328 | specific_files = [] | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 329 | if old_url is None: | 
| 330 | old_url = default_location | |
| 331 | working_tree, branch, relpath = \ | |
| 332 | bzrdir.BzrDir.open_containing_tree_or_branch(old_url) | |
| 3164.1.1
by Ian Clatworthy diff without arguments means the current tree, not the current directory | 333 | if consider_relpath and relpath != '': | 
| 3072.1.2
by Ian Clatworthy Test various --old and --new combinations | 334 | specific_files.append(relpath) | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 335 | old_tree = _get_tree_to_diff(old_revision_spec, working_tree, branch) | 
| 336 | ||
| 337 |     # Get the new location
 | |
| 338 | if new_url is None: | |
| 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) | |
| 3164.1.1
by Ian Clatworthy diff without arguments means the current tree, not the current directory | 343 | if consider_relpath and relpath != '': | 
| 3072.1.2
by Ian Clatworthy Test various --old and --new combinations | 344 | specific_files.append(relpath) | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 345 | new_tree = _get_tree_to_diff(new_revision_spec, working_tree, branch, | 
| 346 | basis_is_default=working_tree is None) | |
| 347 | ||
| 3072.1.2
by Ian Clatworthy Test various --old and --new combinations | 348 |     # Get the specific files (all files is None, no files is [])
 | 
| 3072.1.5
by Ian Clatworthy more good ideas from abentley | 349 | if make_paths_wt_relative and working_tree is not None: | 
| 3072.1.2
by Ian Clatworthy Test various --old and --new combinations | 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 | |
| 354 | ||
| 355 |     # Get extra trees that ought to be searched for file-ids
 | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 356 | extra_trees = None | 
| 3072.1.5
by Ian Clatworthy more good ideas from abentley | 357 | if working_tree is not None and working_tree not in (old_tree, new_tree): | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 358 | extra_trees = (working_tree,) | 
| 359 | return old_tree, new_tree, specific_files, extra_trees | |
| 360 | ||
| 361 | ||
| 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: | |
| 364 | branch = tree.branch | |
| 365 | if spec is None or spec.spec is None: | |
| 366 | if basis_is_default: | |
| 3072.1.5
by Ian Clatworthy more good ideas from abentley | 367 | if tree is not None: | 
| 368 | return tree.basis_tree() | |
| 369 | else: | |
| 370 | return branch.basis_tree() | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 371 | else: | 
| 372 | return tree | |
| 3655.3.1
by Lukáš Lalinský Fix `bzr st -rbranch:PATH_TO_BRANCH` | 373 | return spec.as_tree(branch) | 
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 374 | |
| 375 | ||
| 3072.1.2
by Ian Clatworthy Test various --old and --new combinations | 376 | def _relative_paths_in_tree(tree, paths): | 
| 377 | """Get the relative paths within a working tree. | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 378 | |
| 3072.1.6
by Ian Clatworthy Fix docstring for _relative_paths_in_tree | 379 |     Each path may be either an absolute path or a path relative to the
 | 
| 380 |     current working directory.
 | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 381 |     """
 | 
| 3072.1.2
by Ian Clatworthy Test various --old and --new combinations | 382 | result = [] | 
| 383 | for filename in paths: | |
| 384 | try: | |
| 385 | result.append(tree.relpath(osutils.dereference_path(filename))) | |
| 386 | except errors.PathNotChild: | |
| 387 | raise errors.BzrCommandError("Files are in different branches") | |
| 388 | return result | |
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 389 | |
| 390 | ||
| 571
by Martin Pool - new --diff-options to pass options through to external | 391 | def show_diff_trees(old_tree, new_tree, to_file, specific_files=None, | 
| 1684.1.6
by Martin Pool (patch) --diff-prefix option (goffredo, alexander) | 392 | external_diff_options=None, | 
| 1551.7.17
by Aaron Bentley Switch to PathsNotVersioned, accept extra_trees | 393 | old_label='a/', new_label='b/', | 
| 2598.6.12
by ghigo Move the encoding of the commit message at the command line level | 394 | extra_trees=None, | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 395 | path_encoding='utf8', | 
| 396 | using=None): | |
| 550
by Martin Pool - Refactor diff code into one that works purely on | 397 | """Show in text form the changes from one tree to another. | 
| 398 | ||
| 3072.1.1
by Ian Clatworthy Improved diff based on feedback from abentley | 399 |     to_file
 | 
| 400 |         The output stream.
 | |
| 401 | ||
| 402 |     specific_files
 | |
| 403 |         Include only changes to these files - None for all changes.
 | |
| 571
by Martin Pool - new --diff-options to pass options through to external | 404 | |
| 405 |     external_diff_options
 | |
| 406 |         If set, use an external GNU diff and pass these options.
 | |
| 1551.7.18
by Aaron Bentley Indentation and documentation fixes | 407 | |
| 408 |     extra_trees
 | |
| 409 |         If set, more Trees to use for looking up file ids
 | |
| 2598.6.12
by ghigo Move the encoding of the commit message at the command line level | 410 | |
| 411 |     path_encoding
 | |
| 2598.6.24
by ghigo update on the basis of Aaron suggestions | 412 |         If set, the path will be encoded as specified, otherwise is supposed
 | 
| 413 |         to be utf8
 | |
| 550
by Martin Pool - Refactor diff code into one that works purely on | 414 |     """
 | 
| 1543.1.1
by Denys Duchier lock operations for trees - use them for diff | 415 | old_tree.lock_read() | 
| 416 | try: | |
| 2255.7.38
by John Arbash Meinel show_diff_trees() should lock any extra trees it is passed. | 417 | if extra_trees is not None: | 
| 418 | for tree in extra_trees: | |
| 419 | tree.lock_read() | |
| 1543.1.1
by Denys Duchier lock operations for trees - use them for diff | 420 | new_tree.lock_read() | 
| 421 | try: | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 422 | differ = DiffTree.from_trees_options(old_tree, new_tree, to_file, | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 423 | path_encoding, | 
| 424 | external_diff_options, | |
| 425 | old_label, new_label, using) | |
| 3009.2.12
by Aaron Bentley Associate labels with text diffing only | 426 | return differ.show_diff(specific_files, extra_trees) | 
| 1543.1.1
by Denys Duchier lock operations for trees - use them for diff | 427 | finally: | 
| 428 | new_tree.unlock() | |
| 2255.7.38
by John Arbash Meinel show_diff_trees() should lock any extra trees it is passed. | 429 | if extra_trees is not None: | 
| 430 | for tree in extra_trees: | |
| 431 | tree.unlock() | |
| 1543.1.1
by Denys Duchier lock operations for trees - use them for diff | 432 | finally: | 
| 433 | old_tree.unlock() | |
| 434 | ||
| 435 | ||
| 1740.2.5
by Aaron Bentley Merge from bzr.dev | 436 | def _patch_header_date(tree, file_id, path): | 
| 437 | """Returns a timestamp suitable for use in a patch header.""" | |
| 2405.1.2
by John Arbash Meinel Fix bug #103870 by passing None instead of a (sometimes wrong) path | 438 | mtime = tree.get_file_mtime(file_id, path) | 
| 439 | return timestamp.format_patch_date(mtime) | |
| 1740.2.5
by Aaron Bentley Merge from bzr.dev | 440 | |
| 441 | ||
| 1662.1.9
by Martin Pool Give a clear error for bzr status of an unversioned, nonexistent file. (Malone #3619) | 442 | def _raise_if_nonexistent(paths, old_tree, new_tree): | 
| 443 | """Complain if paths are not in either inventory or tree. | |
| 444 | ||
| 445 |     It's OK with the files exist in either tree's inventory, or 
 | |
| 446 |     if they exist in the tree but are not versioned.
 | |
| 447 |     
 | |
| 448 |     This can be used by operations such as bzr status that can accept
 | |
| 449 |     unknown or ignored files.
 | |
| 450 |     """
 | |
| 451 | mutter("check paths: %r", paths) | |
| 452 | if not paths: | |
| 453 |         return
 | |
| 454 | s = old_tree.filter_unversioned_files(paths) | |
| 455 | s = new_tree.filter_unversioned_files(s) | |
| 456 | s = [path for path in s if not new_tree.has_filename(path)] | |
| 457 | if s: | |
| 458 | raise errors.PathsDoNotExist(sorted(s)) | |
| 459 | ||
| 460 | ||
| 3268.1.1
by C Miller Describe the property changes in diffs. Currently, this is the executable-bit | 461 | @deprecated_function(one_three) | 
| 1398
by Robert Collins integrate in Gustavos x-bit patch | 462 | def get_prop_change(meta_modified): | 
| 463 | if meta_modified: | |
| 464 | return " (properties changed)" | |
| 465 | else: | |
| 466 | return "" | |
| 467 | ||
| 3268.1.1
by C Miller Describe the property changes in diffs. Currently, this is the executable-bit | 468 | def get_executable_change(old_is_x, new_is_x): | 
| 469 | descr = { True:"+x", False:"-x", None:"??" } | |
| 470 | if old_is_x != new_is_x: | |
| 471 | return ["%s to %s" % (descr[old_is_x], descr[new_is_x],)] | |
| 472 | else: | |
| 473 | return [] | |
| 474 | ||
| 1398
by Robert Collins integrate in Gustavos x-bit patch | 475 | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 476 | class DiffPath(object): | 
| 3009.2.14
by Aaron Bentley Update return type handling | 477 | """Base type for command object that compare files""" | 
| 3009.2.17
by Aaron Bentley Update docs | 478 | |
| 3009.2.14
by Aaron Bentley Update return type handling | 479 |     # The type or contents of the file were unsuitable for diffing
 | 
| 3009.2.29
by Aaron Bentley Change constants to strings | 480 | CANNOT_DIFF = 'CANNOT_DIFF' | 
| 3009.2.14
by Aaron Bentley Update return type handling | 481 |     # The file has changed in a semantic way
 | 
| 3009.2.29
by Aaron Bentley Change constants to strings | 482 | CHANGED = 'CHANGED' | 
| 483 |     # The file content may have changed, but there is no semantic change
 | |
| 484 | UNCHANGED = 'UNCHANGED' | |
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 485 | |
| 3009.2.13
by Aaron Bentley Refactor differ to support registering differ factories | 486 | def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8'): | 
| 3009.2.17
by Aaron Bentley Update docs | 487 | """Constructor. | 
| 488 | ||
| 489 |         :param old_tree: The tree to show as the old tree in the comparison
 | |
| 490 |         :param new_tree: The tree to show as new in the comparison
 | |
| 491 |         :param to_file: The file to write comparison data to
 | |
| 492 |         :param path_encoding: The character encoding to write paths in
 | |
| 493 |         """
 | |
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 494 | self.old_tree = old_tree | 
| 495 | self.new_tree = new_tree | |
| 496 | self.to_file = to_file | |
| 3009.2.13
by Aaron Bentley Refactor differ to support registering differ factories | 497 | self.path_encoding = path_encoding | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 498 | |
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 499 | def finish(self): | 
| 500 |         pass
 | |
| 501 | ||
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 502 |     @classmethod
 | 
| 503 | def from_diff_tree(klass, diff_tree): | |
| 504 | return klass(diff_tree.old_tree, diff_tree.new_tree, | |
| 505 | diff_tree.to_file, diff_tree.path_encoding) | |
| 506 | ||
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 507 |     @staticmethod
 | 
| 508 | def _diff_many(differs, file_id, old_path, new_path, old_kind, new_kind): | |
| 509 | for file_differ in differs: | |
| 510 | result = file_differ.diff(file_id, old_path, new_path, old_kind, | |
| 511 | new_kind) | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 512 | if result is not DiffPath.CANNOT_DIFF: | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 513 | return result | 
| 514 | else: | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 515 | return DiffPath.CANNOT_DIFF | 
| 516 | ||
| 517 | ||
| 518 | class DiffKindChange(object): | |
| 3009.2.17
by Aaron Bentley Update docs | 519 | """Special differ for file kind changes. | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 520 | |
| 3009.2.17
by Aaron Bentley Update docs | 521 |     Represents kind change as deletion + creation.  Uses the other differs
 | 
| 522 |     to do this.
 | |
| 523 |     """
 | |
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 524 | def __init__(self, differs): | 
| 525 | self.differs = differs | |
| 526 | ||
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 527 | def finish(self): | 
| 528 |         pass
 | |
| 529 | ||
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 530 |     @classmethod
 | 
| 531 | def from_diff_tree(klass, diff_tree): | |
| 532 | return klass(diff_tree.differs) | |
| 533 | ||
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 534 | def diff(self, file_id, old_path, new_path, old_kind, new_kind): | 
| 3009.2.17
by Aaron Bentley Update docs | 535 | """Perform comparison | 
| 536 | ||
| 537 |         :param file_id: The file_id of the file to compare
 | |
| 538 |         :param old_path: Path of the file in the old tree
 | |
| 539 |         :param new_path: Path of the file in the new tree
 | |
| 540 |         :param old_kind: Old file-kind of the file
 | |
| 541 |         :param new_kind: New file-kind of the file
 | |
| 542 |         """
 | |
| 3009.2.18
by Aaron Bentley Change KindChangeDiffer's anti-recursion to avoid kind pairs with None | 543 | if None in (old_kind, new_kind): | 
| 3009.2.22
by Aaron Bentley Update names & docstring | 544 | return DiffPath.CANNOT_DIFF | 
| 545 | result = DiffPath._diff_many(self.differs, file_id, old_path, | |
| 3009.2.18
by Aaron Bentley Change KindChangeDiffer's anti-recursion to avoid kind pairs with None | 546 | new_path, old_kind, None) | 
| 3009.2.22
by Aaron Bentley Update names & docstring | 547 | if result is DiffPath.CANNOT_DIFF: | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 548 | return result | 
| 3009.2.22
by Aaron Bentley Update names & docstring | 549 | return DiffPath._diff_many(self.differs, file_id, old_path, new_path, | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 550 | None, new_kind) | 
| 551 | ||
| 552 | ||
| 3009.2.22
by Aaron Bentley Update names & docstring | 553 | class DiffDirectory(DiffPath): | 
| 3009.2.19
by Aaron Bentley Implement directory diffing | 554 | |
| 555 | def diff(self, file_id, old_path, new_path, old_kind, new_kind): | |
| 556 | """Perform comparison between two directories. (dummy) | |
| 557 | ||
| 558 |         """
 | |
| 559 | if 'directory' not in (old_kind, new_kind): | |
| 560 | return self.CANNOT_DIFF | |
| 561 | if old_kind not in ('directory', None): | |
| 562 | return self.CANNOT_DIFF | |
| 563 | if new_kind not in ('directory', None): | |
| 564 | return self.CANNOT_DIFF | |
| 565 | return self.CHANGED | |
| 566 | ||
| 3009.2.20
by Aaron Bentley PEP8 | 567 | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 568 | class DiffSymlink(DiffPath): | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 569 | |
| 570 | def diff(self, file_id, old_path, new_path, old_kind, new_kind): | |
| 3009.2.17
by Aaron Bentley Update docs | 571 | """Perform comparison between two symlinks | 
| 572 | ||
| 573 |         :param file_id: The file_id of the file to compare
 | |
| 574 |         :param old_path: Path of the file in the old tree
 | |
| 575 |         :param new_path: Path of the file in the new tree
 | |
| 576 |         :param old_kind: Old file-kind of the file
 | |
| 577 |         :param new_kind: New file-kind of the file
 | |
| 578 |         """
 | |
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 579 | if 'symlink' not in (old_kind, new_kind): | 
| 3009.2.14
by Aaron Bentley Update return type handling | 580 | return self.CANNOT_DIFF | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 581 | if old_kind == 'symlink': | 
| 582 | old_target = self.old_tree.get_symlink_target(file_id) | |
| 583 | elif old_kind is None: | |
| 584 | old_target = None | |
| 585 | else: | |
| 3009.2.14
by Aaron Bentley Update return type handling | 586 | return self.CANNOT_DIFF | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 587 | if new_kind == 'symlink': | 
| 588 | new_target = self.new_tree.get_symlink_target(file_id) | |
| 589 | elif new_kind is None: | |
| 590 | new_target = None | |
| 591 | else: | |
| 3009.2.14
by Aaron Bentley Update return type handling | 592 | return self.CANNOT_DIFF | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 593 | return self.diff_symlink(old_target, new_target) | 
| 594 | ||
| 595 | def diff_symlink(self, old_target, new_target): | |
| 596 | if old_target is None: | |
| 597 | self.to_file.write('=== target is %r\n' % new_target) | |
| 598 | elif new_target is None: | |
| 599 | self.to_file.write('=== target was %r\n' % old_target) | |
| 600 | else: | |
| 601 | self.to_file.write('=== target changed %r => %r\n' % | |
| 602 | (old_target, new_target)) | |
| 3009.2.14
by Aaron Bentley Update return type handling | 603 | return self.CHANGED | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 604 | |
| 605 | ||
| 3009.2.22
by Aaron Bentley Update names & docstring | 606 | class DiffText(DiffPath): | 
| 3009.2.2
by Aaron Bentley Implement Differ object for abstracting diffing | 607 | |
| 3009.2.7
by Aaron Bentley Move responsibility for generating diff labels into Differ.diff | 608 |     # GNU Patch uses the epoch date to detect files that are being added
 | 
| 609 |     # or removed in a diff.
 | |
| 610 | EPOCH_DATE = '1970-01-01 00:00:00 +0000' | |
| 611 | ||
| 3009.2.13
by Aaron Bentley Refactor differ to support registering differ factories | 612 | def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8', | 
| 613 | old_label='', new_label='', text_differ=internal_diff): | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 614 | DiffPath.__init__(self, old_tree, new_tree, to_file, path_encoding) | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 615 | self.text_differ = text_differ | 
| 616 | self.old_label = old_label | |
| 617 | self.new_label = new_label | |
| 3009.2.12
by Aaron Bentley Associate labels with text diffing only | 618 | self.path_encoding = path_encoding | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 619 | |
| 620 | def diff(self, file_id, old_path, new_path, old_kind, new_kind): | |
| 3009.2.17
by Aaron Bentley Update docs | 621 | """Compare two files in unified diff format | 
| 622 | ||
| 623 |         :param file_id: The file_id of the file to compare
 | |
| 624 |         :param old_path: Path of the file in the old tree
 | |
| 625 |         :param new_path: Path of the file in the new tree
 | |
| 626 |         :param old_kind: Old file-kind of the file
 | |
| 627 |         :param new_kind: New file-kind of the file
 | |
| 628 |         """
 | |
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 629 | if 'file' not in (old_kind, new_kind): | 
| 3009.2.14
by Aaron Bentley Update return type handling | 630 | return self.CANNOT_DIFF | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 631 | from_file_id = to_file_id = file_id | 
| 632 | if old_kind == 'file': | |
| 633 | old_date = _patch_header_date(self.old_tree, file_id, old_path) | |
| 634 | elif old_kind is None: | |
| 635 | old_date = self.EPOCH_DATE | |
| 3009.2.12
by Aaron Bentley Associate labels with text diffing only | 636 | from_file_id = None | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 637 | else: | 
| 3009.2.14
by Aaron Bentley Update return type handling | 638 | return self.CANNOT_DIFF | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 639 | if new_kind == 'file': | 
| 640 | new_date = _patch_header_date(self.new_tree, file_id, new_path) | |
| 641 | elif new_kind is None: | |
| 642 | new_date = self.EPOCH_DATE | |
| 643 | to_file_id = None | |
| 644 | else: | |
| 3009.2.14
by Aaron Bentley Update return type handling | 645 | return self.CANNOT_DIFF | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 646 | from_label = '%s%s\t%s' % (self.old_label, old_path, old_date) | 
| 647 | to_label = '%s%s\t%s' % (self.new_label, new_path, new_date) | |
| 648 | return self.diff_text(from_file_id, to_file_id, from_label, to_label) | |
| 649 | ||
| 650 | def diff_text(self, from_file_id, to_file_id, from_label, to_label): | |
| 651 | """Diff the content of given files in two trees | |
| 652 | ||
| 653 |         :param from_file_id: The id of the file in the from tree.  If None,
 | |
| 654 |             the file is not present in the from tree.
 | |
| 655 |         :param to_file_id: The id of the file in the to tree.  This may refer
 | |
| 656 |             to a different file from from_file_id.  If None,
 | |
| 657 |             the file is not present in the to tree.
 | |
| 658 |         """
 | |
| 659 | def _get_text(tree, file_id): | |
| 660 | if file_id is not None: | |
| 661 | return tree.get_file(file_id).readlines() | |
| 662 | else: | |
| 663 | return [] | |
| 664 | try: | |
| 665 | from_text = _get_text(self.old_tree, from_file_id) | |
| 666 | to_text = _get_text(self.new_tree, to_file_id) | |
| 667 | self.text_differ(from_label, from_text, to_label, to_text, | |
| 668 | self.to_file) | |
| 669 | except errors.BinaryFile: | |
| 670 | self.to_file.write( | |
| 671 | ("Binary files %s and %s differ\n" % | |
| 672 | (from_label, to_label)).encode(self.path_encoding)) | |
| 3009.2.14
by Aaron Bentley Update return type handling | 673 | return self.CHANGED | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 674 | |
| 675 | ||
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 676 | class DiffFromTool(DiffPath): | 
| 677 | ||
| 678 | def __init__(self, command_template, old_tree, new_tree, to_file, | |
| 679 | path_encoding='utf-8'): | |
| 680 | DiffPath.__init__(self, old_tree, new_tree, to_file, path_encoding) | |
| 681 | self.command_template = command_template | |
| 3638.3.2
by Vincent Ladeuil Fix all calls to tempfile.mkdtemp to osutils.mkdtemp. | 682 | self._root = osutils.mkdtemp(prefix='bzr-diff-') | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 683 | |
| 684 |     @classmethod
 | |
| 685 | def from_string(klass, command_string, old_tree, new_tree, to_file, | |
| 686 | path_encoding='utf-8'): | |
| 687 | command_template = commands.shlex_split_unicode(command_string) | |
| 688 | command_template.extend(['%(old_path)s', '%(new_path)s']) | |
| 689 | return klass(command_template, old_tree, new_tree, to_file, | |
| 690 | path_encoding) | |
| 691 | ||
| 692 |     @classmethod
 | |
| 693 | def make_from_diff_tree(klass, command_string): | |
| 694 | def from_diff_tree(diff_tree): | |
| 695 | return klass.from_string(command_string, diff_tree.old_tree, | |
| 696 | diff_tree.new_tree, diff_tree.to_file) | |
| 697 | return from_diff_tree | |
| 698 | ||
| 699 | def _get_command(self, old_path, new_path): | |
| 700 | my_map = {'old_path': old_path, 'new_path': new_path} | |
| 3221.6.6
by Matt McClure Cleans up previous change. | 701 | return [t % my_map for t in self.command_template] | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 702 | |
| 703 | def _execute(self, old_path, new_path): | |
| 3145.1.1
by Aaron Bentley Handle missing tools gracefully in diff --using | 704 | command = self._get_command(old_path, new_path) | 
| 705 | try: | |
| 706 | proc = subprocess.Popen(command, stdout=subprocess.PIPE, | |
| 707 | cwd=self._root) | |
| 708 | except OSError, e: | |
| 709 | if e.errno == errno.ENOENT: | |
| 710 | raise errors.ExecutableMissing(command[0]) | |
| 3145.1.2
by Aaron Bentley Don't swallow other OSErrors | 711 | else: | 
| 712 |                 raise
 | |
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 713 | self.to_file.write(proc.stdout.read()) | 
| 714 | return proc.wait() | |
| 715 | ||
| 3123.6.5
by Aaron Bentley Symlink to real files if possible | 716 | def _try_symlink_root(self, tree, prefix): | 
| 3287.18.3
by Matt McClure Toward a more acceptable patch for bug 209281. | 717 | if (getattr(tree, 'abspath', None) is None | 
| 3287.18.14
by Matt McClure Extracted a host_os_dereferences_symlinks method. | 718 | or not osutils.host_os_dereferences_symlinks()): | 
| 3123.6.5
by Aaron Bentley Symlink to real files if possible | 719 | return False | 
| 720 | try: | |
| 721 | os.symlink(tree.abspath(''), osutils.pathjoin(self._root, prefix)) | |
| 722 | except OSError, e: | |
| 723 | if e.errno != errno.EEXIST: | |
| 724 |                 raise
 | |
| 725 | return True | |
| 726 | ||
| 3123.6.4
by Aaron Bentley Set mtime (and atime) on files for --using | 727 | def _write_file(self, file_id, tree, prefix, relpath): | 
| 728 | full_path = osutils.pathjoin(self._root, prefix, relpath) | |
| 3123.6.5
by Aaron Bentley Symlink to real files if possible | 729 | if self._try_symlink_root(tree, prefix): | 
| 730 | return full_path | |
| 3123.6.4
by Aaron Bentley Set mtime (and atime) on files for --using | 731 | parent_dir = osutils.dirname(full_path) | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 732 | try: | 
| 733 | os.makedirs(parent_dir) | |
| 734 | except OSError, e: | |
| 735 | if e.errno != errno.EEXIST: | |
| 736 |                 raise
 | |
| 3123.6.6
by Aaron Bentley Use relpath for get_file | 737 | source = tree.get_file(file_id, relpath) | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 738 | try: | 
| 3123.6.4
by Aaron Bentley Set mtime (and atime) on files for --using | 739 | target = open(full_path, 'wb') | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 740 | try: | 
| 741 | osutils.pumpfile(source, target) | |
| 742 | finally: | |
| 743 | target.close() | |
| 744 | finally: | |
| 745 | source.close() | |
| 3123.6.7
by Aaron Bentley Mark temporary files read-only | 746 | osutils.make_readonly(full_path) | 
| 3123.6.4
by Aaron Bentley Set mtime (and atime) on files for --using | 747 | mtime = tree.get_file_mtime(file_id) | 
| 748 | os.utime(full_path, (mtime, mtime)) | |
| 749 | return full_path | |
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 750 | |
| 751 | def _prepare_files(self, file_id, old_path, new_path): | |
| 752 | old_disk_path = self._write_file(file_id, self.old_tree, 'old', | |
| 753 | old_path) | |
| 754 | new_disk_path = self._write_file(file_id, self.new_tree, 'new', | |
| 755 | new_path) | |
| 756 | return old_disk_path, new_disk_path | |
| 757 | ||
| 758 | def finish(self): | |
| 3123.6.8
by Aaron Bentley Us osutils.rmtree instead of shutil.rmtree, now that some files are readonly | 759 | osutils.rmtree(self._root) | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 760 | |
| 761 | def diff(self, file_id, old_path, new_path, old_kind, new_kind): | |
| 762 | if (old_kind, new_kind) != ('file', 'file'): | |
| 763 | return DiffPath.CANNOT_DIFF | |
| 764 | self._prepare_files(file_id, old_path, new_path) | |
| 3287.18.2
by Matt McClure Reverts to 3290. | 765 | self._execute(osutils.pathjoin('old', old_path), | 
| 766 | osutils.pathjoin('new', new_path)) | |
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 767 | |
| 768 | ||
| 3009.2.22
by Aaron Bentley Update names & docstring | 769 | class DiffTree(object): | 
| 770 | """Provides textual representations of the difference between two trees. | |
| 771 | ||
| 772 |     A DiffTree examines two trees and where a file-id has altered
 | |
| 773 |     between them, generates a textual representation of the difference.
 | |
| 774 |     DiffTree uses a sequence of DiffPath objects which are each
 | |
| 775 |     given the opportunity to handle a given altered fileid. The list
 | |
| 776 |     of DiffPath objects can be extended globally by appending to
 | |
| 777 |     DiffTree.diff_factories, or for a specific diff operation by
 | |
| 3009.2.27
by Aaron Bentley Use extra_factories instead of extra_diffs | 778 |     supplying the extra_factories option to the appropriate method.
 | 
| 3009.2.22
by Aaron Bentley Update names & docstring | 779 |     """
 | 
| 780 | ||
| 781 |     # list of factories that can provide instances of DiffPath objects
 | |
| 3009.2.17
by Aaron Bentley Update docs | 782 |     # may be extended by plugins.
 | 
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 783 | diff_factories = [DiffSymlink.from_diff_tree, | 
| 784 | DiffDirectory.from_diff_tree] | |
| 3009.2.13
by Aaron Bentley Refactor differ to support registering differ factories | 785 | |
| 3009.2.12
by Aaron Bentley Associate labels with text diffing only | 786 | def __init__(self, old_tree, new_tree, to_file, path_encoding='utf-8', | 
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 787 | diff_text=None, extra_factories=None): | 
| 3009.2.17
by Aaron Bentley Update docs | 788 | """Constructor | 
| 789 | ||
| 790 |         :param old_tree: Tree to show as old in the comparison
 | |
| 791 |         :param new_tree: Tree to show as new in the comparison
 | |
| 792 |         :param to_file: File to write comparision to
 | |
| 793 |         :param path_encoding: Character encoding to write paths in
 | |
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 794 |         :param diff_text: DiffPath-type object to use as a last resort for
 | 
| 3009.2.17
by Aaron Bentley Update docs | 795 |             diffing text files.
 | 
| 3009.2.27
by Aaron Bentley Use extra_factories instead of extra_diffs | 796 |         :param extra_factories: Factories of DiffPaths to try before any other
 | 
| 797 |             DiffPaths"""
 | |
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 798 | if diff_text is None: | 
| 799 | diff_text = DiffText(old_tree, new_tree, to_file, path_encoding, | |
| 800 | '', '', internal_diff) | |
| 3009.2.4
by Aaron Bentley Make old_tree/new_tree construction parameters of Differ | 801 | self.old_tree = old_tree | 
| 802 | self.new_tree = new_tree | |
| 3009.2.2
by Aaron Bentley Implement Differ object for abstracting diffing | 803 | self.to_file = to_file | 
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 804 | self.path_encoding = path_encoding | 
| 3009.2.13
by Aaron Bentley Refactor differ to support registering differ factories | 805 | self.differs = [] | 
| 3009.2.27
by Aaron Bentley Use extra_factories instead of extra_diffs | 806 | if extra_factories is not None: | 
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 807 | self.differs.extend(f(self) for f in extra_factories) | 
| 808 | self.differs.extend(f(self) for f in self.diff_factories) | |
| 809 | self.differs.extend([diff_text, DiffKindChange.from_diff_tree(self)]) | |
| 3009.2.6
by Aaron Bentley Convert show_diff_trees into a Differ method | 810 | |
| 811 |     @classmethod
 | |
| 812 | def from_trees_options(klass, old_tree, new_tree, to_file, | |
| 3009.2.17
by Aaron Bentley Update docs | 813 | path_encoding, external_diff_options, old_label, | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 814 | new_label, using): | 
| 3009.2.22
by Aaron Bentley Update names & docstring | 815 | """Factory for producing a DiffTree. | 
| 3009.2.17
by Aaron Bentley Update docs | 816 | |
| 817 |         Designed to accept options used by show_diff_trees.
 | |
| 818 |         :param old_tree: The tree to show as old in the comparison
 | |
| 819 |         :param new_tree: The tree to show as new in the comparison
 | |
| 820 |         :param to_file: File to write comparisons to
 | |
| 821 |         :param path_encoding: Character encoding to use for writing paths
 | |
| 822 |         :param external_diff_options: If supplied, use the installed diff
 | |
| 823 |             binary to perform file comparison, using supplied options.
 | |
| 824 |         :param old_label: Prefix to use for old file labels
 | |
| 825 |         :param new_label: Prefix to use for new file labels
 | |
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 826 |         :param using: Commandline to use to invoke an external diff tool
 | 
| 3009.2.17
by Aaron Bentley Update docs | 827 |         """
 | 
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 828 | if using is not None: | 
| 829 | extra_factories = [DiffFromTool.make_from_diff_tree(using)] | |
| 830 | else: | |
| 831 | extra_factories = [] | |
| 3009.2.6
by Aaron Bentley Convert show_diff_trees into a Differ method | 832 | if external_diff_options: | 
| 833 | opts = external_diff_options.split() | |
| 834 | def diff_file(olab, olines, nlab, nlines, to_file): | |
| 835 | external_diff(olab, olines, nlab, nlines, to_file, opts) | |
| 836 | else: | |
| 837 | diff_file = internal_diff | |
| 3009.2.28
by Aaron Bentley Add from_diff_tree factories | 838 | diff_text = DiffText(old_tree, new_tree, to_file, path_encoding, | 
| 839 | old_label, new_label, diff_file) | |
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 840 | return klass(old_tree, new_tree, to_file, path_encoding, diff_text, | 
| 841 | extra_factories) | |
| 3009.2.6
by Aaron Bentley Convert show_diff_trees into a Differ method | 842 | |
| 3009.2.12
by Aaron Bentley Associate labels with text diffing only | 843 | def show_diff(self, specific_files, extra_trees=None): | 
| 3009.2.17
by Aaron Bentley Update docs | 844 | """Write tree diff to self.to_file | 
| 845 | ||
| 846 |         :param sepecific_files: the specific files to compare (recursive)
 | |
| 847 |         :param extra_trees: extra trees to use for mapping paths to file_ids
 | |
| 848 |         """
 | |
| 3123.6.2
by Aaron Bentley Implement diff --using natively | 849 | try: | 
| 850 | return self._show_diff(specific_files, extra_trees) | |
| 851 | finally: | |
| 852 | for differ in self.differs: | |
| 853 | differ.finish() | |
| 854 | ||
| 855 | def _show_diff(self, specific_files, extra_trees): | |
| 3009.2.6
by Aaron Bentley Convert show_diff_trees into a Differ method | 856 |         # TODO: Generation of pseudo-diffs for added/deleted files could
 | 
| 857 |         # be usefully made into a much faster special case.
 | |
| 3254.1.1
by Aaron Bentley Make Tree.iter_changes a public method | 858 | iterator = self.new_tree.iter_changes(self.old_tree, | 
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 859 | specific_files=specific_files, | 
| 860 | extra_trees=extra_trees, | |
| 861 | require_versioned=True) | |
| 3009.2.6
by Aaron Bentley Convert show_diff_trees into a Differ method | 862 | has_changes = 0 | 
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 863 | def changes_key(change): | 
| 864 | old_path, new_path = change[1] | |
| 865 | path = new_path | |
| 866 | if path is None: | |
| 867 | path = old_path | |
| 868 | return path | |
| 869 | def get_encoded_path(path): | |
| 870 | if path is not None: | |
| 871 | return path.encode(self.path_encoding, "replace") | |
| 872 | for (file_id, paths, changed_content, versioned, parent, name, kind, | |
| 873 | executable) in sorted(iterator, key=changes_key): | |
| 3619.4.2
by Robert Collins Change bzrlib.diff.DiffTree.show_diff to skip entries missing in both trees. | 874 |             # The root does not get diffed, and items with no known kind (that
 | 
| 875 |             # is, missing) in both trees are skipped as well.
 | |
| 876 | if parent == (None, None) or kind == (None, None): | |
| 3123.4.3
by Aaron Bentley Tweak path handling | 877 |                 continue
 | 
| 878 | oldpath, newpath = paths | |
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 879 | oldpath_encoded = get_encoded_path(paths[0]) | 
| 880 | newpath_encoded = get_encoded_path(paths[1]) | |
| 881 | old_present = (kind[0] is not None and versioned[0]) | |
| 882 | new_present = (kind[1] is not None and versioned[1]) | |
| 883 | renamed = (parent[0], name[0]) != (parent[1], name[1]) | |
| 3268.1.1
by C Miller Describe the property changes in diffs. Currently, this is the executable-bit | 884 | |
| 885 | properties_changed = [] | |
| 886 | properties_changed.extend(get_executable_change(executable[0], executable[1])) | |
| 887 | ||
| 888 | if properties_changed: | |
| 889 | prop_str = " (properties changed: %s)" % (", ".join(properties_changed),) | |
| 890 | else: | |
| 891 | prop_str = "" | |
| 892 | ||
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 893 | if (old_present, new_present) == (True, False): | 
| 894 | self.to_file.write("=== removed %s '%s'\n" % | |
| 895 | (kind[0], oldpath_encoded)) | |
| 3123.4.3
by Aaron Bentley Tweak path handling | 896 | newpath = oldpath | 
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 897 | elif (old_present, new_present) == (False, True): | 
| 898 | self.to_file.write("=== added %s '%s'\n" % | |
| 899 | (kind[1], newpath_encoded)) | |
| 3123.4.3
by Aaron Bentley Tweak path handling | 900 | oldpath = newpath | 
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 901 | elif renamed: | 
| 902 | self.to_file.write("=== renamed %s '%s' => '%s'%s\n" % | |
| 903 | (kind[0], oldpath_encoded, newpath_encoded, prop_str)) | |
| 3123.4.2
by Aaron Bentley Handle diff with property change correctly | 904 | else: | 
| 3254.1.1
by Aaron Bentley Make Tree.iter_changes a public method | 905 |                 # if it was produced by iter_changes, it must be
 | 
| 3123.4.2
by Aaron Bentley Handle diff with property change correctly | 906 |                 # modified *somehow*, either content or execute bit.
 | 
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 907 | self.to_file.write("=== modified %s '%s'%s\n" % (kind[0], | 
| 908 | newpath_encoded, prop_str)) | |
| 909 | if changed_content: | |
| 3123.4.3
by Aaron Bentley Tweak path handling | 910 | self.diff(file_id, oldpath, newpath) | 
| 3123.4.1
by Aaron Bentley Diff sorts files in alphabetical order | 911 | has_changes = 1 | 
| 912 | if renamed: | |
| 913 | has_changes = 1 | |
| 3009.2.6
by Aaron Bentley Convert show_diff_trees into a Differ method | 914 | return has_changes | 
| 3009.2.2
by Aaron Bentley Implement Differ object for abstracting diffing | 915 | |
| 3009.2.12
by Aaron Bentley Associate labels with text diffing only | 916 | def diff(self, file_id, old_path, new_path): | 
| 3009.2.17
by Aaron Bentley Update docs | 917 | """Perform a diff of a single file | 
| 918 | ||
| 919 |         :param file_id: file-id of the file
 | |
| 920 |         :param old_path: The path of the file in the old tree
 | |
| 921 |         :param new_path: The path of the file in the new tree
 | |
| 922 |         """
 | |
| 3009.2.2
by Aaron Bentley Implement Differ object for abstracting diffing | 923 | try: | 
| 3009.2.8
by Aaron Bentley Support diffing without indirecting through inventory entries | 924 | old_kind = self.old_tree.kind(file_id) | 
| 3087.1.1
by Aaron Bentley Diff handles missing files correctly, with no tracebacks | 925 | except (errors.NoSuchId, errors.NoSuchFile): | 
| 3009.2.8
by Aaron Bentley Support diffing without indirecting through inventory entries | 926 | old_kind = None | 
| 3009.2.3
by Aaron Bentley Detect missing files from inv operation | 927 | try: | 
| 3009.2.8
by Aaron Bentley Support diffing without indirecting through inventory entries | 928 | new_kind = self.new_tree.kind(file_id) | 
| 3087.1.1
by Aaron Bentley Diff handles missing files correctly, with no tracebacks | 929 | except (errors.NoSuchId, errors.NoSuchFile): | 
| 3009.2.8
by Aaron Bentley Support diffing without indirecting through inventory entries | 930 | new_kind = None | 
| 3009.2.7
by Aaron Bentley Move responsibility for generating diff labels into Differ.diff | 931 | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 932 | result = DiffPath._diff_many(self.differs, file_id, old_path, | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 933 | new_path, old_kind, new_kind) | 
| 3009.2.22
by Aaron Bentley Update names & docstring | 934 | if result is DiffPath.CANNOT_DIFF: | 
| 3009.2.11
by Aaron Bentley Refactor diff to be more pluggable | 935 | error_path = new_path | 
| 936 | if error_path is None: | |
| 937 | error_path = old_path | |
| 3009.2.22
by Aaron Bentley Update names & docstring | 938 | raise errors.NoDiffFound(error_path) |