An _environment source is added that looks in os.environ for values. Using the environment is on by default, but can be shut down by setting `use_env` to False when __call__ is called. The enviroment is inspected before any other sources of config data but the value is used after command line arguments and before config file options. This is done by checking both the command line and config files and then inspecting the location of the result. If it is command_line, we use it. If not, we use the environment value (if any). If there's no environment value, the config file value is used. If checking the command line and config file results in a KeyError, the environment value is used, if set. The names of the environment variables follow the rules described in oslo_config.sources._environment. A new exception has been added: ConfigSourceValueError, this is the superclass of the existing ConfigFileValueError. The code in _do_get has been updated to only use ConfigFileValueError when it is in fact a file from whence a ValueError came. Documentation has been updated and a rlease note created to indicate the new functionality. Change-Id: I3245c40ebdcc96f8e3b2dc0bab3b4aa71d07ad15
10 KiB
Defining Options
Configuration options may be set on the command line, in the environment <oslo_config.sources._environment>
,
or in config files. Options are processed in that order.
The schema for each option is defined using the Opt
class or its
sub-classes, for example:
from oslo_config import cfg
from oslo_config import types
= types.Integer(1, 65535)
PortType
= [
common_opts 'bind_host',
cfg.StrOpt(='0.0.0.0',
defaulthelp='IP address to listen on.'),
'bind_port',
cfg.Opt(type=PortType,
=9292,
defaulthelp='Port number to listen on.')
]
Option Types
Options can have arbitrary types via the type parameter to the Opt
constructor. The type parameter is a callable object that takes
a string and either returns a value of that particular type or raises
ValueError
if the
value can not be converted.
For convenience, there are predefined option subclasses in oslo_config.cfg
that set the
option type as in the following
table:
Type | Option |
---|---|
oslo_config.types.String |
oslo_config.cfg.StrOpt |
oslo_config.types.String |
oslo_config.cfg.SubCommandOpt |
oslo_config.types.Boolean |
oslo_config.cfg.BoolOpt |
oslo_config.types.Integer |
oslo_config.cfg.IntOpt |
oslo_config.types.Float |
oslo_config.cfg.FloatOpt |
oslo_config.types.Port |
oslo_config.cfg.PortOpt |
oslo_config.types.List |
oslo_config.cfg.ListOpt |
oslo_config.types.Dict |
oslo_config.cfg.DictOpt |
oslo_config.types.IPAddress |
oslo_config.cfg.IPOpt |
oslo_config.types.Hostname |
oslo_config.cfg.HostnameOpt |
oslo_config.types.HostAddress |
oslo_config.cfg.HostAddressOpt |
oslo_config.types.URI |
oslo_config.cfg.URIOpt |
For oslo_config.cfg.MultiOpt
the item_type parameter defines the type of the
values. For convenience, oslo_config.cfg.MultiStrOpt
is ~oslo_config.cfg.MultiOpt
with the item_type parameter set to oslo_config.types.MultiString
.
The following example defines options using the convenience classes:
= cfg.ListOpt('enabled_apis',
enabled_apis_opt =['ec2', 'osapi_compute'],
defaulthelp='List of APIs to enable by default.')
= [
DEFAULT_EXTENSIONS 'nova.api.openstack.compute.contrib.standard_extensions'
]= cfg.MultiStrOpt('osapi_compute_extension',
osapi_compute_extension_opt =DEFAULT_EXTENSIONS) default
Registering Options
Option schemas are registered with the config manager at runtime, but before the option is referenced:
class ExtensionManager(object):
= cfg.ListOpt(...)
enabled_apis_opt
def __init__(self, conf):
self.conf = conf
self.conf.register_opt(enabled_apis_opt)
...
def _load_extensions(self):
for ext_factory in self.conf.osapi_compute_extension:
....
A common usage pattern is for each option schema to be defined in the module or class which uses the option:
= ...
opts
def add_common_opts(conf):
conf.register_opts(opts)
def get_bind_host(conf):
return conf.bind_host
def get_bind_port(conf):
return conf.bind_port
An option may optionally be made available via the command line. Such options must be registered with the config manager before the command line is parsed (for the purposes of --help and CLI arg validation):
= [
cli_opts 'verbose',
cfg.BoolOpt(='v',
short=False,
defaulthelp='Print more verbose output.'),
'debug',
cfg.BoolOpt(='d',
short=False,
defaulthelp='Print debugging output.'),
]
def add_common_opts(conf):
conf.register_cli_opts(cli_opts)
Option Groups
Options can be registered as belonging to a group:
= cfg.OptGroup(name='rabbit',
rabbit_group ='RabbitMQ options')
title
= cfg.StrOpt('host',
rabbit_host_opt ='localhost',
defaulthelp='IP/hostname to listen on.'),
= cfg.PortOpt('port',
rabbit_port_opt =5672,
defaulthelp='Port number to listen on.')
def register_rabbit_opts(conf):
conf.register_group(rabbit_group)# options can be registered under a group in either of these ways:
=rabbit_group)
conf.register_opt(rabbit_host_opt, group='rabbit') conf.register_opt(rabbit_port_opt, group
If no group attributes are required other than the group name, the group need not be explicitly registered for example:
def register_rabbit_opts(conf):
# The group will automatically be created, equivalent calling:
# conf.register_group(OptGroup(name='rabbit'))
='rabbit') conf.register_opt(rabbit_port_opt, group
If no group is specified, options belong to the 'DEFAULT' section of config files:
glance-api.conf:
[DEFAULT]
bind_port = 9292
...
[rabbit]
host = localhost
port = 5672
use_ssl = False
userid = guest
password = guest
virtual_host = /
Command-line options in a group are automatically prefixed with the group name:
--rabbit-host localhost --rabbit-port 9999
Dynamic Groups
Groups can be registered dynamically by application code. This
introduces a challenge for the sample generator, discovery mechanisms,
and validation tools, since they do not know in advance the names of all
of the groups. The dynamic_group_owner
parameter to the
constructor specifies the full name of an option registered in another
group that controls repeated instances of a dynamic group. This option
is usually a MultiStrOpt.
For example, Cinder supports multiple storage backend devices and
services. To configure Cinder to communicate with multiple backends, the
enabled_backends
option is set to the list of names of
backends. Each backend group includes the options for communicating with
that device or service.
Driver Groups
Groups can have dynamic sets of options, usually based on a driver that has unique requirements. This works at runtime because the code registers options before it uses them, but it introduces a challenge for the sample generator, discovery mechanisms, and validation tools because they do not know in advance the correct options for a group.
To address this issue, the driver option for a group can be named
using the driver_option
parameter. Each driver option
should define its own discovery entry point namespace to return the set
of options for that driver, named using the prefix
"oslo.config.opts."
followed by the driver option name.
In the Cinder case described above, a
volume_backend_name
option is part of the static definition
of the group, so driver_option
should be set to
"volume_backend_name"
. And plugins should be registered
under "oslo.config.opts.volume_backend_name"
using the same
names as the main plugin registered with
"oslo.config.opts"
. The drivers residing within the Cinder
code base have an entry point named "cinder"
registered.
Special Handling Instructions
Options may be declared as required so that an error is raised if the user does not supply a value for the option:
= [
opts 'service_name', required=True),
cfg.StrOpt('image_id', required=True),
cfg.StrOpt(
... ]
Options may be declared as secret so that their values are not leaked into log files:
= [
opts 's3_store_access_key', secret=True),
cfg.StrOpt('s3_store_secret_key', secret=True),
cfg.StrOpt(
... ]
Dictionary Options
If you need end users to specify a dictionary of key/value pairs, then you can use the DictOpt:
= [
opts 'foo',
cfg.DictOpt(={})
default ]
The end users can then specify the option foo in their configuration file as shown below:
[DEFAULT]
foo = k1:v1,k2:v2
Advanced Option
Use if you need to label an option as advanced in sample files, indicating the option is not normally used by the majority of users and might have a significant effect on stability and/or performance:
from oslo_config import cfg
= [
opts 'option1', default='default_value',
cfg.StrOpt(=True, help='This is help '
advanced'text.'),
'option2', default='default_value',
cfg.PortOpt(help='This is help text.'),
]
= cfg.CONF
CONF CONF.register_opts(opts)
This will result in the option being pushed to the bottom of the namespace and labeled as advanced in the sample files, with a notation about possible effects:
[DEFAULT]
...
# This is help text. (string value)
# option2 = default_value
...
<pushed to bottom of section>
...
# This is help text. (string value)
# Advanced Option: intended for advanced users and not used
# by the majority of users, and might have a significant
# effect on stability and/or performance.
# option1 = default_value