Source code for woob.tools.backend

# Copyright(C) 2010-2023 Romain Bignon
#
# This file is part of woob.
#
# woob is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# woob 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 Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with woob. If not, see <http://www.gnu.org/licenses/>.

from __future__ import annotations

import logging
import importlib
import os
from copy import copy
from threading import RLock
import warnings
from urllib.request import getproxies
from typing import List, Any, Dict, Tuple, Iterator, Type, ClassVar, TYPE_CHECKING

from packaging.version import Version

from woob import __version__
from woob.capabilities.base import BaseObject, Capability, FieldNotFound, NotAvailable, NotLoaded
from woob.tools.json import json
from woob.tools.log import getLogger
from woob.tools.misc import iter_fields
from woob.tools.storage import IStorage
from woob.tools.value import ValuesDict, ValueBool

if TYPE_CHECKING:
    from woob.core import WoobBase
    from woob.browser import Browser


__all__ = ['BackendStorage', 'BackendConfig', 'Module']


[docs]class BackendStorage: """ This is an abstract layer to store data in storages (:mod:`woob.tools.storage`) easily. It is instancied automatically in constructor of :class:`Module`, in the :attr:`Module.storage` attribute. :param name: name of backend :type name: str :param storage: storage object :type storage: :class:`woob.tools.storage.IStorage` """ def __init__(self, name: str, storage: IStorage | None): self.name = name self.storage = storage
[docs] def set(self, *args): """ Set value in the storage. Example: >>> from woob.tools.storage import StandardStorage >>> backend = BackendStorage('blah', StandardStorage('/tmp/cfg')) >>> backend.storage.set('config', 'nb_of_threads', 10) # doctest: +SKIP >>> :param args: the path where to store value """ if self.storage: self.storage.set('backends', self.name, *args)
[docs] def delete(self, *args): """ Delete a value from the storage. :param args: path to delete. """ if self.storage: self.storage.delete('backends', self.name, *args)
[docs] def get(self, *args, **kwargs) -> Any: """ Get a value or a dict of values in storage. Example: >>> from woob.tools.storage import StandardStorage >>> backend = BackendStorage('blah', StandardStorage('/tmp/cfg')) >>> backend.storage.get('config', 'nb_of_threads') # doctest: +SKIP 10 >>> backend.storage.get('config', 'unexistant', 'path', default='lol') # doctest: +SKIP 'lol' >>> backend.storage.get('config') # doctest: +SKIP {'nb_of_threads': 10, 'other_things': 'blah'} :param args: path to get :param default: if specified, default value when path is not found """ if self.storage: return self.storage.get('backends', self.name, *args, **kwargs) return kwargs.get('default', None)
[docs] def load(self, default: Dict): """ Load storage. It is made automatically when your backend is created, and use the ``STORAGE`` class attribute as default. :param default: this is the default tree if storage is empty :type default: :class:`dict` """ if self.storage: self.storage.load('backends', self.name, default)
[docs] def save(self): """ Save storage. """ if self.storage: self.storage.save('backends', self.name)
[docs]class BackendConfig(ValuesDict): """ Configuration of a backend. This class is firstly instanced as a :class:`woob.tools.value.ValuesDict`, containing some :class:`woob.tools.value.Value` (and derivated) objects. Then, using the :func:`load` method will load configuration from file and create a copy of the :class:`BackendConfig` object with the loaded values. """ modname: str instname: str woob: WoobBase
[docs] def load( self, woob: WoobBase, modname: str, instname: str, config: Dict, nofail: bool = False ) -> BackendConfig: """ Load configuration from dict to create an instance. :param woob: woob object :type woob: :class:`woob.core.woob.WoobBase` :param modname: name of the module :type modname: :class:`str` :param instname: name of this backend :type instname: :class:`str` :param params: parameters to load :type params: :class:`dict` :param nofail: if true, this call can't fail :type nofail: :class:`bool` :rtype: :class:`BackendConfig` """ cfg = self.__class__() cfg.modname = modname cfg.instname = instname cfg.woob = woob for name, field in self.items(): value = config.get(name, None) if value is None: if not nofail and field.required: raise Module.ConfigError( f'Backend({cfg.instname}): Configuration error: Missing parameter {name} ({field.description})', bad_fields=[name] ) value = field.default field = copy(field) try: field.load(cfg.instname, value, cfg.woob.requests) except ValueError as v: if not nofail: raise Module.ConfigError( f'Backend({cfg.instname}): Configuration error for field "{name}": {v}', bad_fields=[name] ) cfg[name] = field return cfg
[docs] def dump(self) -> dict: """ Dump config in a dictionary. :rtype: :class:`dict` """ settings = {} for name, value in self.items(): if not value.transient: settings[name] = value.dump() return settings
[docs] def save(self, edit: bool = True, params: Dict | None = None): """ Save backend config. :param edit: if true, it changes config of an existing backend :type edit: :class:`bool` :param params: if supplied, params to merge with the ones of the current object :type params: :class:`dict` """ assert self.modname is not None assert self.instname is not None assert self.woob is not None dump = self.dump() if params is not None: dump.update(params) if edit: self.woob.backends_config.edit_backend(self.instname, dump) else: self.woob.backends_config.add_backend(self.instname, self.modname, dump)
[docs]class Module: """ Base class for modules. You may derivate it, and also all capabilities you want to implement. :param woob: woob instance :type woob: :class:`woob.core.woob.Woob` :param name: name of backend :type name: :class:`str` :param config: configuration of backend (optional) :type config: :class:`dict` :param storage: storage object (optional) :type storage: :class:`woob.tools.storage.IStorage` :param logger: parent logger (optional) :type logger: :class:`logging.Logger` """ NAME: ClassVar[str] """Name of the maintainer of this module.""" MAINTAINER: ClassVar[str] = '<unspecified>' """Name of the maintainer.""" EMAIL: ClassVar[str] = '<unspecified>' """Email address of the maintainer.""" DESCRIPTION: ClassVar[str] = '<unspecified>' """Description""" LICENSE: ClassVar[str] = '<unspecified>' """License of the module""" CONFIG: ClassVar[BackendConfig] = BackendConfig() """Configuration required for backends. Values must be :class:`woob.tools.value.Value` objects. """ STORAGE: ClassVar[Dict] = {} """Storage""" BROWSER: Browser | None = None """Browser class""" ICON: ClassVar[str | None] = None """URL to an optional icon. If you want to create your own icon, create a 'favicon.png' icon in the module's directory, and keep the ICON value to None. """ OBJECTS: ClassVar[Dict] = {} """Supported objects to fill The key is the class and the value the method to call to fill Method prototype: method(object, fields) When the method is called, fields are only the one which are NOT yet filled. """ DEPENDENCIES: ClassVar[Tuple[str]] = () """Tuple of module names on which this module depends."""
[docs] class ConfigError(Exception): """ Raised when the config can't be loaded. """ def __init__(self, message, bad_fields=None): """ :type message: str :param message: message of the exception :type bad_fields: list[str] :param bad_fields: names of the config fields which are incorrect """ super().__init__(message) self.bad_fields = bad_fields or ()
def __enter__(self): self.lock.acquire() def __exit__(self, t, v, tb): self.lock.release() def __repr__(self): return f"<Backend {self.name}>" def __new__(cls, *args, **kwargs): """ Accept any arguments, necessary for AbstractModule __new__ override. AbstractModule, in its overridden __new__, removes itself from class hierarchy so its __new__ is called only once. In python 3, default (object) __new__ is then used for next instantiations but it's a slot/"fixed" version supporting only one argument (type to instanciate). """ return object.__new__(cls) @property def VERSION(self): warnings.warn( 'Attribute Module.VERSION will be removed in woob 4, do not use it.', DeprecationWarning, stacklevel=3 ) return Version(__version__).base_version def __init__( self, woob: WoobBase, name: str, config: Dict | None = None, storage: IStorage | None = None, logger: logging.Logger | None = None, nofail: bool = False ): if ( hasattr(self.__class__, 'VERSION') and not isinstance(self.__class__.VERSION, property) ): warnings.warn( f'Class attribute {self.__class__.__name__}.VERSION is now ' 'unused and deprecated, you can remove it. ' 'If you do so, do not forget to increase the woob version to at ' 'least 3.4 in requirements.txt.', DeprecationWarning, ) self.logger = getLogger(name, parent=logger) self.woob = woob self.name = name self.lock = RLock() if config is None: config = {} # Private fields (which start with '_') self._private_config = dict((key, value) for key, value in config.items() if key.startswith('_')) # Load configuration of backend. self.config = self.CONFIG.load(woob, self.NAME, self.name, config, nofail) self.storage = BackendStorage(self.name, storage) self.storage.load(self.STORAGE)
[docs] def dump_state(self): """ Dump module state into storage. """ if hasattr(self.browser, 'dump_state'): self.storage.set('browser_state', self.browser.dump_state()) self.storage.save()
[docs] def deinit(self): """ This abstract method is called when the backend is unloaded. """ if self._browser is None: return try: self.dump_state() finally: if hasattr(self.browser, 'deinit'): self.browser.deinit()
@property def weboob(self): """ .. deprecated:: 3.4 Don't use this attribute, but :attr:`woob` instead. """ warnings.warn('Use Module.woob instead.', DeprecationWarning, stacklevel=2) return self.woob _browser = None @property def browser(self) -> Browser: """ Attribute 'browser'. The browser is created at the first call of this attribute, to avoid useless pages access. Note that the :func:`create_default_browser` method is called to create it. """ if self._browser is None: self._browser = self.create_default_browser() return self._browser
[docs] def create_default_browser(self) -> Browser | None: """ Method to overload to build the default browser in attribute 'browser'. """ return self.create_browser()
[docs] def create_browser(self, *args, **kwargs) -> Browser | None: """ Build a browser from the BROWSER class attribute and the given arguments. :param klass: optional parameter to give another browser class to instanciate :type klass: :class:`woob.browser.browsers.Browser` :param load_state: Whether to load the browser state if it supports the feature. :type load_state: bool """ klass = kwargs.pop('klass', self.BROWSER) if not klass: return None should_load_state = bool(kwargs.pop('load_state', True)) kwargs['proxy'] = self.get_proxy() if '_proxy_headers' in self._private_config: kwargs['proxy_headers'] = self._private_config['_proxy_headers'] if isinstance(kwargs['proxy_headers'], str): kwargs['proxy_headers'] = json.loads(kwargs['proxy_headers']) if '_ssl_verify' in self._private_config: # value can be either a boolean or a string (path) value = ValueBool() try: value.set(self._private_config['_ssl_verify']) except ValueError: kwargs.setdefault('verify', self._private_config['_ssl_verify']) else: kwargs.setdefault('verify', value.get()) kwargs['logger'] = self.logger if self.logger.settings['responses_dirname']: kwargs.setdefault('responses_dirname', os.path.join(self.logger.settings['responses_dirname'], self._private_config.get('_debug_dir', self.name))) elif os.path.isabs(self._private_config.get('_debug_dir', '')): kwargs.setdefault('responses_dirname', self._private_config['_debug_dir']) if '_highlight_el' in self._private_config: value = ValueBool() try: value.set(self._private_config['_highlight_el']) except ValueError as e: raise Module.ConfigError( f'Backend({self.name}): Configuration error: _highlight_el must be a boolean' ) from e kwargs.setdefault('highlight_el', value.get()) browser = klass(*args, **kwargs) if should_load_state and hasattr(browser, 'load_state'): browser.load_state(self.storage.get('browser_state', default={})) return browser
[docs] def get_proxy(self) -> Dict[str, str]: """ Get proxy to use. It will read in environment variables, then in backend config. Proxy keys in backend config are: * ``_proxy`` for HTTP requests * ``_proxy_ssl`` for HTTPS requests """ # Get proxies from environment variables proxies = getproxies() # Override them with backend-specific config if '_proxy' in self._private_config: proxies['http'] = self._private_config['_proxy'] if '_proxy_ssl' in self._private_config: proxies['https'] = self._private_config['_proxy_ssl'] # Remove empty values for key in list(proxies.keys()): if not proxies[key]: del proxies[key] return proxies
[docs] @classmethod def iter_caps(cls) -> Iterator[Type[Capability]]: """ Iter capabilities implemented by this backend. :rtype: iter[:class:`woob.capabilities.base.Capability`] """ for base in cls.mro(): if issubclass(base, Capability) and base != Capability and base != cls and not issubclass(base, Module): yield base
[docs] def has_caps(self, *caps) -> bool: """ Check if this backend implements at least one of these capabilities. `caps` should be list of :class:`Capability` objects (e.g. :class:`CapBank`) or capability names (e.g. 'bank'). """ available_cap_names = [cap.__name__ for cap in self.iter_caps()] return any( (isinstance(c, str) and c in available_cap_names) or isinstance(self, c) for c in caps )
[docs] def fillobj(self, obj: BaseObject, fields: List[str] | None = None): """ Fill an object with the wanted fields. :param fields: what fields to fill; if None, all fields are filled :type fields: :class:`list` """ if obj is None: return obj def not_loaded_or_incomplete(v): return (v is NotLoaded or isinstance(v, BaseObject) and not v.__iscomplete__()) def not_loaded(v): return v is NotLoaded def filter_missing_fields(obj, fields, check_cb): missing_fields = [] if fields is None: # Select all fields if isinstance(obj, BaseObject): fields = [item[0] for item in obj.iter_fields()] else: fields = [item[0] for item in iter_fields(obj)] for field in fields: if not hasattr(obj, field): raise FieldNotFound(obj, field) value = getattr(obj, field) missing = False if hasattr(value, '__iter__'): for v in (value.values() if isinstance(value, dict) else value): if check_cb(v): missing = True break elif check_cb(value): missing = True if missing: missing_fields.append(field) return missing_fields if isinstance(fields, str): fields = (fields,) missing_fields = filter_missing_fields(obj, fields, not_loaded_or_incomplete) if not missing_fields: return obj for key, value in self.OBJECTS.items(): if isinstance(obj, key): self.logger.debug('Fill %r with fields: %s', obj, missing_fields) obj = value(self, obj, missing_fields) or obj break missing_fields = filter_missing_fields(obj, fields, not_loaded) # Object is not supported by backend. Do not notice it to avoid flooding user. # That's not so bad. for field in missing_fields: setattr(obj, field, NotAvailable) return obj
class AbstractModuleMissingParentError(Exception): pass class MetaModule(type): # we can remove this class as soon as we get rid of Abstract* def __new__(mcs, name, bases, dct): if name != 'AbstractModule' and AbstractModule in bases: warnings.warn('AbstractModule is deprecated and will be removed in woob 4.0. ' 'Use standard "from woob_modules.other_module import Module" instead.', DeprecationWarning, stacklevel=2) module = importlib.import_module('woob_modules.%s' % dct['PARENT']) symbols = [getattr(module, attr) for attr in dir(module) if not attr.startswith('__')] klass = next( symbol for symbol in symbols if isinstance(symbol, type) and issubclass(symbol, Module) and symbol != Module ) bases = tuple(klass if isinstance(base, mcs) else base for base in bases) additional_config = dct.pop('ADDITIONAL_CONFIG', None) if additional_config: dct['CONFIG'] = BackendConfig(*(list(klass.CONFIG.values()) + list(additional_config.values()))) return super().__new__(mcs, name, bases, dct) class AbstractModule(metaclass=MetaModule): """ .. deprecated:: 3.4 Don't use this class, import woob_modules.other_module.etc instead """