/configparser_extended

A python configparser extension which implements a few addons like distinct key values depending on the "configuration" of an application (DEV, TESTS, PRODUCTION), group inheritance...

Primary LanguagePythonMIT LicenseMIT

configparser_extended

WARNING: this project is replaced by opinionated_configparser

What is it ?

A python configparser extension which implements a few addons like:

  • distinct option values depending on the "configuration" of an application (DEV, TESTS, PRODUCTION...)
  • section inheritance

This extension is supposed to implement these new functionnalities without changing the base behavior of the original ConfigParser. You should be able to switch from the original ConfigParser to ConfigParserExtended without changing your code too much.

As this project is open source, feel free to add new functionnalities that could help the community and report any bug that you encounter.

Option specification : values depending on the configuration's name

Basics

This functionnality, as the name suggests it, allows you to define option values for a specific configuration name :

[section1]
option1=foo
option1[server]=bar

In this case, the value of option1 will be foo for any configuration name except for the server configuration : the value of option1 will be bar for the server configuration only.

The configuration name can be set in 2 ways : either in the constructor with config_name= or via the set_config_name() method.

Inheritance

Inheritance is also available for configuration names as well. This allows you to get specific values without having to define an option exclusively for that :

[section1]
option1=val1
option1[server]=server1
option1[server_eu]=server_eu1
option1[server_eu_fr]=server_eu_fr1
option2=val2
option2[server]=server2
option2[server_eu]=server_eu2
option3=val3
option3[server]=server3

From this config file, we can obtain different values depending on the config name :

  • for the server configuration, the values for the 3 options will be :

      [section1]
      option1[server]=server1
      option2[server]=server2
      option3[server]=server3
    
  • for the server_usa configuration, the values for the 3 options will be :

      [section1]
      option1[server]=server1
      option2[server]=server2
      option3[server]=server3
    
  • for the server_eu_gb configuration, the values for the 3 options will be :

      [section1]
      option1[server_eu]=server_eu1
      option2[server_eu]=server_eu2
      option3[server]=server3
    
  • for the server_eu_fr configuration, the values for the 3 options will be :

      [section1]
      option1[server_eu_fr]=server_eu_fr1
      option2[server_eu]=server_eu2
      option3[server]=server3
    

Basically, with a foo_bar_baz configuration name, ConfigParserExtended will search for :

  1. option[foo_bar_baz]
  2. option[foo_bar]
  3. option[foo]
  4. option

There is also another "version" of this function which will search for :

  1. option[foo_bar_baz]
  2. option[bar_baz]
  3. option[baz]
  4. option[foo_bar]
  5. option[bar]
  6. option[foo]
  7. option

You can use this version via the get() method by setting the cfg_plus parameter to True.

Syntax

Syntax : option[grandparent_parent_child]=value

Note that the separator _ is customizable via the constructor (config_separator=) and set_config_separator().

Section inheritance

Basics

This functionnality allows you to define "fallback sections" for any particular section :

[section1:section2:section3]
option1=val1

[section2]
option2=val2

[section3]
option2=val2_sect3
option3=val3

[DEFAULT]
option1=default1
option2=default2
option3=default3
option4=default4

In this case, section2 and section3 serve as fallback sections, prioritized over the DEFAULT section, for section1. The list of items in section1 will be :

[section1]
option1=val1
option2=val2
option3=val3
option4=default4

As you can see, it is possible to have multiple fallback sections but they are prioritized from left to right (see option2). You can consider that every section "inherits" from the DEFAULT section. Warning, writing section1:section2:section3 doesn't imply that section2:section3!!! It only implies that section1:section2 and section1:section3

There is also a mode that allows Object-like inheritance : in this case, you do not define fallback sections, but parents for the section you want. Here is an example :

Diamond Inheritance

A file that would reproduce this situation would be :

[D:B:C]
key1=D

[B:A]
key2=B

[C:A]
key2=C

[A]
key3=A

There is a little twist here though : technically, there should be an ambiguity about the value of key2 here, but, since B was defined as a parent before C in the code, B will have the priority over C. This also applies for their respective parents. Therefore, the values in D will be :

[D]
key1=D
key2=B
key3=A

To switch from section fallback (default) to Object-like inheritance, the attribute inheritance must be set to either im, impl or implicit. This can be done from the constructor (inheriance=) or using the set_intheritance() function. Any other value given to inheritance will switch to the default mode.

Usage with option specification

In ConfigParserExtended, by default, the search using configuration names is done on a section scale, not on the whole file. Thus, if an option has been found in a child section, the option will be returned and the parent section will not be searched. Example with a dev configuration :

[section1:section2]
option1=val1

[section2]
option1[dev]=dev1

The value of option1 for section1 will be val1 here. However, if you want to get dev1 instead of val1 with the same parameters, you can set the sect_first parameter to False in the get() method, which will prioritize the configuration name over the section name.

Syntax

Syntax : [child:parent:grandparent]

Note that the separator : is customizable via the constructor (section_separator=) and set_section_separator().

Miscellaneous

Some methods have been adapted to these new functionnalities by adding options to take into account option specification and inheritance. These options allow you to activate or deactivate the new functionalities. Keep in mind that these methods have the exact same roles as in the original ConfigParser (if not, feel free to report it).

has_option()

This function has now 3 additional optional parameters :

  • config : enter a configuration name to use has_option for this specific configuration name

  • cfg_ind : if True, the function will return True if the option has been found, regardless of the specified configuration name. If False (default), the function will return True only if the option has been found either without specification, or specified with the current configuration name (or with config, if set).

  • strict : if True, the function will only search the given section, neither the parent section(s) nor the DEFAULT section will be searched. If False (default), the given section will be searched as well as its parent(s) and the DEFAULT section.

A little example with a aye config name using has_option('section1',option1):

[section1:section2]
option1[nope]=nope1

[section2]
option1=val1
  • strict=False ; cfg_ind=False => return True (found in section2)

  • strict=True ; cfg_ind=False => return False (didn't find option1[aye] or option1 in section 1)

  • strict=True ; cfg_ind=False ; config=nope => return True (found option[nope] in section 1)

  • strict=False ; cfg_ind=True => return True (found an option1 in section1)

  • strict=True ; cfg_ind=True => return True (found an option1 in section1)

has_section()

This function has got an additional boolean parameter, strict, which will decide if the function searches for the exact given section name (strict=True) or if the function searches for eventual inheritance signs(strict=True, default). Here's a small example using has_section('section1') :

[section1:section2]
random=stuff

[section2]
really=random
  • strict=False => return True because section1: has been found

  • strict=True => return False because section1 != section1:section2 here

items()

This function now has the same strict optional parameter as has_option() which plays the exact same role : if strict=True, the function will return a list of option-value tuples from the section only, and, if strict=False (default), will return a list of option-value tuples from the section, its parents and the DEFAULT section.

There is another optional parameter defaults which is intended to work with strict=True : if defaults=True, the DEFAULT section's items will be returned as well as the given section's items.

options()

This function now has the same strict optional parameter as has_option() which plays the exact same role : if strict=True, the function will return a list of options from the section only, and, if strict=False (default), will return a list of options from the section, its parents and the DEFAULT section.

If strict is True, you can set the optional parameter defaults to True to get the list of options from the section and the DEFAULT section, excluding the section's parents

You can also obtain a list of options without the option specifications (without [config_name]) by setting the optional parameter cfg_ind to True.

List values

You can now have a list of values associated with an option : just by separating the values with ;, you can have a list of values bound to an option :

[section1]
option1=a;very;long;list

The get() function now has a new optionnal parameter isList that, if set to True, allows the function to return a value as a list by splitting the value using the list separator (; by default). This separator is customizable via the constructor (list_separator=) and set_list_separator().

There are also new functions which combine this new functionnalities with the ones from their predecessors : getintlist(), getfloatlist(), getbooleanlist()

Special thanks

  • The configparser dev team
  • The six dev team
  • The Python dev team
  • thefab
  • Météo France