This repository has been archived by the owner on Mar 16, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathhooks.py
233 lines (180 loc) · 7.98 KB
/
hooks.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
"""
hooks - git hooks provided by gitzilla.
"""
import re
import sys
from .bugwrap import BugzillaWrapper
from .utils import get_changes, notify_and_exit
from gitzilla import sDefaultSeparator, sDefaultFormatSpec, oDefaultBugRegex, sDefaultRefPrefix
from gitzilla import NullLogger
import traceback
def post_receive(sBZUrl, sBZUser=None, sBZPasswd=None, sFormatSpec=None, oBugRegex=None, sSeparator=None, logger=None, bz_wrap=None, sRefPrefix=None, bIncludeDiffStat=True, aasPushes=None):
"""
a post-recieve hook handler which extracts bug ids and adds the commit
info to the comment. If multiple bug ids are found, the comment is added
to each of those bugs.
sBZUrl is the base URL for the Bugzilla installation. sBZUser and
sBZPassword should be the login and password of a Bugzilla user
allowed to check a bug's status and comment on them. If they are
None, what happens is defined by the bz_wrap function.
oBugRegex specifies the regex used to search for the bug id in the commit
messages. It MUST provide a named group called 'bug' which contains the bug
id (all digits only). If oBugRegex is None, a default bug regex is used,
which is:
r"bug\s*(?:#|)\s*(?P<bug>\d+)"
This matches forms such as:
- bug 123
- bug #123
- BUG # 123
- Bug#123
- bug123
The format spec is appended to "--pretty=format:" and passed to
"git whatchanged". See the git whatchanged manpage for more info on the
format spec. Newlines are automatically converted to the "--pretty"
equivalent, which is '%n'.
If sFormatSpec is None, a default format spec is used.
The separator is a string that would never occur in a commit message.
If sSeparator is None, a default separator is used, which should be
good enough for everyone.
If a logger is provided, it would be used for all the logging. If logger
is None, logging will be disabled. The logger must be a Python
logging.Logger instance.
The function bz_wrap(url, username, password) is invoked to
instantiate the bugwrap.BugzillaWrapper instance. If this is None,
the default method is used.
sRefPrefix is the string prefix of the git reference. If a git reference
does not start with this, its commits will be ignored. 'refs/heads/' by default.
aasPushes is a list of (sOldRev, sNewRev, sRefName) tuples, for when these
aren't read from stdin (gerrit integration).
"""
if sFormatSpec is None:
sFormatSpec = sDefaultFormatSpec
if sSeparator is None:
sSeparator = sDefaultSeparator
if oBugRegex is None:
oBugRegex = oDefaultBugRegex
if logger is None:
logger = NullLogger
if bz_wrap is None:
bz_wrap = BugzillaWrapper
if sRefPrefix is None:
sRefPrefix = sDefaultRefPrefix
oBZ = bz_wrap(sBZUrl, sBZUser, sBZPasswd)
def gPushes():
for sLine in iter(sys.stdin.readline, ""):
yield sLine.strip().split(" ")
if not aasPushes:
aasPushes = gPushes()
sPrevRev = None
for asPush in aasPushes:
(sOldRev, sNewRev, sRefName) = asPush
if not sRefName.startswith(sRefPrefix):
logger.debug("ignoring ref: '%s'" % (sRefName,))
continue
if sPrevRev is None:
sPrevRev = sOldRev
logger.debug("oldrev: '%s', newrev: '%s'" % (sOldRev, sNewRev))
asChangeLogs = get_changes(sOldRev, sNewRev, sFormatSpec, sSeparator, bIncludeDiffStat, sRefName, sRefPrefix)
for sMessage in asChangeLogs:
logger.debug("Considering commit:\n%s" % (sMessage,))
oMatch = re.search(oBugRegex, sMessage)
if oMatch is None:
logger.info("Bug id not found in commit:\n%s" % (sMessage,))
continue
for oMatch in re.finditer(oBugRegex, sMessage):
iBugId = int(oMatch.group("bug"))
logger.debug("Found bugid %d" % (iBugId,))
try:
oBZ.add_bug_comment(iBugId, sMessage)
except Exception as e:
logger.exception("Could not add comment to bug %d" % (iBugId,))
def update(oBugRegex=None, asAllowedStatuses=None, sSeparator=None, sBZUrl=None, sBZUser=None, sBZPasswd=None, logger=None, bz_wrap=None, sRefPrefix=None, bRequireBugNumber=True):
"""
an update hook handler which rejects commits without a bug reference.
This looks at the sys.argv array, so make sure you don't modify it before
calling this function.
oBugRegex specifies the regex used to search for the bug id in the commit
messages. It MUST provide a named group called 'bug' which contains the bug
id (all digits only). If oBugRegex is None, a default bug regex is used,
which is:
r"bug\s*(?:#|)\s*(?P<bug>\d+)"
This matches forms such as:
- bug 123
- bug #123
- BUG # 123
- Bug#123
- bug123
asAllowedStatuses is an array containing allowed statuses for the found
bugs. If a bug is not in one of these states, the commit will be rejected.
If asAllowedStatuses is None, status checking is diabled.
The separator is a string that would never occur in a commit message.
If sSeparator is None, a default separator is used, which should be
good enough for everyone.
sBZUrl specifies the base URL for the Bugzilla installation. sBZUser and
sBZPasswd are the bugzilla credentials.
If a logger is provided, it would be used for all the logging. If logger
is None, logging will be disabled. The logger must be a Python
logging.Logger instance.
The function bz_wrap(url, username, password) is invoked to
instantiate the bugwrap.BugzillaWrapper instance. If this is None,
the default method is used.
sRefPrefix is the string prefix of the git reference. If a git reference
does not start with this, its commits will be ignored. 'refs/heads/' by default.
bRequireBugNumber, if True, requires that a bug number appears in the
commit message (otherwise it will be rejected).
"""
if oBugRegex is None:
oBugRegex = oDefaultBugRegex
if sSeparator is None:
sSeparator = sDefaultSeparator
if logger is None:
logger = NullLogger
if bz_wrap is None:
bz_wrap = BugzillaWrapper
if sRefPrefix is None:
sRefPrefix = sDefaultRefPrefix
sFormatSpec = sDefaultFormatSpec
if asAllowedStatuses is not None:
# sanity checking
if sBZUrl is None:
raise ValueError("Bugzilla info required for status checks")
# create and cache bugzilla instance
oBZ = bz_wrap(sBZUrl, sBZUser, sBZPasswd)
# check auth
try:
oBZ.auth()
except:
logger.error("Could not login to Bugzilla", exc_info=1)
notify_and_exit("Could not login to Bugzilla. Check your auth details and settings")
(sRefName, sOldRev, sNewRev) = sys.argv[1:4]
if not sRefName.startswith(sRefPrefix):
logger.debug("ignoring ref: '%s'" % (sRefName,))
return
logger.debug("oldrev: '%s', newrev: '%s'" % (sOldRev, sNewRev))
asChangeLogs = get_changes(sOldRev, sNewRev, sFormatSpec, sSeparator, False, sRefName, sRefPrefix)
for sMessage in asChangeLogs:
logger.debug("Checking for bug refs in commit:\n%s" % (sMessage,))
oMatch = re.search(oBugRegex, sMessage)
if oMatch is None:
if bRequireBugNumber:
logger.error("No bug ref found in commit:\n%s" % (sMessage,))
notify_and_exit("No bug ref found in commit:\n%s" % (sMessage,))
else:
logger.debug("No bug ref found, but none required.")
else:
if asAllowedStatuses is not None:
# check all bug statuses
for oMatch in re.finditer(oBugRegex, sMessage):
iBugId = int(oMatch.group("bug"))
logger.debug("Found bug id %d" % (iBugId,))
try:
sStatus = oBZ.bug_status(iBugId)
if sStatus is None:
notify_and_exit("Bug %d does not exist" % (iBugId,))
except Exception as e:
logger.exception("Could not get status for bug %d" % (iBugId,))
notify_and_exit("Could not get staus for bug %d" % (iBugId,))
logger.debug("status for bug %d is %s" % (iBugId, sStatus))
if sStatus not in asAllowedStatuses:
logger.info("Cannot accept commit for bug %d in state %s" % (iBugId, sStatus))
notify_and_exit("Bug %d['%s'] is not in %s" % (iBugId, sStatus, asAllowedStatuses))