-
Notifications
You must be signed in to change notification settings - Fork 23
/
generate_updateinfo.py
executable file
·252 lines (224 loc) · 10.1 KB
/
generate_updateinfo.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
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
#!/usr/bin/env python
# Copyright (C) 2013 Kristian K. [http://vmfarms.com] [kris@vmfarms.com]
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see [http://www.gnu.org/licenses/].
import re
import xml.sax.handler
import sys
import os
import errno
import logging
##### START CONFIGURATION HEADER #####
# Set to true if you want to use Sentry
SENTRY = False
# Sentry logging
if SENTRY:
from raven.handlers.logging import SentryHandler
from raven import Client
from raven.conf import setup_logging
sentry_client = Client('INSERT_SENTRY_DSN_HERE')
handler = SentryHandler(sentry_client)
setup_logging(handler)
# What releases would you like to track. 'other' is mandatory
RELEASES = ['6','other']
# Who is this from?
UPDATE_FROM = "you@your_domain.com"
# Directory prefix to build the files under.
BUILD_PREFIX = "/tmp"
##### END CONFIGURATION HEADER #####
def xml2obj(src):
"""
A simple function to converts XML data into native Python object.
"""
non_id_char = re.compile('[^_0-9a-zA-Z]')
def _name_mangle(name):
return non_id_char.sub('_', name)
class DataNode(object):
def __init__(self):
self._attrs = {} # XML attributes and child elements
self.data = None # child text data
def __len__(self):
# treat single element as a list of 1
return 1
def __getitem__(self, key):
if isinstance(key, basestring):
return self._attrs.get(key,None)
else:
return [self][key]
def __contains__(self, name):
return self._attrs.has_key(name)
def __nonzero__(self):
return bool(self._attrs or self.data)
def __getattr__(self, name):
if name.startswith('__'):
# need to do this for Python special methods???
raise AttributeError(name)
return self._attrs.get(name,None)
def _add_xml_attr(self, name, value):
if name in self._attrs:
# multiple attribute of the same name are represented by a list
children = self._attrs[name]
if not isinstance(children, list):
children = [children]
self._attrs[name] = children
children.append(value)
else:
self._attrs[name] = value
def __str__(self):
return self.data or ''
def __repr__(self):
items = sorted(self._attrs.items())
if self.data:
items.append(('data', self.data))
return u'{%s}' % ', '.join([u'%s:%s' % (k,repr(v)) for k,v in items])
class TreeBuilder(xml.sax.handler.ContentHandler):
def __init__(self):
self.stack = []
self.root = DataNode()
self.current = self.root
self.text_parts = []
def startElement(self, name, attrs):
self.stack.append((self.current, self.text_parts))
self.current = DataNode()
self.text_parts = []
# xml attributes --> python attributes
for k, v in attrs.items():
self.current._add_xml_attr(_name_mangle(k), v)
def endElement(self, name):
text = ''.join(self.text_parts).strip()
if text:
self.current.data = text
if self.current._attrs:
obj = self.current
else:
# a text only node is simply represented by the string
obj = text or ''
self.current, self.text_parts = self.stack.pop()
self.current._add_xml_attr(_name_mangle(name), obj)
def characters(self, content):
self.text_parts.append(content)
builder = TreeBuilder()
if isinstance(src,basestring):
xml.sax.parseString(src, builder)
else:
xml.sax.parse(src, builder)
return builder.root._attrs.values()[0]
def build_updateinfo(src):
rel_fd = {}
for rel_num in RELEASES:
try:
os.mkdir("%s/updateinfo-%s" % (BUILD_PREFIX, rel_num))
except OSError, e:
# Directories that exist are fine
if e.errno != errno.EEXIST:
logging.debug("Directory %s/updateinfo-%s already exists." % (BUILD_PREFIX, rel_num))
try:
rel_fd[rel_num] = open("%s/updateinfo-%s/updateinfo.xml" % (BUILD_PREFIX, rel_num), 'w')
except Exception, e:
logging.error("Error opening file: %s" % e)
rel_fd[rel_num].write('<updates>\n')
pkg_parts = re.compile("(?P<name>.*)-(?P<version>.*)-(?P<release>.*)\.(?P<arch>.*).rpm")
pkg_os_rel = re.compile(".*\.el(?P<os_rel>[0-9]*).*")
for i in src._attrs.keys():
# Sometimes it's a dict, sometimes it's a list, where we just take the first element.
if type(src._attrs[i]) is list:
sec_dict = src._attrs[i][0]
else:
sec_dict = src._attrs[i]
# Ignore this entry
if "meta" == i:
continue
# Is this a properly formatted CEBA/CESA entry?
if 'type' not in sec_dict:
logging.warning("Improperly formatted CEBA/CESA entry: %s" % (i))
continue
# Is this a security advisory?
# Sometimes 'types'field merges multiple purposes e.g. "security and bugfix.."
#to make sure everything gets detected "find" is used.
if (( sec_dict._attrs['type'].find('Security') == -1 )
and( sec_dict._attrs['type'].find('Bug Fix') == -1 ) ):
continue
# More than one OS release? Generate multiple entries
if sec_dict.os_release == None:
sec_dict.os_release = ""
if isinstance(sec_dict.os_release, basestring):
releases = [sec_dict.os_release]
else:
releases = sec_dict.os_release
for release in releases:
# If we can't pull the release, then we infer it from the package names
p_release = release
packages = []
for pkg in sec_dict.packages:
package = None
# Parse the package name
try:
pkg_match = pkg_parts.match(pkg)
package = pkg_match.groupdict()
packages.append(package)
except Exception, err:
logging.warning("Package name '%s' couldn't be matched against regex" % (pkg))
continue
# Extract the el release from here, otherwise it has no discernable release
if not p_release:
if ".el" in package['release']:
try:
p_rel_match = pkg_os_rel.match(package['release'])
p_release = p_rel_match.groupdict()['os_rel']
except Exception, err:
logging.warning("Package release '%s' couldn't be matched against regex" % (package['release']))
continue
# Place unidentifiable or uninteresting releases in an alternate updateinfo.xml
if p_release not in RELEASES:
p_release = "other"
TYP = "none"
if sec_dict._attrs['type'].find( "Security") != -1:
TYP = "security"
elif sec_dict._attrs['type'].find("Bug Fix") != -1:
TYP = "bugfix"
else:
print sec_dict._attrs['type']
rel_fd[p_release].write(' <update from="%s" status="stable" type="%s" version="1.4">\n' % ( UPDATE_FROM, TYP) )
rel_fd[p_release].write(" <id>%s</id>\n" % i)
rel_fd[p_release].write(" <title>%s</title>\n" % sec_dict._attrs['synopsis'])
rel_fd[p_release].write(" <release>CentOS %s</release>\n" % p_release)
rel_fd[p_release].write(" <issued date=\"%s\" />\n" % sec_dict._attrs['issue_date'])
rel_fd[p_release].write(" <references>\n")
for ref in sec_dict._attrs['references'].split():
rel_fd[p_release].write(" <reference href=\"%s\"/>\n" % ref)
rel_fd[p_release].write(" </references>\n")
rel_fd[p_release].write(" <description>%s</description>\n" % sec_dict._attrs['synopsis'])
rel_fd[p_release].write(" <pkglist>\n")
rel_fd[p_release].write(" <collection short=\"EL-%s\">\n" % p_release)
rel_fd[p_release].write(" <name>CentOS %s</name>\n" % p_release)
for pkg in packages:
rel_fd[p_release].write(" <package arch=\"%s\" epoch=\"%s\" name=\"%s\" release=\"%s\" src=\"%s\" version=\"%s\">\n" % (pkg['arch'], "0", pkg['name'], pkg['release'], "", pkg['version']))
rel_fd[p_release].write(" <filename>%s</filename>\n" % (pkg))
rel_fd[p_release].write(" </package>\n")
rel_fd[p_release].write(" </collection>\n")
rel_fd[p_release].write(" </pkglist>\n")
rel_fd[p_release].write(" </update>\n")
for rel_num in RELEASES:
rel_fd[rel_num].write("</updates>\n")
rel_fd[rel_num].close()
if __name__ == "__main__":
try:
if len(sys.argv) < 2:
sys.exit("Usage: %s <path to errata.xml>\n")
errata_file = open(sys.argv[1], 'r')
errata_xml = errata_file.read()
errata = xml2obj(errata_xml)
build_updateinfo(errata)
except Exception, e:
logging.critical("Caught exception: %s" % e, exc_info=True)