diff options
-rw-r--r-- | ipalib/config.py | 120 | ||||
-rw-r--r-- | ipalib/constants.py | 4 | ||||
-rw-r--r-- | tests/test_ipalib/test_config.py | 64 |
3 files changed, 137 insertions, 51 deletions
diff --git a/ipalib/config.py b/ipalib/config.py index 7f12b4256..4631d8996 100644 --- a/ipalib/config.py +++ b/ipalib/config.py @@ -31,13 +31,95 @@ import os from os import path import sys from constants import CONFIG_SECTION -from constants import TYPE_ERROR, OVERRIDE_ERROR, LOCK_ERROR - +from constants import TYPE_ERROR, OVERRIDE_ERROR, SET_ERROR, DEL_ERROR class Env(object): """ - A mapping object used to store the environment variables. + Store and retrieve environment variables. + + First an foremost, the `Env` class provides a handy container for + environment variables. These variables can be both set and retrieved as + either attributes or as dictionary items. + + For example, we can set a variable as an attribute: + + >>> env = Env() + >>> env.attr = 'I was set as an attribute.' + >>> env.attr # Retrieve as an attribute + 'I was set as an attribute.' + >>> env['attr'] # Also retrieve as a dictionary item + 'I was set as an attribute.' + + Or we can set a variable as a dictionary item: + + >>> env['item'] = 'I was set as a dictionary item.' + >>> env['item'] # Retrieve as a dictionary item + 'I was set as a dictionary item.' + >>> env.item # Also retrieve as an attribute + 'I was set as a dictionary item.' + + The variable values can be ``str`` or ``int`` instances, or the ``True``, + ``False``, or ``None`` constants. When the value provided is an ``str`` + instance, some limited automatic type conversion is performed, which allows + values of specific types to be set easily from configuration files and from + command-line options. + + The ``True``, ``False``, and ``None`` constants can be specified with a + string that matches what ``repr()`` would return. For example: + + >>> env.true = 'True' + >>> env.true + True + + Note that the automatic type conversion is case sensitive. For example: + + >>> env.false = 'false' # Doesn't match repr(False) + >>> env.false + 'false' + + If an ``str`` value looks like an integer, it's automatically converted to + the ``int`` type. For example: + + >>> env.lucky = '7' + >>> env.lucky + 7 + + Also, leading and trailing white-space is automatically stripped from + ``str`` values. For example: + + >>> env.message = ' Hello! ' # Surrounded by double spaces + >>> env.message + 'Hello!' + >>> env.number = '42 ' # Still converted to an int + >>> env.number + 42 + >>> env.actually_false = ' False' # Still matches repr(False) + >>> env.actually_false + False + + `Env` is set-once, first-one-wins. Once a variable has been set, trying to + override it will raise an ``AttributeError``. For example: + + >>> env.my_var = 'first' + >>> env.my_var = 'second' + Traceback (most recent call last): + ... + AttributeError: cannot override Env.my_var value 'first' with 'second' + + An `Env` instance can also be *locked*, after which no further variables can + be set. Trying to set variables on a locked `Env` instance will also raise + an ``AttributeError``. For example: + + >>> env = Env() + >>> env.var1 = 'This will work.' + >>> env.__lock__() + >>> env.var2 = 'This wont work!' + Traceback (most recent call last): + ... + AttributeError: locked: cannot set Env.var2 to 'This wont work!' + + Finish me! """ __locked = False @@ -67,12 +149,16 @@ class Env(object): # FIXME: the key should be checked with check_name() if self.__locked: raise AttributeError( - LOCK_ERROR % (self.__class__.__name__, key, value) + SET_ERROR % (self.__class__.__name__, key, value) ) if key in self.__d: raise AttributeError(OVERRIDE_ERROR % (self.__class__.__name__, key, self.__d[key], value) ) + if hasattr(self, key): + raise AttributeError(OVERRIDE_ERROR % + (self.__class__.__name__, key, getattr(self, key), value) + ) if isinstance(value, basestring): value = str(value.strip()) m = { @@ -88,6 +174,20 @@ class Env(object): object.__setattr__(self, key, value) self.__d[key] = value + def __getitem__(self, key): + """ + Return the value corresponding to ``key``. + """ + return self.__d[key] + + def __delattr__(self, name): + """ + Raise AttributeError (deletion is never allowed). + """ + raise AttributeError( + DEL_ERROR % (self.__class__.__name__, name) + ) + def __doing(self, name): if name in self.__done: raise StandardError( @@ -214,19 +314,7 @@ class Env(object): - def __delattr__(self, name): - """ - Raise AttributeError (deletion is not allowed). - """ - raise AttributeError('cannot del %s.%s' % - (self.__class__.__name__, name) - ) - def __getitem__(self, key): - """ - Return the value corresponding to ``key``. - """ - return self.__d[key] diff --git a/ipalib/constants.py b/ipalib/constants.py index c74808d6d..dc23b1091 100644 --- a/ipalib/constants.py +++ b/ipalib/constants.py @@ -36,8 +36,8 @@ OVERRIDE_ERROR = 'cannot override %s.%s value %r with %r' # Standard format for AttributeError message when a read-only attribute is # already locked: -LOCK_ERROR = 'locked: cannot set %s.%s to %r' -DEL_ERROR = 'locked: cannot set %s.%s to %r' +SET_ERROR = 'locked: cannot set %s.%s to %r' +DEL_ERROR = 'locked: cannot del %s.%s' # Used for a tab (or indentation level) when formatting for CLI: CLI_TAB = ' ' # Two spaces diff --git a/tests/test_ipalib/test_config.py b/tests/test_ipalib/test_config.py index c1a5faaa0..740ef6cf6 100644 --- a/tests/test_ipalib/test_config.py +++ b/tests/test_ipalib/test_config.py @@ -28,7 +28,7 @@ import sys from tests.util import raises, setitem, delitem, ClassChecker from tests.util import getitem, setitem, delitem from tests.util import TempDir, TempHome -from ipalib.constants import TYPE_ERROR, OVERRIDE_ERROR, LOCK_ERROR +from ipalib.constants import TYPE_ERROR, OVERRIDE_ERROR, SET_ERROR, DEL_ERROR from ipalib import config, constants @@ -177,7 +177,7 @@ class test_Env(ClassChecker): o.__lock__() for (name, raw, value) in good_vars: e = raises(AttributeError, setattr, o, name, raw) - assert str(e) == LOCK_ERROR % ('Env', name, raw) + assert str(e) == SET_ERROR % ('Env', name, raw) def test_setitem(self): """ @@ -201,7 +201,35 @@ class test_Env(ClassChecker): o.__lock__() for (key, raw, value) in good_vars: e = raises(AttributeError, o.__setitem__, key, raw) - assert str(e) == LOCK_ERROR % ('Env', key, raw) + assert str(e) == SET_ERROR % ('Env', key, raw) + + def test_getitem(self): + """ + Test the `ipalib.config.Env.__getitem__` method. + """ + o = self.cls() + value = 'some value' + o.key = value + assert o.key is value + assert o['key'] is value + for name in ('one', 'two'): + e = raises(KeyError, getitem, o, name) + assert str(e) == repr(name) + + def test_delattr(self): + """ + Test the `ipalib.config.Env.__delattr__` method. + + This also tests that ``__delitem__`` is not implemented. + """ + o = self.cls() + o.one = 1 + assert o.one == 1 + for key in ('one', 'two'): + e = raises(AttributeError, delattr, o, key) + assert str(e) == DEL_ERROR % ('Env', key) + e = raises(AttributeError, delitem, o, key) + assert str(e) == '__delitem__' def bootstrap(self, **overrides): (o, home) = self.new() @@ -445,36 +473,6 @@ class test_Env(ClassChecker): e = raises(StandardError, o.__lock__) assert str(e) == 'Env.__lock__() already called' - def test_getitem(self): - """ - Test the `ipalib.config.Env.__getitem__` method. - """ - o = self.cls() - value = 'some value' - o.key = value - assert o.key is value - assert o['key'] is value - for name in ('one', 'two'): - e = raises(KeyError, getitem, o, name) - assert str(e) == repr(name) - - - - def test_delattr(self): - """ - Test the `ipalib.config.Env.__delattr__` method. - - This also tests that ``__delitem__`` is not implemented. - """ - o = self.cls() - o.one = 1 - assert o.one == 1 - for key in ('one', 'two'): - e = raises(AttributeError, delattr, o, key) - assert str(e) == 'cannot del Env.%s' % key - e = raises(AttributeError, delitem, o, key) - assert str(e) == '__delitem__' - def test_contains(self): """ Test the `ipalib.config.Env.__contains__` method. |