/usr/share/pyshared/yum/parser.py is in yum 3.2.25-1ubuntu2.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
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 | import re
import urlparse
import urlgrabber
import os.path
import Errors
_KEYCRE = re.compile(r"\$(\w+)")
def varReplace(raw, vars):
'''Perform variable replacement
@param raw: String to perform substitution on.
@param vars: Dictionary of variables to replace. Key is variable name
(without $ prefix). Value is replacement string.
@return: Input raw string with substituted values.
'''
done = [] # Completed chunks to return
while raw:
m = _KEYCRE.search(raw)
if not m:
done.append(raw)
break
# Determine replacement value (if unknown variable then preserve
# original)
varname = m.group(1).lower()
replacement = vars.get(varname, m.group())
start, end = m.span()
done.append(raw[:start]) # Keep stuff leading up to token
done.append(replacement) # Append replacement value
raw = raw[end:] # Continue with remainder of string
return ''.join(done)
class ConfigPreProcessor:
"""
ConfigParser Include Pre-Processor
File-like Object capable of pre-processing include= lines for
a ConfigParser.
The readline function expands lines matching include=(url)
into lines from the url specified. Includes may occur in
included files as well.
Suggested Usage::
cfg = ConfigParser.ConfigParser()
fileobj = confpp( fileorurl )
cfg.readfp(fileobj)
"""
def __init__(self, configfile, vars=None):
# put the vars away in a helpful place
self._vars = vars
# used to track the current ini-section
self._section = None
# set some file-like object attributes for ConfigParser
# these just make confpp look more like a real file object.
self.mode = 'r'
# first make configfile a url even if it points to
# a local file
scheme = urlparse.urlparse(configfile)[0]
if scheme == '':
# check it to make sure it's not a relative file url
if configfile[0] != '/':
configfile = os.getcwd() + '/' + configfile
url = 'file://' + configfile
else:
url = configfile
# these are used to maintain the include stack and check
# for recursive/duplicate includes
self._incstack = []
self._alreadyincluded = []
# _pushfile will return None if he couldn't open the file
fo = self._pushfile( url )
if fo is None:
raise Errors.ConfigError, 'Error accessing file: %s' % url
def readline( self, size=0 ):
"""
Implementation of File-Like Object readline function. This should be
the only function called by ConfigParser according to the python docs.
We maintain a stack of real FLOs and delegate readline calls to the
FLO on top of the stack. When EOF occurs on the topmost FLO, it is
popped off the stack and the next FLO takes over. include= lines
found anywhere cause a new FLO to be opened and pushed onto the top
of the stack. Finally, we return EOF when the bottom-most (configfile
arg to __init__) FLO returns EOF.
Very Technical Pseudo Code::
def confpp.readline() [this is called by ConfigParser]
open configfile, push on stack
while stack has some stuff on it
line = readline from file on top of stack
pop and continue if line is EOF
if line starts with 'include=' then
error if file is recursive or duplicate
otherwise open file, push on stack
continue
else
return line
return EOF
"""
# set line to EOF initially.
line=''
while len(self._incstack) > 0:
# peek at the file like object on top of the stack
fo = self._incstack[-1]
line = fo.readline()
if len(line) > 0:
m = re.match( r'\s*include\s*=\s*(?P<url>.*)', line )
if m:
url = m.group('url')
if len(url) == 0:
raise Errors.ConfigError, \
'Error parsing config %s: include must specify file to include.' % (self.name)
else:
# whooohoo a valid include line.. push it on the stack
fo = self._pushfile( url )
else:
# check if the current line starts a new section
secmatch = re.match( r'\s*\[(?P<section>.*)\]', line )
if secmatch:
self._section = secmatch.group('section')
# line didn't match include=, just return it as is
# for the ConfigParser
break
else:
# the current file returned EOF, pop it off the stack.
self._popfile()
# at this point we have a line from the topmost file on the stack
# or EOF if the stack is empty
if self._vars:
return varReplace(line, self._vars)
return line
def _absurl( self, url ):
"""
Returns an absolute url for the (possibly) relative
url specified. The base url used to resolve the
missing bits of url is the url of the file currently
being included (i.e. the top of the stack).
"""
if len(self._incstack) == 0:
# it's the initial config file. No base url to resolve against.
return url
else:
return urlparse.urljoin( self.geturl(), url )
def _pushfile( self, url ):
"""
Opens the url specified, pushes it on the stack, and
returns a file like object. Returns None if the url
has previously been included.
If the file can not be opened this function exits.
"""
# absolutize this url using the including files url
# as a base url.
absurl = self._absurl(url)
# get the current section to add it to the included
# url's name.
includetuple = (absurl, self._section)
# check if this has previously been included.
if self._isalreadyincluded(includetuple):
return None
try:
fo = urlgrabber.grabber.urlopen(absurl)
except urlgrabber.grabber.URLGrabError, e:
fo = None
if fo is not None:
self.name = absurl
self._incstack.append( fo )
self._alreadyincluded.append(includetuple)
else:
raise Errors.ConfigError, \
'Error accessing file for config %s' % (absurl)
return fo
def _popfile( self ):
"""
Pop a file off the stack signaling completion of including that file.
"""
fo = self._incstack.pop()
fo.close()
if len(self._incstack) > 0:
self.name = self._incstack[-1].geturl()
else:
self.name = None
def _isalreadyincluded( self, tuple ):
"""
Checks if the tuple describes an include that was already done.
This does not necessarily have to be recursive
"""
for etuple in self._alreadyincluded:
if etuple == tuple: return 1
return 0
def geturl(self): return self.name
|