2013-02-07 16:20:21 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
2013-02-08 11:50:22 +01:00
|
|
|
"takes care of the option's values and multi values"
|
2013-02-21 17:07:00 +01:00
|
|
|
# Copyright (C) 2013 Team tiramisu (see AUTHORS for all contributors)
|
2013-02-07 16:20:21 +01:00
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program 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 General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
#
|
|
|
|
# ____________________________________________________________
|
2013-03-25 11:45:21 +01:00
|
|
|
from tiramisu.error import NoValueReturned, MandatoryError, MultiTypeError, \
|
2013-04-03 12:20:26 +02:00
|
|
|
ConfigError # , OptionValueError
|
2013-02-22 11:09:17 +01:00
|
|
|
from tiramisu.setting import owners, multitypes
|
2013-02-07 16:20:21 +01:00
|
|
|
|
2013-04-03 12:20:26 +02:00
|
|
|
|
2013-02-22 11:09:17 +01:00
|
|
|
class Values(object):
|
2013-04-03 12:20:26 +02:00
|
|
|
__slots__ = ('values', 'context')
|
|
|
|
|
2013-02-08 11:50:22 +01:00
|
|
|
def __init__(self, context):
|
2013-02-21 17:07:00 +01:00
|
|
|
"""
|
|
|
|
Initializes the values's dict.
|
|
|
|
|
2013-04-03 12:20:26 +02:00
|
|
|
:param context: the context is the home config's values
|
2013-02-21 17:07:00 +01:00
|
|
|
"""
|
2013-02-07 16:20:21 +01:00
|
|
|
"Config's root indeed is in charge of the `Option()`'s values"
|
|
|
|
self.values = {}
|
2013-02-08 11:50:22 +01:00
|
|
|
self.context = context
|
2013-02-07 16:20:21 +01:00
|
|
|
|
2013-02-08 11:50:22 +01:00
|
|
|
def _get_value(self, opt):
|
|
|
|
"special case for the multis: they never return None"
|
2013-02-22 11:09:17 +01:00
|
|
|
if opt not in self.values:
|
|
|
|
if opt.is_multi():
|
2013-04-03 12:20:26 +02:00
|
|
|
value = Multi(opt.getdefault(), self.context, opt)
|
|
|
|
if opt.multitype == multitypes.slave:
|
|
|
|
masterpath = self.context._cfgimpl_descr.get_path_by_opt(opt.master_slaves)
|
|
|
|
mastervalue = getattr(self.context, masterpath)
|
|
|
|
masterlen = len(mastervalue)
|
|
|
|
if len(value) < masterlen:
|
|
|
|
for num in range(0, masterlen - len(value)):
|
|
|
|
value.append(None, force=True)
|
2013-02-22 11:09:17 +01:00
|
|
|
else:
|
2013-03-06 17:17:33 +01:00
|
|
|
value = opt.getdefault()
|
|
|
|
|
|
|
|
return value
|
2013-04-03 12:20:26 +02:00
|
|
|
return self.values[opt][1]
|
2013-02-07 16:20:21 +01:00
|
|
|
|
2013-02-21 17:07:00 +01:00
|
|
|
def reset(self, opt):
|
|
|
|
if opt in self.values:
|
|
|
|
del(self.values[opt])
|
2013-02-26 14:56:15 +01:00
|
|
|
|
2013-02-21 17:07:00 +01:00
|
|
|
def _is_empty(self, opt, value=None):
|
2013-02-08 11:50:22 +01:00
|
|
|
"convenience method to know if an option is empty"
|
2013-04-03 12:20:26 +02:00
|
|
|
#FIXME: buggy ?
|
|
|
|
#if value is not None:
|
|
|
|
# return False
|
|
|
|
if (not opt.is_multi() and value is None) or \
|
|
|
|
(opt.is_multi() and (value == [] or
|
|
|
|
None in self._get_value(opt))):
|
2013-02-08 11:50:22 +01:00
|
|
|
return True
|
|
|
|
return False
|
2013-02-07 16:20:21 +01:00
|
|
|
|
2013-02-26 14:31:45 +01:00
|
|
|
def is_empty(self, opt):
|
2013-04-03 12:20:26 +02:00
|
|
|
#FIXME that not empty ... just no value!
|
2013-02-26 14:31:45 +01:00
|
|
|
if opt not in self.values:
|
|
|
|
return True
|
2013-04-03 12:20:26 +02:00
|
|
|
value = self.values[opt][1]
|
2013-02-26 14:31:45 +01:00
|
|
|
if not opt.is_multi():
|
2013-04-03 12:20:26 +02:00
|
|
|
if self._get_value(opt) is None:
|
2013-02-26 14:31:45 +01:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
value = list(value)
|
|
|
|
for val in value:
|
2013-04-03 12:20:26 +02:00
|
|
|
if val is not None:
|
2013-02-26 14:31:45 +01:00
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2013-04-03 12:20:26 +02:00
|
|
|
def _test_mandatory(self, opt, value, force_properties=None):
|
|
|
|
setting = self.context.cfgimpl_get_settings()
|
|
|
|
if force_properties is None:
|
|
|
|
set_mandatory = setting.has_property('mandatory')
|
|
|
|
else:
|
|
|
|
set_mandatory = ('mandatory' in force_properties or
|
|
|
|
setting.has_property('mandatory'))
|
|
|
|
if setting.has_property('mandatory', opt) and set_mandatory:
|
2013-02-21 17:07:00 +01:00
|
|
|
if self._is_empty(opt, value) and \
|
2013-02-08 11:50:22 +01:00
|
|
|
opt.is_empty_by_default():
|
|
|
|
raise MandatoryError("option: {0} is mandatory "
|
2013-04-03 12:20:26 +02:00
|
|
|
"and shall have a value".format(opt._name))
|
|
|
|
#empty value
|
|
|
|
if opt.is_multi():
|
|
|
|
for val in value:
|
|
|
|
if val == '':
|
|
|
|
raise MandatoryError("option: {0} is mandatory "
|
|
|
|
"and shall have not empty value".format(opt._name))
|
|
|
|
else:
|
|
|
|
if value == '':
|
|
|
|
raise MandatoryError("option: {0} is mandatory "
|
|
|
|
"and shall have not empty value".format(opt._name))
|
2013-02-08 11:50:22 +01:00
|
|
|
|
2013-02-21 17:07:00 +01:00
|
|
|
def fill_multi(self, opt, result):
|
2013-02-08 11:50:22 +01:00
|
|
|
"""fills a multi option with default and calculated values
|
|
|
|
"""
|
|
|
|
if not isinstance(result, list):
|
|
|
|
_result = [result]
|
|
|
|
else:
|
|
|
|
_result = result
|
2013-04-03 12:20:26 +02:00
|
|
|
#multitype = self._get_multitype(opt)
|
|
|
|
return Multi(_result, self.context, opt) # , multitype)
|
2013-02-08 11:50:22 +01:00
|
|
|
|
|
|
|
def __getitem__(self, opt):
|
2013-04-03 12:20:26 +02:00
|
|
|
return self._getitem(opt)
|
|
|
|
|
|
|
|
def _getitem(self, opt, force_properties=None):
|
2013-02-08 11:50:22 +01:00
|
|
|
# options with callbacks
|
2013-02-21 17:07:00 +01:00
|
|
|
value = self._get_value(opt)
|
2013-04-04 11:24:00 +02:00
|
|
|
setting = self.context.cfgimpl_get_settings()
|
2013-02-08 11:50:22 +01:00
|
|
|
if opt.has_callback():
|
2013-04-04 11:24:00 +02:00
|
|
|
is_frozen = setting.has_property('frozen', opt)
|
|
|
|
if (not is_frozen or (is_frozen and
|
|
|
|
not setting.has_property('force_default_on_freeze', opt)
|
|
|
|
)) and not self.context.cfgimpl_get_values().is_default_owner(opt):
|
|
|
|
return value
|
2013-02-08 11:50:22 +01:00
|
|
|
try:
|
2013-04-03 12:20:26 +02:00
|
|
|
result = opt.getcallback_value(self.context)
|
|
|
|
except NoValueReturned:
|
2013-02-08 11:50:22 +01:00
|
|
|
pass
|
|
|
|
else:
|
|
|
|
if opt.is_multi():
|
2013-02-25 14:27:12 +01:00
|
|
|
value = self.fill_multi(opt, result)
|
2013-02-08 11:50:22 +01:00
|
|
|
else:
|
|
|
|
# this result **shall not** be a list
|
|
|
|
if isinstance(result, list):
|
2013-02-21 17:07:00 +01:00
|
|
|
raise ConfigError('invalid calculated value returned '
|
2013-04-03 12:20:26 +02:00
|
|
|
'for option {0} : shall not be a list'
|
|
|
|
''.format(opt._name))
|
2013-02-21 17:07:00 +01:00
|
|
|
value = result
|
2013-04-03 12:20:26 +02:00
|
|
|
if value is not None and \
|
|
|
|
not opt.validate(value, setting.has_property('validator')):
|
2013-02-08 11:50:22 +01:00
|
|
|
raise ConfigError('invalid calculated value returned'
|
2013-04-03 12:20:26 +02:00
|
|
|
' for option {0}'.format(opt._name))
|
2013-02-08 11:50:22 +01:00
|
|
|
# frozen and force default
|
2013-04-04 11:24:00 +02:00
|
|
|
elif setting.has_property('force_default_on_freeze', opt):
|
2013-02-08 11:50:22 +01:00
|
|
|
value = opt.getdefault()
|
|
|
|
if opt.is_multi():
|
2013-02-21 17:07:00 +01:00
|
|
|
value = self.fill_multi(opt, value)
|
2013-04-03 12:20:26 +02:00
|
|
|
self._test_mandatory(opt, value, force_properties)
|
2013-02-08 11:50:22 +01:00
|
|
|
return value
|
|
|
|
|
|
|
|
def __setitem__(self, opt, value):
|
2013-04-03 12:20:26 +02:00
|
|
|
if opt.is_multi():
|
|
|
|
if opt.multitype == multitypes.master:
|
|
|
|
masterlen = len(value)
|
2013-04-03 17:05:41 +02:00
|
|
|
for slave in opt.master_slaves:
|
2013-04-03 12:20:26 +02:00
|
|
|
value_slave = self._get_value(slave)
|
|
|
|
if len(value_slave) > masterlen:
|
|
|
|
raise MultiTypeError("invalid len for the slave: {0}"
|
|
|
|
" which has {1} as master".format(
|
|
|
|
slave._name, opt._name))
|
|
|
|
elif len(value_slave) < masterlen:
|
|
|
|
for num in range(0, masterlen - len(value_slave)):
|
|
|
|
value_slave.append(None, force=True)
|
2013-02-25 15:52:10 +01:00
|
|
|
|
2013-04-03 12:20:26 +02:00
|
|
|
elif opt.multitype == multitypes.slave:
|
2013-04-03 17:05:41 +02:00
|
|
|
if len(self._get_value(opt.master_slaves)) != len(value):
|
2013-04-03 12:20:26 +02:00
|
|
|
raise MultiTypeError("invalid len for the slave: {0}"
|
|
|
|
" which has {1} as master".format(
|
2013-04-03 17:05:41 +02:00
|
|
|
opt._name, opt.master_slaves._name))
|
2013-04-03 12:20:26 +02:00
|
|
|
if not isinstance(value, Multi):
|
|
|
|
value = Multi(value, self.context, opt)
|
2013-02-22 11:09:17 +01:00
|
|
|
self.setitem(opt, value)
|
|
|
|
|
|
|
|
def setitem(self, opt, value):
|
2013-02-26 16:58:44 +01:00
|
|
|
if type(value) == list:
|
|
|
|
raise MultiTypeError("the type of the value {0} which is multi shall "
|
|
|
|
"be Multi and not list".format(str(value)))
|
2013-04-03 12:20:26 +02:00
|
|
|
self._test_mandatory(opt, value)
|
|
|
|
self.values[opt] = (self.context.cfgimpl_get_settings().getowner(), value)
|
2013-02-08 11:50:22 +01:00
|
|
|
|
|
|
|
def __contains__(self, opt):
|
|
|
|
return opt in self.values
|
|
|
|
|
|
|
|
def getowner(self, opt):
|
2013-04-03 12:20:26 +02:00
|
|
|
return self.values.get(opt, (owners.default, None))[0]
|
|
|
|
|
|
|
|
def setowner(self, opt, owner):
|
|
|
|
if opt not in self.values:
|
|
|
|
raise ConfigError('no value for {} cannot change owner to {}'.format(opt))
|
|
|
|
if not isinstance(owner, owners.Owner):
|
|
|
|
raise TypeError("invalid generic owner {0}".format(str(owner)))
|
|
|
|
self.values[opt] = (owner, self.values[opt][1])
|
|
|
|
|
|
|
|
def is_default_owner(self, opt):
|
|
|
|
"""
|
|
|
|
:param config: *must* be only the **parent** config
|
|
|
|
(not the toplevel config)
|
|
|
|
:return: boolean
|
|
|
|
"""
|
|
|
|
return self.getowner(opt) == owners.default
|
2013-02-21 17:07:00 +01:00
|
|
|
|
2013-02-08 11:50:22 +01:00
|
|
|
# ____________________________________________________________
|
|
|
|
# multi types
|
2013-04-03 12:20:26 +02:00
|
|
|
|
|
|
|
|
2013-02-08 11:50:22 +01:00
|
|
|
class Multi(list):
|
|
|
|
"""multi options values container
|
|
|
|
that support item notation for the values of multi options"""
|
2013-04-03 12:20:26 +02:00
|
|
|
__slots__ = ('opt', 'context')
|
|
|
|
|
|
|
|
def __init__(self, lst, context, opt):
|
2013-02-08 11:50:22 +01:00
|
|
|
"""
|
|
|
|
:param lst: the Multi wraps a list value
|
2013-04-03 12:20:26 +02:00
|
|
|
:param context: the home config that has the values
|
2013-02-08 11:50:22 +01:00
|
|
|
:param opt: the option object that have this Multi value
|
|
|
|
"""
|
|
|
|
self.opt = opt
|
2013-04-03 12:20:26 +02:00
|
|
|
self.context = context
|
2013-02-08 11:50:22 +01:00
|
|
|
super(Multi, self).__init__(lst)
|
2013-04-03 12:20:26 +02:00
|
|
|
|
2013-02-08 11:50:22 +01:00
|
|
|
def __setitem__(self, key, value):
|
2013-02-22 11:09:17 +01:00
|
|
|
self._validate(value)
|
2013-04-03 12:20:26 +02:00
|
|
|
self.context.cfgimpl_get_values()[self.opt] = self
|
2013-02-22 11:09:17 +01:00
|
|
|
super(Multi, self).__setitem__(key, value)
|
2013-02-08 11:50:22 +01:00
|
|
|
|
2013-02-22 11:09:17 +01:00
|
|
|
def append(self, value, force=False):
|
2013-02-08 11:50:22 +01:00
|
|
|
"""the list value can be updated (appened)
|
|
|
|
only if the option is a master
|
|
|
|
"""
|
2013-02-22 11:09:17 +01:00
|
|
|
if not force:
|
2013-04-03 12:20:26 +02:00
|
|
|
if self.opt.multitype == multitypes.slave:
|
2013-02-22 11:09:17 +01:00
|
|
|
raise MultiTypeError("cannot append a value on a multi option {0}"
|
2013-04-03 12:20:26 +02:00
|
|
|
" which is a slave".format(self.opt._name))
|
|
|
|
elif self.opt.multitype == multitypes.master:
|
|
|
|
for slave in self.opt.master_slaves:
|
|
|
|
self.context.cfgimpl_get_values()[slave].append(None, force=True)
|
2013-02-22 11:09:17 +01:00
|
|
|
self._validate(value)
|
2013-04-03 12:20:26 +02:00
|
|
|
self.context.cfgimpl_get_values().setitem(self.opt, self)
|
2013-02-22 11:09:17 +01:00
|
|
|
super(Multi, self).append(value)
|
|
|
|
|
|
|
|
def _validate(self, value):
|
2013-04-03 12:20:26 +02:00
|
|
|
if value is not None and not self.opt._validate(value):
|
2013-02-22 11:09:17 +01:00
|
|
|
raise ConfigError("invalid value {0} "
|
2013-04-03 12:20:26 +02:00
|
|
|
"for option {1}".format(str(value),
|
|
|
|
self.opt._name))
|
2013-02-22 11:09:17 +01:00
|
|
|
|
|
|
|
def pop(self, key, force=False):
|
2013-02-08 11:50:22 +01:00
|
|
|
"""the list value can be updated (poped)
|
|
|
|
only if the option is a master
|
|
|
|
|
|
|
|
:param key: index of the element to pop
|
|
|
|
:return: the requested element
|
|
|
|
"""
|
2013-02-22 11:09:17 +01:00
|
|
|
if not force:
|
2013-04-03 12:20:26 +02:00
|
|
|
if self.opt.multitype == multitypes.slave:
|
2013-02-22 11:09:17 +01:00
|
|
|
raise MultiTypeError("cannot append a value on a multi option {0}"
|
2013-04-03 12:20:26 +02:00
|
|
|
" which is a slave".format(self.opt._name))
|
|
|
|
elif self.opt.multitype == multitypes.master:
|
|
|
|
for slave in self.opt.master_slaves:
|
|
|
|
self.context.cfgimpl_get_values()[slave].pop(key, force=True)
|
|
|
|
self.context.cfgimpl_get_values().setitem(self.opt, self)
|
2013-02-08 11:50:22 +01:00
|
|
|
return super(Multi, self).pop(key)
|