支持部分继承的Python配置解析器?

时间:2011-06-29 22:45:17

标签: python config ini

我正在寻找一个Python中的ini样式配置解析器,它支持类似于Zend_Config_Ini在PHP中的部分继承。

这样的模块是否存在或者我需要自己动手?

3 个答案:

答案 0 :(得分:14)

Python的ConfigParser可以加载多个文件。稍后阅读的文件可以 覆盖第一个文件中的设置。

例如,我的应用程序在其内部默认值中具有数据库设置 配置文件:

[database]
server = 127.0.0.1
port = 1234
...

我使用" environment.ini"在不同的服务器上覆盖它们。包含的文件 相同的部分,但不同的值:

[database]
server = 192.168.0.12
port = 2345
...

在Python中:

import os
from ConfigParser import ConfigParser
dbconf = ConfigParser()
dbconf.readfp(open('default.ini'))
if os.path.exists('environment.ini'):
    dbconf.readfp(open('environment.ini'))
dbconf.get('database', 'server') # Returns 192.168.0.12

答案 1 :(得分:1)

这是我使用的。 extended_get方法就是您所需要的 - 它支持分层部分。

import re
import io
import ConfigParser

class ZendConfigParser(ConfigParser.ConfigParser):
    def extended_get(self, section, key):
        if self.has_option(section, key):
            return self.get(section, key)
        else:
            orig_section, parent_section = self._get_orig_section(section)
            if orig_section != None:
                if self.has_option(orig_section,key):
                    return self.get(orig_section,key)
                else:
                    return self.extended_get(parent_section,key)
            else:
                return None



    def _get_orig_section(self, zend_section):
        orig_section = None
        parent_section = None
        for section in self.sections():
            if re.search(r'^[ \t]*' + zend_section + '\\b', section) != None:
                orig_section = section
                #look for a parent section
                match = re.match(r'\w+[ \t]*:[ \t]*(\w+)$', section)
                if match != None:
                    parent_section = match.groups()[0]
                break

        return (orig_section, parent_section)

config = ZendConfigParser()
config.read(file)
print(config.extended_get('production', 'database.params.host'))

答案 2 :(得分:0)

我也没有找到任何现成的解决方案。为了解决这个问题,我调整了ConfigParser的get函数来搜索子节,然后在父节中搜索:

config = SafeConfigParser()
config.read(filenames)
required_environment = "mysection"

# determine fallback requirement in case parameter is not found in required environment
fallback_environment = "default"
# loop through all sections of config files
for environment in config.sections():
    # check whether we find an inheritance based on the required section
    if re.search(required_environment + " *: *\w+", environment):
        # found inheritance, take parent as fallback section
        fallback_environment = re.sub(required_environment + r" : (\w+)", r"\1", environment)
        # take this name as requested section
        required_environment = environment

# override get method
_config_parser_get = config.get
def myConfigParserGet(id):
    # check different sections for desired value
    if config.has_option(required_environment, id):
        return _config_parser_get(required_environment, id)
    else:
        return _config_parser_get(fallback_environment, id)

config.get = myConfigParserGet

限制:

  • 仅支持对配置的只读访问
  • 只有一个级别的继承