Source code for qudi.core.configoption
# -*- coding: utf-8 -*-
"""
ConfigOption object to be used in qudi modules. The value of each ConfigOption can
(if it has a default value) or must be specified by the user in the config file.
Usually these values should be constant for the duration of a qudi session.
Copyright (c) 2021, the qudi developers. See the AUTHORS.md file at the top-level directory of this
distribution and on <https://github.com/Ulm-IQO/qudi-core/>
This file is part of qudi.
Qudi is free software: you can redistribute it and/or modify it under the terms of
the GNU Lesser General Public License as published by the Free Software Foundation,
either version 3 of the License, or (at your option) any later version.
Qudi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License along with qudi.
If not, see <https://www.gnu.org/licenses/>.
"""
__all__ = ['ConfigOption', 'MissingOption']
import copy
import inspect
from enum import Enum
from typing import Any, Optional, Callable
[docs]
class MissingOption(Enum):
""" Representation for missing ConfigOption """
error = -3
warn = -2
info = -1
nothing = 0
[docs]
class ConfigOption:
"""This class represents a configuration entry in the config file that is loaded before
module initalisation.
"""
[docs]
def __init__(self, name: Optional[str] = None, default: Optional[Any] = None, *,
missing: Optional[str] = 'nothing', constructor: Optional[Callable] = None,
checker: Optional[Callable] = None, converter: Optional[Callable] = None):
""" Create a ConfigOption object.
Parameters
----------
name : Optional[str], optional
Identifier of the option in the configuration file.
default : Optional[Any], optional
Default value for the case that the option is not set in the config file.
missing : Optional[str], optional
Action to take when the option is not set.
'nothing' does nothing,
'warn' logs a warning,
'error' logs an error and prevents the module from loading.
constructor : Optional[Callable], optional
Constructor function for complex config option behavior.
checker : Optional[Callable], optional
Static function that checks if value is okay.
converter : Optional[Callable], optional
Static function that forces type interpretation.
"""
self.missing = MissingOption[missing]
self.name = name
self.default = default
self.checker = checker
self.converter = converter
self.constructor_function = None
if constructor is not None:
self.constructor(constructor)
def __set_name__(self, owner, name):
if self.name is None:
self.name = name
def __copy__(self):
return self.copy()
def __deepcopy__(self, memodict={}):
return self.copy()
@property
def optional(self) -> bool:
return self.missing != MissingOption.error
def copy(self, **kwargs):
"""Create a new instance of ConfigOption with copied values and update.
Parameters
----------
**kwargs
Extra arguments or overrides for the constructor of this class.
"""
newargs = {'name': self.name,
'default': copy.deepcopy(self.default),
'missing': self.missing.name,
'constructor': self.constructor_function,
'checker': self.checker,
'converter': self.converter}
newargs.update(kwargs)
return ConfigOption(**newargs)
def check(self, value: Any) -> bool:
"""If checker function set, check value. Assume everything is ok otherwise.
"""
if callable(self.checker):
return self.checker(value)
return True
def convert(self, value: Any) -> Any:
"""If converter function set, convert value (pass-through otherwise).
"""
if callable(self.converter):
return self.converter(value)
return value
def constructor(self, func: Callable) -> Callable:
"""Decorator for declaring a constructor function for this ConfigOption.
Parameters
----------
func : Callable
Constructor function for this ConfigOption.
Returns
-------
Callable
Returns the original function so it can be used as a decorator.
"""
self.constructor_function = self._assert_func_signature(func)
return func
@staticmethod
def _assert_func_signature(func: Callable) -> Callable:
assert callable(func), 'ConfigOption constructor must be callable'
params = tuple(inspect.signature(func).parameters)
assert 0 < len(params) < 3, 'ConfigOption constructor must be function with ' \
'1 (static) or 2 (bound method) parameters.'
if len(params) == 1:
def wrapper(instance, value):
return func(value)
return wrapper
return func