mirror of
https://salsa.debian.org/freeipa-team/freeipa.git
synced 2025-01-11 00:31:56 -06:00
Implemented placeholder API.bootstrap() method; added API __doing(), __do_if_not_done(), isdone() methods borrowed from Env; API.finalize() now cascades call to API.bootstrap()
This commit is contained in:
parent
ff5cb4cf6f
commit
6b8abb0d78
@ -707,19 +707,40 @@ class API(DictProxy):
|
|||||||
"""
|
"""
|
||||||
Dynamic API object through which `Plugin` instances are accessed.
|
Dynamic API object through which `Plugin` instances are accessed.
|
||||||
"""
|
"""
|
||||||
__finalized = False
|
|
||||||
|
|
||||||
def __init__(self, *allowed):
|
def __init__(self, *allowed):
|
||||||
self.__d = dict()
|
self.__d = dict()
|
||||||
|
self.__done = set()
|
||||||
self.register = Registrar(*allowed)
|
self.register = Registrar(*allowed)
|
||||||
self.env = Environment()
|
self.env = Environment()
|
||||||
super(API, self).__init__(self.__d)
|
super(API, self).__init__(self.__d)
|
||||||
|
|
||||||
|
def __doing(self, name):
|
||||||
|
if name in self.__done:
|
||||||
|
raise StandardError(
|
||||||
|
'%s.%s() already called' % (self.__class__.__name__, name)
|
||||||
|
)
|
||||||
|
self.__done.add(name)
|
||||||
|
|
||||||
|
def __do_if_not_done(self, name):
|
||||||
|
if name not in self.__done:
|
||||||
|
getattr(self, name)()
|
||||||
|
|
||||||
|
def isdone(self, name):
|
||||||
|
return name in self.__done
|
||||||
|
|
||||||
|
def bootstrap(self, **overrides):
|
||||||
|
"""
|
||||||
|
Initialize environment variables needed by built-in plugins.
|
||||||
|
"""
|
||||||
|
self.__doing('bootstrap')
|
||||||
|
|
||||||
def finalize(self):
|
def finalize(self):
|
||||||
"""
|
"""
|
||||||
Finalize the registration, instantiate the plugins.
|
Finalize the registration, instantiate the plugins.
|
||||||
"""
|
"""
|
||||||
assert not self.__finalized, 'finalize() can only be called once'
|
self.__doing('finalize')
|
||||||
|
self.__do_if_not_done('bootstrap')
|
||||||
|
|
||||||
class PluginInstance(object):
|
class PluginInstance(object):
|
||||||
"""
|
"""
|
||||||
|
@ -764,99 +764,122 @@ def test_Registrar():
|
|||||||
assert issubclass(klass, base)
|
assert issubclass(klass, base)
|
||||||
|
|
||||||
|
|
||||||
def test_API():
|
class test_API(ClassChecker):
|
||||||
"""
|
"""
|
||||||
Test the `ipalib.plugable.API` class.
|
Test the `ipalib.plugable.API` class.
|
||||||
"""
|
"""
|
||||||
assert issubclass(plugable.API, plugable.ReadOnly)
|
|
||||||
|
|
||||||
# Setup the test bases, create the API:
|
_cls = plugable.API
|
||||||
class base0(plugable.Plugin):
|
|
||||||
__public__ = frozenset((
|
|
||||||
'method',
|
|
||||||
))
|
|
||||||
|
|
||||||
def method(self, n):
|
def test_API(self):
|
||||||
return n
|
"""
|
||||||
|
Test the `ipalib.plugable.API` class.
|
||||||
|
"""
|
||||||
|
assert issubclass(plugable.API, plugable.ReadOnly)
|
||||||
|
|
||||||
class base1(plugable.Plugin):
|
# Setup the test bases, create the API:
|
||||||
__public__ = frozenset((
|
class base0(plugable.Plugin):
|
||||||
'method',
|
__public__ = frozenset((
|
||||||
))
|
'method',
|
||||||
|
))
|
||||||
|
|
||||||
def method(self, n):
|
def method(self, n):
|
||||||
return n + 1
|
return n
|
||||||
|
|
||||||
api = plugable.API(base0, base1)
|
class base1(plugable.Plugin):
|
||||||
r = api.register
|
__public__ = frozenset((
|
||||||
assert isinstance(r, plugable.Registrar)
|
'method',
|
||||||
assert read_only(api, 'register') is r
|
))
|
||||||
|
|
||||||
class base0_plugin0(base0):
|
def method(self, n):
|
||||||
pass
|
return n + 1
|
||||||
r(base0_plugin0)
|
|
||||||
|
|
||||||
class base0_plugin1(base0):
|
api = plugable.API(base0, base1)
|
||||||
pass
|
r = api.register
|
||||||
r(base0_plugin1)
|
assert isinstance(r, plugable.Registrar)
|
||||||
|
assert read_only(api, 'register') is r
|
||||||
|
|
||||||
class base0_plugin2(base0):
|
class base0_plugin0(base0):
|
||||||
pass
|
pass
|
||||||
r(base0_plugin2)
|
r(base0_plugin0)
|
||||||
|
|
||||||
class base1_plugin0(base1):
|
class base0_plugin1(base0):
|
||||||
pass
|
pass
|
||||||
r(base1_plugin0)
|
r(base0_plugin1)
|
||||||
|
|
||||||
class base1_plugin1(base1):
|
class base0_plugin2(base0):
|
||||||
pass
|
pass
|
||||||
r(base1_plugin1)
|
r(base0_plugin2)
|
||||||
|
|
||||||
class base1_plugin2(base1):
|
class base1_plugin0(base1):
|
||||||
pass
|
pass
|
||||||
r(base1_plugin2)
|
r(base1_plugin0)
|
||||||
|
|
||||||
# Test API instance:
|
class base1_plugin1(base1):
|
||||||
api.finalize()
|
pass
|
||||||
|
r(base1_plugin1)
|
||||||
|
|
||||||
def get_base(b):
|
class base1_plugin2(base1):
|
||||||
return 'base%d' % b
|
pass
|
||||||
|
r(base1_plugin2)
|
||||||
|
|
||||||
def get_plugin(b, p):
|
# Test API instance:
|
||||||
return 'base%d_plugin%d' % (b, p)
|
assert api.isdone('bootstrap') is False
|
||||||
|
assert api.isdone('finalize') is False
|
||||||
|
api.finalize()
|
||||||
|
assert api.isdone('bootstrap') is True
|
||||||
|
assert api.isdone('finalize') is True
|
||||||
|
|
||||||
for b in xrange(2):
|
def get_base(b):
|
||||||
base_name = get_base(b)
|
return 'base%d' % b
|
||||||
ns = getattr(api, base_name)
|
|
||||||
assert isinstance(ns, plugable.NameSpace)
|
|
||||||
assert read_only(api, base_name) is ns
|
|
||||||
assert len(ns) == 3
|
|
||||||
for p in xrange(3):
|
|
||||||
plugin_name = get_plugin(b, p)
|
|
||||||
proxy = ns[plugin_name]
|
|
||||||
assert isinstance(proxy, plugable.PluginProxy)
|
|
||||||
assert proxy.name == plugin_name
|
|
||||||
assert read_only(ns, plugin_name) is proxy
|
|
||||||
assert read_only(proxy, 'method')(7) == 7 + b
|
|
||||||
|
|
||||||
# Test that calling finilize again raises AssertionError:
|
def get_plugin(b, p):
|
||||||
raises(AssertionError, api.finalize)
|
return 'base%d_plugin%d' % (b, p)
|
||||||
|
|
||||||
# Test with base class that doesn't request a proxy
|
for b in xrange(2):
|
||||||
class NoProxy(plugable.Plugin):
|
base_name = get_base(b)
|
||||||
__proxy__ = False
|
ns = getattr(api, base_name)
|
||||||
api = plugable.API(NoProxy)
|
assert isinstance(ns, plugable.NameSpace)
|
||||||
class plugin0(NoProxy):
|
assert read_only(api, base_name) is ns
|
||||||
pass
|
assert len(ns) == 3
|
||||||
api.register(plugin0)
|
for p in xrange(3):
|
||||||
class plugin1(NoProxy):
|
plugin_name = get_plugin(b, p)
|
||||||
pass
|
proxy = ns[plugin_name]
|
||||||
api.register(plugin1)
|
assert isinstance(proxy, plugable.PluginProxy)
|
||||||
api.finalize()
|
assert proxy.name == plugin_name
|
||||||
names = ['plugin0', 'plugin1']
|
assert read_only(ns, plugin_name) is proxy
|
||||||
assert list(api.NoProxy) == names
|
assert read_only(proxy, 'method')(7) == 7 + b
|
||||||
for name in names:
|
|
||||||
plugin = api.NoProxy[name]
|
# Test that calling finilize again raises AssertionError:
|
||||||
assert getattr(api.NoProxy, name) is plugin
|
e = raises(StandardError, api.finalize)
|
||||||
assert isinstance(plugin, plugable.Plugin)
|
assert str(e) == 'API.finalize() already called', str(e)
|
||||||
assert not isinstance(plugin, plugable.PluginProxy)
|
|
||||||
|
# Test with base class that doesn't request a proxy
|
||||||
|
class NoProxy(plugable.Plugin):
|
||||||
|
__proxy__ = False
|
||||||
|
api = plugable.API(NoProxy)
|
||||||
|
class plugin0(NoProxy):
|
||||||
|
pass
|
||||||
|
api.register(plugin0)
|
||||||
|
class plugin1(NoProxy):
|
||||||
|
pass
|
||||||
|
api.register(plugin1)
|
||||||
|
api.finalize()
|
||||||
|
names = ['plugin0', 'plugin1']
|
||||||
|
assert list(api.NoProxy) == names
|
||||||
|
for name in names:
|
||||||
|
plugin = api.NoProxy[name]
|
||||||
|
assert getattr(api.NoProxy, name) is plugin
|
||||||
|
assert isinstance(plugin, plugable.Plugin)
|
||||||
|
assert not isinstance(plugin, plugable.PluginProxy)
|
||||||
|
|
||||||
|
def test_bootstrap(self):
|
||||||
|
"""
|
||||||
|
Test the `ipalib.plugable.API.bootstrap` method.
|
||||||
|
"""
|
||||||
|
o = self.cls()
|
||||||
|
assert o.isdone('bootstrap') is False
|
||||||
|
o.bootstrap()
|
||||||
|
assert o.isdone('bootstrap') is True
|
||||||
|
e = raises(StandardError, o.bootstrap)
|
||||||
|
assert str(e) == 'API.bootstrap() already called'
|
||||||
|
Loading…
Reference in New Issue
Block a user