# Part of Odoo. See LICENSE file for full copyright and licensing details. import hashlib import hmac import logging import pprint import requests from odoo import _, fields, models from odoo.exceptions import ValidationError from odoo.addons.payment_razorpay import const _logger = logging.getLogger(__name__) class PaymentProvider(models.Model): _inherit = 'payment.provider' code = fields.Selection( selection_add=[('razorpay', "Razorpay")], ondelete={'razorpay': 'set default'} ) razorpay_key_id = fields.Char( string="Razorpay Key Id", help="The key solely used to identify the account with Razorpay.", required_if_provider='razorpay', ) razorpay_key_secret = fields.Char( string="Razorpay Key Secret", required_if_provider='razorpay', groups='base.group_system', ) razorpay_webhook_secret = fields.Char( string="Razorpay Webhook Secret", required_if_provider='razorpay', groups='base.group_system', ) #=== 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 == 'razorpay').update({ 'support_manual_capture': 'full_only', 'support_refund': 'partial', '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 == 'razorpay': supported_currencies = supported_currencies.filtered( lambda c: c.name in const.SUPPORTED_CURRENCIES ) return supported_currencies def _razorpay_make_request(self, endpoint, payload=None, method='POST'): """ Make a request to Razorpay API at the specified endpoint. Note: self.ensure_one() :param str endpoint: The endpoint to be reached by the request. :param dict payload: The payload of the request. :param str method: The HTTP method of the request. :return The JSON-formatted content of the response. :rtype: dict :raise ValidationError: If an HTTP error occurs. """ self.ensure_one() # TODO: Make api_version a kwarg in master. api_version = self.env.context.get('razorpay_api_version', 'v1') url = f'https://api.razorpay.com/{api_version}/{endpoint}' headers = None if access_token := self._razorpay_get_access_token(): headers = {'Authorization': f'Bearer {access_token}'} auth = (self.razorpay_key_id, self.razorpay_key_secret) if self.razorpay_key_id else None try: if method == 'GET': response = requests.get( url, params=payload, headers=headers, auth=auth, timeout=10, ) else: response = requests.post( url, json=payload, headers=headers, auth=auth, timeout=10, ) try: response.raise_for_status() except requests.exceptions.HTTPError: _logger.exception( "Invalid API request at %s with data:\n%s", url, pprint.pformat(payload), ) raise ValidationError("Razorpay: " + _( "Razorpay gave us the following information: '%s'", response.json().get('error', {}).get('description') )) except (requests.exceptions.ConnectionError, requests.exceptions.Timeout): _logger.exception("Unable to reach endpoint at %s", url) raise ValidationError( "Razorpay: " + _("Could not establish the connection to the API.") ) return response.json() def _razorpay_calculate_signature(self, data): """ Compute the signature for the request's data according to the Razorpay documentation. See https://razorpay.com/docs/webhooks/validate-test#validate-webhooks. :param bytes data: The data to sign. :return: The calculated signature. :rtype: str """ secret = self.razorpay_webhook_secret return hmac.new(secret.encode(), msg=data, digestmod=hashlib.sha256).hexdigest() 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 != 'razorpay': return default_codes return const.DEFAULT_PAYMENT_METHOD_CODES def _get_validation_amount(self): """ Override of `payment` to return the amount for Razorpay validation operations. :return: The validation amount. :rtype: float """ res = super()._get_validation_amount() if self.code != 'razorpay': return res return 1.0 # === BUSINESS METHODS - OAUTH === # def _razorpay_get_public_token(self): # TODO: remove in master self.ensure_one() return None def _razorpay_get_access_token(self): # TODO: remove in master self.ensure_one() return None