config.py
183 lines
| 6.3 KiB
| text/x-python
|
PythonLexer
/ mercurial / config.py
Martin Geisler
|
r8229 | # config.py - configuration parsing for Mercurial | ||
# | ||||
# Copyright 2009 Matt Mackall <mpm@selenic.com> and others | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Martin Geisler
|
r8229 | |||
Matt Mackall
|
r8144 | from i18n import _ | ||
Chad Dombrova
|
r11224 | import error, util | ||
Matt Mackall
|
r14486 | import re, os, errno | ||
Matt Mackall
|
r8144 | |||
class sortdict(dict): | ||||
Matt Mackall
|
r8184 | 'a simple sorted dictionary' | ||
Matt Mackall
|
r8144 | def __init__(self, data=None): | ||
self._list = [] | ||||
if data: | ||||
self.update(data) | ||||
def copy(self): | ||||
return sortdict(self) | ||||
def __setitem__(self, key, val): | ||||
if key in self: | ||||
self._list.remove(key) | ||||
self._list.append(key) | ||||
dict.__setitem__(self, key, val) | ||||
def __iter__(self): | ||||
return self._list.__iter__() | ||||
def update(self, src): | ||||
for k in src: | ||||
self[k] = src[k] | ||||
Sune Foldager
|
r14696 | def clear(self): | ||
dict.clear(self) | ||||
self._list = [] | ||||
Matt Mackall
|
r8144 | def items(self): | ||
Dirkjan Ochtman
|
r8222 | return [(k, self[k]) for k in self._list] | ||
Matt Mackall
|
r8184 | def __delitem__(self, key): | ||
dict.__delitem__(self, key) | ||||
self._list.remove(key) | ||||
Angel Ezquerra
|
r16865 | def keys(self): | ||
return self._list | ||||
def iterkeys(self): | ||||
return self._list.__iter__() | ||||
Matt Mackall
|
r8144 | |||
Matt Mackall
|
r8186 | class config(object): | ||
Matt Mackall
|
r8144 | def __init__(self, data=None): | ||
self._data = {} | ||||
Matt Mackall
|
r8185 | self._source = {} | ||
Matt Mackall
|
r8144 | if data: | ||
for k in data._data: | ||||
self._data[k] = data[k].copy() | ||||
Matt Mackall
|
r8185 | self._source = data._source.copy() | ||
Matt Mackall
|
r8144 | def copy(self): | ||
return config(self) | ||||
def __contains__(self, section): | ||||
return section in self._data | ||||
Matt Mackall
|
r8186 | def __getitem__(self, section): | ||
return self._data.get(section, {}) | ||||
def __iter__(self): | ||||
for d in self.sections(): | ||||
yield d | ||||
Matt Mackall
|
r8193 | def update(self, src): | ||
for s in src: | ||||
Matt Mackall
|
r8144 | if s not in self: | ||
self._data[s] = sortdict() | ||||
Matt Mackall
|
r8193 | self._data[s].update(src._data[s]) | ||
self._source.update(src._source) | ||||
Matt Mackall
|
r8144 | def get(self, section, item, default=None): | ||
Matt Mackall
|
r8185 | return self._data.get(section, {}).get(item, default) | ||
Pierre-Yves David
|
r15919 | |||
def backup(self, section, item): | ||||
"""return a tuple allowing restore to reinstall a previous valuesi | ||||
The main reason we need it is because it handle the "no data" case. | ||||
""" | ||||
try: | ||||
value = self._data[section][item] | ||||
source = self.source(section, item) | ||||
return (section, item, value, source) | ||||
except KeyError: | ||||
return (section, item) | ||||
Matt Mackall
|
r8198 | def source(self, section, item): | ||
Matt Mackall
|
r8185 | return self._source.get((section, item), "") | ||
Matt Mackall
|
r8144 | def sections(self): | ||
return sorted(self._data.keys()) | ||||
def items(self, section): | ||||
Matt Mackall
|
r8185 | return self._data.get(section, {}).items() | ||
Matt Mackall
|
r8144 | def set(self, section, item, value, source=""): | ||
if section not in self: | ||||
self._data[section] = sortdict() | ||||
Matt Mackall
|
r8185 | self._data[section][item] = value | ||
self._source[(section, item)] = source | ||||
Matt Mackall
|
r8144 | |||
Pierre-Yves David
|
r15919 | def restore(self, data): | ||
"""restore data returned by self.backup""" | ||||
if len(data) == 4: | ||||
# restore old data | ||||
section, item, value, source = data | ||||
self._data[section][item] = value | ||||
self._source[(section, item)] = source | ||||
else: | ||||
# no data before, remove everything | ||||
section, item = data | ||||
if section in self._data: | ||||
del self._data[section][item] | ||||
self._source.pop((section, item), None) | ||||
Matt Mackall
|
r8265 | def parse(self, src, data, sections=None, remap=None, include=None): | ||
Matt Mackall
|
r8144 | sectionre = re.compile(r'\[([^\[]+)\]') | ||
Matt Mackall
|
r8263 | itemre = re.compile(r'([^=\s][^=]*?)\s*=\s*(.*\S|)') | ||
Matt Mackall
|
r10295 | contre = re.compile(r'\s+(\S|\S.*\S)\s*$') | ||
Matt Mackall
|
r14659 | emptyre = re.compile(r'(;|#|\s*$)') | ||
Matt Mackall
|
r14642 | commentre = re.compile(r'(;|#)') | ||
Matt Mackall
|
r8192 | unsetre = re.compile(r'%unset\s+(\S+)') | ||
Matt Mackall
|
r10295 | includere = re.compile(r'%include\s+(\S|\S.*\S)\s*$') | ||
Matt Mackall
|
r8144 | section = "" | ||
item = None | ||||
line = 0 | ||||
r9339 | cont = False | |||
Matt Mackall
|
r8180 | |||
Nicolas Dumazet
|
r9136 | for l in data.splitlines(True): | ||
Matt Mackall
|
r8144 | line += 1 | ||
Matt Mackall
|
r16348 | if line == 1 and l.startswith('\xef\xbb\xbf'): | ||
# Someone set us up the BOM | ||||
l = l[3:] | ||||
Matt Mackall
|
r8144 | if cont: | ||
Matt Mackall
|
r14642 | if commentre.match(l): | ||
continue | ||||
Matt Mackall
|
r8144 | m = contre.match(l) | ||
if m: | ||||
Matt Mackall
|
r8193 | if sections and section not in sections: | ||
continue | ||||
Matt Mackall
|
r8144 | v = self.get(section, item) + "\n" + m.group(1) | ||
Matt Mackall
|
r8265 | self.set(section, item, v, "%s:%d" % (src, line)) | ||
Matt Mackall
|
r8144 | continue | ||
item = None | ||||
Nicolas Dumazet
|
r9469 | cont = False | ||
Matt Mackall
|
r8183 | m = includere.match(l) | ||
if m: | ||||
Chad Dombrova
|
r11224 | inc = util.expandpath(m.group(1)) | ||
Matt Mackall
|
r8265 | base = os.path.dirname(src) | ||
Matt Mackall
|
r8183 | inc = os.path.normpath(os.path.join(base, inc)) | ||
Matt Mackall
|
r8265 | if include: | ||
Martin Geisler
|
r10042 | try: | ||
include(inc, remap=remap, sections=sections) | ||||
except IOError, inst: | ||||
Matt Mackall
|
r14486 | if inst.errno != errno.ENOENT: | ||
raise error.ParseError(_("cannot include %s (%s)") | ||||
% (inc, inst.strerror), | ||||
"%s:%s" % (src, line)) | ||||
Matt Mackall
|
r8183 | continue | ||
Matt Mackall
|
r8144 | if emptyre.match(l): | ||
continue | ||||
m = sectionre.match(l) | ||||
if m: | ||||
section = m.group(1) | ||||
Matt Mackall
|
r8298 | if remap: | ||
section = remap.get(section, section) | ||||
Matt Mackall
|
r8144 | if section not in self: | ||
self._data[section] = sortdict() | ||||
continue | ||||
m = itemre.match(l) | ||||
if m: | ||||
item = m.group(1) | ||||
r9339 | cont = True | |||
Matt Mackall
|
r8193 | if sections and section not in sections: | ||
continue | ||||
Matt Mackall
|
r8265 | self.set(section, item, m.group(2), "%s:%d" % (src, line)) | ||
Matt Mackall
|
r8144 | continue | ||
Matt Mackall
|
r8184 | m = unsetre.match(l) | ||
if m: | ||||
name = m.group(1) | ||||
Matt Mackall
|
r8193 | if sections and section not in sections: | ||
continue | ||||
Martin Geisler
|
r13031 | if self.get(section, name) is not None: | ||
Matt Mackall
|
r8184 | del self._data[section][name] | ||
continue | ||||
Matt Mackall
|
r11288 | raise error.ParseError(l.rstrip(), ("%s:%s" % (src, line))) | ||
Matt Mackall
|
r8265 | |||
def read(self, path, fp=None, sections=None, remap=None): | ||||
if not fp: | ||||
Adrian Buehlmann
|
r13664 | fp = util.posixfile(path) | ||
Matt Mackall
|
r8265 | self.parse(path, fp.read(), sections, remap, self.read) | ||