246 lines
9.7 KiB
Python
246 lines
9.7 KiB
Python
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
|
|
|
import logging
|
|
import pprint
|
|
|
|
from werkzeug import urls
|
|
|
|
from odoo import _, models
|
|
from odoo.exceptions import ValidationError
|
|
from odoo.tools import float_round
|
|
|
|
from odoo.addons.payment import utils as payment_utils
|
|
from odoo.addons.payment_xendit import const
|
|
|
|
|
|
_logger = logging.getLogger(__name__)
|
|
|
|
|
|
class PaymentTransaction(models.Model):
|
|
_inherit = 'payment.transaction'
|
|
|
|
def _get_specific_processing_values(self, processing_values):
|
|
""" Override of payment to return Xendit-specific processing values.
|
|
|
|
Note: self.ensure_one() from `_get_processing_values`
|
|
|
|
:param dict processing_values: The generic processing values of the transaction
|
|
:return: The dict of provider-specific processing values
|
|
:rtype: dict
|
|
"""
|
|
res = super()._get_specific_processing_values(processing_values)
|
|
if self.provider_code != 'xendit':
|
|
return res
|
|
|
|
if self.currency_id.name in const.CURRENCY_DECIMALS:
|
|
rounding = const.CURRENCY_DECIMALS.get(self.currency_id.name)
|
|
else:
|
|
rounding = self.currency_id.decimal_places
|
|
rounded_amount = float_round(self.amount, rounding, rounding_method='DOWN')
|
|
return {
|
|
'rounded_amount': rounded_amount
|
|
}
|
|
|
|
def _get_specific_rendering_values(self, processing_values):
|
|
""" Override of `payment` to return Xendit-specific rendering values.
|
|
|
|
Note: self.ensure_one() from `_get_processing_values`
|
|
|
|
:param dict processing_values: The generic and specific processing values of the transaction
|
|
:return: The dict of provider-specific processing values.
|
|
:rtype: dict
|
|
"""
|
|
res = super()._get_specific_rendering_values(processing_values)
|
|
if self.provider_code != 'xendit' or self.payment_method_code == 'card':
|
|
return res
|
|
|
|
# Initiate the payment and retrieve the invoice data.
|
|
payload = self._xendit_prepare_invoice_request_payload()
|
|
_logger.info("Sending invoice request for link creation:\n%s", pprint.pformat(payload))
|
|
invoice_data = self.provider_id._xendit_make_request('v2/invoices', payload=payload)
|
|
_logger.info("Received invoice request response:\n%s", pprint.pformat(invoice_data))
|
|
|
|
# Extract the payment link URL and embed it in the redirect form.
|
|
rendering_values = {
|
|
'api_url': invoice_data.get('invoice_url')
|
|
}
|
|
return rendering_values
|
|
|
|
def _xendit_prepare_invoice_request_payload(self):
|
|
""" Create the payload for the invoice request based on the transaction values.
|
|
|
|
:return: The request payload.
|
|
:rtype: dict
|
|
"""
|
|
base_url = self.provider_id.get_base_url()
|
|
redirect_url = urls.url_join(base_url, '/payment/status')
|
|
payload = {
|
|
'external_id': self.reference,
|
|
'amount': self.amount,
|
|
'description': self.reference,
|
|
'customer': {
|
|
'given_names': self.partner_name,
|
|
},
|
|
'success_redirect_url': redirect_url,
|
|
'failure_redirect_url': redirect_url,
|
|
'payment_methods': [const.PAYMENT_METHODS_MAPPING.get(
|
|
self.payment_method_code, self.payment_method_code.upper())
|
|
],
|
|
'currency': self.currency_id.name,
|
|
}
|
|
# Extra payload values that must not be included if empty.
|
|
if self.partner_email:
|
|
payload['customer']['email'] = self.partner_email
|
|
if phone := self.partner_id.mobile or self.partner_id.phone:
|
|
payload['customer']['mobile_number'] = phone
|
|
address_details = {}
|
|
if self.partner_city:
|
|
address_details['city'] = self.partner_city
|
|
if self.partner_country_id.name:
|
|
address_details['country'] = self.partner_country_id.name
|
|
if self.partner_zip:
|
|
address_details['postal_code'] = self.partner_zip
|
|
if self.partner_state_id.name:
|
|
address_details['state'] = self.partner_state_id.name
|
|
if self.partner_address:
|
|
address_details['street_line1'] = self.partner_address
|
|
if address_details:
|
|
payload['customer']['addresses'] = [address_details]
|
|
|
|
return payload
|
|
|
|
def _send_payment_request(self):
|
|
""" Override of `payment` to send a payment request to Xendit.
|
|
|
|
Note: self.ensure_one()
|
|
|
|
:return: None
|
|
:raise UserError: If the transaction is not linked to a token.
|
|
"""
|
|
super()._send_payment_request()
|
|
if self.provider_code != 'xendit':
|
|
return
|
|
|
|
if not self.token_id:
|
|
raise ValidationError("Xendit: " + _("The transaction is not linked to a token."))
|
|
|
|
self._xendit_create_charge(self.token_id.provider_ref)
|
|
|
|
def _xendit_create_charge(self, token_ref):
|
|
""" Create a charge on Xendit using the `credit_card_charges` endpoint.
|
|
|
|
:param str token_ref: The reference of the Xendit token to use to make the payment.
|
|
:return: None
|
|
"""
|
|
if self.currency_id.name in const.CURRENCY_DECIMALS:
|
|
rounding = const.CURRENCY_DECIMALS.get(self.currency_id.name)
|
|
else:
|
|
rounding = self.currency_id.decimal_places
|
|
rounded_amount = float_round(self.amount, rounding, rounding_method='DOWN')
|
|
payload = {
|
|
'token_id': token_ref,
|
|
'external_id': self.reference,
|
|
'amount': rounded_amount,
|
|
'currency': self.currency_id.name,
|
|
}
|
|
charge_notification_data = self.provider_id._xendit_make_request(
|
|
'credit_card_charges', payload=payload
|
|
)
|
|
self._handle_notification_data('xendit', charge_notification_data)
|
|
|
|
def _get_tx_from_notification_data(self, provider_code, notification_data):
|
|
""" Override of `payment` to find the transaction based on the notification data.
|
|
|
|
:param str provider_code: The code of the provider that handled the transaction.
|
|
:param dict notification_data: The notification data sent by the provider.
|
|
:return: The transaction if found.
|
|
:rtype: payment.transaction
|
|
:raise ValidationError: If inconsistent data were received.
|
|
:raise ValidationError: If the data match no transaction.
|
|
"""
|
|
tx = super()._get_tx_from_notification_data(provider_code, notification_data)
|
|
if provider_code != 'xendit' or len(tx) == 1:
|
|
return tx
|
|
|
|
reference = notification_data.get('external_id')
|
|
if not reference:
|
|
raise ValidationError("Xendit: " + _("Received data with missing reference."))
|
|
|
|
tx = self.search([('reference', '=', reference), ('provider_code', '=', 'xendit')])
|
|
if not tx:
|
|
raise ValidationError(
|
|
"Xendit: " + _("No transaction found matching reference %s.", reference)
|
|
)
|
|
return tx
|
|
|
|
def _process_notification_data(self, notification_data):
|
|
""" Override of `payment` to process the transaction based on Xendit data.
|
|
|
|
Note: self.ensure_one()
|
|
|
|
:param dict notification_data: The notification data sent by the provider.
|
|
:return: None
|
|
:raise ValidationError: If inconsistent data were received.
|
|
"""
|
|
self.ensure_one()
|
|
|
|
super()._process_notification_data(notification_data)
|
|
if self.provider_code != 'xendit':
|
|
return
|
|
|
|
# Update the provider reference.
|
|
self.provider_reference = notification_data.get('id')
|
|
|
|
# Update payment method.
|
|
payment_method_code = notification_data.get('payment_method', '')
|
|
payment_method = self.env['payment.method']._get_from_code(
|
|
payment_method_code, mapping=const.PAYMENT_METHODS_MAPPING
|
|
)
|
|
self.payment_method_id = payment_method or self.payment_method_id
|
|
|
|
# Update the payment state.
|
|
payment_status = notification_data.get('status')
|
|
if payment_status in const.PAYMENT_STATUS_MAPPING['pending']:
|
|
self._set_pending()
|
|
elif payment_status in const.PAYMENT_STATUS_MAPPING['done']:
|
|
if self.tokenize:
|
|
self._xendit_tokenize_from_notification_data(notification_data)
|
|
self._set_done()
|
|
elif payment_status in const.PAYMENT_STATUS_MAPPING['cancel']:
|
|
self._set_canceled()
|
|
elif payment_status in const.PAYMENT_STATUS_MAPPING['error']:
|
|
failure_reason = notification_data.get('failure_reason')
|
|
self._set_error(_(
|
|
"An error occurred during the processing of your payment (%s). Please try again.",
|
|
failure_reason,
|
|
))
|
|
|
|
def _xendit_tokenize_from_notification_data(self, notification_data):
|
|
""" Create a new token based on the notification data.
|
|
|
|
:param dict notification_data: Xendit's response to a charge API request.
|
|
:return: None
|
|
"""
|
|
card_info = notification_data['masked_card_number'][-4:] # Xendit pads details with X's.
|
|
token_id = notification_data['credit_card_token_id']
|
|
token = self.env['payment.token'].create({
|
|
"provider_id": self.provider_id.id,
|
|
"payment_method_id": self.payment_method_id.id,
|
|
"payment_details": card_info,
|
|
"partner_id": self.partner_id.id,
|
|
"provider_ref": token_id,
|
|
})
|
|
self.write({
|
|
'token_id': token.id,
|
|
'tokenize': False,
|
|
})
|
|
_logger.info(
|
|
"created token with id %(token_id)s for partner with id %(partner_id)s from "
|
|
"transaction with reference %(ref)s",
|
|
{
|
|
'token_id': token.id,
|
|
'partner_id': self.partner_id.id,
|
|
'ref': self.reference,
|
|
},
|
|
)
|