~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/diff.py

  • Committer: Martin Pool
  • Date: 2005-09-01 02:34:38 UTC
  • Revision ID: mbp@sourcefrog.net-20050901023437-bf791a0ef5edae8d
- old docs: clarify that this is not mainly descended from arch anymore

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
#! /usr/bin/env python
1
2
# -*- coding: UTF-8 -*-
2
3
 
3
4
# This program is free software; you can redistribute it and/or modify
44
45
    ud = difflib.unified_diff(oldlines, newlines,
45
46
                              fromfile=old_label, tofile=new_label)
46
47
 
47
 
    ud = list(ud)
48
48
    # work-around for difflib being too smart for its own good
49
49
    # if /dev/null is "1,0", patch won't recognize it as /dev/null
50
50
    if not oldlines:
 
51
        ud = list(ud)
51
52
        ud[2] = ud[2].replace('-1,0', '-0,0')
52
53
    elif not newlines:
 
54
        ud = list(ud)
53
55
        ud[2] = ud[2].replace('+1,0', '+0,0')
54
 
    # work around for difflib emitting random spaces after the label
55
 
    ud[0] = ud[0][:-2] + '\n'
56
 
    ud[1] = ud[1][:-2] + '\n'
57
56
 
58
57
    for line in ud:
59
58
        to_file.write(line)
62
61
    print >>to_file
63
62
 
64
63
 
 
64
 
 
65
 
65
66
def external_diff(old_label, oldlines, new_label, newlines, to_file,
66
67
                  diff_opts):
67
68
    """Display a diff by calling out to the external diff program."""
139
140
    finally:
140
141
        oldtmpf.close()                 # and delete
141
142
        newtmpf.close()
142
 
 
143
 
def show_diff(b, from_spec, specific_files, external_diff_options=None,
144
 
              revision2=None, output=None):
 
143
    
 
144
 
 
145
 
 
146
def show_diff(b, revision, specific_files, external_diff_options=None,
 
147
              revision2=None):
145
148
    """Shortcut for showing the diff to the working tree.
146
149
 
147
150
    b
148
151
        Branch.
149
152
 
150
153
    revision
151
 
        None for 'basis tree', or otherwise the old revision to compare against.
 
154
        None for each, or otherwise the old revision to compare against.
152
155
    
153
156
    The more general form is show_diff_trees(), where the caller
154
157
    supplies any two trees.
155
158
    """
156
 
    if output is None:
157
 
        import sys
158
 
        output = sys.stdout
 
159
    import sys
159
160
 
160
 
    if from_spec is None:
 
161
    if revision == None:
161
162
        old_tree = b.basis_tree()
162
163
    else:
163
 
        old_tree = b.revision_tree(from_spec.in_history(b).rev_id)
 
164
        old_tree = b.revision_tree(b.lookup_revision(revision))
164
165
 
165
 
    if revision2 is None:
 
166
    if revision2 == None:
166
167
        new_tree = b.working_tree()
167
168
    else:
168
 
        new_tree = b.revision_tree(revision2.in_history(b).rev_id)
 
169
        new_tree = b.revision_tree(b.lookup_revision(revision2))
169
170
 
170
 
    return show_diff_trees(old_tree, new_tree, output, specific_files,
171
 
                           external_diff_options)
 
171
    show_diff_trees(old_tree, new_tree, sys.stdout, specific_files,
 
172
                    external_diff_options)
172
173
 
173
174
 
174
175
 
207
208
    delta = compare_trees(old_tree, new_tree, want_unchanged=False,
208
209
                          specific_files=specific_files)
209
210
 
210
 
    has_changes = 0
211
211
    for path, file_id, kind in delta.removed:
212
 
        has_changes = 1
213
 
        print >>to_file, '=== removed %s %r' % (kind, path)
214
 
        old_tree.inventory[file_id].diff(diff_file, old_label + path, old_tree,
215
 
                                         DEVNULL, None, None, to_file)
 
212
        print >>to_file, '*** removed %s %r' % (kind, path)
 
213
        if kind == 'file':
 
214
            diff_file(old_label + path,
 
215
                      old_tree.get_file(file_id).readlines(),
 
216
                      DEVNULL, 
 
217
                      [],
 
218
                      to_file)
 
219
 
216
220
    for path, file_id, kind in delta.added:
217
 
        has_changes = 1
218
 
        print >>to_file, '=== added %s %r' % (kind, path)
219
 
        new_tree.inventory[file_id].diff(diff_file, new_label + path, new_tree,
220
 
                                         DEVNULL, None, None, to_file, 
221
 
                                         reverse=True)
222
 
    for (old_path, new_path, file_id, kind,
223
 
         text_modified, meta_modified) in delta.renamed:
224
 
        has_changes = 1
225
 
        prop_str = get_prop_change(meta_modified)
226
 
        print >>to_file, '=== renamed %s %r => %r%s' % (
227
 
                          kind, old_path, new_path, prop_str)
228
 
        _maybe_diff_file_or_symlink(old_label, old_path, old_tree, file_id,
229
 
                                    new_label, new_path, new_tree,
230
 
                                    text_modified, kind, to_file, diff_file)
231
 
    for path, file_id, kind, text_modified, meta_modified in delta.modified:
232
 
        has_changes = 1
233
 
        prop_str = get_prop_change(meta_modified)
234
 
        print >>to_file, '=== modified %s %r%s' % (kind, path, prop_str)
 
221
        print >>to_file, '*** added %s %r' % (kind, path)
 
222
        if kind == 'file':
 
223
            diff_file(DEVNULL,
 
224
                      [],
 
225
                      new_label + path,
 
226
                      new_tree.get_file(file_id).readlines(),
 
227
                      to_file)
 
228
 
 
229
    for old_path, new_path, file_id, kind, text_modified in delta.renamed:
 
230
        print >>to_file, '*** renamed %s %r => %r' % (kind, old_path, new_path)
235
231
        if text_modified:
236
 
            _maybe_diff_file_or_symlink(old_label, path, old_tree, file_id,
237
 
                                        new_label, path, new_tree,
238
 
                                        True, kind, to_file, diff_file)
239
 
    return has_changes
240
 
    
241
 
 
242
 
def get_prop_change(meta_modified):
243
 
    if meta_modified:
244
 
        return " (properties changed)"
245
 
    else:
246
 
        return  ""
247
 
 
248
 
 
249
 
def _maybe_diff_file_or_symlink(old_label, old_path, old_tree, file_id,
250
 
                                new_label, new_path, new_tree, text_modified,
251
 
                                kind, to_file, diff_file):
252
 
    if text_modified:
253
 
        new_entry = new_tree.inventory[file_id]
254
 
        old_tree.inventory[file_id].diff(diff_file,
255
 
                                         old_label + old_path, old_tree,
256
 
                                         new_label + new_path, new_entry, 
257
 
                                         new_tree, to_file)
 
232
            diff_file(old_label + old_path,
 
233
                      old_tree.get_file(file_id).readlines(),
 
234
                      new_label + new_path,
 
235
                      new_tree.get_file(file_id).readlines(),
 
236
                      to_file)
 
237
 
 
238
    for path, file_id, kind in delta.modified:
 
239
        print >>to_file, '*** modified %s %r' % (kind, path)
 
240
        if kind == 'file':
 
241
            diff_file(old_label + path,
 
242
                      old_tree.get_file(file_id).readlines(),
 
243
                      new_label + path,
 
244
                      new_tree.get_file(file_id).readlines(),
 
245
                      to_file)
 
246
 
 
247
 
 
248
 
 
249
 
 
250