3 A script for reporting the results of an invirtibuild. Supports any
4 combination of the following reporting mechanisms.
6 Note that all strings are interpolated with a dictionary containing
9 build_id, commit, failed_stage, inserted_at, package,
10 pocket, principal, result, short_commit, traceback, version
14 To configure zephyr, add something like the following to your invirt config:
19 zephyr: &post_build_zephyr
20 class: myclass [required]
21 instance: myinstance [optional]
22 zsig: myzsig [optional]
24 zephyr: *post_build_zephyr
29 To configure email notifications, add something like the following to your invirt config:
34 mail: &post_build_mail
35 to: myemail@example.com [required]
36 from: myemail@example.com [required]
37 subject: My Subject [optional]
39 mail: *post_build_mail
42 The script chooses which configuration option to use based off the
43 name it is called with. This name also determines which command-line
44 arguments the script takes, as well as how they are formatted. When
47 post-build: uses post_build option
48 failed-build: uses failed_build option
49 post-submit: uses post_submit option
50 failed-submit: uses failed_submit option
51 post-add-repo: uses post_add_repo option
60 from email.mime import text
62 from invirt import common, database, builder
63 from invirt.config import structs as config
65 def build_completion_msg(succeeded, values, verbose=True, success=lambda x: x, failure=lambda x: x,
67 """Format a message reporting the results of a build"""
69 if not verbose and values['traceback'] is not None:
70 split = values['traceback'].split('\n')
71 # Here, have a hackish heuristic
73 for i in xrange(2, len(split)):
74 truncated = textwrap.fill('\n'.join(split[-i:]))
75 if len(truncated) >= 10:
77 values['traceback'] = truncated
79 for key in ['package', 'version', 'pocket', 'principal', 'inserted_at', 'short_commit']:
80 values[key] = escape(values[key])
83 values['result'] = success(values['result'])
84 msg = """Build of %(package)s %(version)s in %(pocket)s %(result)s.
85 Job submitted by %(principal)s at %(inserted_at)s.
86 Branch %(pocket)s has been advanced to %(short_commit)s.""" % values
88 values['result'] = failure(values['result'])
89 msg = """Build of %(package)s version %(version)s in %(pocket)s %(result)s while %(failed_stage)s.
90 Job submitted by %(principal)s at %(inserted_at)s.
91 Error: %(traceback)s""" % values
94 def submit_completion_msg(succeeded, values, verbose=True, success=lambda x: x, failure=lambda x: x,
98 values['result'] = success(values['result'])
101 values['result'] = failure(values['result'])
102 values['_extra'] = '\nError: %s' % escape(sys.stdin.read())
103 for key in ['commit', 'pocket', 'principal']:
104 values[key] = escape(values[key])
105 msg = """Submission of %(commit)s (%(package)s) to be built in %(pocket)s %(result)s.
106 Build submitted by %(principal)s.%(_extra)s""" % values
109 def repo_creation_msg(succeeded, values, verbose=True, success=lambda x: x, failure=lambda x: x,
111 values = dict(values)
113 for key in ['category', 'name', 'principal']:
114 values[key] = escape(values[key])
115 msg = '%(principal)s just created a new repository, %(category)s/%(name)s.git' % values
118 def prebuild_msg(succeeded, values, verbose=True, success=lambda x: x, failure=lambda x: x,
120 for key in ['build_id', 'pocket', 'package', 'commit', 'principal', 'version', 'inserted_at']:
121 values[key] = escape(values[key])
122 msg = """Build started: %(package)s %(version)s in %(pocket)s.
123 Base commit %(commit)s .
124 Job submitted by %(principal)s at %(inserted_at)s.""" % values
128 POST_BUILD = 'post-build'
129 FAILED_BUILD = 'failed-build'
130 POST_SUBMIT = 'post-submit'
131 FAILED_SUBMIT = 'failed-submit'
132 POST_ADD_REPO = 'post-add-repo'
133 PRE_BUILD = 'pre-build'
135 # Types of communication
140 message_generators = {
141 ZEPHYR : { POST_BUILD : build_completion_msg,
142 FAILED_BUILD : build_completion_msg,
143 POST_SUBMIT : submit_completion_msg,
144 FAILED_SUBMIT : submit_completion_msg,
145 POST_ADD_REPO : repo_creation_msg,
146 PRE_BUILD : prebuild_msg },
147 MAIL : { POST_BUILD : build_completion_msg,
148 FAILED_BUILD : build_completion_msg,
149 POST_SUBMIT : submit_completion_msg,
150 FAILED_SUBMIT : submit_completion_msg,
151 POST_ADD_REPO : repo_creation_msg,
152 PRE_BUILD : prebuild_msg }
155 def zephyr_escape(m):
157 m = re.sub('@', '@@', m)
158 m = re.sub('}', '@(})', m)
161 def zephyr_success(m):
162 return '}@{@color(green)%s}@{' % zephyr_escape(m)
164 def zephyr_failure(m):
165 return '}@{@color(red)%s}@{' % zephyr_escape(m)
168 parser = optparse.OptionParser('Usage: %prog [options] [arguments]')
169 opts, args = parser.parse_args()
170 prog = os.path.basename(sys.argv[0])
173 if prog == POST_BUILD:
174 hook_config = config.build.hooks.post_build
175 elif prog == FAILED_BUILD:
176 hook_config = config.build.hooks.failed_build
177 elif prog == POST_SUBMIT:
178 hook_config = config.build.hooks.post_submit
179 elif prog == FAILED_SUBMIT:
180 hook_config = config.build.hooks.failed_submit
181 elif prog == POST_ADD_REPO:
182 hook_config = config.build.hooks.post_add_repo
183 elif prog == PRE_BUILD:
184 hook_config = config.build.hooks.pre_build
186 parser.error('hook script invoked with unrecognized name %s' % prog)
188 except common.InvirtConfigError:
189 print >>sys.stderr, 'No hook configuration found for %s.' % prog
192 if prog in [POST_BUILD, FAILED_BUILD, PRE_BUILD]:
194 parser.set_usage('Usage: %prog [options] build_id')
198 build = database.Build.query().get(args[0])
199 short_commit = builder.canonicalize_commit(build.package, build.commit, shorten=True)
200 values = { 'build_id' : build.build_id,
201 'commit' : build.commit,
202 'failed_stage' : build.failed_stage,
203 'inserted_at' : build.inserted_at,
204 'package' : build.package,
205 'pocket' : build.pocket,
206 'principal' : build.principal,
207 'short_commit' : short_commit,
208 'traceback' : build.traceback,
209 'version' : build.version,
210 'default_instance' : 'b%(build_id)s',
211 'default_subject' : 'Build %(build_id)d %(result)s: %(package)s %(version)s in %(pocket)s'}
212 if prog == PRE_BUILD:
214 elif build.succeeded:
215 assert prog == POST_BUILD
216 values['result'] = 'succeeded'
219 assert prog == FAILED_BUILD
220 values['result'] = 'failed'
222 elif prog in [POST_SUBMIT, FAILED_SUBMIT]:
224 parser.set_usage('Usage: %prog [options] pocket package commit principal')
227 values = { 'pocket' : args[0],
230 'principal' : args[3],
231 'default_instance' : 'submission',
232 'default_subject' : 'Submission %(result)s: %(package)s %(version)s in %(pocket)s'}
233 if prog == POST_SUBMIT:
234 values['result'] = 'succeeded'
237 values['result'] = 'failed'
239 elif prog in [POST_ADD_REPO]:
241 parser.set_usage('Usage: %prog [options] category name principal')
244 values = { 'category' : args[0],
246 'principal' : args[2],
247 'default_instance' : 'new-repo',
248 'default_subject' : 'New repository %(category)s/%(name)s'}
251 raise AssertionError('Impossible state')
254 zephyr_config = hook_config.zephyr
255 klass = zephyr_config['class'] % values
256 except common.InvirtConfigError:
257 print >>sys.stderr, 'No zephyr configuration specified for %s.' % prog
259 make_msg = message_generators[ZEPHYR][prog]
260 msg = '@{%s}' % make_msg(succeeded, values, verbose=False,
261 success=zephyr_success, failure=zephyr_failure,
262 escape=zephyr_escape)
263 instance = zephyr_config.get('instance', values['default_instance']) % values
264 zsig = zephyr_config.get('zsig', 'XVM Buildbot') % values
265 common.captureOutput(['zwrite', '-c', klass, '-i', instance, '-s',
266 zsig, '-d', '-m', msg],
267 stdout=None, stderr=None)
270 mail_config = hook_config.mail
271 to = mail_config.to % values
272 sender = mail_config['from'] % values
273 except common.InvirtConfigError:
274 print >>sys.stderr, 'No email configuration specified for %s.' % prog
276 make_msg = message_generators[MAIL][prog]
277 msg = make_msg(succeeded, values)
278 email = text.MIMEText(msg)
279 email['To'] = to % values
280 email['From'] = sender % values
281 email['Subject'] = mail_config.get('subject', values['default_subject']) % values
282 common.captureOutput(['sendmail', '-t'], email.as_string(),
283 stdout=None, stderr=None)
285 if __name__ == '__main__':