config.py
144 lines
| 5.0 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 _ | ||
Simon Heimberg
|
r8312 | import error | ||
import re, os | ||||
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] | ||||
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) | ||||
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) | ||
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 | |||
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
|
r8192 | contre = re.compile(r'\s+(\S.*\S)') | ||
Matt Mackall
|
r8144 | emptyre = re.compile(r'(;|#|\s*$)') | ||
Matt Mackall
|
r8192 | unsetre = re.compile(r'%unset\s+(\S+)') | ||
includere = re.compile(r'%include\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 | ||
if cont: | ||||
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: | ||||
inc = 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: | ||||
msg = _("config error at %s:%d: " | ||||
"cannot include %s (%s)") \ | ||||
% (src, line, inc, inst.strerror) | ||||
raise error.ConfigError(msg) | ||||
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 | ||||
Matt Mackall
|
r8184 | if self.get(section, name) != None: | ||
del self._data[section][name] | ||||
continue | ||||
r9339 | raise error.ConfigError(_("config error at %s:%d: '%s'") | |||
Matt Mackall
|
r8265 | % (src, line, l.rstrip())) | ||
def read(self, path, fp=None, sections=None, remap=None): | ||||
if not fp: | ||||
fp = open(path) | ||||
self.parse(path, fp.read(), sections, remap, self.read) | ||||