~abentley/bzrtools/bzrtools.dev

« back to all changes in this revision

Viewing changes to patch.py

  • Committer: Michael Ellerman
  • Date: 2005-11-29 07:12:26 UTC
  • mto: (0.3.1 shelf-dev) (325.1.2 bzrtools)
  • mto: This revision was merged to the branch mainline in revision 334.
  • Revision ID: michael@ellerman.id.au-20051129071226-a04b3f827880025d
Unshelve --pick was broken, because we deleted the whole patch, even when only
part of it was unshelved. So now if we unshelve part of a patch, the patch is
replaced with a new patch that has just the unshelved parts. That's a long way
of saying it does what you'd expect.

Implementing this required changing HunkSelector to return both the selected,
and unselected hunks (ie. patches to shelve, and patches to keep).

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2008 Aaron Bentley, 2006 Michael Ellerman
2
 
# <aaron@aaronbentley.com>
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
 
import sys
18
 
import subprocess
19
 
 
20
 
from bzrlib.workingtree import WorkingTree
21
 
import bzrlib.add
22
 
 
23
 
from bzrlib.plugins.bzrtools.bzrtools import open_from_url
24
 
from errors import CommandError, PatchFailed, PatchInvokeError
25
 
 
26
 
def patch(tree, location, strip, quiet=False):
27
 
    """Apply a patch to a branch, using patch(1).  URLs may be used."""
28
 
    my_file = None
29
 
    if location is None:
30
 
        my_file = sys.stdin
31
 
    else:
32
 
        my_file = open_from_url(location)
33
 
    patches = [my_file.read()]
34
 
    return run_patch(tree.basedir, patches, strip, quiet=quiet)
35
 
 
36
 
 
37
 
def run_patch(directory, patches, strip=0, reverse=False, dry_run=False,
38
 
              quiet=False, _patch_cmd='patch', target_file=None):
39
 
    args = [_patch_cmd, '-d', directory, '-s', '-p%d' % strip, '-f']
40
 
    if quiet:
41
 
        args.append('--quiet')
42
 
 
43
 
    if sys.platform == "win32":
44
 
        args.append('--binary')
45
 
 
46
 
    if reverse:
47
 
        args.append('-R')
48
 
    if dry_run:
49
 
        if sys.platform.startswith('freebsd'):
50
 
            args.append('--check')
51
 
        else:
52
 
            args.append('--dry-run')
53
 
        stderr = subprocess.PIPE
54
 
    else:
55
 
        stderr = None
56
 
    if target_file is not None:
57
 
        args.append(target_file)
58
 
 
59
 
    try:
60
 
        process = subprocess.Popen(args, stdin=subprocess.PIPE,
61
 
                                   stdout=subprocess.PIPE, stderr=stderr)
62
 
    except OSError, e:
63
 
        raise PatchInvokeError(e)
64
 
    try:
65
 
        for patch in patches:
66
 
            process.stdin.write(str(patch))
67
 
        process.stdin.close()
68
 
 
69
 
    except IOError, e:
70
 
        raise PatchInvokeError(e, process.stderr.read())
71
 
 
72
 
    result = process.wait()
73
 
    if not dry_run:
74
 
        sys.stdout.write(process.stdout.read())
75
 
    if result != 0:
76
 
        raise PatchFailed()
77
 
 
78
 
    return result