# Part of Odoo. See LICENSE file for full copyright and licensing details. import logging import pprint import requests from odoo import _, fields, models from odoo.exceptions import ValidationError from odoo.addons.payment_xendit import const _logger = logging.getLogger(__name__) class PaymentProvider(models.Model): _inherit = 'payment.provider' code = fields.Selection( selection_add=[('xendit', "Xendit")], ondelete={'xendit': 'set default'} ) xendit_public_key = fields.Char( string="Xendit Public Key", groups='base.group_system', required_if_provider='xendit' ) xendit_secret_key = fields.Char( string="Xendit Secret Key", groups='base.group_system', required_if_provider='xendit' ) xendit_webhook_token = fields.Char( string="Xendit Webhook Token", groups='base.group_system', required_if_provider='xendit' ) # === COMPUTE METHODS === # def _compute_feature_support_fields(self): """ Override of `payment` to enable additional features. """ super()._compute_feature_support_fields() self.filtered(lambda p: p.code == 'xendit').support_tokenization = True # === BUSINESS METHODS - PAYMENT FLOW ===# def _get_supported_currencies(self): """ Override of `payment` to return the supported currencies. """ supported_currencies = super()._get_supported_currencies() if self.code == 'xendit': supported_currencies = supported_currencies.filtered( lambda c: c.name in const.SUPPORTED_CURRENCIES ) return supported_currencies def _get_default_payment_method_codes(self): """ Override of `payment` to return the default payment method codes. """ default_codes = super()._get_default_payment_method_codes() if self.code != 'xendit': return default_codes return const.DEFAULT_PAYMENT_METHOD_CODES def _xendit_make_request(self, endpoint, payload=None): """ Make a request to Xendit API and return the JSON-formatted content of the response. Note: self.ensure_one() :param str endpoint: The endpoint to be reached by the request. :param dict payload: The payload of the request. :return The JSON-formatted content of the response. :rtype: dict :raise ValidationError: If an HTTP error occurs. """ self.ensure_one() url = f'https://api.xendit.co/{endpoint}' auth = (self.xendit_secret_key, '') try: response = requests.post(url, json=payload, auth=auth, timeout=10) response.raise_for_status() except (requests.exceptions.ConnectionError, requests.exceptions.Timeout): _logger.exception("Unable to reach endpoint at %s", url) raise ValidationError("Xendit: " + _("Could not establish the connection to the API.")) except requests.exceptions.HTTPError as err: error_message = err.response.json().get('message') _logger.exception( "Invalid API request at %s with data:\n%s", url, pprint.pformat(payload) ) raise ValidationError( "Xendit: " + _( "The communication with the API failed. Xendit gave us the following" " information: '%s'", error_message ) ) return response.json() # === BUSINESS METHODS - GETTERS === # def _get_redirect_form_view(self, is_validation=False): """ Override of `payment` to avoid rendering the form view for validation operations. Unlike other compatible payment methods in Xendit, `Card` is implemented using a direct flow. To avoid rendering a useless template, and also to avoid computing wrong values, this method returns `None` for Xendit's validation operations (Card is and will always be the sole tokenizable payment method for Xendit). Note: `self.ensure_one()` :param bool is_validation: Whether the operation is a validation. :return: The view of the redirect form template or None. :rtype: ir.ui.view | None """ self.ensure_one() if self.code == 'xendit' and is_validation: return None return super()._get_redirect_form_view(is_validation)