Allow ':' in the subject of imported messages
[stgit] / stgit / commands / imprt.py
1 __copyright__ = """
2 Copyright (C) 2005, Catalin Marinas <catalin.marinas@gmail.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 version 2 as
6 published by the Free Software Foundation.
7
8 This program is distributed in the hope that it will be useful,
9 but WITHOUT ANY WARRANTY; without even the implied warranty of
10 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 GNU General Public License for more details.
12
13 You should have received a copy of the GNU General Public License
14 along with this program; if not, write to the Free Software
15 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
16 """
17
18 import sys, os
19 from optparse import OptionParser, make_option
20
21 from stgit.commands.common import *
22 from stgit.utils import *
23 from stgit import stack, git
24
25
26 help = 'import a GNU diff file as a new patch'
27 usage = """%prog [options] [<file>]
28
29 Create a new patch and apply the given GNU diff file (or the standard
30 input). By default, the file name is used as the patch name but this
31 can be overriden with the '--name' option. The patch can either be a
32 normal file with the description at the top or it can have standard
33 mail format, the Subject, From and Date headers being used for
34 generating the patch information.
35
36 The patch description has to be separated from the data with a '---'
37 line. For a normal file, if no author information is given, the first
38 'Signed-off-by:' line is used."""
39
40 options = [make_option('-m', '--mail',
41 help = 'import the patch from a standard e-mail file',
42 action = 'store_true'),
43 make_option('-n', '--name',
44 help = 'use NAME as the patch name'),
45 make_option('-a', '--author', metavar = '"NAME <EMAIL>"',
46 help = 'use "NAME <EMAIL>" as the author details'),
47 make_option('--authname',
48 help = 'use AUTHNAME as the author name'),
49 make_option('--authemail',
50 help = 'use AUTHEMAIL as the author e-mail'),
51 make_option('--authdate',
52 help = 'use AUTHDATE as the author date'),
53 make_option('--commname',
54 help = 'use COMMNAME as the committer name'),
55 make_option('--commemail',
56 help = 'use COMMEMAIL as the committer e-mail')]
57
58
59 def __parse_mail(filename = None):
60 """Parse the input file in a mail format and return (description,
61 authname, authemail, authdate)
62 """
63 if filename:
64 f = file(filename)
65 else:
66 f = sys.stdin
67
68 descr = authname = authemail = authdate = None
69
70 # parse the headers
71 for line in f:
72 line = line.strip()
73 if re.match('from:\s+', line, re.I):
74 auth = re.findall('^.*?:\s+(.*)$', line)[0]
75 authname, authemail = name_email(auth)
76 elif re.match('date:\s+', line, re.I):
77 authdate = re.findall('^.*?:\s+(.*)$', line)[0]
78 elif re.match('subject:\s+', line, re.I):
79 descr = re.findall('^.*?:\s+(.*)$', line)[0]
80 elif line == '':
81 # end of headers
82 break
83
84 # remove the '[*PATCH*]' expression in the subject
85 if descr:
86 descr = re.findall('^(\[[^\s]*PATCH.*?\])?\s*(.*)$', descr)[0][1]
87 descr += '\n\n'
88 else:
89 raise CmdException, 'Subject: line not found'
90
91 # the rest of the patch description
92 for line in f:
93 if re.match('----*\s*$', line) or re.match('diff -', line):
94 break
95 else:
96 descr += line
97 descr.rstrip()
98
99 if filename:
100 f.close()
101
102 return (descr, authname, authemail, authdate)
103
104 def __parse_patch(filename = None):
105 """Parse the input file and return (description, authname,
106 authemail, authdate)
107 """
108 if filename:
109 f = file(filename)
110 else:
111 f = sys.stdin
112
113 authname = authemail = authdate = None
114
115 descr = ''
116 for line in f:
117 # the first 'Signed-of-by:' is the author
118 if not authname and re.match('signed-off-by:\s+', line, re.I):
119 auth = re.findall('^.*?:\s+(.*)$', line)[0]
120 authname, authemail = name_email(auth)
121
122 if re.match('----*\s*$', line) or re.match('diff -', line):
123 break
124 else:
125 descr += line
126 descr.rstrip()
127
128 if descr == '':
129 descr = None
130
131 if filename:
132 f.close()
133
134 return (descr, authname, authemail, authdate)
135
136 def func(parser, options, args):
137 """Import a GNU diff file as a new patch
138 """
139 if len(args) > 1:
140 parser.error('incorrect number of arguments')
141
142 check_local_changes()
143 check_conflicts()
144 check_head_top_equal()
145
146 if len(args) == 1:
147 filename = args[0]
148 patch = os.path.basename(filename)
149 elif options.name:
150 filename = None
151 patch = options.name
152 else:
153 raise CmdException, 'Unkown patch name'
154
155 # the defaults
156 message = author_name = author_email = author_date = committer_name = \
157 committer_email = None
158
159 if options.author:
160 options.authname, options.authemail = name_email(options.author)
161
162 if options.mail:
163 message, author_name, author_email, author_date = \
164 __parse_mail(filename)
165 else:
166 message, author_name, author_email, author_date = \
167 __parse_patch(filename)
168
169 # override the automatically parsed settings
170 if options.authname:
171 author_name = options.authname
172 if options.authemail:
173 author_email = options.authemail
174 if options.authdate:
175 author_date = options.authdate
176 if options.commname:
177 committer_name = options.commname
178 if options.commemail:
179 committer_email = options.commemail
180
181 crt_series.new_patch(patch, message = message,
182 author_name = author_name,
183 author_email = author_email,
184 author_date = author_date,
185 committer_name = committer_name,
186 committer_email = committer_email)
187
188 print 'Importing patch %s...' % patch,
189 sys.stdout.flush()
190
191 git.apply_patch(filename)
192 crt_series.refresh_patch()
193
194 print 'done'
195 print_crt_patch()