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
|
#!/usr/bin/python
# Simple script that will check which bugs mentioned in NEWS
# are not yet marked Fix Released in Launchpad
import getopt, re, sys
try:
from launchpadlib.launchpad import Launchpad
from lazr.restfulclient import errors
except ImportError:
print "Please install launchpadlib from lp:launchpadlib"
sys.exit(1)
try:
import hydrazine
except ImportError:
print "Please install hydrazine from lp:hydrazine"
sys.exit(1)
options, args = getopt.gnu_getopt(sys.argv, "lw", ["launchpad", 'webbrowser'])
options = dict(options)
if len(args) == 1:
print ("Usage: check-newsbugs [--launchpad][--webbrowser] "
"doc/en/release-notes/bzr-x.y.txt")
print "Options:"
print "--launchpad Print out Launchpad mail commands for closing bugs "
print " that are already fixed."
print "--webbrowser Open launchpad bug pages for bugs that are already "
print " fixed."
sys.exit(1)
def report_notmarked(bug, task, section):
print
print "Bug %d was mentioned in NEWS but is not marked fix released:" % (bug.id, )
print "Launchpad title: %s" % bug.title
print "NEWS summary: "
print section
if "--launchpad" in options or "-l" in options:
print " bug %d" % bug.id
print " affects %s" % task.bug_target_name
print " status fixreleased"
if "--webbrowser" in options or "-w" in options:
import webbrowser
webbrowser.open('http://pad.lv/%s>' % (bug.id,))
def read_news_bugnos(path):
"""Read the bug numbers closed by a particular NEWS file
:param path: Path to the NEWS file
:return: list of bug numbers that were closed.
"""
# Pattern to find bug numbers
bug_pattern = re.compile("\#([0-9]+)")
ret = set()
f = open(path, 'r')
try:
section = ""
for l in f.readlines():
if l.strip() == "":
try:
parenthesed = section.rsplit("(", 1)[1]
except IndexError:
parenthesed = ""
# Empty line, next section begins
for bugno in [int(m) for m in bug_pattern.findall(parenthesed)]:
ret.add((bugno, section))
section = ""
else:
section += l
return ret
finally:
f.close()
def print_bug_url(bugno):
print '<URL:http://pad.lv/%s>' % (bugno,)
launchpad = hydrazine.create_session()
bugnos = read_news_bugnos(args[1])
for bugno, section in bugnos:
try:
bug = launchpad.bugs[bugno]
except errors.HTTPError, e:
if e.response.status == 401:
print_bug_url(bugno)
# Private, we can't access the bug content
print '%s is private and cannot be accessed' % (bugno,)
continue
raise
found_bzr = False
fix_released = False
for task in bug.bug_tasks:
parts = task.bug_target_name.split('/')
if len(parts) == 1:
project = parts[0]
distribution = None
else:
project = parts[0]
distribution = parts[1]
if project == "bzr":
found_bzr = True
if not fix_released and task.status == "Fix Released":
# We could check that the NEWS section and task_status are in
# sync, but that would be overkill. (case at hand: bug #416732)
fix_released = True
if not found_bzr:
print_bug_url(bugno)
print "Bug %d was mentioned in NEWS but is not marked as affecting bzr" % bugno
elif not fix_released:
print_bug_url(bugno)
report_notmarked(bug, task, section)
|