Source code for openerp.addons.magentoerpconnect.unit.backend_adapter

# -*- 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/>.
#
##############################################################################

import socket
import logging
import xmlrpclib

from openerp.addons.connector.unit.backend_adapter import CRUDAdapter
from openerp.addons.connector.exception import (NetworkRetryableError,
                                                RetryableJobError)
from datetime import datetime
_logger = logging.getLogger(__name__)

try:
    import magento as magentolib
except ImportError as err:
    _logger.debug('Cannot `import magento`.')


MAGENTO_DATETIME_FORMAT = '%Y/%m/%d %H:%M:%S'


recorder = {}


[docs]def call_to_key(method, arguments): """ Used to 'freeze' the method and arguments of a call to Magento so they can be hashable; they will be stored in a dict. Used in both the recorder and the tests. """ def freeze(arg): if isinstance(arg, dict): items = dict((key, freeze(value)) for key, value in arg.iteritems()) return frozenset(items.iteritems()) elif isinstance(arg, list): return tuple([freeze(item) for item in arg]) else: return arg new_args = [] for arg in arguments: new_args.append(freeze(arg)) return (method, tuple(new_args))
[docs]def record(method, arguments, result): """ Utility function which can be used to record test data during synchronisations. Call it from MagentoCRUDAdapter._call Then ``output_recorder`` can be used to write the data recorded to a file. """ recorder[call_to_key(method, arguments)] = result
[docs]def output_recorder(filename): import pprint with open(filename, 'w') as f: pprint.pprint(recorder, f) _logger.debug('recorder written to file %s', filename)
[docs]class MagentoLocation(object): def __init__(self, location, username, password, use_custom_api_path=False): self._location = location self.username = username self.password = password self.use_custom_api_path = use_custom_api_path self.use_auth_basic = False self.auth_basic_username = None self.auth_basic_password = None @property def location(self): location = self._location if not self.use_auth_basic: return location assert self.auth_basic_username and self.auth_basic_password replacement = "%s:%s@" % (self.auth_basic_username, self.auth_basic_password) location = location.replace('://', '://' + replacement) return location
[docs]class MagentoCRUDAdapter(CRUDAdapter): """ External Records Adapter for Magento """ def __init__(self, connector_env): """ :param connector_env: current environment (backend, session, ...) :type connector_env: :class:`connector.connector.ConnectorEnvironment` """ super(MagentoCRUDAdapter, self).__init__(connector_env) backend = self.backend_record magento = MagentoLocation( backend.location, backend.username, backend.password, use_custom_api_path=backend.use_custom_api_path) if backend.use_auth_basic: magento.use_auth_basic = True magento.auth_basic_username = backend.auth_basic_username magento.auth_basic_password = backend.auth_basic_password self.magento = magento
[docs] def search(self, filters=None): """ Search records according to some criterias and returns a list of ids """ raise NotImplementedError
[docs] def read(self, id, attributes=None): """ Returns the information of a record """ raise NotImplementedError
[docs] def search_read(self, filters=None): """ Search records according to some criterias and returns their information""" raise NotImplementedError
[docs] def create(self, data): """ Create a record on the external system """ raise NotImplementedError
[docs] def write(self, id, data): """ Update records on the external system """ raise NotImplementedError
[docs] def delete(self, id): """ Delete a record on the external system """ raise NotImplementedError
def _call(self, method, arguments): try: custom_url = self.magento.use_custom_api_path _logger.debug("Start calling Magento api %s", method) with magentolib.API(self.magento.location, self.magento.username, self.magento.password, full_url=custom_url) as api: # When Magento is installed on PHP 5.4+, the API # may return garble data if the arguments contain # trailing None. if isinstance(arguments, list): while arguments and arguments[-1] is None: arguments.pop() start = datetime.now() try: result = api.call(method, arguments) except: _logger.error("api.call(%s, %s) failed", method, arguments) raise else: _logger.debug("api.call(%s, %s) returned %s in %s seconds", method, arguments, result, (datetime.now() - start).seconds) # Uncomment to record requests/responses in ``recorder`` # record(method, arguments, result) return result except (socket.gaierror, socket.error, socket.timeout) as err: raise NetworkRetryableError( 'A network error caused the failure of the job: ' '%s' % err) except xmlrpclib.ProtocolError as err: if err.errcode in [502, # Bad gateway 503, # Service unavailable 504]: # Gateway timeout raise RetryableJobError( 'A protocol error caused the failure of the job:\n' 'URL: %s\n' 'HTTP/HTTPS headers: %s\n' 'Error code: %d\n' 'Error message: %s\n' % (err.url, err.headers, err.errcode, err.errmsg)) else: raise
[docs]class GenericAdapter(MagentoCRUDAdapter): _model_name = None _magento_model = None _admin_path = None
[docs] def search(self, filters=None): """ Search records according to some criterias and returns a list of ids :rtype: list """ return self._call('%s.search' % self._magento_model, [filters] if filters else [{}])
[docs] def read(self, id, attributes=None): """ Returns the information of a record :rtype: dict """ arguments = [int(id)] if attributes: # Avoid to pass Null values in attributes. Workaround for # https://bugs.launchpad.net/openerp-connector-magento/+bug/1210775 # When Magento is installed on PHP 5.4 and the compatibility patch # http://magento.com/blog/magento-news/magento-now-supports-php-54 # is not installed, calling info() with None in attributes # would return a wrong result (almost empty list of # attributes). The right correction is to install the # compatibility patch on Magento. arguments.append(attributes) return self._call('%s.info' % self._magento_model, arguments)
[docs] def search_read(self, filters=None): """ Search records according to some criterias and returns their information""" return self._call('%s.list' % self._magento_model, [filters])
[docs] def create(self, data): """ Create a record on the external system """ return self._call('%s.create' % self._magento_model, [data])
[docs] def write(self, id, data): """ Update records on the external system """ return self._call('%s.update' % self._magento_model, [int(id), data])
[docs] def delete(self, id): """ Delete a record on the external system """ return self._call('%s.delete' % self._magento_model, [int(id)])
[docs] def admin_url(self, id): """ Return the URL in the Magento admin for a record """ if self._admin_path is None: raise ValueError('No admin path is defined for this record') backend = self.backend_record url = backend.admin_location if not url: raise ValueError('No admin URL configured on the backend.') path = self._admin_path.format(model=self._magento_model, id=id) url = url.rstrip('/') path = path.lstrip('/') url = '/'.join((url, path)) return url