change way to describe fields of CapBaseObject, and lot of documentation
This commit is contained in:
parent
99391a95ef
commit
c6a141595c
35 changed files with 1630 additions and 638 deletions
|
|
@ -36,6 +36,19 @@ __all__ = ['Weboob']
|
|||
|
||||
|
||||
class Weboob(object):
|
||||
"""
|
||||
The main class of Weboob, used to manage backends and call methods.
|
||||
|
||||
:param workdir: optional parameter to set path of the working directory
|
||||
:type workdir: str
|
||||
:param backends_filename: name of the *backends* file, where configuration of
|
||||
backends is stored
|
||||
:type backends_filename: str
|
||||
:param scheduler: what scheduler to use; default is :class:`weboob.core.scheduler.Scheduler`
|
||||
:type scheduler: :class:`weboob.core.scheduler.IScheduler`
|
||||
:param storage: provide a storage where backends can save data
|
||||
:type storage: :class:`weboob.tools.storage.IStorage`
|
||||
"""
|
||||
VERSION = '0.c'
|
||||
BACKENDS_FILENAME = 'backends'
|
||||
|
||||
|
|
@ -64,8 +77,8 @@ class Weboob(object):
|
|||
if os.path.isdir(old_workdir):
|
||||
self.logger.warning('You are using "%s" as working directory. Files are moved into %s and %s.'
|
||||
% (old_workdir, xdg_config_home, xdg_data_home))
|
||||
self.create_dir(xdg_config_home)
|
||||
self.create_dir(xdg_data_home)
|
||||
self._create_dir(xdg_config_home)
|
||||
self._create_dir(xdg_data_home)
|
||||
for f in os.listdir(old_workdir):
|
||||
if f in Repositories.SHARE_DIRS:
|
||||
dest = xdg_data_home
|
||||
|
|
@ -77,7 +90,7 @@ class Weboob(object):
|
|||
datadir = xdg_data_home
|
||||
|
||||
self.workdir = os.path.realpath(workdir)
|
||||
self.create_dir(workdir)
|
||||
self._create_dir(workdir)
|
||||
|
||||
# Repositories management
|
||||
self.repositories = Repositories(workdir, datadir, self.VERSION)
|
||||
|
|
@ -95,7 +108,7 @@ class Weboob(object):
|
|||
# Storage
|
||||
self.storage = storage
|
||||
|
||||
def create_dir(self, name):
|
||||
def _create_dir(self, name):
|
||||
if not os.path.exists(name):
|
||||
os.makedirs(name)
|
||||
elif not os.path.isdir(name):
|
||||
|
|
@ -105,11 +118,15 @@ class Weboob(object):
|
|||
self.deinit()
|
||||
|
||||
def deinit(self):
|
||||
"""
|
||||
Call this method when you stop using Weboob, to
|
||||
properly unload all correctly.
|
||||
"""
|
||||
self.unload_backends()
|
||||
|
||||
def update(self, progress=IProgress()):
|
||||
"""
|
||||
Update modules.
|
||||
Update modules from repositories.
|
||||
"""
|
||||
self.repositories.update(progress)
|
||||
|
||||
|
|
@ -120,13 +137,27 @@ class Weboob(object):
|
|||
self.repositories.install(minfo, progress)
|
||||
|
||||
class LoadError(Exception):
|
||||
"""
|
||||
Raised when a backend is unabled to load.
|
||||
|
||||
:param backend_name: name of backend we can't load
|
||||
:param exception: exception object
|
||||
"""
|
||||
def __init__(self, backend_name, exception):
|
||||
Exception.__init__(self, unicode(exception))
|
||||
self.backend_name = backend_name
|
||||
|
||||
def load_backend(self, module_name, params=None, storage=None):
|
||||
def build_backend(self, module_name, params=None, storage=None):
|
||||
"""
|
||||
Load a single backend.
|
||||
Create a single backend which is not listed
|
||||
in configuration.
|
||||
|
||||
:param module_name: name of module
|
||||
:param params: parameters to give to backend
|
||||
:type params: :class:`dict`
|
||||
:param storage: storage to use
|
||||
:type storage: :class:`weboob.tools.storage.IStorage`
|
||||
:rtype: :class:`weboob.tools.backend.BaseBackend`
|
||||
"""
|
||||
minfo = self.repositories.get_module_info(module_name)
|
||||
if minfo is None:
|
||||
|
|
@ -143,14 +174,20 @@ class Weboob(object):
|
|||
|
||||
def load_backends(self, caps=None, names=None, modules=None, storage=None, errors=None):
|
||||
"""
|
||||
Load backends.
|
||||
Load backends listed in config file.
|
||||
|
||||
@param caps [tuple(ICapBase)] load backends which implement all of caps
|
||||
@param names [tuple(unicode)] load backends with instance name in list
|
||||
@param modules [tuple(unicode)] load backends which module is in list
|
||||
@param storage [IStorage] use the storage if specified
|
||||
@param errors [list] if specified, store every errors in
|
||||
@return [dict(str,BaseBackend)] return loaded backends
|
||||
:param caps: load backends which implement all of specified caps
|
||||
:type caps: tuple[:class:`weboob.capabilities.base.ICapBase`]
|
||||
:param names: load backends with instance name in list
|
||||
:type names: tuple[:class:`str`]
|
||||
:param modules: load backends which module is in list
|
||||
:type modules: tuple[:class:`str`]
|
||||
:param storage: use this storage if specified
|
||||
:type storage: :class:`weboob.tools.storage.IStorage`
|
||||
:param errors: if specified, store every errors in this list
|
||||
:type errors: list[:class:`LoadError`]
|
||||
:returns: loaded backends
|
||||
:rtype: dict[:class:`str`, :class:`weboob.tools.backend.BaseBackend`]
|
||||
"""
|
||||
loaded = {}
|
||||
if storage is None:
|
||||
|
|
@ -195,6 +232,12 @@ class Weboob(object):
|
|||
return loaded
|
||||
|
||||
def unload_backends(self, names=None):
|
||||
"""
|
||||
Unload backends.
|
||||
|
||||
:param names: if specified, only unload that backends
|
||||
:type names: :class:`list`
|
||||
"""
|
||||
unloaded = {}
|
||||
if isinstance(names, basestring):
|
||||
names = [names]
|
||||
|
|
@ -213,8 +256,11 @@ class Weboob(object):
|
|||
"""
|
||||
Get a backend from its name.
|
||||
|
||||
It raises a KeyError if not found. If you set the 'default' parameter,
|
||||
the default value is returned instead.
|
||||
:param name: name of backend to get
|
||||
:type name: str
|
||||
:param default: if specified, get this value when the backend is not found
|
||||
:type default: whatever you want
|
||||
:raises: :class:`KeyError` if not found.
|
||||
"""
|
||||
try:
|
||||
return self.backend_instances[name]
|
||||
|
|
@ -225,6 +271,9 @@ class Weboob(object):
|
|||
raise
|
||||
|
||||
def count_backends(self):
|
||||
"""
|
||||
Get number of loaded backends.
|
||||
"""
|
||||
return len(self.backend_instances)
|
||||
|
||||
def iter_backends(self, caps=None):
|
||||
|
|
@ -233,8 +282,9 @@ class Weboob(object):
|
|||
|
||||
Note: each backend is locked when it is returned.
|
||||
|
||||
@param caps Optional list of capabilities to select backends
|
||||
@return iterator on selected backends.
|
||||
:param caps: optional list of capabilities to select backends
|
||||
:type caps: tuple[:class:`weboob.capabilities.base.IBaseCap`]
|
||||
:rtype: iter[:class:`weboob.tools.backend.BaseBackend`]
|
||||
"""
|
||||
for name, backend in sorted(self.backend_instances.iteritems()):
|
||||
if caps is None or backend.has_caps(caps):
|
||||
|
|
@ -248,17 +298,21 @@ class Weboob(object):
|
|||
|
||||
This function has two modes:
|
||||
|
||||
- If 'function' is a string, it calls the method with this name on
|
||||
- If *function* is a string, it calls the method with this name on
|
||||
each backends with the specified arguments;
|
||||
- If 'function' is a callable, it calls it in a separated thread with
|
||||
- If *function* is a callable, it calls it in a separated thread with
|
||||
the locked backend instance at first arguments, and \*args and
|
||||
\*\*kwargs.
|
||||
|
||||
@param function backend's method name, or callable object
|
||||
@param backends list of backends to iterate on
|
||||
@param caps iterate on backends with this caps
|
||||
@param condition a condition to validate to keep the result
|
||||
@return the BackendsCall object (iterable)
|
||||
:param function: backend's method name, or a callable object
|
||||
:type function: :class:`str`
|
||||
:param backends: list of backends to iterate on
|
||||
:type backends: list[:class:`str`]
|
||||
:param caps: iterate on backends which implement this caps
|
||||
:type caps: list[:class:`weboob.capabilities.base.IBaseCap`]
|
||||
:param condition: a condition to validate results
|
||||
:type condition: :class:`weboob.core.bcall.IResultsCondition`
|
||||
:rtype: A :class:`weboob.core.bcall.BackendsCall` object (iterable)
|
||||
"""
|
||||
backends = self.backend_instances.values()
|
||||
_backends = kwargs.pop('backends', None)
|
||||
|
|
@ -296,16 +350,47 @@ class Weboob(object):
|
|||
return BackendsCall(backends, condition, function, *args, **kwargs)
|
||||
|
||||
def schedule(self, interval, function, *args):
|
||||
"""
|
||||
Schedule an event.
|
||||
|
||||
:param interval: delay before calling the function
|
||||
:type interval: int
|
||||
:param function: function to call
|
||||
:type function: callabale
|
||||
:param args: arguments to give to function
|
||||
:returns: an event identificator
|
||||
"""
|
||||
return self.scheduler.schedule(interval, function, *args)
|
||||
|
||||
def repeat(self, interval, function, *args):
|
||||
"""
|
||||
Repeat a call to a function
|
||||
|
||||
:param interval: interval between two calls
|
||||
:type interval: int
|
||||
:param function: function to call
|
||||
:type function: callable
|
||||
:param args: arguments to give to function
|
||||
:returns: an event identificator
|
||||
"""
|
||||
return self.scheduler.repeat(interval, function, *args)
|
||||
|
||||
def cancel(self, ev):
|
||||
"""
|
||||
Cancel an event
|
||||
|
||||
:param ev: the event identificator
|
||||
"""
|
||||
return self.scheduler.cancel(ev)
|
||||
|
||||
def want_stop(self):
|
||||
"""
|
||||
Plan to stop the scheduler.
|
||||
"""
|
||||
return self.scheduler.want_stop()
|
||||
|
||||
def loop(self):
|
||||
"""
|
||||
Run the scheduler loop
|
||||
"""
|
||||
return self.scheduler.run()
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue