PHP前端开发

迈向轻松的 Python 配置文件版本 3

百变鹏仔 3小时前 #Python
文章标签 配置文件

介绍

这是本系列的最后一篇文章。 此实现旨在修复我在上一篇文章中描述的样板代码的主要缺点。 我将此实现称为动态属性类。

班级代表

以下类图显示了 dynamicconfiguration 可重用类以及开发人员使用此功能所需的支持数据结构。 它仍然提供版本 2 的所有基本功能,包括自动启动捆绑、创建缺失部分和键值。

开发者代码说明

我将展示寻求使用此类的应用程序的完整源代码。 我正在使用我们在前 3 篇文章中讨论过的属性。

from codeallybasic.dynamicconfiguration import dynamicconfigurationfrom codeallybasic.dynamicconfiguration import keynamefrom codeallybasic.dynamicconfiguration import sectionnamefrom codeallybasic.dynamicconfiguration import sectionsfrom codeallybasic.dynamicconfiguration import valuedescriptionfrom codeallybasic.dynamicconfiguration import valuedescriptionsfrom codeallybasic.secureconversions import secureconversionsfrom codeallybasic.singletonv3 import singletonv3from bytesizedpython.impostorenumbyname import impostorenumbynamefrom bytesizedpython.phoneyenumbyvalue import phoneyenumbyvaluelogger_name:       str = 'tutorial'base_file_name: str = 'config.ini'module_name:       str = 'version3properties'default_phoney_enum_by_value:  phoneyenumbyvalue  = phoneyenumbyvalue.fakebrendadefault_impostor_enum_by_name: impostorenumbyname = impostorenumbyname.highgeneral_properties: valuedescriptions = valuedescriptions(    {        keyname('debug'):    valuedescription(defaultvalue='false', deserializer=secureconversions.secureboolean),        keyname('loglevel'): valuedescription(defaultvalue='info'),        keyname('phoneyenumbyvalue'):  valuedescription(defaultvalue=default_phoney_enum_by_value.value,  enumusevalue=true),        keyname('impostorenumbyname'): valuedescription(defaultvalue=default_impostor_enum_by_name.name,  enumusename=true),    })database_properties: valuedescriptions = valuedescriptions(    {        keyname('dbname'): valuedescription(defaultvalue='dbname'),        keyname('dbhost'): valuedescription(defaultvalue='localhost'),        keyname('dbport'): valuedescription(defaultvalue='5342', deserializer=secureconversions.secureinteger),    })configuration_sections: sections = sections(    {        sectionname('general'):  general_properties,        sectionname('database'): database_properties,    })class configurationpropertiesversion3(dynamicconfiguration, metaclass=singletonv3):    def __init__(self):        self._logger: logger = getlogger(logger_name)        super().__init__(basefilename=base_file_name, modulename=module_name, sections=configuration_sections)

第 45-50 行是您必须编写的代码。 本质上,您只需确保传递文件名、模块名称和配置部分。 这个sections类型来自dynamicconfiguration模块。

第 21-28 行和第 30-36 行是 valuedescriptions 字典。 keyname 是属性并指向 valuedescription。 请注意,有关如何持久化枚举的指示符已从先前实现的装饰器移至 valuedescription 中的布尔属性。

立即学习“Python免费学习笔记(深入)”;

实现代码说明

如果仔细查看 dynamicconfiguration 的类图,您会发现它实现了两个 python magic 方法。 它们是 __getattr__(self, name)__ 和 __setattr__(self, name, value)__ 方法。

以下是 __getattr__ 的代码。 这看起来非常像我们在版本 2 中使用的装饰器。关键工作发生在第 14 行对受保护方法 _lookupkey() 的调用上。 它返回属性的完整描述,以便我们可以模拟属性检索。

    def __getattr__(self, attrname: str) -> any:        """        does the work of retrieving the named attribute from the configuration parser        args:            attrname:        returns:  the correctly typed value        """        self._logger.info(f'{attrname}')        configparser: configparser     = self._configparser        result:       lookupresult     = self._lookupkey(searchkeyname=keyname(attrname))        valuedescription: valuedescription = result.keydescription        valuestr: str = configparser.get(result.sectionname, attrname)        if valuedescription.deserializer is not none:            value: any = valuedescription.deserializer(valuestr)        else:            value = valuestr        return value

以下是 __setattr__() 的实现。 请注意第 22-27 行中对枚举的支持以及第 30 行中的 直写 功能。

    def __setattr__(self, key: str, value: any):        """        do the work of writing this back to the configuration/settings/preferences file        ignores protected and private variables uses by this class        does a "write through" to the backing configuration file (.ini)        args:            key:    the property name            value:  its new value        """        if key.startswith(protected_property_indicator) or key.startswith(private_property_indicator):            super(dynamicconfiguration, self).__setattr__(key, value)        else:            self._logger.debug(f'writing `{key}` with `{value}` to configuration file')            configparser: configparser  = self._configparser            result:       lookupresult  = self._lookupkey(searchkeyname=keyname(key))            valuedescription: valuedescription = result.keydescription            if valuedescription.enumusevalue is true:                valuestr: str = value.value                configparser.set(result.sectionname, key, valuestr)            elif valuedescription.enumusename is true:                configparser.set(result.sectionname, key, value.name)            else:                configparser.set(result.sectionname, key, str(value))            self.saveconfiguration()

访问和修改属性

访问和修改属性与版本 2 完全相同。

    basicconfig(level=info)    config: configurationpropertiesversion2 = configurationpropertiesversion2()    logger: logger = getlogger(logger_name)    logger.info(f'{config.debug=}')    logger.info(f'{config.loglevel=}')    logger.info(f'{config.phoneyenumbyvalue=}')    logger.info(f'{config.impostorenumbyname=}')    logger.info('database properties follow')    logger.info(f'{config.dbname=}')    logger.info(f'{config.dbhost=}')    logger.info(f'{config.dbport=}')    logger.info('mutate enumeration properties')    config.phoneyenumbyvalue = phoneyenumbyvalue.thewanderer    logger.info(f'{config.phoneyenumbyvalue=}')    config.impostorenumbyname = impostorenumbyname.low    logger.info(f'{config.impostorenumbyname=}')

上面的代码片段产生以下输出。

INFO:Tutorial:config.debug='False'INFO:Tutorial:config.logLevel='Info'INFO:Tutorial:config.phoneyEnumByValue=<PhoneyEnumByValue.FakeBrenda: 'Faker Extraordinaire'>INFO:Tutorial:config.impostorEnumByName='High'INFO:Tutorial:Database Properties FollowINFO:Tutorial:config.dbName='example_db'INFO:Tutorial:config.dbHost='localhost'INFO:Tutorial:config.dbPort=5432INFO:Tutorial:Mutate Enumeration PropertiesINFO:Tutorial:config.phoneyEnumByValue=<PhoneyEnumByValue.TheWanderer: 'The Wanderer'>INFO:Tutorial:config.impostorEnumByName='Low'

结论

本文的源代码在这里。 请参阅支持类 singletonv3。 查看

的实现

优点

缺点