2008-07-27 23:34:25 -05:00
|
|
|
# Authors:
|
|
|
|
# Jason Gerard DeRose <jderose@redhat.com>
|
|
|
|
#
|
|
|
|
# Copyright (C) 2008 Red Hat
|
|
|
|
# see file 'COPYING' for use and warranty information
|
|
|
|
#
|
|
|
|
# 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; version 2 only
|
|
|
|
#
|
|
|
|
# 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
|
|
|
|
|
|
|
|
"""
|
2008-08-06 09:22:38 -05:00
|
|
|
Base classes for plug-in architecture and generative API.
|
2008-07-27 23:34:25 -05:00
|
|
|
"""
|
|
|
|
|
2008-08-05 01:33:09 -05:00
|
|
|
import re
|
2008-07-27 23:34:25 -05:00
|
|
|
import inspect
|
2008-07-31 13:57:10 -05:00
|
|
|
import errors
|
2008-07-27 23:34:25 -05:00
|
|
|
|
|
|
|
|
2008-08-08 16:49:09 -05:00
|
|
|
class ReadOnly(object):
|
|
|
|
"""
|
|
|
|
Base class for classes with read-only attributes.
|
2008-08-08 23:35:06 -05:00
|
|
|
|
|
|
|
Be forewarned that Python does not offer true read-only user defined
|
|
|
|
classes. In particular, do not rely upon the read-only-ness of this
|
|
|
|
class for security purposes.
|
|
|
|
|
|
|
|
The point of this class is not to make it impossible to set or delete
|
2008-08-08 23:37:37 -05:00
|
|
|
attributes, but to make it impossible to accidentally do so. The plugins
|
2008-08-08 23:35:06 -05:00
|
|
|
are not thread-safe: in the server, they are loaded once and the same
|
|
|
|
instances will be used to process many requests. Therefore, it is
|
|
|
|
imperative that they not set any instance attributes after they have
|
|
|
|
been initialized. This base class enforces that policy.
|
|
|
|
|
|
|
|
For example:
|
|
|
|
|
2008-08-12 18:45:36 -05:00
|
|
|
>>> class givenname(ReadOnly):
|
2008-08-08 23:35:06 -05:00
|
|
|
>>> def __init__(self):
|
|
|
|
>>> self.whatever = 'some value' # Hasn't been locked yet
|
|
|
|
>>> self.__lock__()
|
|
|
|
>>>
|
|
|
|
>>> def finalize(self, api):
|
|
|
|
>>> # After the instance has been locked, attributes can still be
|
|
|
|
>>> # set, but only in a round-about, unconventional way:
|
|
|
|
>>> object.__setattr__(self, 'api', api)
|
|
|
|
>>>
|
|
|
|
>>> def normalize(self, value):
|
|
|
|
>>> # After the instance has been locked, trying to set an
|
|
|
|
>>> # attribute in the normal way will raise AttributeError.
|
|
|
|
>>> self.value = value # Not thread safe!
|
|
|
|
>>> return self.actually_normalize()
|
|
|
|
>>>
|
|
|
|
>>> def actually_normalize(self):
|
|
|
|
>>> # Again, this is not thread safe:
|
|
|
|
>>> return unicode(self.value).strip()
|
2008-08-08 16:49:09 -05:00
|
|
|
"""
|
|
|
|
__locked = False
|
|
|
|
|
|
|
|
def __lock__(self):
|
2008-08-08 17:45:09 -05:00
|
|
|
"""
|
|
|
|
Puts this instance into a read-only state, after which attempting to
|
|
|
|
set or delete an attribute will raise AttributeError.
|
|
|
|
"""
|
|
|
|
assert self.__locked is False, '__lock__() can only be called once'
|
2008-08-08 16:49:09 -05:00
|
|
|
self.__locked = True
|
|
|
|
|
2008-08-08 23:35:06 -05:00
|
|
|
def __islocked__(self):
|
|
|
|
"""
|
|
|
|
Returns True if this instance is locked, False otherwise.
|
|
|
|
"""
|
|
|
|
return self.__locked
|
|
|
|
|
2008-08-08 16:49:09 -05:00
|
|
|
def __setattr__(self, name, value):
|
|
|
|
"""
|
|
|
|
Raises an AttributeError if ReadOnly.__lock__() has already been called;
|
|
|
|
otherwise calls object.__setattr__()
|
|
|
|
"""
|
|
|
|
if self.__locked:
|
|
|
|
raise AttributeError('read-only: cannot set %s.%s' %
|
|
|
|
(self.__class__.__name__, name)
|
|
|
|
)
|
|
|
|
return object.__setattr__(self, name, value)
|
|
|
|
|
|
|
|
def __delattr__(self, name):
|
|
|
|
"""
|
|
|
|
Raises an AttributeError if ReadOnly.__lock__() has already been called;
|
|
|
|
otherwise calls object.__delattr__()
|
|
|
|
"""
|
|
|
|
if self.__locked:
|
|
|
|
raise AttributeError('read-only: cannot del %s.%s' %
|
|
|
|
(self.__class__.__name__, name)
|
|
|
|
)
|
|
|
|
return object.__delattr__(self, name)
|
|
|
|
|
|
|
|
|
2008-08-14 02:10:07 -05:00
|
|
|
class Plugin(ReadOnly):
|
|
|
|
"""
|
|
|
|
Base class for all plugins.
|
|
|
|
"""
|
|
|
|
__public__ = frozenset()
|
|
|
|
__api = None
|
|
|
|
|
|
|
|
def __get_name(self):
|
|
|
|
"""
|
|
|
|
Convenience property to return the class name.
|
|
|
|
"""
|
|
|
|
return self.__class__.__name__
|
|
|
|
name = property(__get_name)
|
|
|
|
|
|
|
|
def __get_doc(self):
|
|
|
|
"""
|
|
|
|
Convenience property to return the class docstring.
|
|
|
|
"""
|
|
|
|
return self.__class__.__doc__
|
|
|
|
doc = property(__get_doc)
|
|
|
|
|
|
|
|
def __get_api(self):
|
|
|
|
"""
|
|
|
|
Returns the `API` instance passed to `finalize`, or
|
|
|
|
or returns None if `finalize` has not yet been called.
|
|
|
|
"""
|
|
|
|
return self.__api
|
|
|
|
api = property(__get_api)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def implements(cls, arg):
|
|
|
|
"""
|
|
|
|
Returns True if this cls.__public__ frozenset contains `arg`;
|
|
|
|
returns False otherwise.
|
|
|
|
|
|
|
|
There are three different ways this can be called:
|
|
|
|
|
|
|
|
1. With a <type 'str'> argument, e.g.:
|
|
|
|
|
|
|
|
>>> class base(ProxyTarget):
|
|
|
|
>>> __public__ = frozenset(['some_attr', 'another_attr'])
|
|
|
|
>>> base.implements('some_attr')
|
|
|
|
True
|
|
|
|
>>> base.implements('an_unknown_attribute')
|
|
|
|
False
|
|
|
|
|
|
|
|
2. With a <type 'frozenset'> argument, e.g.:
|
|
|
|
|
|
|
|
>>> base.implements(frozenset(['some_attr']))
|
|
|
|
True
|
|
|
|
>>> base.implements(frozenset(['some_attr', 'an_unknown_attribute']))
|
|
|
|
False
|
|
|
|
|
|
|
|
3. With any object that has a `__public__` attribute that is
|
|
|
|
<type 'frozenset'>, e.g.:
|
|
|
|
|
|
|
|
>>> class whatever(object):
|
|
|
|
>>> __public__ = frozenset(['another_attr'])
|
|
|
|
>>> base.implements(whatever)
|
|
|
|
True
|
|
|
|
|
|
|
|
Unlike ProxyTarget.implemented_by(), this returns an abstract answer
|
|
|
|
because only the __public__ frozenset is checked... a ProxyTarget
|
|
|
|
need not itself have attributes for all names in __public__
|
|
|
|
(subclasses might provide them).
|
|
|
|
"""
|
|
|
|
assert type(cls.__public__) is frozenset
|
|
|
|
if isinstance(arg, str):
|
|
|
|
return arg in cls.__public__
|
|
|
|
if type(getattr(arg, '__public__', None)) is frozenset:
|
|
|
|
return cls.__public__.issuperset(arg.__public__)
|
|
|
|
if type(arg) is frozenset:
|
|
|
|
return cls.__public__.issuperset(arg)
|
|
|
|
raise TypeError(
|
|
|
|
"must be str, frozenset, or have frozenset '__public__' attribute"
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def implemented_by(cls, arg):
|
|
|
|
"""
|
|
|
|
Returns True if (1) `arg` is an instance of or subclass of this class,
|
|
|
|
and (2) `arg` (or `arg.__class__` if instance) has an attribute for
|
|
|
|
each name in this class's __public__ frozenset; returns False
|
|
|
|
otherwise.
|
|
|
|
|
|
|
|
Unlike ProxyTarget.implements(), this returns a concrete answer
|
|
|
|
because the attributes of the subclass are checked.
|
|
|
|
"""
|
|
|
|
if inspect.isclass(arg):
|
|
|
|
subclass = arg
|
|
|
|
else:
|
|
|
|
subclass = arg.__class__
|
|
|
|
assert issubclass(subclass, cls), 'must be subclass of %r' % cls
|
|
|
|
for name in cls.__public__:
|
|
|
|
if not hasattr(subclass, name):
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
def finalize(self, api):
|
|
|
|
"""
|
|
|
|
After all the plugins are instantiated, `API` calls this method,
|
|
|
|
passing itself as the only argument. This is where plugins should
|
|
|
|
check that other plugins they depend upon have actually been loaded.
|
|
|
|
|
|
|
|
:param api: An `API` instance.
|
|
|
|
"""
|
|
|
|
assert self.__api is None, 'finalize() can only be called once'
|
|
|
|
assert api is not None, 'finalize() argument cannot be None'
|
|
|
|
self.__api = api
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
"""
|
|
|
|
Returns a fully qualified module_name.class_name() representation that
|
|
|
|
could be used to construct this Plugin instance.
|
|
|
|
"""
|
|
|
|
return '%s.%s()' % (
|
|
|
|
self.__class__.__module__,
|
|
|
|
self.__class__.__name__
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2008-07-31 22:12:17 -05:00
|
|
|
class Proxy(ReadOnly):
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-14 02:10:07 -05:00
|
|
|
Allows access to only certain attributes on a `Plugin`.
|
2008-08-08 18:07:22 -05:00
|
|
|
|
2008-08-08 20:06:42 -05:00
|
|
|
Think of a proxy as an agreement that "I will have at most these
|
2008-08-08 18:07:22 -05:00
|
|
|
attributes". This is different from (although similar to) an interface,
|
|
|
|
which can be thought of as an agreement that "I will have at least these
|
|
|
|
attributes".
|
|
|
|
"""
|
2008-08-08 12:11:29 -05:00
|
|
|
__slots__ = (
|
2008-08-08 16:40:03 -05:00
|
|
|
'__base',
|
|
|
|
'__target',
|
|
|
|
'__name_attr',
|
|
|
|
'__public__',
|
2008-08-09 00:19:40 -05:00
|
|
|
'name',
|
2008-08-12 21:34:36 -05:00
|
|
|
'doc',
|
2008-08-08 12:11:29 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
def __init__(self, base, target, name_attr='name'):
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-14 01:53:05 -05:00
|
|
|
:param base: A subclass of `Plugin`.
|
|
|
|
:param target: An instance ``base`` or a subclass of ``base``.
|
|
|
|
:param name_attr: The name of the attribute on ``target`` from which
|
|
|
|
to derive ``self.name``.
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-08 16:40:03 -05:00
|
|
|
if not inspect.isclass(base):
|
2008-08-14 01:53:05 -05:00
|
|
|
raise TypeError(
|
|
|
|
'`base` must be a class, got %r' % base
|
|
|
|
)
|
2008-08-08 16:40:03 -05:00
|
|
|
if not isinstance(target, base):
|
2008-08-14 01:53:05 -05:00
|
|
|
raise ValueError(
|
|
|
|
'`target` must be an instance of `base`, got %r' % target
|
|
|
|
)
|
2008-08-08 16:28:56 -05:00
|
|
|
self.__base = base
|
2008-08-08 16:40:03 -05:00
|
|
|
self.__target = target
|
|
|
|
self.__name_attr = name_attr
|
|
|
|
self.__public__ = base.__public__
|
2008-08-09 00:19:40 -05:00
|
|
|
self.name = getattr(target, name_attr)
|
2008-08-12 21:34:36 -05:00
|
|
|
self.doc = target.doc
|
2008-08-08 16:49:09 -05:00
|
|
|
self.__lock__()
|
2008-08-12 21:34:36 -05:00
|
|
|
assert type(self.__public__) is frozenset
|
2008-08-08 16:28:56 -05:00
|
|
|
|
2008-08-08 20:06:42 -05:00
|
|
|
def implements(self, arg):
|
2008-08-09 13:58:46 -05:00
|
|
|
"""
|
|
|
|
Returns True if this proxy implements `arg`. Calls the corresponding
|
|
|
|
classmethod on ProxyTarget.
|
|
|
|
|
|
|
|
Unlike ProxyTarget.implements(), this is not a classmethod as a Proxy
|
|
|
|
only implements anything as an instance.
|
|
|
|
"""
|
2008-08-08 20:06:42 -05:00
|
|
|
return self.__base.implements(arg)
|
|
|
|
|
2008-08-08 20:46:12 -05:00
|
|
|
def __clone__(self, name_attr):
|
|
|
|
"""
|
|
|
|
Returns a Proxy instance identical to this one except the proxy name
|
|
|
|
might be derived from a different attribute on the target. The same
|
|
|
|
base and target will be used.
|
|
|
|
"""
|
|
|
|
return self.__class__(self.__base, self.__target, name_attr)
|
|
|
|
|
2008-08-08 12:11:29 -05:00
|
|
|
def __iter__(self):
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-08 20:06:42 -05:00
|
|
|
Iterates (in ascending order) though the attribute names this proxy is
|
|
|
|
allowing access to.
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-08 16:40:03 -05:00
|
|
|
for name in sorted(self.__public__):
|
|
|
|
yield name
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __getitem__(self, key):
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-09 13:58:46 -05:00
|
|
|
If this proxy allows access to an attribute named `key`, return that
|
|
|
|
attribute.
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-08 16:40:03 -05:00
|
|
|
if key in self.__public__:
|
|
|
|
return getattr(self.__target, key)
|
|
|
|
raise KeyError('no proxy attribute %r' % key)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __getattr__(self, name):
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-09 13:58:46 -05:00
|
|
|
If this proxy allows access to an attribute named `name`, return that
|
|
|
|
attribute.
|
2008-08-08 18:07:22 -05:00
|
|
|
"""
|
2008-08-08 16:40:03 -05:00
|
|
|
if name in self.__public__:
|
|
|
|
return getattr(self.__target, name)
|
|
|
|
raise AttributeError('no proxy attribute %r' % name)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __call__(self, *args, **kw):
|
2008-08-09 13:58:46 -05:00
|
|
|
"""
|
|
|
|
Attempts to call target.__call__(); raises KeyError if `__call__` is
|
|
|
|
not an attribute this proxy allows access to.
|
|
|
|
"""
|
2008-08-08 16:40:03 -05:00
|
|
|
return self['__call__'](*args, **kw)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __repr__(self):
|
2008-08-09 13:58:46 -05:00
|
|
|
"""
|
|
|
|
Returns a Python expression that could be used to construct this Proxy
|
|
|
|
instance given the appropriate environment.
|
|
|
|
"""
|
2008-08-08 16:40:03 -05:00
|
|
|
return '%s(%s, %r, %r)' % (
|
|
|
|
self.__class__.__name__,
|
|
|
|
self.__base.__name__,
|
|
|
|
self.__target,
|
|
|
|
self.__name_attr,
|
|
|
|
)
|
2008-08-06 19:14:38 -05:00
|
|
|
|
2008-08-06 15:38:07 -05:00
|
|
|
|
2008-08-08 17:45:09 -05:00
|
|
|
|
|
|
|
|
2008-08-14 00:46:20 -05:00
|
|
|
def check_name(name):
|
2008-08-08 12:11:29 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
Raises errors.NameSpaceError if `name` is not a valid Python identifier
|
|
|
|
suitable for use in a NameSpace.
|
|
|
|
"""
|
|
|
|
assert type(name) is str, 'must be %r' % str
|
|
|
|
regex = r'^[a-z][_a-z0-9]*[a-z0-9]$'
|
|
|
|
if re.match(regex, name) is None:
|
|
|
|
raise errors.NameSpaceError(name, regex)
|
|
|
|
return name
|
2008-08-09 14:28:01 -05:00
|
|
|
|
2008-08-14 00:46:20 -05:00
|
|
|
|
|
|
|
class NameSpace(ReadOnly):
|
|
|
|
"""
|
|
|
|
A read-only namespace with handy container behaviours.
|
|
|
|
|
|
|
|
Each member of a NameSpace instance must have a `name` attribute whose
|
|
|
|
value 1) is unique among the members and 2) passes the check_name()
|
|
|
|
function. Beyond that, no restrictions are placed on the members: they can
|
|
|
|
be classes or instances, and of any type.
|
|
|
|
|
|
|
|
The members can be accessed as attributes on the NameSpace instance or
|
|
|
|
through a dictionary interface. For example, assuming `obj` is a member in
|
|
|
|
the NameSpace instance `namespace`:
|
|
|
|
|
|
|
|
>>> obj is getattr(namespace, obj.name) # As attribute
|
|
|
|
True
|
|
|
|
>>> obj is namespace[obj.name] # As dictionary item
|
|
|
|
True
|
|
|
|
|
|
|
|
Here is a more detailed example:
|
|
|
|
|
|
|
|
>>> class member(object):
|
|
|
|
... def __init__(self, i):
|
|
|
|
... self.name = 'member_%d' % i
|
|
|
|
...
|
|
|
|
>>> def get_members(cnt):
|
|
|
|
... for i in xrange(cnt):
|
|
|
|
... yield member(i)
|
|
|
|
...
|
|
|
|
>>> namespace = NameSpace(get_members(2))
|
|
|
|
>>> namespace.member_0 is namespace['member_0']
|
|
|
|
True
|
|
|
|
>>> len(namespace) # Returns the number of members in namespace
|
|
|
|
2
|
|
|
|
>>> list(namespace) # As iterable, iterates through the member names
|
|
|
|
['member_0', 'member_1']
|
|
|
|
>>> list(namespace()) # Calling a NameSpace iterates through the members
|
|
|
|
[<__main__.member object at 0x836710>, <__main__.member object at 0x836750>]
|
|
|
|
>>> 'member_1' in namespace # NameSpace.__contains__()
|
|
|
|
True
|
2008-08-08 12:11:29 -05:00
|
|
|
"""
|
|
|
|
|
2008-08-14 00:46:20 -05:00
|
|
|
def __init__(self, members):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
@type members: iterable
|
|
|
|
@param members: An iterable providing the members.
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
self.__d = dict()
|
2008-08-14 00:46:20 -05:00
|
|
|
self.__names = tuple(self.__member_iter(members))
|
2008-08-08 16:49:09 -05:00
|
|
|
self.__lock__()
|
2008-08-14 00:46:20 -05:00
|
|
|
assert set(self.__d) == set(self.__names)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
2008-08-14 00:46:20 -05:00
|
|
|
def __member_iter(self, members):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
Helper method used only from __init__().
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
for member in members:
|
|
|
|
name = check_name(member.name)
|
|
|
|
assert not (
|
|
|
|
name in self.__d or hasattr(self, name)
|
|
|
|
), 'already has member named %r' % name
|
|
|
|
self.__d[name] = member
|
|
|
|
setattr(self, name, member)
|
|
|
|
yield name
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __len__(self):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
Returns the number of members in this NameSpace.
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
return len(self.__d)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
2008-08-14 00:46:20 -05:00
|
|
|
def __contains__(self, name):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
Returns True if this NameSpace contains a member named `name`; returns
|
|
|
|
False otherwise.
|
|
|
|
|
|
|
|
@type name: str
|
|
|
|
@param name: The name of a potential member
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
return name in self.__d
|
2008-08-08 12:11:29 -05:00
|
|
|
|
2008-08-14 00:46:20 -05:00
|
|
|
def __getitem__(self, name):
|
|
|
|
"""
|
|
|
|
If this NameSpace contains a member named `name`, returns that member;
|
|
|
|
otherwise raises KeyError.
|
|
|
|
|
|
|
|
@type name: str
|
|
|
|
@param name: The name of member to retrieve
|
|
|
|
"""
|
|
|
|
if name in self.__d:
|
|
|
|
return self.__d[name]
|
|
|
|
raise KeyError('NameSpace has no member named %r' % name)
|
|
|
|
|
|
|
|
def __iter__(self):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
Iterates through the member names in the same order as the members
|
|
|
|
were passed to the constructor.
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
2008-08-14 00:46:20 -05:00
|
|
|
for name in self.__names:
|
|
|
|
yield name
|
|
|
|
|
|
|
|
def __call__(self):
|
|
|
|
"""
|
|
|
|
Iterates through the members in the same order they were passed to the
|
|
|
|
constructor.
|
|
|
|
"""
|
|
|
|
for name in self.__names:
|
|
|
|
yield self.__d[name]
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __repr__(self):
|
2008-08-14 00:46:20 -05:00
|
|
|
"""
|
|
|
|
Returns pseudo-valid Python expression that could be used to construct
|
|
|
|
this NameSpace instance.
|
|
|
|
"""
|
|
|
|
return '%s(<%d members>)' % (self.__class__.__name__, len(self))
|
2008-08-06 17:59:50 -05:00
|
|
|
|
|
|
|
|
2008-08-09 14:33:13 -05:00
|
|
|
class Registrar(ReadOnly):
|
2008-08-08 12:11:29 -05:00
|
|
|
def __init__(self, *allowed):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
`*allowed` is a list of the base classes plugins can be subclassed
|
|
|
|
from.
|
|
|
|
"""
|
|
|
|
self.__allowed = frozenset(allowed)
|
|
|
|
self.__d = {}
|
|
|
|
self.__registered = set()
|
|
|
|
assert len(self.__allowed) == len(allowed)
|
|
|
|
for base in self.__allowed:
|
|
|
|
assert inspect.isclass(base)
|
|
|
|
assert base.__name__ not in self.__d
|
|
|
|
self.__d[base.__name__] = {}
|
2008-08-09 14:33:13 -05:00
|
|
|
self.__lock__()
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __findbase(self, cls):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
If `cls` is a subclass of a base in self.__allowed, returns that
|
|
|
|
base; otherwise raises SubclassError.
|
|
|
|
"""
|
|
|
|
assert inspect.isclass(cls)
|
|
|
|
found = False
|
|
|
|
for base in self.__allowed:
|
|
|
|
if issubclass(cls, base):
|
|
|
|
found = True
|
|
|
|
yield base
|
|
|
|
if not found:
|
|
|
|
raise errors.SubclassError(cls, self.__allowed)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __call__(self, cls, override=False):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
Register the plugin `cls`.
|
|
|
|
"""
|
|
|
|
if not inspect.isclass(cls):
|
|
|
|
raise TypeError('plugin must be a class: %r' % cls)
|
|
|
|
|
|
|
|
# Raise DuplicateError if this exact class was already registered:
|
|
|
|
if cls in self.__registered:
|
|
|
|
raise errors.DuplicateError(cls)
|
|
|
|
|
|
|
|
# Find the base class or raise SubclassError:
|
|
|
|
for base in self.__findbase(cls):
|
|
|
|
sub_d = self.__d[base.__name__]
|
|
|
|
|
|
|
|
# Check override:
|
|
|
|
if cls.__name__ in sub_d:
|
|
|
|
# Must use override=True to override:
|
|
|
|
if not override:
|
|
|
|
raise errors.OverrideError(base, cls)
|
|
|
|
else:
|
|
|
|
# There was nothing already registered to override:
|
|
|
|
if override:
|
|
|
|
raise errors.MissingOverrideError(base, cls)
|
|
|
|
|
|
|
|
# The plugin is okay, add to sub_d:
|
|
|
|
sub_d[cls.__name__] = cls
|
|
|
|
|
|
|
|
# The plugin is okay, add to __registered:
|
|
|
|
self.__registered.add(cls)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __getitem__(self, item):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
Returns a copy of the namespace dict of the base class named `name`.
|
|
|
|
"""
|
|
|
|
if inspect.isclass(item):
|
|
|
|
if item not in self.__allowed:
|
|
|
|
raise KeyError(repr(item))
|
|
|
|
key = item.__name__
|
|
|
|
else:
|
|
|
|
key = item
|
|
|
|
return dict(self.__d[key])
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __contains__(self, item):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
Returns True if a base class named `name` is in this Registrar.
|
|
|
|
"""
|
|
|
|
if inspect.isclass(item):
|
|
|
|
return item in self.__allowed
|
|
|
|
return item in self.__d
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __iter__(self):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
Iterates through a (base, registered_plugins) tuple for each allowed
|
|
|
|
base.
|
|
|
|
"""
|
|
|
|
for base in self.__allowed:
|
|
|
|
sub_d = self.__d[base.__name__]
|
|
|
|
yield (base, tuple(sub_d[k] for k in sorted(sub_d)))
|
2008-08-01 16:25:46 -05:00
|
|
|
|
|
|
|
|
|
|
|
class API(ReadOnly):
|
2008-08-12 18:40:36 -05:00
|
|
|
__finalized = False
|
|
|
|
|
2008-08-08 12:11:29 -05:00
|
|
|
def __init__(self, *allowed):
|
2008-08-09 14:48:47 -05:00
|
|
|
self.__keys = tuple(b.__name__ for b in allowed)
|
2008-08-08 16:28:56 -05:00
|
|
|
self.register = Registrar(*allowed)
|
2008-08-08 16:49:09 -05:00
|
|
|
self.__lock__()
|
2008-08-08 12:11:29 -05:00
|
|
|
|
2008-08-12 17:52:37 -05:00
|
|
|
def finalize(self):
|
2008-08-08 16:40:03 -05:00
|
|
|
"""
|
|
|
|
Finalize the registration, instantiate the plugins.
|
|
|
|
"""
|
2008-08-12 18:40:36 -05:00
|
|
|
assert not self.__finalized, 'finalize() can only be called once'
|
2008-08-08 16:40:03 -05:00
|
|
|
d = {}
|
|
|
|
def plugin_iter(base, classes):
|
|
|
|
for cls in classes:
|
|
|
|
if cls not in d:
|
|
|
|
d[cls] = cls()
|
|
|
|
plugin = d[cls]
|
|
|
|
yield Proxy(base, plugin)
|
|
|
|
|
|
|
|
for (base, classes) in self.register:
|
|
|
|
ns = NameSpace(plugin_iter(base, classes))
|
|
|
|
assert not hasattr(self, base.__name__)
|
|
|
|
object.__setattr__(self, base.__name__, ns)
|
|
|
|
for plugin in d.values():
|
|
|
|
plugin.finalize(self)
|
2008-08-09 14:39:58 -05:00
|
|
|
plugin.__lock__()
|
|
|
|
assert plugin.__islocked__() is True
|
2008-08-08 16:40:03 -05:00
|
|
|
assert plugin.api is self
|
2008-08-12 18:40:36 -05:00
|
|
|
object.__setattr__(self, '_API__finalized', True)
|
2008-08-08 12:11:29 -05:00
|
|
|
|
|
|
|
def __iter__(self):
|
2008-08-08 16:40:03 -05:00
|
|
|
for key in self.__keys:
|
|
|
|
yield key
|