1
# Copyright (C) 2004 - 2006 Aaron Bentley
1
# Copyright (C) 2004 - 2006 Aaron Bentley, Canonical Ltd
2
2
# <aaron.bentley@utoronto.ca>
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.
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.
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
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.
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.
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
19
20
class PatchSyntax(Exception):
96
97
def hunk_from_header(line):
97
if not line.startswith("@@") or not line.endswith("@@\n") \
99
raise MalformedHunkHeader("Does not start and end with @@.", line)
98
matches = re.match(r'\@\@ ([^@]*) \@\@( (.*))?\n', line)
100
raise MalformedHunkHeader("Does not match format.", line)
101
(orig, mod) = line[3:-4].split(" ")
102
(orig, mod) = matches.group(1).split(" ")
103
except (ValueError, IndexError), e:
103
104
raise MalformedHunkHeader(str(e), line)
104
105
if not orig.startswith('-') or not mod.startswith('+'):
105
106
raise MalformedHunkHeader("Positions don't start with + or -.", line)
107
108
(orig_pos, orig_range) = parse_range(orig[1:])
108
109
(mod_pos, mod_range) = parse_range(mod[1:])
110
except (ValueError, IndexError), e:
110
111
raise MalformedHunkHeader(str(e), line)
111
112
if mod_range < 0 or orig_range < 0:
112
113
raise MalformedHunkHeader("Hunk range is negative", line)
113
return Hunk(orig_pos, orig_range, mod_pos, mod_range)
114
tail = matches.group(3)
115
return Hunk(orig_pos, orig_range, mod_pos, mod_range, tail)
173
def __init__(self, orig_pos, orig_range, mod_pos, mod_range):
175
def __init__(self, orig_pos, orig_range, mod_pos, mod_range, tail=None):
174
176
self.orig_pos = orig_pos
175
177
self.orig_range = orig_range
176
178
self.mod_pos = mod_pos
177
179
self.mod_range = mod_range
180
183
def get_header(self):
181
return "@@ -%s +%s @@\n" % (self.range_str(self.orig_pos,
183
self.range_str(self.mod_pos,
184
if self.tail is None:
187
tail_str = ' ' + self.tail
188
return "@@ -%s +%s @@%s\n" % (self.range_str(self.orig_pos,
190
self.range_str(self.mod_pos,
186
194
def range_str(self, pos, range):
187
195
"""Return a file range, special-casing for 1-line files.
212
220
return self.shift_to_mod_lines(pos)
214
222
def shift_to_mod_lines(self, pos):
215
assert (pos >= self.orig_pos-1 and pos <= self.orig_pos+self.orig_range)
216
223
position = self.orig_pos-1
218
225
for line in self.lines:
317
324
def iter_file_patch(iter_lines):
319
327
for line in iter_lines:
320
328
if line.startswith('=== ') or line.startswith('*** '):
322
330
if line.startswith('#'):
333
if line.startswith('-') or line.startswith(' '):
324
335
elif line.startswith('--- '):
325
336
if len(saved_lines) > 0:
326
337
yield saved_lines
339
elif line.startswith('@@'):
340
hunk = hunk_from_header(line)
341
orig_range = hunk.orig_range
328
342
saved_lines.append(line)
329
343
if len(saved_lines) > 0:
330
344
yield saved_lines
400
415
if isinstance(hunk_line, ContextLine):
403
assert isinstance(hunk_line, RemoveLine)
418
if not isinstance(hunk_line, RemoveLine):
419
raise AssertionError(hunk_line)
405
421
if orig_lines is not None:
406
422
for line in orig_lines: