# # This file is part of pretix (Community Edition). # # Copyright (C) 2014-2020 Raphael Michel and contributors # Copyright (C) 2020-2021 rami.io GmbH and contributors # # 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 in version 3 of the License. # # ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are # applicable granting you additional permissions and placing additional restrictions on your usage of this software. # Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive # this file, see . # # 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 # . # import json from collections import OrderedDict from django import forms from django.conf import settings from django.dispatch import receiver from django.http import HttpRequest, HttpResponse from django.template.loader import get_template from django.urls import resolve from django.utils.crypto import get_random_string from django.utils.translation import gettext_lazy as _, pgettext_lazy from pretix.base.forms import SecretKeySettingsField from pretix.base.middleware import _merge_csp, _parse_csp, _render_csp from pretix.base.settings import settings_hierarkey from pretix.base.signals import ( logentry_display, register_global_settings, register_payment_providers, ) from pretix.plugins.paypal2.payment import PaypalMethod from pretix.presale.signals import html_head, process_response @receiver(register_payment_providers, dispatch_uid="payment_paypal2") def register_payment_provider(sender, **kwargs): from .payment import PaypalAPM, PaypalSettingsHolder, PaypalWallet return [PaypalSettingsHolder, PaypalWallet, PaypalAPM] @receiver(signal=logentry_display, dispatch_uid="paypal2_logentry_display") def pretixcontrol_logentry_display(sender, logentry, **kwargs): if logentry.action_type != 'pretix.plugins.paypal.event': return data = json.loads(logentry.data) event_type = data.get('event_type') text = None plains = { 'PAYMENT.SALE.COMPLETED': _('Payment completed.'), 'PAYMENT.SALE.DENIED': _('Payment denied.'), 'PAYMENT.SALE.REFUNDED': _('Payment refunded.'), 'PAYMENT.SALE.REVERSED': _('Payment reversed.'), 'PAYMENT.SALE.PENDING': _('Payment pending.'), 'CHECKOUT.ORDER.APPROVED': pgettext_lazy('paypal', 'Order approved.'), 'CHECKOUT.ORDER.COMPLETED': pgettext_lazy('paypal', 'Order completed.'), 'PAYMENT.CAPTURE.COMPLETED': pgettext_lazy('paypal', 'Capture completed.'), 'PAYMENT.CAPTURE.PENDING': pgettext_lazy('paypal', 'Capture pending.'), } if event_type in plains: text = plains[event_type] else: text = event_type if text: return _('PayPal reported an event: {}').format(text) @receiver(register_global_settings, dispatch_uid='paypal2_global_settings') def register_global_settings(sender, **kwargs): return OrderedDict([ ('payment_paypal_connect_client_id', forms.CharField( label=_('PayPal ISU/Connect: Client ID'), required=False, )), ('payment_paypal_connect_secret_key', SecretKeySettingsField( label=_('PayPal ISU/Connect: Secret key'), required=False, )), ('payment_paypal_connect_partner_merchant_id', forms.CharField( label=_('PayPal ISU/Connect: Partner Merchant ID'), help_text=_('This is not the BN-code, but rather the ID of the merchant account which holds branding information for ISU.'), required=False, )), ('payment_paypal_connect_endpoint', forms.ChoiceField( label=_('PayPal ISU/Connect Endpoint'), initial='live', choices=( ('live', 'Live'), ('sandbox', 'Sandbox'), ), )), ]) @receiver(html_head, dispatch_uid="payment_paypal2_html_head") def html_head_presale(sender, request=None, **kwargs): provider = PaypalMethod(sender) url = resolve(request.path_info) if provider.settings.get('_enabled', as_type=bool) and ( url.url_name == "event.order.pay.change" or url.url_name == "event.order.pay" or (url.url_name == "event.checkout" and url.kwargs['step'] == "payment") or (url.namespace == "plugins:paypal2" and url.url_name == "pay") ): provider.init_api() template = get_template('pretixplugins/paypal2/presale_head.html') ctx = { 'client_id': provider.client.environment.client_id, 'merchant_id': provider.client.environment.merchant_id or '', 'csp_nonce': _nonce(request), 'debug': settings.DEBUG, 'settings': provider.settings, # If we ever have more APMs that can be disabled, we should iterate over the # disable_method_*/enable_method*-keys 'disable_funding': 'sepa' if provider.settings.get('disable_method_sepa', as_type=bool) else '', 'enable_funding': 'paylater' if provider.settings.get('enable_method_paylater', as_type=bool) else '' } return template.render(ctx) else: return "" @receiver(signal=process_response, dispatch_uid="payment_paypal2_middleware_resp") def signal_process_response(sender, request: HttpRequest, response: HttpResponse, **kwargs): provider = PaypalMethod(sender) url = resolve(request.path_info) if provider.settings.get('_enabled', as_type=bool) and ( url.url_name == "event.order.pay.change" or url.url_name == "event.order.pay" or (url.url_name == "event.checkout" and url.kwargs['step'] == "payment") or (url.namespace == "plugins:paypal2" and url.url_name == "pay") ): if 'Content-Security-Policy' in response: h = _parse_csp(response['Content-Security-Policy']) else: h = {} csps = { 'script-src': ['https://www.paypal.com', "'nonce-{}'".format(_nonce(request))], # When the stars align in an unpredictable manner and the temperature is just right, the PayPal SDK might # decide to not open a popup for the payment process (which is in turn not tied to our CSP) but instead # use a popover directly on the purchase page. Unfortunately, the latter will be tied to our CSP even when # trying to iframe banking pages such as giropay and SOFORT. # Until PayPal figures a way around this (or at least provides a way to inhibit the popover), we'll allow to # frame any https page only on the pay-page. # Ref: Z#23111577 # 'frame-src': ['https://www.paypal.com', 'https://www.sandbox.paypal.com', "'nonce-{}'".format(_nonce(request))], 'frame-src': ['https:', "'nonce-{}'".format(_nonce(request))], 'connect-src': ['https://www.paypal.com', 'https://www.sandbox.paypal.com'], # Or not - seems to only affect PayPal logging... 'img-src': ['https://t.paypal.com'], 'style-src': ["'unsafe-inline'"] # PayPal does not comply with our nonce unfortunately, see Z#23113213 } _merge_csp(h, csps) if h: response['Content-Security-Policy'] = _render_csp(h) return response settings_hierarkey.add_default('payment_paypal_debug_buyer_country', '', str) settings_hierarkey.add_default('payment_paypal_method_wallet', True, bool) def _nonce(request): if not hasattr(request, "_paypal_nonce"): request._paypal_nonce = get_random_string(32) return request._paypal_nonce