summaryrefslogtreecommitdiffstats
path: root/ipalib
diff options
context:
space:
mode:
Diffstat (limited to 'ipalib')
-rw-r--r--ipalib/__init__.py27
-rw-r--r--ipalib/backend.py14
-rw-r--r--ipalib/cli.py2
-rw-r--r--ipalib/plugable.py21
4 files changed, 51 insertions, 13 deletions
diff --git a/ipalib/__init__.py b/ipalib/__init__.py
index 4593e581..a6664f73 100644
--- a/ipalib/__init__.py
+++ b/ipalib/__init__.py
@@ -30,15 +30,18 @@ To learn about the ``ipalib`` library, you should read the code in this order:
"""
import plugable
-import frontend
-import backend
-import config
-
-api = plugable.API(
- frontend.Command,
- frontend.Object,
- frontend.Method,
- frontend.Property,
- frontend.Application,
- backend.Backend,
-)
+from backend import Backend, Context
+from frontend import Command, Object, Method, Property, Application
+from ipa_types import Bool, Int, Unicode, Enum
+from frontend import Param, DefaultFrom
+
+def get_standard_api(unit_test=False):
+ api = plugable.API(
+ Command, Object, Method, Property, Application,
+ Backend, Context,
+ )
+ if unit_test is True:
+ api.env.mode = 'unit_test'
+ return api
+
+api = get_standard_api()
diff --git a/ipalib/backend.py b/ipalib/backend.py
index 82ed14f3..b1e15f33 100644
--- a/ipalib/backend.py
+++ b/ipalib/backend.py
@@ -23,9 +23,23 @@ Base classes for all backed-end plugins.
import plugable
+
class Backend(plugable.Plugin):
"""
Base class for all backend plugins.
"""
__proxy__ = False # Backend plugins are not wrapped in a PluginProxy
+
+
+class Context(plugable.Plugin):
+ """
+ Base class for plugable context components.
+ """
+
+ __proxy__ = False # Backend plugins are not wrapped in a PluginProxy
+
+ def get_value(self):
+ raise NotImplementedError(
+ '%s.get_value()' % self.__class__.__name__
+ )
diff --git a/ipalib/cli.py b/ipalib/cli.py
index 021e01ad..39773d73 100644
--- a/ipalib/cli.py
+++ b/ipalib/cli.py
@@ -331,7 +331,7 @@ class CLI(object):
if len(a) < 2:
parser.error('badly specified environment string,'\
'use var1=val1[,var2=val2]..')
- overrides[a[0].strip()] = a[1].strip()
+ overrides[str(a[0].strip())] = a[1].strip()
overrides['context'] = 'cli'
self.api.bootstrap(**overrides)
diff --git a/ipalib/plugable.py b/ipalib/plugable.py
index b0ba32b7..9ddcb30f 100644
--- a/ipalib/plugable.py
+++ b/ipalib/plugable.py
@@ -27,6 +27,7 @@ http://docs.python.org/ref/sequence-types.html
import re
import inspect
+import threading
import errors
from errors import check_type, check_isinstance
from config import Environment, Env
@@ -705,6 +706,25 @@ class Registrar(DictProxy):
self.__registered.add(klass)
+class LazyContext(object):
+ """
+ On-demand creation of thread-local context attributes.
+ """
+
+ def __init__(self, api):
+ self.__api = api
+ self.__context = threading.local()
+
+ def __getattr__(self, name):
+ if name not in self.__context.__dict__:
+ if name not in self.__api.Context:
+ raise AttributeError('no Context plugin for %r' % name)
+ value = self.__api.Context[name].get_value()
+ self.__context.__dict__[name] = value
+ return self.__context.__dict__[name]
+
+
+
class API(DictProxy):
"""
Dynamic API object through which `Plugin` instances are accessed.
@@ -715,6 +735,7 @@ class API(DictProxy):
self.__done = set()
self.register = Registrar(*allowed)
self.env = Env()
+ self.context = LazyContext(self)
super(API, self).__init__(self.__d)
def __doing(self, name):