diff -r 36e05c0da8f7 -r f55ca49f7f44 hg_hooks/bugzilla/versiontobugzilla.py --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/hg_hooks/bugzilla/versiontobugzilla.py Thu Jul 23 12:50:28 2009 +0100 @@ -0,0 +1,136 @@ +'''Bugzilla integration for adding versions from tags + +The hook updates the Bugzilla database directly. Only Bugzilla installations +using MySQL are supported. + +This hook uses the same .hgrc parameters as the default Bugzilla hook. There +is no need for configuring the same stuff twice. (connection, etc.) + +Configuring the extension: (same as Bugzilla -hook) + + [bugzilla] + host Hostname of the MySQL server holding the Bugzilla database. + db Name of the Bugzilla database in MySQL. Default 'bugs'. + user Username to use to access MySQL server. Default 'bugs'. + password Password to use to access MySQL server. + timeout Database connection timeout (seconds). Default 5. + +Additional elements under Bugzilla -section: (new items) + [bugzilla] + product The name on the Bugzilla product that is used for adding + the new versions. + +Activating the extension: + + [extensions] + hgext.versiontobugzilla = + + [hooks] + incoming.versiontobugzilla = python:hgext.versiontobugzilla.hook + +Example configuration in hgrc: + [bugzilla] + host = localhost + user = bugs + password = password + product = my_product + + [extensions] + hgext.versiontobugzilla = + + [hooks] + incoming.versiontobugzilla = python:hgext.versiontobugzilla.hook +''' + +from mercurial import util +import re + +MySQLdb = None + +class BugzillaClient: + + def __init__(self, ui, repo, node): + self.tag = None + self.ui = ui + self.repo = repo + self.node = node + self.product = ui.config('bugzilla', 'product') + self.host = ui.config('bugzilla', 'host', 'localhost') + self.user = ui.config('bugzilla', 'user', 'bugs') + self.passwd = ui.config('bugzilla', 'password') + self.db = ui.config('bugzilla', 'db', 'bugs') + self.timeout = int(ui.config('bugzilla', 'timeout', 10)) + self.connection = MySQLdb.connect(host=self.host, user=self.user, passwd=self.passwd, + db=self.db, connect_timeout=self.timeout) + self.cursor = self.connection.cursor() + + def printMessageInVerboseMode(self, message): + '''Prints a message to console if hg has been executed with -v option.''' + self.ui.note(message) + + def executeDatabaseQuery(self, *args, **kwargs): + self.printMessageInVerboseMode('Bugzilla: query: %s %s\n' % (args, kwargs)) + try: + self.cursor.execute(*args, **kwargs) + except MySQLdb.MySQLError: + self.printMessageInVerboseMode('Bugzilla: failed query: %s %s\n' % (args, kwargs)) + raise + + def commitContainsTag(self): + self.parseTagFromCommitMessage() + if self.tag: + return True + else: + return False + + def parseTagFromCommitMessage(self): + ctx = self.repo[self.node] + version_re = re.compile(('Added tag (.+) for changeset [0-9a-h]+'), re.IGNORECASE) + m = version_re.search(ctx.description()) + if m: + self.tag = m.group(1) + + def insertTagIntoDatabase(self): + self.makeSureThatProductExists() + if not self.doesVersionAlreadyExist(): + self.printMessageInVerboseMode("Bugzilla: adding version '%s' to product '%s' in database.\n" % (self.tag, self.product)) + self.insertNewVersionIntoDatabase() + else: + self.printMessageInVerboseMode("Bugzilla: product '%s' already has a version '%s' in database. Not trying to add it again." % (self.product, self.tag)) + + def makeSureThatProductExists(self): + self.executeDatabaseQuery('select id from products where name = %s', (self.product,)) + ids = self.cursor.fetchall() + if len(ids) != 1: + raise util.Abort("Product '%s' does not exist in database, please check the [bugzilla] -section in hgrc." % self.product) + + def doesVersionAlreadyExist(self): + self.executeDatabaseQuery('select * from versions where value = %s and product_id in (select id from products where name=%s )', (self.tag, self.product)) + ids = self.cursor.fetchall() + if len(ids) == 1: + return True + else: + return False + + def insertNewVersionIntoDatabase(self): + self.executeDatabaseQuery('insert into versions (value, product_id) values (%s, (select id from products where name=%s ))', (self.tag, self.product)) + self.connection.commit() + +def hook(ui, repo, hooktype, node=None, **kwargs): + + try: + import MySQLdb as mysql + global MySQLdb + MySQLdb = mysql + except ImportError, err: + raise util.Abort('MySQL driver not installed: %s' % err) + + if node is None: + raise util.Abort('Only hooks that have changesetid''s can be used.') + + try: + bzClient = BugzillaClient(ui, repo, node) + if bzClient.commitContainsTag(): + bzClient.insertTagIntoDatabase() + except MySQLdb.MySQLError, err: + raise util.Abort('Database error: %s' % err[1])