1 # -*- coding: utf-8 -*-
3 from trac.core import *
4 from trac.ticket import ITicketChangeListener
10 quoted_re = re.compile('^(?:> ?\n)*> .+\n(?:>(?: .*)?\n)*', re.MULTILINE)
12 class ZephyrPlugin(Component):
13 implements(ITicketChangeListener)
15 def zwrite(self, id, message):
16 zclass = self.config.get('ZephyrPlugin', 'class')
19 command = shlex.split(self.config.get('ZephyrPlugin', 'command').encode('utf-8'))
21 command = ['zwrite', '-q', '-l', '-d']
22 opcode = self.config.get('ZephyrPlugin', 'opcode')
24 command += ['-O', opcode]
25 p = subprocess.Popen(command +
27 '-i', 'trac-#%s' % id],
28 stdin=subprocess.PIPE)
29 p.stdin.write(message.replace('@', '@@').encode('utf-8', 'replace'))
33 def format_text(self, text):
34 text = re.sub(quoted_re, u'> […]\n', text)
35 lines = textwrap.fill(text).split('\n')
37 lines = lines[:5] + [u'[…]']
38 return '\n'.join(lines)
40 def ticket_created(self, ticket):
42 if ticket['type'] != 'defect':
44 message = "%s filed a new %s %s:\n%s\n\n%s" % (ticket['reporter'],
48 self.format_text(ticket['description']))
49 self.zwrite(ticket.id, message)
51 def ticket_changed(self, ticket, comment, author, old_values):
52 message = "(%s)\n" % ticket['summary']
53 for field in ticket.fields:
55 if name not in old_values:
57 elif field['type'] == 'textarea':
58 message += "%s changed %s to:\n%s\n" % (author, name, self.format_text(ticket[name]))
59 elif ticket[name] and old_values[name]:
60 message += "%s changed %s from %s to %s.\n" % (author, name, old_values[name], ticket[name])
62 message += "%s set %s to %s.\n" % (author, name, ticket[name])
63 elif old_values[name]:
64 message += "%s deleted %s.\n" % (author, name)
66 message += "%s changed %s.\n" % (author, name)
68 message += "%s commented:\n%s\n" % (author, self.format_text(comment))
69 self.zwrite(ticket.id, message)
71 def ticket_deleted(self, ticket):
72 message = "%s deleted ticket %d" % (author, ticket.id)
73 self.zwrite(ticket.id, message)