~bzr-pqm/bzr/bzr.dev

1 by mbp at sourcefrog
import from baz patch-364
1
#! /usr/bin/env python
2
# -*- coding: UTF-8 -*-
3
4
# This program is free software; you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation; either version 2 of the License, or
7
# (at your option) any later version.
8
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
# GNU General Public License for more details.
13
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17
478 by Martin Pool
- put back support for running diff or status on
18
from sets import Set, ImmutableSet
1 by mbp at sourcefrog
import from baz patch-364
19
20
from trace import mutter
356 by Martin Pool
- pychecker fixes in bzrlib.diff
21
from errors import BzrError
1 by mbp at sourcefrog
import from baz patch-364
22
23
329 by Martin Pool
- refactor command functions into command classes
24
475 by Martin Pool
- rewrite diff using compare_trees()
25
def _diff_one(oldlines, newlines, to_file, **kw):
26
    import difflib
27
    
28
    # FIXME: difflib is wrong if there is no trailing newline.
29
    # The syntax used by patch seems to be "\ No newline at
30
    # end of file" following the last diff line from that
31
    # file.  This is not trivial to insert into the
32
    # unified_diff output and it might be better to just fix
33
    # or replace that function.
34
35
    # In the meantime we at least make sure the patch isn't
36
    # mangled.
37
38
39
    # Special workaround for Python2.3, where difflib fails if
40
    # both sequences are empty.
41
    if not oldlines and not newlines:
42
        return
43
44
    nonl = False
45
46
    if oldlines and (oldlines[-1][-1] != '\n'):
47
        oldlines[-1] += '\n'
48
        nonl = True
49
    if newlines and (newlines[-1][-1] != '\n'):
50
        newlines[-1] += '\n'
51
        nonl = True
52
53
    ud = difflib.unified_diff(oldlines, newlines, **kw)
54
55
    # work-around for difflib being too smart for its own good
56
    # if /dev/null is "1,0", patch won't recognize it as /dev/null
57
    if not oldlines:
58
        ud = list(ud)
59
        ud[2] = ud[2].replace('-1,0', '-0,0')
60
    elif not newlines:
61
        ud = list(ud)
62
        ud[2] = ud[2].replace('+1,0', '+0,0')
63
64
    to_file.writelines(ud)
65
    if nonl:
66
        print >>to_file, "\\ No newline at end of file"
67
    print >>to_file
68
69
329 by Martin Pool
- refactor command functions into command classes
70
def show_diff(b, revision, file_list):
475 by Martin Pool
- rewrite diff using compare_trees()
71
    import sys
72
329 by Martin Pool
- refactor command functions into command classes
73
    if revision == None:
74
        old_tree = b.basis_tree()
75
    else:
76
        old_tree = b.revision_tree(b.lookup_revision(revision))
77
        
78
    new_tree = b.working_tree()
79
80
    # TODO: Options to control putting on a prefix or suffix, perhaps as a format string
81
    old_label = ''
82
    new_label = ''
83
84
    DEVNULL = '/dev/null'
85
    # Windows users, don't panic about this filename -- it is a
86
    # special signal to GNU patch that the file should be created or
87
    # deleted respectively.
88
89
    # TODO: Generation of pseudo-diffs for added/deleted files could
90
    # be usefully made into a much faster special case.
91
478 by Martin Pool
- put back support for running diff or status on
92
    delta = compare_trees(old_tree, new_tree, want_unchanged=False,
93
                          file_list=file_list)
475 by Martin Pool
- rewrite diff using compare_trees()
94
95
    for path, file_id, kind in delta.removed:
96
        print '*** removed %s %r' % (kind, path)
97
        if kind == 'file':
98
            _diff_one(old_tree.get_file(file_id).readlines(),
99
                   [],
100
                   sys.stdout,
101
                   fromfile=old_label + path,
102
                   tofile=DEVNULL)
103
104
    for path, file_id, kind in delta.added:
105
        print '*** added %s %r' % (kind, path)
106
        if kind == 'file':
107
            _diff_one([],
108
                   new_tree.get_file(file_id).readlines(),
109
                   sys.stdout,
110
                   fromfile=DEVNULL,
111
                   tofile=new_label + path)
112
113
    for old_path, new_path, file_id, kind, text_modified in delta.renamed:
114
        print '*** renamed %s %r => %r' % (kind, old_path, new_path)
115
        if text_modified:
116
            _diff_one(old_tree.get_file(file_id).readlines(),
117
                   new_tree.get_file(file_id).readlines(),
118
                   sys.stdout,
119
                   fromfile=old_label + old_path,
120
                   tofile=new_label + new_path)
121
122
    for path, file_id, kind in delta.modified:
123
        print '*** modified %s %r' % (kind, path)
124
        if kind == 'file':
125
            _diff_one(old_tree.get_file(file_id).readlines(),
126
                   new_tree.get_file(file_id).readlines(),
127
                   sys.stdout,
128
                   fromfile=old_label + path,
129
                   tofile=new_label + path)
329 by Martin Pool
- refactor command functions into command classes
130
131
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
132
133
class TreeDelta:
134
    """Describes changes from one tree to another.
135
136
    Contains four lists:
137
138
    added
475 by Martin Pool
- rewrite diff using compare_trees()
139
        (path, id, kind)
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
140
    removed
475 by Martin Pool
- rewrite diff using compare_trees()
141
        (path, id, kind)
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
142
    renamed
475 by Martin Pool
- rewrite diff using compare_trees()
143
        (oldpath, newpath, id, kind, text_modified)
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
144
    modified
475 by Martin Pool
- rewrite diff using compare_trees()
145
        (path, id, kind)
463 by Martin Pool
- compare_trees() also reports unchanged files
146
    unchanged
475 by Martin Pool
- rewrite diff using compare_trees()
147
        (path, id, kind)
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
148
460 by Martin Pool
- new testing command compare-trees
149
    Each id is listed only once.
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
150
460 by Martin Pool
- new testing command compare-trees
151
    Files that are both modified and renamed are listed only in
152
    renamed, with the text_modified flag true.
463 by Martin Pool
- compare_trees() also reports unchanged files
153
154
    The lists are normally sorted when the delta is created.
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
155
    """
156
    def __init__(self):
157
        self.added = []
158
        self.removed = []
159
        self.renamed = []
160
        self.modified = []
463 by Martin Pool
- compare_trees() also reports unchanged files
161
        self.unchanged = []
379 by Martin Pool
- Simpler compare_inventories() to possibly replace diff_trees
162
465 by Martin Pool
- Move show_status() out of Branch into a new function in
163
    def show(self, to_file, show_ids=False, show_unchanged=False):
164
        def show_list(files):
475 by Martin Pool
- rewrite diff using compare_trees()
165
            for path, fid, kind in files:
166
                if kind == 'directory':
167
                    path += '/'
168
                elif kind == 'symlink':
169
                    path += '@'
170
                    
465 by Martin Pool
- Move show_status() out of Branch into a new function in
171
                if show_ids:
172
                    print >>to_file, '  %-30s %s' % (path, fid)
173
                else:
174
                    print >>to_file, ' ', path
175
            
460 by Martin Pool
- new testing command compare-trees
176
        if self.removed:
475 by Martin Pool
- rewrite diff using compare_trees()
177
            print >>to_file, 'removed:'
465 by Martin Pool
- Move show_status() out of Branch into a new function in
178
            show_list(self.removed)
179
                
460 by Martin Pool
- new testing command compare-trees
180
        if self.added:
475 by Martin Pool
- rewrite diff using compare_trees()
181
            print >>to_file, 'added:'
465 by Martin Pool
- Move show_status() out of Branch into a new function in
182
            show_list(self.added)
183
460 by Martin Pool
- new testing command compare-trees
184
        if self.renamed:
475 by Martin Pool
- rewrite diff using compare_trees()
185
            print >>to_file, 'renamed:'
186
            for oldpath, newpath, fid, kind, text_modified in self.renamed:
460 by Martin Pool
- new testing command compare-trees
187
                if show_ids:
188
                    print >>to_file, '  %s => %s %s' % (oldpath, newpath, fid)
189
                else:
190
                    print >>to_file, '  %s => %s' % (oldpath, newpath)
465 by Martin Pool
- Move show_status() out of Branch into a new function in
191
                    
460 by Martin Pool
- new testing command compare-trees
192
        if self.modified:
475 by Martin Pool
- rewrite diff using compare_trees()
193
            print >>to_file, 'modified:'
465 by Martin Pool
- Move show_status() out of Branch into a new function in
194
            show_list(self.modified)
195
            
196
        if show_unchanged and self.unchanged:
475 by Martin Pool
- rewrite diff using compare_trees()
197
            print >>to_file, 'unchanged:'
465 by Martin Pool
- Move show_status() out of Branch into a new function in
198
            show_list(self.unchanged)
460 by Martin Pool
- new testing command compare-trees
199
200
201
478 by Martin Pool
- put back support for running diff or status on
202
def compare_trees(old_tree, new_tree, want_unchanged, file_list=None):
203
    """Describe changes from one tree to another.
204
205
    Returns a TreeDelta with details of added, modified, renamed, and
206
    deleted entries.
207
208
    The root entry is specifically exempt.
209
210
    This only considers versioned files.
211
212
    want_unchanged
213
        If true, also list files unchanged from one version to the next.
214
215
    file_list
216
        If true, only check for changes to specified files.        
217
    """
460 by Martin Pool
- new testing command compare-trees
218
    old_inv = old_tree.inventory
219
    new_inv = new_tree.inventory
220
    delta = TreeDelta()
475 by Martin Pool
- rewrite diff using compare_trees()
221
    mutter('start compare_trees')
478 by Martin Pool
- put back support for running diff or status on
222
223
    if file_list:
224
        file_list = ImmutableSet(file_list)
225
462 by Martin Pool
- New form 'file_id in tree' to check if the file is present
226
    for file_id in old_tree:
227
        if file_id in new_tree:
460 by Martin Pool
- new testing command compare-trees
228
            kind = old_inv.get_file_kind(file_id)
475 by Martin Pool
- rewrite diff using compare_trees()
229
            assert kind == new_inv.get_file_kind(file_id)
230
            
460 by Martin Pool
- new testing command compare-trees
231
            assert kind in ('file', 'directory', 'symlink', 'root_directory'), \
232
                   'invalid file kind %r' % kind
477 by Martin Pool
- fix header for listing of unknown files
233
234
            if kind == 'root_directory':
235
                continue
236
            
237
            old_path = old_inv.id2path(file_id)
238
            new_path = new_inv.id2path(file_id)
239
478 by Martin Pool
- put back support for running diff or status on
240
            if file_list:
241
                if (old_path not in file_list
242
                    and new_path not in file_list):
243
                    continue
244
460 by Martin Pool
- new testing command compare-trees
245
            if kind == 'file':
246
                old_sha1 = old_tree.get_file_sha1(file_id)
247
                new_sha1 = new_tree.get_file_sha1(file_id)
248
                text_modified = (old_sha1 != new_sha1)
249
            else:
250
                ## mutter("no text to check for %r %r" % (file_id, kind))
251
                text_modified = False
471 by Martin Pool
- actually avoid reporting unchanged files if not required
252
253
            # TODO: Can possibly avoid calculating path strings if the
254
            # two files are unchanged and their names and parents are
255
            # the same and the parents are unchanged all the way up.
256
            # May not be worthwhile.
460 by Martin Pool
- new testing command compare-trees
257
            
258
            if old_path != new_path:
475 by Martin Pool
- rewrite diff using compare_trees()
259
                delta.renamed.append((old_path, new_path, file_id, kind,
260
                                      text_modified))
460 by Martin Pool
- new testing command compare-trees
261
            elif text_modified:
475 by Martin Pool
- rewrite diff using compare_trees()
262
                delta.modified.append((new_path, file_id, kind))
471 by Martin Pool
- actually avoid reporting unchanged files if not required
263
            elif want_unchanged:
475 by Martin Pool
- rewrite diff using compare_trees()
264
                delta.unchanged.append((new_path, file_id, kind))
460 by Martin Pool
- new testing command compare-trees
265
        else:
475 by Martin Pool
- rewrite diff using compare_trees()
266
            delta.removed.append((old_inv.id2path(file_id), file_id, kind))
267
268
    mutter('start looking for new files')
460 by Martin Pool
- new testing command compare-trees
269
    for file_id in new_inv:
270
        if file_id in old_inv:
271
            continue
478 by Martin Pool
- put back support for running diff or status on
272
        new_path = new_inv.id2path(file_id)
273
        if file_list:
274
            if new_path not in file_list:
275
                continue
475 by Martin Pool
- rewrite diff using compare_trees()
276
        kind = new_inv.get_file_kind(file_id)
478 by Martin Pool
- put back support for running diff or status on
277
        delta.added.append((new_path, file_id, kind))
460 by Martin Pool
- new testing command compare-trees
278
            
279
    delta.removed.sort()
280
    delta.added.sort()
281
    delta.renamed.sort()
282
    delta.modified.sort()
474 by Martin Pool
- sort unchanged files
283
    delta.unchanged.sort()
460 by Martin Pool
- new testing command compare-trees
284
285
    return delta