summaryrefslogtreecommitdiff
path: root/addons/pos_adyen/models/pos_payment_method.py
diff options
context:
space:
mode:
authorstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
committerstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
commit3751379f1e9a4c215fb6eb898b4ccc67659b9ace (patch)
treea44932296ef4a9b71d5f010906253d8c53727726 /addons/pos_adyen/models/pos_payment_method.py
parent0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (diff)
initial commit 2
Diffstat (limited to 'addons/pos_adyen/models/pos_payment_method.py')
-rw-r--r--addons/pos_adyen/models/pos_payment_method.py148
1 files changed, 148 insertions, 0 deletions
diff --git a/addons/pos_adyen/models/pos_payment_method.py b/addons/pos_adyen/models/pos_payment_method.py
new file mode 100644
index 00000000..73c09287
--- /dev/null
+++ b/addons/pos_adyen/models/pos_payment_method.py
@@ -0,0 +1,148 @@
+# coding: utf-8
+# Part of Odoo. See LICENSE file for full copyright and licensing details.
+import json
+import logging
+import pprint
+import random
+import requests
+import string
+from werkzeug.exceptions import Forbidden
+
+from odoo import fields, models, api, _
+from odoo.exceptions import ValidationError
+
+_logger = logging.getLogger(__name__)
+
+class PosPaymentMethod(models.Model):
+ _inherit = 'pos.payment.method'
+
+ def _get_payment_terminal_selection(self):
+ return super(PosPaymentMethod, self)._get_payment_terminal_selection() + [('odoo_adyen', 'Odoo Payments by Adyen'), ('adyen', 'Adyen')]
+
+ # Adyen
+ adyen_api_key = fields.Char(string="Adyen API key", help='Used when connecting to Adyen: https://docs.adyen.com/user-management/how-to-get-the-api-key/#description', copy=False)
+ adyen_terminal_identifier = fields.Char(help='[Terminal model]-[Serial number], for example: P400Plus-123456789', copy=False)
+ adyen_test_mode = fields.Boolean(help='Run transactions in the test environment.')
+
+ # Odoo Payments by Adyen
+ adyen_account_id = fields.Many2one('adyen.account', related='company_id.adyen_account_id')
+ adyen_payout_id = fields.Many2one('adyen.payout', string='Adyen Payout', domain="[('adyen_account_id', '=', adyen_account_id)]")
+ adyen_terminal_id = fields.Many2one('adyen.terminal', string='Adyen Terminal', domain="[('adyen_account_id', '=', adyen_account_id)]")
+
+ adyen_latest_response = fields.Char(help='Technical field used to buffer the latest asynchronous notification from Adyen.', copy=False, groups='base.group_erp_manager')
+ adyen_latest_diagnosis = fields.Char(help='Technical field used to determine if the terminal is still connected.', copy=False, groups='base.group_erp_manager')
+
+ @api.constrains('adyen_terminal_identifier')
+ def _check_adyen_terminal_identifier(self):
+ for payment_method in self:
+ if not payment_method.adyen_terminal_identifier:
+ continue
+ existing_payment_method = self.search([('id', '!=', payment_method.id),
+ ('adyen_terminal_identifier', '=', payment_method.adyen_terminal_identifier)],
+ limit=1)
+ if existing_payment_method:
+ raise ValidationError(_('Terminal %s is already used on payment method %s.')
+ % (payment_method.adyen_terminal_identifier, existing_payment_method.display_name))
+
+ def _get_adyen_endpoints(self):
+ return {
+ 'terminal_request': 'https://terminal-api-%s.adyen.com/async',
+ }
+
+ @api.onchange('adyen_terminal_id')
+ def onchange_use_payment_terminal(self):
+ for payment_method in self:
+ if payment_method.use_payment_terminal == 'odoo_adyen' and payment_method.adyen_terminal_id:
+ payment_method.adyen_terminal_identifier = payment_method.adyen_terminal_id.terminal_uuid
+
+ def _is_write_forbidden(self, fields):
+ whitelisted_fields = set(('adyen_latest_response', 'adyen_latest_diagnosis'))
+ return super(PosPaymentMethod, self)._is_write_forbidden(fields - whitelisted_fields)
+
+ def _adyen_diagnosis_request_data(self, pos_config_name):
+ service_id = ''.join(random.choices(string.ascii_letters + string.digits, k=10))
+ return {
+ "SaleToPOIRequest": {
+ "MessageHeader": {
+ "ProtocolVersion": "3.0",
+ "MessageClass": "Service",
+ "MessageCategory": "Diagnosis",
+ "MessageType": "Request",
+ "ServiceID": service_id,
+ "SaleID": pos_config_name,
+ "POIID": self.adyen_terminal_identifier,
+ },
+ "DiagnosisRequest": {
+ "HostDiagnosisFlag": False
+ }
+ }
+ }
+
+ def get_latest_adyen_status(self, pos_config_name):
+ self.ensure_one()
+
+ # Poll the status of the terminal if there's no new
+ # notification we received. This is done so we can quickly
+ # notify the user if the terminal is no longer reachable due
+ # to connectivity issues.
+ self.proxy_adyen_request(self._adyen_diagnosis_request_data(pos_config_name))
+
+ latest_response = self.sudo().adyen_latest_response
+ latest_response = json.loads(latest_response) if latest_response else False
+ self.sudo().adyen_latest_response = '' # avoid handling old responses multiple times
+
+ return {
+ 'latest_response': latest_response,
+ 'last_received_diagnosis_id': self.sudo().adyen_latest_diagnosis,
+ }
+
+ def proxy_adyen_request(self, data, operation=False):
+ ''' Necessary because Adyen's endpoints don't have CORS enabled '''
+ if not operation:
+ operation = 'terminal_request'
+
+ if self.use_payment_terminal == 'odoo_adyen':
+ return self._proxy_adyen_request_odoo_proxy(data, operation)
+ else:
+ return self._proxy_adyen_request_direct(data, operation)
+
+ def _proxy_adyen_request_direct(self, data, operation):
+ self.ensure_one()
+ TIMEOUT = 10
+
+ _logger.info('request to adyen\n%s', pprint.pformat(data))
+
+ environment = 'test' if self.adyen_test_mode else 'live'
+ endpoint = self._get_adyen_endpoints()[operation] % environment
+ headers = {
+ 'x-api-key': self.adyen_api_key,
+ }
+ req = requests.post(endpoint, json=data, headers=headers, timeout=TIMEOUT)
+
+ # Authentication error doesn't return JSON
+ if req.status_code == 401:
+ return {
+ 'error': {
+ 'status_code': req.status_code,
+ 'message': req.text
+ }
+ }
+
+ if req.text == 'ok':
+ return True
+
+ return req.json()
+
+ def _proxy_adyen_request_odoo_proxy(self, data, operation):
+ try:
+ return self.env.company.sudo().adyen_account_id._adyen_rpc(operation, {
+ 'request_data': data,
+ 'account_code': self.sudo().adyen_payout_id.code,
+ 'notification_url': self.env['ir.config_parameter'].sudo().get_param('web.base.url'),
+ })
+ except Forbidden:
+ return {
+ 'error': {
+ 'status_code': 401,
+ }
+ }