Stgit: allow importing series files where patch names include slashes
[stgit] / stgit / commands / imprt.py
CommitLineData
0d2cd1e4
CM
1__copyright__ = """
2Copyright (C) 2005, Catalin Marinas <catalin.marinas@gmail.com>
3
4This program is free software; you can redistribute it and/or modify
5it under the terms of the GNU General Public License version 2 as
6published by the Free Software Foundation.
7
8This program is distributed in the hope that it will be useful,
9but WITHOUT ANY WARRANTY; without even the implied warranty of
10MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11GNU General Public License for more details.
12
13You should have received a copy of the GNU General Public License
14along with this program; if not, write to the Free Software
15Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
16"""
17
9417ece4 18import sys, os, re
0d2cd1e4
CM
19from optparse import OptionParser, make_option
20
21from stgit.commands.common import *
22from stgit.utils import *
23from stgit import stack, git
24
25
26help = 'import a GNU diff file as a new patch'
b8a0986f 27usage = """%prog [options] [<file>]
0d2cd1e4 28
b8a0986f
CM
29Create a new patch and apply the given GNU diff file (or the standard
30input). By default, the file name is used as the patch name but this
388f63b6 31can be overridden with the '--name' option. The patch can either be a
b8a0986f
CM
32normal file with the description at the top or it can have standard
33mail format, the Subject, From and Date headers being used for
34generating the patch information.
0d2cd1e4 35
b8a0986f 36The patch description has to be separated from the data with a '---'
99e73103 37line."""
0d2cd1e4
CM
38
39options = [make_option('-m', '--mail',
40 help = 'import the patch from a standard e-mail file',
41 action = 'store_true'),
42 make_option('-n', '--name',
43 help = 'use NAME as the patch name'),
b0cdad5e
CM
44 make_option('-t', '--strip',
45 help = 'strip numbering and extension from patch name',
46 action = 'store_true'),
9417ece4
CM
47 make_option('-s', '--series',
48 help = 'import a series of patches',
49 action = 'store_true'),
50 make_option('-i', '--ignore',
51 help = 'ignore the applied patches in the series',
52 action = 'store_true'),
034db15c
CM
53 make_option('--replace',
54 help = 'replace the unapplied patches in the series',
55 action = 'store_true'),
b21bc8d1 56 make_option('-b', '--base',
35344f86 57 help = 'use BASE instead of HEAD for file importing'),
33e580e0
CM
58 make_option('-e', '--edit',
59 help = 'invoke an editor for the patch description',
60 action = 'store_true'),
9417ece4 61 make_option('-p', '--showpatch',
6ad48e48
PBG
62 help = 'show the patch content in the editor buffer',
63 action = 'store_true'),
0d2cd1e4
CM
64 make_option('-a', '--author', metavar = '"NAME <EMAIL>"',
65 help = 'use "NAME <EMAIL>" as the author details'),
66 make_option('--authname',
67 help = 'use AUTHNAME as the author name'),
68 make_option('--authemail',
69 help = 'use AUTHEMAIL as the author e-mail'),
70 make_option('--authdate',
71 help = 'use AUTHDATE as the author date'),
72 make_option('--commname',
73 help = 'use COMMNAME as the committer name'),
74 make_option('--commemail',
75 help = 'use COMMEMAIL as the committer e-mail')]
76
77
d4c43e19
PBG
78def __end_descr(line):
79 return re.match('---\s*$', line) or re.match('diff -', line) or \
80 re.match('Index: ', line)
99e73103 81
b0cdad5e 82def __strip_patch_name(name):
bcb6d890
CM
83 stripped = re.sub('^[0-9]+-(.*)$', '\g<1>', name)
84 stripped = re.sub('^(.*)\.(diff|patch)$', '\g<1>', stripped)
85
86 return stripped
b0cdad5e 87
613a2f16
PBG
88def __replace_slashes_with_dashes(name):
89 stripped = name.replace('/', '-')
90
91 return stripped
92
99e73103
CM
93def __parse_description(descr):
94 """Parse the patch description and return the new description and
95 author information (if any).
96 """
97 subject = body = ''
0543bc5f 98 authname = authemail = authdate = None
99e73103 99
0543bc5f 100 descr_lines = [line.rstrip() for line in descr.split('\n')]
99e73103
CM
101 if not descr_lines:
102 raise CmdException, "Empty patch description"
103
0543bc5f 104 lasthdr = 0
99e73103
CM
105 end = len(descr_lines)
106
0543bc5f 107 # Parse the patch header
61dabd0e 108 for pos in range(0, end):
0543bc5f
TM
109 if not descr_lines[pos]:
110 continue
111 # check for a "From|Author:" line
112 if re.match('\s*(?:from|author):\s+', descr_lines[pos], re.I):
113 auth = re.findall('^.*?:\s+(.*)$', descr_lines[pos])[0]
114 authname, authemail = name_email(auth)
115 lasthdr = pos + 1
116 continue
117 # check for a "Date:" line
118 if re.match('\s*date:\s+', descr_lines[pos], re.I):
119 authdate = re.findall('^.*?:\s+(.*)$', descr_lines[pos])[0]
120 lasthdr = pos + 1
121 continue
122 if subject:
123 break
124 # get the subject
125 subject = descr_lines[pos]
126 lasthdr = pos + 1
99e73103
CM
127
128 # get the body
0543bc5f
TM
129 if lasthdr < end:
130 body = reduce(lambda x, y: x + '\n' + y, descr_lines[lasthdr:], '')
99e73103 131
0543bc5f 132 return (subject + body, authname, authemail, authdate)
99e73103 133
0d2cd1e4
CM
134def __parse_mail(filename = None):
135 """Parse the input file in a mail format and return (description,
136 authname, authemail, authdate)
137 """
138 if filename:
139 f = file(filename)
140 else:
141 f = sys.stdin
142
143 descr = authname = authemail = authdate = None
144
145 # parse the headers
6fe6b1bd
CM
146 while True:
147 line = f.readline()
148 if not line:
149 break
0d2cd1e4
CM
150 line = line.strip()
151 if re.match('from:\s+', line, re.I):
152 auth = re.findall('^.*?:\s+(.*)$', line)[0]
153 authname, authemail = name_email(auth)
154 elif re.match('date:\s+', line, re.I):
155 authdate = re.findall('^.*?:\s+(.*)$', line)[0]
156 elif re.match('subject:\s+', line, re.I):
157 descr = re.findall('^.*?:\s+(.*)$', line)[0]
158 elif line == '':
159 # end of headers
160 break
161
186e6b6b 162 # remove the '[*PATCH*]' expression in the subject
0d2cd1e4 163 if descr:
7c02f338
CM
164 descr = re.findall('^(\[[^\s]*[Pp][Aa][Tt][Cc][Hh].*?\])?\s*(.*)$',
165 descr)[0][1]
0d2cd1e4
CM
166 descr += '\n\n'
167 else:
168 raise CmdException, 'Subject: line not found'
169
170 # the rest of the patch description
6fe6b1bd
CM
171 while True:
172 line = f.readline()
173 if not line:
174 break
d4c43e19 175 if __end_descr(line):
0d2cd1e4
CM
176 break
177 else:
178 descr += line
179 descr.rstrip()
180
181 if filename:
182 f.close()
183
99e73103 184 # parse the description for author information
0543bc5f 185 descr, descr_authname, descr_authemail, descr_authdate = __parse_description(descr)
99e73103
CM
186 if descr_authname:
187 authname = descr_authname
188 if descr_authemail:
189 authemail = descr_authemail
0543bc5f
TM
190 if descr_authdate:
191 authdate = descr_authdate
99e73103 192
0d2cd1e4
CM
193 return (descr, authname, authemail, authdate)
194
195def __parse_patch(filename = None):
196 """Parse the input file and return (description, authname,
197 authemail, authdate)
198 """
199 if filename:
200 f = file(filename)
201 else:
202 f = sys.stdin
203
0d2cd1e4 204 descr = ''
6fe6b1bd
CM
205 while True:
206 line = f.readline()
207 if not line:
208 break
209
d4c43e19 210 if __end_descr(line):
0d2cd1e4
CM
211 break
212 else:
213 descr += line
214 descr.rstrip()
215
0d2cd1e4
CM
216 if filename:
217 f.close()
218
0543bc5f 219 descr, authname, authemail, authdate = __parse_description(descr)
99e73103
CM
220
221 # we don't yet have an agreed place for the creation date.
222 # Just return None
0543bc5f 223 return (descr, authname, authemail, authdate)
0d2cd1e4 224
9417ece4
CM
225def __import_patch(patch, filename, options):
226 """Import a patch from a file or standard input
0d2cd1e4 227 """
0d2cd1e4
CM
228 # the defaults
229 message = author_name = author_email = author_date = committer_name = \
230 committer_email = None
231
232 if options.author:
233 options.authname, options.authemail = name_email(options.author)
234
235 if options.mail:
236 message, author_name, author_email, author_date = \
237 __parse_mail(filename)
238 else:
239 message, author_name, author_email, author_date = \
240 __parse_patch(filename)
241
fff9bce5
CM
242 if not patch:
243 patch = make_patch_name(message)
244 if not patch:
245 raise CmdException, 'Unknown patch name'
246
95742cfc
PBG
247 # refresh_patch() will invoke the editor in this case, with correct
248 # patch content
9d15ccd8 249 if not message:
95742cfc 250 can_edit = False
9d15ccd8 251
0d2cd1e4
CM
252 # override the automatically parsed settings
253 if options.authname:
254 author_name = options.authname
255 if options.authemail:
256 author_email = options.authemail
257 if options.authdate:
258 author_date = options.authdate
259 if options.commname:
260 committer_name = options.commname
261 if options.commemail:
262 committer_email = options.commemail
263
034db15c
CM
264 if options.replace and patch in crt_series.get_unapplied():
265 crt_series.delete_patch(patch)
266
95742cfc 267 crt_series.new_patch(patch, message = message, can_edit = False,
0d2cd1e4
CM
268 author_name = author_name,
269 author_email = author_email,
270 author_date = author_date,
271 committer_name = committer_name,
272 committer_email = committer_email)
273
9417ece4 274 print 'Importing patch "%s"...' % patch,
0d2cd1e4
CM
275 sys.stdout.flush()
276
35344f86 277 if options.base:
be3e6bd9 278 git.apply_patch(filename, git_id(options.base))
35344f86
CM
279 else:
280 git.apply_patch(filename)
281
6ad48e48
PBG
282 crt_series.refresh_patch(edit = options.edit,
283 show_patch = options.showpatch)
0d2cd1e4
CM
284
285 print 'done'
9417ece4
CM
286
287def __import_series(filename, options):
288 """Import a series of patches
289 """
290 applied = crt_series.get_applied()
291
292 if filename:
293 f = file(filename)
294 patchdir = os.path.dirname(filename)
295 else:
296 f = sys.stdin
297 patchdir = ''
298
299 for line in f:
300 patch = re.sub('#.*$', '', line).strip()
301 if not patch:
302 continue
bcb6d890
CM
303 patchfile = os.path.join(patchdir, patch)
304
b0cdad5e
CM
305 if options.strip:
306 patch = __strip_patch_name(patch)
613a2f16 307 patch = __replace_slashes_with_dashes(patch);
9417ece4
CM
308 if options.ignore and patch in applied:
309 print 'Ignoring already applied patch "%s"' % patch
310 continue
311
9417ece4
CM
312 __import_patch(patch, patchfile, options)
313
314def func(parser, options, args):
315 """Import a GNU diff file as a new patch
316 """
317 if len(args) > 1:
318 parser.error('incorrect number of arguments')
319
320 check_local_changes()
321 check_conflicts()
322 check_head_top_equal()
323
324 if len(args) == 1:
325 filename = args[0]
326 else:
327 filename = None
328
329 if options.series:
330 __import_series(filename, options)
331 else:
332 if options.name:
333 patch = options.name
334 elif filename:
335 patch = os.path.basename(filename)
336 else:
fff9bce5 337 patch = ''
b0cdad5e
CM
338 if options.strip:
339 patch = __strip_patch_name(patch)
9417ece4
CM
340
341 __import_patch(patch, filename, options)
342
0d2cd1e4 343 print_crt_patch()