From c3fc3a538f8fea4c078751bee2bb5e1fc4da5fbe Mon Sep 17 00:00:00 2001 From: Can Demiralp Date: Tue, 5 Nov 2024 16:37:18 +0100 Subject: [PATCH] [ECP-9352] Implement onError handler on Amazon Pay component (#2768) * [ECP-9352] Update csp allowed domains * [ECP-9352] Update returnUrl and cancelUrl * [ECP-9352] Add Amazon Pay assets to CSP allow list * [ECP-9352] Implement handleOnError and handleOnFailure callbacks * [ECP-9352] Update error handling --------- Co-authored-by: Can Demiralp --- etc/csp_whitelist.xml | 2 +- view/frontend/web/js/model/adyen-checkout.js | 8 +- .../method-renderer/adyen-amazonpay-method.js | 109 ++++++++++++------ .../method-renderer/adyen-pm-method.js | 28 ++--- 4 files changed, 95 insertions(+), 52 deletions(-) diff --git a/etc/csp_whitelist.xml b/etc/csp_whitelist.xml index 385265dd22..4a6688e513 100644 --- a/etc/csp_whitelist.xml +++ b/etc/csp_whitelist.xml @@ -40,7 +40,7 @@ *.adyen.com pay.google.com payments-eu.amazon.com - payments.amazon.de + *.amazon.de diff --git a/view/frontend/web/js/model/adyen-checkout.js b/view/frontend/web/js/model/adyen-checkout.js index 8b5a138c41..3fbdcc7233 100644 --- a/view/frontend/web/js/model/adyen-checkout.js +++ b/view/frontend/web/js/model/adyen-checkout.js @@ -19,7 +19,13 @@ define( ) { 'use strict'; return { - buildCheckoutComponent: function (paymentMethodsResponse, handleOnAdditionalDetails, handleOnCancel = undefined, handleOnSubmit = undefined, handleOnError = undefined) { + buildCheckoutComponent: function ( + paymentMethodsResponse, + handleOnAdditionalDetails, + handleOnCancel = undefined, + handleOnSubmit = undefined, + handleOnError = undefined + ) { if (!!paymentMethodsResponse.paymentMethodsResponse) { return AdyenCheckout({ locale: adyenConfiguration.getLocale(), diff --git a/view/frontend/web/js/view/payment/method-renderer/adyen-amazonpay-method.js b/view/frontend/web/js/view/payment/method-renderer/adyen-amazonpay-method.js index b8fe8d38df..3111c8b501 100644 --- a/view/frontend/web/js/view/payment/method-renderer/adyen-amazonpay-method.js +++ b/view/frontend/web/js/view/payment/method-renderer/adyen-amazonpay-method.js @@ -11,24 +11,33 @@ define( [ 'Magento_Checkout/js/model/quote', 'Adyen_Payment/js/view/payment/method-renderer/adyen-pm-method', - 'Adyen_Payment/js/model/adyen-checkout' + 'Adyen_Payment/js/model/adyen-checkout', + 'Adyen_Payment/js/model/adyen-payment-service', + 'mage/url' ], function( quote, adyenPaymentMethod, - adyenCheckout + adyenCheckout, + adyenPaymentService, + urlBuilder ) { const amazonSessionKey = 'amazonCheckoutSessionId'; return adyenPaymentMethod.extend({ placeOrderButtonVisible: false, - initialize: function () { - this._super(); - }, + amazonPayComponent: null, + buildComponentConfiguration: function (paymentMethod, paymentMethodsExtraInfo) { let self = this; let formattedShippingAddress = {}; let formattedBillingAddress = {}; let baseComponentConfiguration = this._super(); + + baseComponentConfiguration = Object.assign( + baseComponentConfiguration, + paymentMethodsExtraInfo[paymentMethod.type].configuration + ); + if (!quote.isVirtual() && !!quote.shippingAddress()) { formattedShippingAddress = self.getFormattedAddress(quote.shippingAddress()); } @@ -36,7 +45,7 @@ define( if (!!quote.billingAddress()) { formattedBillingAddress = self.getFormattedAddress(quote.billingAddress()); } - baseComponentConfiguration.showPayButton = true; + baseComponentConfiguration.onClick = function(resolve,reject) { if (self.validate()) { resolve(); @@ -44,19 +53,15 @@ define( reject(); } } - baseComponentConfiguration = Object.assign(baseComponentConfiguration, paymentMethodsExtraInfo[paymentMethod.type].configuration); + baseComponentConfiguration.productType = 'PayAndShip'; baseComponentConfiguration.checkoutMode = 'ProcessOrder'; - let url = new URL(location.href); - url.searchParams.delete(amazonSessionKey); - baseComponentConfiguration.returnUrl = url.href; - baseComponentConfiguration.onSubmit = async (state, amazonPayComponent) => { - try { - await self.handleOnSubmit(state.data, amazonPayComponent); - } catch (error) { - amazonPayComponent.handleDeclineFlow(); - } - }; + baseComponentConfiguration.showPayButton = true; + + // Redirect shoppers to the cart page if they cancel the payment on Amazon Pay hosted page. + baseComponentConfiguration.cancelUrl = urlBuilder.build('checkout/cart'); + // Redirect shoppers to the checkout if they complete the payment on Amazon Pay hosted page. + baseComponentConfiguration.returnUrl = urlBuilder.build('checkout/#payment'); if (formattedShippingAddress && formattedShippingAddress.telephone) { @@ -71,6 +76,7 @@ define( countryCode: formattedShippingAddress.country, phoneNumber: formattedShippingAddress.telephone }; + if (baseComponentConfiguration.addressDetails.countryCode === 'US') { baseComponentConfiguration.addressDetails.stateOrRegion = quote.shippingAddress().regionCode } @@ -88,21 +94,26 @@ define( countryCode: formattedBillingAddress.country, phoneNumber: formattedBillingAddress.telephone }; + if (baseComponentConfiguration.addressDetails.countryCode === 'US') { baseComponentConfiguration.addressDetails.stateOrRegion = quote.billingAddress().regionCode } } + return baseComponentConfiguration; }, mountPaymentMethodComponent: function (paymentMethod, configuration) { - let self = this; const containerId = '#' + paymentMethod.type + 'Container'; - let url = new URL(location.href); - //Handles the redirect back to checkout page with amazonSessionKey in url - if (url.searchParams.has(amazonSessionKey)) { + const currentUrl = new URL(location.href); + + /* + * If the first redirect is successful and URL contains `amazonCheckoutSessionId` parameter, + * don't mount the default component but mount the second component to submit the `/payments` request. + */ + if (currentUrl.searchParams.has(amazonSessionKey)) { let componentConfig = { - amazonCheckoutSessionId: url.searchParams.get(amazonSessionKey), + amazonCheckoutSessionId: currentUrl.searchParams.get(amazonSessionKey), showOrderButton: false, amount: { currency: configuration.amount.currency, @@ -110,23 +121,47 @@ define( }, showChangePaymentDetailsButton: false } - try { - const amazonPayComponent = adyenCheckout.mountPaymentMethodComponent( // This mountPaymentMethodCOmponent isn't up to date. - self.checkoutComponent, - 'amazonpay', - componentConfig, - containerId - ); - amazonPayComponent.submit(); - } catch (err) { - // The component does not exist yet - if ('test' === adyenConfiguration.getCheckoutEnvironment()) { - console.log(err); - } - } - } else{ + + this.amazonPayComponent = adyenCheckout.mountPaymentMethodComponent( + this.checkoutComponent, + 'amazonpay', + componentConfig, + containerId + ); + + // Triggers `onSubmit` event and `handleOnSubmit()` callback in adyen-pm-method.js handles it + this.amazonPayComponent.submit(); + } else { this._super(); } + }, + + /* + * Try to handle decline flow if Amazon Pay session allows in case of `/payments` call fails. + * If decline flow is available, shopper will be redirected to Amazon Pay hosted page again. + * If handle decline flow is not present, the component will throw `onError` event. + */ + handleOnFailure: function (response, component) { + this.amazonPayComponent.handleDeclineFlow(); + }, + + /* + * If `handleDeclineFlow()` can not be handled for any reason, `onError` will be thrown. + * In this case, remove `amazonCheckoutSessionId` from the URL and remount the payment component. + */ + handleOnError: function (error, component) { + this.remountAmazonPayComponent(); + }, + + /* + * Remove `amazonCheckoutSessionId` from the URL and remount the component. + */ + remountAmazonPayComponent: function () { + const checkoutPaymentUrl = "checkout/#payment"; + window.history.pushState({}, document.title, "/" + checkoutPaymentUrl); + + const paymentMethodsResponse = adyenPaymentService.getPaymentMethods(); + this.createCheckoutComponent(paymentMethodsResponse()); } }) } diff --git a/view/frontend/web/js/view/payment/method-renderer/adyen-pm-method.js b/view/frontend/web/js/view/payment/method-renderer/adyen-pm-method.js index a5609a8adc..604f5afe5b 100755 --- a/view/frontend/web/js/view/payment/method-renderer/adyen-pm-method.js +++ b/view/frontend/web/js/view/payment/method-renderer/adyen-pm-method.js @@ -180,6 +180,21 @@ define( self.isPlaceOrderAllowed(true); }); }, + + handleOnError: function (error, component) { + /* + * Passing false as the response to hide the actual error message from the shopper for security. + * This will show a generic error message instead of the actual error message. + */ + this.handleOnFailure(error, component); + }, + + handleOnFailure: function(error, component) { + this.isPlaceOrderAllowed(true); + fullScreenLoader.stopLoader(); + errorProcessor.process(error, this.currentMessageContainer); + }, + renderCheckoutComponent: function() { let methodCode = this.getMethodCode(); @@ -382,19 +397,6 @@ define( } }, - handleOnError: function (error, component) { - /* - * Passing false as the response to hide the actual error message from the shopper for security. - * This will show a generic error message instead of the actual error message. - */ - this.handleOnFailure(error, component); - }, - handleOnFailure: function(response, component) { - this.isPlaceOrderAllowed(true); - fullScreenLoader.stopLoader(); - errorProcessor.process(response, this.currentMessageContainer); - }, - /** * This method is a workaround to close the modal in the right way and reconstruct the ActionModal. * This will solve issues when you cancel the 3DS2 challenge and retry the payment