# -*- coding: utf-8 -*- ############################################################################## # # Author: Guewen Baconnier # Copyright 2013 Camptocamp SA # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as # published by the Free Software Foundation, either version 3 of the # License, or (at your option) any later version. # # 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 Affero General Public License for more details. # # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. # ############################################################################## from functools import partial from collections import namedtuple from .exception import NoConnectorUnitError __all__ = ['Backend'] class BackendRegistry(object): """ Hold a set of backends """ def __init__(self): self.backends = set() def register_backend(self, backend): """ Register an instance of :py:class:`connector.backend.Backend` :param backend: backend to register :type backend: Backend """ self.backends.add(backend) def get_backend(self, service, version=None): """ Return an instance of :py:class:`connector.backend.Backend` for a ``service`` and a ``version`` :param service: name of the service to return :type service: str :param version: version of the service to return :type version: str """ for backend in self.backends: if backend.match(service, version): return backend raise ValueError('No backend found for %s %s' % (service, version)) BACKENDS = BackendRegistry() def get_backend(service, version=None): """ Return the correct instance of :py:class:`connector.backend.Backend` for a ``service`` and a ``version`` :param service: name of the service to return :type service: str :param version: version of the service to return :type version: str """ return BACKENDS.get_backend(service, version) # Represents an entry for a class in a ``Backend`` registry. _ConnectorUnitEntry = namedtuple('_ConnectorUnitEntry', ['cls', 'openerp_module', 'replaced_by']) class Backend(object): """ A backend represents a system to interact with, like Magento, Prestashop, Redmine, ... It owns 3 properties: .. attribute:: service Name of the service, for instance 'magento' .. attribute:: version The version of the service. For instance: '1.7' .. attribute:: parent A parent backend. When no :py:class:`~connector.connector.ConnectorUnit` is found for a backend, it will search it in the `parent`. The Backends structure is a key part of the framework, but is rather simple. * A ``Backend`` instance holds a registry of :py:class:`~connector.connector.ConnectorUnit` classes * It can return the appropriate :py:class:`~connector.connector.ConnectorUnit` to use for a task * If no :py:class:`~connector.connector.ConnectorUnit` is registered for a task, it will ask it to its direct parent (and so on) The Backends support 2 different extension mechanisms. One is more vertical - across the versions - and the other would be more horizontal as it allows to modify the behavior for 1 version of backend. For the sake of the example, let's say we have theses backend versions:: <Magento> | ----------------- | | <Magento 1.7> <Magento 2.0> | <Magento with specific> And here is the way they are declared in Python:: magento = Backend('magento') magento1700 = Backend(parent=magento, version='1.7') magento2000 = Backend(parent=magento, version='2.0') magento_specific = Backend(parent=magento1700, version='1.7-specific') In the graph above, ``<Magento>`` will hold all the classes shared between all the versions. Each Magento version (``<Magento 1.7>``, ``<Magento 2.0>``) will use the classes defined on ``<Magento>``, excepted if they registered their own ones instead. That's the same for ``<Magento with specific>`` but this one contains customizations which are specific to an instance (typically you want specific mappings for one instance). Here is how you would register classes on ``<Magento>`` and another on ``<Magento 1.7>``:: @magento class Synchronizer(ConnectorUnit): _model_name = 'res.partner' @magento class Mapper(ConnectorUnit): _model_name = 'res.partner' @magento1700 class Synchronizer1700(Synchronizer): _model_name = 'res.partner' Here, the :py:meth:`~get_class` called on ``magento1700`` would return:: magento1700.get_class(Synchronizer, session, 'res.partner') # => Synchronizer1700 magento1700.get_class(Mapper, session, 'res.partner') # => Mapper This is the vertical extension mechanism, it says that each child version is able to extend or replace the behavior of its parent. .. note:: when using the framework, you won't need to call :py:meth:`~get_class`, usually, you will call :py:meth:`connector.connector.ConnectorEnvironment.get_connector_unit`. The vertical extension is the one you will probably use the most, because most of the things you will change concern your custom adaptations or different behaviors between the versions of the backend. However, some time, we need to change the behavior of a connector, by installing an addon. For example, say that we already have an ``ImportMapper`` for the products in the Magento Connector. We create a - generic - addon to handle the catalog in a more advanced manner. We redefine an ``AdvancedImportMapper``, which should be used when the addon is installed. This is the horizontal extension mechanism. Replace a :py:class:`~connector.connector.ConnectorUnit` by another one in a backend:: @backend(replacing=ImportMapper) class AdvancedImportMapper(ImportMapper): _model_name = 'product.product' .. warning:: The horizontal extension should be used sparingly and cautiously since as soon as 2 addons want to replace the same class, you'll have a conflict (which would need to create a third addon to glue them, ``replacing`` can take a tuple of classes to replace and this is exponential). This mechanism should be used only in some well placed circumstances for generic addons. """ def __init__(self, service=None, version=None, parent=None, registry=None): if service is None and parent is None: raise ValueError('A service or a parent service is expected') self._service = service self.version = version self.parent = parent self._class_entries = [] if registry is None: registry = BACKENDS registry.register_backend(self) def match(self, service, version): """Used to find the backend for a service and a version""" return (self.service == service and self.version == version) @property def service(self): return self._service or self.parent.service def __str__(self): if self.version: return 'Backend(\'%s\', \'%s\')' % (self.service, self.version) return 'Backend(\'%s\')' % self.service def __repr__(self): if self.version: return '<Backend \'%s\', \'%s\'>' % (self.service, self.version) return '<Backend \'%s\'>' % self.service def _get_classes(self, base_class, session, model_name): def follow_replacing(entries): candidates = set() for entry in entries: replacings = None if entry.replaced_by: replacings = follow_replacing(entry.replaced_by) if replacings: candidates.update(replacings) # If all the classes supposed to replace the current class # have been discarded, the current class is a candidate. # It happens when the entries in 'replaced_by' are # in modules not installed. if not replacings: if (session.is_module_installed(entry.openerp_module) and issubclass(entry.cls, base_class) and entry.cls.match(session, model_name)): candidates.add(entry.cls) return candidates matching_classes = follow_replacing(self._class_entries) if not matching_classes and self.parent: matching_classes = self.parent._get_classes(base_class, session, model_name) return matching_classes def get_class(self, base_class, session, model_name): """ Find a matching subclass of ``base_class`` in the registered classes. :param base_class: class (and its subclass) to search in the registry :type base_class: :py:class:`connector.connector.MetaConnectorUnit` :param session: current session :type session: :py:class:`connector.session.ConnectorSession` """ matching_classes = self._get_classes(base_class, session, model_name) if not matching_classes: raise NoConnectorUnitError('No matching class found for %s ' 'with session: %s, ' 'model name: %s' % (base_class, session, model_name)) assert len(matching_classes) == 1, ( 'Several classes found for %s ' 'with session %s, model name: %s. Found: %s' % (base_class, session, model_name, matching_classes)) return matching_classes.pop() def register_class(self, cls, replacing=None): """ Register a class in the backend. :param cls: the ConnectorUnit class class to register :type cls: :py:class:`connector.connector.MetaConnectorUnit` :param replacing: optional, the ConnectorUnit class to replace :type replacing: :py:class:`connector.connector.MetaConnectorUnit` """ def register_replace(replacing_cls): found = False for replaced_entry in self._class_entries: if replaced_entry.cls is replacing_cls: replaced_entry.replaced_by.append(entry) found = True break if not found: raise ValueError('%s replaces an unregistered class: %s' % (cls, replacing)) entry = _ConnectorUnitEntry(cls=cls, openerp_module=cls._openerp_module_, replaced_by=[]) if replacing is not None: if replacing is cls: raise ValueError('%r cannot replace itself' % replacing) if hasattr(replacing, '__iter__'): for replacing_cls in replacing: register_replace(replacing_cls) else: register_replace(replacing) self._class_entries.append(entry) def __call__(self, cls=None, replacing=None): """ Backend decorator For a backend ``magento`` declared like this:: magento = Backend('magento') A :py:class:`connector.connector.ConnectorUnit` (like a binder, a synchronizer, a mapper, ...) can be registered as follows:: @magento class MagentoBinder(Binder): _model_name = 'a.model' # other stuff Thus, by doing:: magento.get_class(Binder, 'a.model') We get the correct class ``MagentoBinder``. Any ``ConnectorUnit`` can be replaced by another doing:: @magento(replacing=MagentoBinder) class MagentoBinder2(Binder): _model_name = 'a.model' # other stuff This is useful when working on an OpenERP module which should alter the original behavior of a connector for an existing backend. :param cls: the ConnectorUnit class class to register :type cls: :py:class:`connector.connector.MetaConnectorUnit` :param replacing: optional, the ConnectorUnit class to replace :type replacing: :py:class:`connector.connector.MetaConnectorUnit` """ if cls is None: return partial(self, replacing=replacing) def with_subscribe(): self.register_class(cls, replacing=replacing) return cls return with_subscribe()