2017-08-24 12:26:55 +02:00
|
|
|
from typing import Any, Callable, Dict, Iterable, Optional, TypeVar, Union
|
2017-08-22 14:17:20 +02:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
from django.core.exceptions import ValidationError as DjangoValidationError
|
|
|
|
from django.utils.translation import ugettext as _
|
2017-08-22 14:17:20 +02:00
|
|
|
from mypy_extensions import TypedDict
|
2015-06-17 18:32:05 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
from ..utils.collection import CollectionElement
|
2013-03-01 17:13:12 +01:00
|
|
|
from .exceptions import ConfigError, ConfigNotFound
|
2013-09-25 10:01:01 +02:00
|
|
|
from .models import ConfigStore
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
INPUT_TYPE_MAPPING = {
|
|
|
|
'string': str,
|
2016-01-09 12:51:26 +01:00
|
|
|
'text': str,
|
2017-01-06 10:00:31 +01:00
|
|
|
'markupText': str,
|
2015-06-17 18:32:05 +02:00
|
|
|
'integer': int,
|
|
|
|
'boolean': bool,
|
2016-02-21 22:03:25 +01:00
|
|
|
'choice': str,
|
2017-08-23 14:34:15 +02:00
|
|
|
'comments': dict,
|
2016-09-23 14:03:07 +02:00
|
|
|
'colorpicker': str,
|
2016-10-12 17:28:18 +02:00
|
|
|
'datetimepicker': int,
|
2016-10-15 18:16:22 +02:00
|
|
|
'majorityMethod': str,
|
2018-01-30 16:12:02 +01:00
|
|
|
'static': dict,
|
2017-08-30 12:17:07 +02:00
|
|
|
'translations': list,
|
2016-10-15 18:16:22 +02:00
|
|
|
}
|
2015-06-17 18:32:05 +02:00
|
|
|
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
class ConfigHandler:
|
2013-03-01 17:13:12 +01:00
|
|
|
"""
|
2015-06-17 18:32:05 +02:00
|
|
|
A simple object class to wrap the config variables. It is a container
|
2013-03-01 17:13:12 +01:00
|
|
|
object. To get a config variable use x = config[...], to set it use
|
|
|
|
config[...] = x.
|
|
|
|
"""
|
2016-06-02 12:47:01 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
def __init__(self) -> None:
|
2016-06-02 12:47:01 +02:00
|
|
|
# Dict, that keeps all ConfigVariable objects. Has to be set at statup.
|
2017-08-22 14:17:20 +02:00
|
|
|
# See the ready() method in openslides.core.apps.
|
|
|
|
self.config_variables = {} # type: Dict[str, ConfigVariable]
|
|
|
|
|
|
|
|
# Index to get the database id from a given config key
|
|
|
|
self.key_to_id = {} # type: Dict[str, int]
|
2016-06-02 12:47:01 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
def __getitem__(self, key: str) -> Any:
|
2015-06-17 18:32:05 +02:00
|
|
|
"""
|
2017-08-22 14:17:20 +02:00
|
|
|
Returns the value of the config variable.
|
2015-06-17 18:32:05 +02:00
|
|
|
"""
|
2017-08-22 14:17:20 +02:00
|
|
|
# Build the key_to_id dict
|
|
|
|
self.save_default_values()
|
|
|
|
|
|
|
|
if not self.exists(key):
|
2016-06-02 12:47:01 +02:00
|
|
|
raise ConfigNotFound(_('The config variable {} was not found.').format(key))
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
return CollectionElement.from_values(
|
|
|
|
self.get_collection_string(),
|
|
|
|
self.key_to_id[key]).get_full_data()['value']
|
2015-06-17 18:32:05 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
def exists(self, key: str) -> bool:
|
2016-06-02 12:47:01 +02:00
|
|
|
"""
|
2017-08-22 14:17:20 +02:00
|
|
|
Returns True, if the config varialbe was defined.
|
2016-06-02 12:47:01 +02:00
|
|
|
"""
|
2015-06-17 18:32:05 +02:00
|
|
|
try:
|
2016-06-02 12:47:01 +02:00
|
|
|
self.config_variables[key]
|
|
|
|
except KeyError:
|
2015-06-17 18:32:05 +02:00
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
# TODO: Remove the any by using right types in INPUT_TYPE_MAPPING
|
|
|
|
def __setitem__(self, key: str, value: Any) -> None:
|
2015-06-17 18:32:05 +02:00
|
|
|
"""
|
|
|
|
Sets the new value. First it validates the input.
|
|
|
|
"""
|
2015-06-16 14:03:42 +02:00
|
|
|
# Check if the variable is defined.
|
2015-06-17 18:32:05 +02:00
|
|
|
try:
|
2016-06-02 12:47:01 +02:00
|
|
|
config_variable = self.config_variables[key]
|
2015-06-17 18:32:05 +02:00
|
|
|
except KeyError:
|
2016-06-02 12:47:01 +02:00
|
|
|
raise ConfigNotFound(_('The config variable {} was not found.').format(key))
|
2015-06-17 18:32:05 +02:00
|
|
|
|
|
|
|
# Validate datatype and run validators.
|
|
|
|
expected_type = INPUT_TYPE_MAPPING[config_variable.input_type]
|
2015-06-29 14:17:05 +02:00
|
|
|
|
|
|
|
# Try to convert value into the expected datatype
|
|
|
|
try:
|
|
|
|
value = expected_type(value)
|
|
|
|
except ValueError:
|
2015-11-18 00:15:18 +01:00
|
|
|
raise ConfigError(_('Wrong datatype. Expected %(expected_type)s, got %(got_type)s.') % {
|
|
|
|
'expected_type': expected_type, 'got_type': type(value)})
|
2016-06-02 12:47:01 +02:00
|
|
|
|
|
|
|
if config_variable.input_type == 'choice':
|
|
|
|
# Choices can be a callable. In this case call it at this place
|
|
|
|
if callable(config_variable.choices):
|
|
|
|
choices = config_variable.choices()
|
|
|
|
else:
|
|
|
|
choices = config_variable.choices
|
2017-08-22 14:17:20 +02:00
|
|
|
if choices is None or value not in map(lambda choice: choice['value'], choices):
|
2016-06-02 12:47:01 +02:00
|
|
|
raise ConfigError(_('Invalid input. Choice does not match.'))
|
2017-08-22 14:17:20 +02:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
for validator in config_variable.validators:
|
|
|
|
try:
|
|
|
|
validator(value)
|
|
|
|
except DjangoValidationError as e:
|
|
|
|
raise ConfigError(e.messages[0])
|
2015-06-16 14:03:42 +02:00
|
|
|
|
2016-09-09 15:16:56 +02:00
|
|
|
if config_variable.input_type == 'comments':
|
2017-08-23 14:34:15 +02:00
|
|
|
if not isinstance(value, dict):
|
|
|
|
raise ConfigError(_('motions_comments has to be a dict.'))
|
|
|
|
valuecopy = dict()
|
|
|
|
for id, commentsfield in value.items():
|
|
|
|
try:
|
|
|
|
id = int(id)
|
|
|
|
except ValueError:
|
|
|
|
raise ConfigError(_('Each id has to be an int.'))
|
|
|
|
|
|
|
|
if id < 1:
|
|
|
|
raise ConfigError(_('Each id has to be greater then 0.'))
|
|
|
|
# Deleted commentsfields are saved as None to block the used ids
|
|
|
|
if commentsfield is not None:
|
|
|
|
if not isinstance(commentsfield, dict):
|
|
|
|
raise ConfigError(_('Each commentsfield in motions_comments has to be a dict.'))
|
|
|
|
if commentsfield.get('name') is None or commentsfield.get('public') is None:
|
|
|
|
raise ConfigError(_('A name and a public property have to be given.'))
|
|
|
|
if not isinstance(commentsfield['name'], str):
|
|
|
|
raise ConfigError(_('name has to be string.'))
|
|
|
|
if not isinstance(commentsfield['public'], bool):
|
|
|
|
raise ConfigError(_('public property has to be bool.'))
|
|
|
|
valuecopy[id] = commentsfield
|
|
|
|
value = valuecopy
|
2016-09-09 15:16:56 +02:00
|
|
|
|
2018-01-30 16:12:02 +01:00
|
|
|
if config_variable.input_type == 'static':
|
2017-03-31 13:48:41 +02:00
|
|
|
if not isinstance(value, dict):
|
2018-01-30 16:12:02 +01:00
|
|
|
raise ConfigError(_('This has to be a dict.'))
|
2017-03-31 13:48:41 +02:00
|
|
|
whitelist = (
|
|
|
|
'path',
|
|
|
|
'display_name',
|
|
|
|
)
|
|
|
|
for required_entry in whitelist:
|
|
|
|
if required_entry not in value:
|
|
|
|
raise ConfigError(_('{} has to be given.'.format(required_entry)))
|
|
|
|
if not isinstance(value[required_entry], str):
|
|
|
|
raise ConfigError(_('{} has to be a string.'.format(required_entry)))
|
|
|
|
|
2017-08-30 12:17:07 +02:00
|
|
|
if config_variable.input_type == 'translations':
|
|
|
|
if not isinstance(value, list):
|
|
|
|
raise ConfigError(_('Translations has to be a list.'))
|
|
|
|
for entry in value:
|
|
|
|
if not isinstance(entry, dict):
|
|
|
|
raise ConfigError(_('Every value has to be a dict, not {}.'.format(type(entry))))
|
|
|
|
whitelist = (
|
|
|
|
'original',
|
|
|
|
'translation',
|
|
|
|
)
|
|
|
|
for required_entry in whitelist:
|
|
|
|
if required_entry not in entry:
|
|
|
|
raise ConfigError(_('{} has to be given.'.format(required_entry)))
|
|
|
|
if not isinstance(entry[required_entry], str):
|
|
|
|
raise ConfigError(_('{} has to be a string.'.format(required_entry)))
|
|
|
|
|
2015-06-16 14:03:42 +02:00
|
|
|
# Save the new value to the database.
|
2017-08-22 14:17:20 +02:00
|
|
|
db_value = ConfigStore.objects.get(key=key)
|
2016-10-01 12:58:49 +02:00
|
|
|
db_value.value = value
|
|
|
|
db_value.save(information={'changed_config': key})
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-16 14:03:42 +02:00
|
|
|
# Call on_change callback.
|
2015-06-17 18:32:05 +02:00
|
|
|
if config_variable.on_change:
|
|
|
|
config_variable.on_change()
|
2013-09-08 10:26:51 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
def update_config_variables(self, items: Iterable['ConfigVariable']) -> None:
|
2015-01-24 16:35:50 +01:00
|
|
|
"""
|
2016-06-02 12:47:01 +02:00
|
|
|
Updates the config_variables dict.
|
2015-01-24 16:35:50 +01:00
|
|
|
"""
|
2017-08-22 14:17:20 +02:00
|
|
|
# build an index from variable name to the variable
|
|
|
|
item_index = dict((variable.name, variable) for variable in items)
|
|
|
|
|
2016-06-02 12:47:01 +02:00
|
|
|
# Check that all ConfigVariables are unique. So no key from items can
|
|
|
|
# be in already in self.config_variables
|
2017-08-22 14:17:20 +02:00
|
|
|
intersection = set(item_index.keys()).intersection(self.config_variables.keys())
|
|
|
|
if intersection:
|
|
|
|
raise ConfigError(_('Too many values for config variables {} found.').format(intersection))
|
2015-01-24 16:35:50 +01:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
self.config_variables.update(item_index)
|
2016-06-02 12:47:01 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
def save_default_values(self) -> None:
|
2015-06-16 14:03:42 +02:00
|
|
|
"""
|
2017-08-22 14:17:20 +02:00
|
|
|
Saves the default values to the database.
|
2016-06-02 12:47:01 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
Does also build the dictonary key_to_id.
|
2015-06-16 14:03:42 +02:00
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
Does nothing on a second run.
|
2014-01-31 01:54:41 +01:00
|
|
|
"""
|
2017-08-22 14:17:20 +02:00
|
|
|
if not self.key_to_id:
|
|
|
|
for item in self.config_variables.values():
|
|
|
|
try:
|
|
|
|
db_value = ConfigStore.objects.get(key=item.name)
|
|
|
|
except ConfigStore.DoesNotExist:
|
|
|
|
db_value = ConfigStore()
|
|
|
|
db_value.key = item.name
|
|
|
|
db_value.value = item.default_value
|
|
|
|
db_value.save(skip_autoupdate=True)
|
|
|
|
self.key_to_id[item.name] = db_value.pk
|
|
|
|
|
|
|
|
def get_collection_string(self) -> str:
|
2016-09-17 22:26:23 +02:00
|
|
|
"""
|
|
|
|
Returns the collection_string from the CollectionStore.
|
|
|
|
"""
|
|
|
|
return ConfigStore.get_collection_string()
|
|
|
|
|
2016-11-09 22:18:44 +01:00
|
|
|
|
2013-03-01 17:13:12 +01:00
|
|
|
config = ConfigHandler()
|
|
|
|
"""
|
|
|
|
Final entry point to get an set config variables. To get a config variable
|
|
|
|
use x = config[...], to set it use config[...] = x.
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
2017-08-22 14:17:20 +02:00
|
|
|
T = TypeVar('T')
|
2017-08-24 12:26:55 +02:00
|
|
|
ChoiceType = Optional[Iterable[Dict[str, str]]]
|
2017-08-22 14:17:20 +02:00
|
|
|
ChoiceCallableType = Union[ChoiceType, Callable[[], ChoiceType]]
|
2017-08-24 12:26:55 +02:00
|
|
|
ValidatorsType = Iterable[Callable[[T], None]]
|
2017-08-22 14:17:20 +02:00
|
|
|
OnChangeType = Callable[[], None]
|
|
|
|
ConfigVariableDict = TypedDict('ConfigVariableDict', {
|
|
|
|
'key': str,
|
|
|
|
'default_value': Any,
|
|
|
|
'value': Any,
|
|
|
|
'input_type': str,
|
|
|
|
'label': str,
|
|
|
|
'help_text': str,
|
|
|
|
'choices': ChoiceType,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
class ConfigVariable:
|
2013-03-01 17:13:12 +01:00
|
|
|
"""
|
2015-06-17 18:32:05 +02:00
|
|
|
A simple object class to wrap new config variables.
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
The keyword arguments 'name' and 'default_value' are required.
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2016-02-14 21:38:26 +01:00
|
|
|
The keyword arguments 'input_type', 'label', 'help_text' and 'hidden'
|
|
|
|
are for rendering a HTML form element. The 'input_type is also used for
|
|
|
|
validation. If you set 'input_type' to 'choice' you have to provide
|
|
|
|
'choices', which is a list of dictionaries containing a value and a
|
|
|
|
display_name of every possible choice.
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
The keyword arguments 'weight', 'group' and 'subgroup' are for sorting
|
|
|
|
and grouping.
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
The keyword argument validators expects an interable of validator
|
|
|
|
functions. Such a function gets the value and raises Django's
|
|
|
|
ValidationError if the value is invalid.
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
The keyword argument 'on_change' can be a callback which is called
|
|
|
|
every time, the variable is changed.
|
2013-03-01 17:13:12 +01:00
|
|
|
|
2015-06-17 18:32:05 +02:00
|
|
|
If the argument 'translatable' is set, OpenSlides is able to translate
|
|
|
|
the value during setup of the database if the admin uses the respective
|
|
|
|
command line option.
|
2013-03-01 17:13:12 +01:00
|
|
|
"""
|
2018-08-08 21:09:22 +02:00
|
|
|
def __init__(self, name: str, default_value: T, input_type: str = 'string',
|
|
|
|
label: str = None, help_text: str = None, choices: ChoiceCallableType = None,
|
|
|
|
hidden: bool = False, weight: int = 0, group: str = None, subgroup: str = None,
|
|
|
|
validators: ValidatorsType = None, on_change: OnChangeType = None) -> None:
|
2015-06-17 18:32:05 +02:00
|
|
|
if input_type not in INPUT_TYPE_MAPPING:
|
|
|
|
raise ValueError(_('Invalid value for config attribute input_type.'))
|
|
|
|
if input_type == 'choice' and choices is None:
|
|
|
|
raise ConfigError(_("Either config attribute 'choices' must not be None or "
|
|
|
|
"'input_type' must not be 'choice'."))
|
|
|
|
elif input_type != 'choice' and choices is not None:
|
|
|
|
raise ConfigError(_("Either config attribute 'choices' must be None or "
|
|
|
|
"'input_type' must be 'choice'."))
|
2013-03-01 17:13:12 +01:00
|
|
|
self.name = name
|
|
|
|
self.default_value = default_value
|
2015-06-17 18:32:05 +02:00
|
|
|
self.input_type = input_type
|
|
|
|
self.label = label or name
|
|
|
|
self.help_text = help_text or ''
|
|
|
|
self.choices = choices
|
2016-02-14 21:38:26 +01:00
|
|
|
self.hidden = hidden
|
2015-06-17 18:32:05 +02:00
|
|
|
self.weight = weight
|
|
|
|
self.group = group or _('General')
|
|
|
|
self.subgroup = subgroup
|
|
|
|
self.validators = validators or ()
|
2013-09-08 10:26:51 +02:00
|
|
|
self.on_change = on_change
|
2015-06-17 18:32:05 +02:00
|
|
|
|
|
|
|
@property
|
2017-08-22 14:17:20 +02:00
|
|
|
def data(self) -> ConfigVariableDict:
|
2015-06-17 18:32:05 +02:00
|
|
|
"""
|
2018-01-20 15:58:36 +01:00
|
|
|
Property with all data for AngularJS variable on startup.
|
2015-06-17 18:32:05 +02:00
|
|
|
"""
|
2017-08-22 14:17:20 +02:00
|
|
|
return ConfigVariableDict(
|
|
|
|
key=self.name,
|
|
|
|
default_value=self.default_value,
|
|
|
|
value=config[self.name],
|
|
|
|
input_type=self.input_type,
|
|
|
|
label=self.label,
|
|
|
|
help_text=self.help_text,
|
|
|
|
choices=self.choices() if callable(self.choices) else self.choices
|
|
|
|
)
|
|
|
|
|
|
|
|
def is_hidden(self) -> bool:
|
2016-02-14 21:38:26 +01:00
|
|
|
"""
|
|
|
|
Returns True if the config variable is hidden so it can be removed
|
|
|
|
from response of OPTIONS request.
|
|
|
|
"""
|
|
|
|
return self.hidden
|