diff --git a/Api/Transaction/Data/DataInterface.php b/Api/Transaction/Data/DataInterface.php
old mode 100644
new mode 100755
index 4999ac1..d4fab38
--- a/Api/Transaction/Data/DataInterface.php
+++ b/Api/Transaction/Data/DataInterface.php
@@ -20,6 +20,7 @@ interface DataInterface extends ExtensibleDataInterface
*/
public const ENTITY_ID = 'entity_id';
public const QUOTE_ID = 'quote_id';
+ public const AMOUNT = 'amount';
public const ORDER_ID = 'order_id';
public const UUID = 'uuid';
public const TOKEN = 'token';
@@ -50,6 +51,21 @@ public function getQuoteId(): ?int;
*/
public function setQuoteId(int $quoteId): self;
+ /**
+ * Returns transaction amount.
+ *
+ * @return int quote ID.
+ */
+ public function getAmount(): int;
+
+ /**
+ * Sets transaction amount.
+ *
+ * @param int $amount
+ * @return $this
+ */
+ public function setAmount(int $amount): self;
+
/**
* Returns the order ID.
*
diff --git a/Model/Transaction/DataModel.php b/Model/Transaction/DataModel.php
old mode 100644
new mode 100755
index a373ffe..7bf493b
--- a/Model/Transaction/DataModel.php
+++ b/Model/Transaction/DataModel.php
@@ -51,6 +51,22 @@ public function setQuoteId(int $quoteId): DataInterface
return $this->setData(self::QUOTE_ID, $quoteId);
}
+ /**
+ * @inheritDoc
+ */
+ public function getAmount(): int
+ {
+ return (int)$this->getData(self::AMOUNT);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ public function setAmount(int $amount): DataInterface
+ {
+ return $this->setData(self::AMOUNT, $amount);
+ }
+
/**
* @inheritDoc
*/
diff --git a/Model/Webapi/Checkout.php b/Model/Webapi/Checkout.php
old mode 100644
new mode 100755
index c29473c..cdc4b23
--- a/Model/Webapi/Checkout.php
+++ b/Model/Webapi/Checkout.php
@@ -70,8 +70,8 @@ public function orderRequest(bool $isLoggedIn, string $cartId)
//web api can't return first level associative array
$return = [];
try {
- $paymentUrl = $this->orderRequest->execute($token);
- $return['response'] = ['success' => true, 'payment_page_url' => $paymentUrl];
+ $response = $this->orderRequest->execute($token);
+ $return['response'] = ['success' => true, 'response' => $response];
return $return;
} catch (\Exception $exception) {
$this->logRepository->addErrorLog('Checkout endpoint', $exception->getMessage());
diff --git a/Service/Order/MakeRequest.php b/Service/Order/MakeRequest.php
index bb13f1b..588538e 100644
--- a/Service/Order/MakeRequest.php
+++ b/Service/Order/MakeRequest.php
@@ -124,7 +124,7 @@ public function __construct(
* Executes TrueLayer Api for Order Request and returns redirect to platform Url
*
* @param string $token
- * @return string
+ * @return array
* @throws AuthenticationException
* @throws CouldNotSaveException
* @throws InputException
@@ -133,7 +133,7 @@ public function __construct(
* @throws \TrueLayer\Exceptions\InvalidArgumentException
* @throws \TrueLayer\Exceptions\ValidationException
*/
- public function execute(string $token): string
+ public function execute(string $token)
{
$transaction = $this->transactionRepository->getByToken($token);
$quote = $this->quoteRepository->get($transaction->getQuoteId());
@@ -163,15 +163,15 @@ public function execute(string $token): string
}
if ($payment->getId()) {
- $transaction->setUuid($payment->getId());
+ $transaction->setUuid($payment->getId())
+ ->setAmount($paymentData['amount_in_minor']);
$this->transactionRepository->save($transaction);
- $this->duplicateCurrentQuote($quote);
- return $payment->hostedPaymentsPage()
- ->returnUri($this->getReturnUrl())
- ->primaryColour($this->configProvider->getPaymentPagePrimaryColor())
- ->secondaryColour($this->configProvider->getPaymentPageSecondaryColor())
- ->tertiaryColour($this->configProvider->getPaymentPageTertiaryColor())
- ->toUrl();
+
+ return [
+ 'payment_id' => $payment->getId(),
+ 'resource_token' => $payment->getResourceToken(),
+ 'transaction_id' => $transaction->getUuid(),
+ ];
}
$msg = self::REQUEST_EXCEPTION;
@@ -233,7 +233,8 @@ private function prepareData(Quote $quote, string $merchantAccountId): array
"type" => "merchant_account",
"name" => $this->configProvider->getMerchantAccountName(),
"merchant_account_id" => $merchantAccountId
- ]
+ ],
+ "retry" => []
],
"user" => [
"name" => trim($quote->getBillingAddress()->getFirstname()) .
@@ -252,73 +253,4 @@ private function prepareData(Quote $quote, string $merchantAccountId): array
return $data;
}
-
- /**
- * Duplicate current quote and set this as active session.
- * This prevents quotes to change during checkout process
- *
- * @param Quote $quote
- * @throws NoSuchEntityException
- * @throws CouldNotSaveException
- */
- private function duplicateCurrentQuote(Quote $quote)
- {
- $quote->setIsActive(false);
- $this->quoteRepository->save($quote);
- if ($customerId = $quote->getCustomerId()) {
- $cartId = $this->cartManagement->createEmptyCartForCustomer($customerId);
- } else {
- $cartId = $this->cartManagement->createEmptyCart();
- }
- $newQuote = $this->quoteRepository->get($cartId);
- $newQuote->merge($quote);
-
- $newQuote->removeAllAddresses();
- if (!$quote->getIsVirtual()) {
- $addressData = $this->dataObjectProcessor->buildOutputDataArray(
- $quote->getShippingAddress(),
- AddressInterface::class
- );
- unset($addressData['id']);
- $shippingAddress = $this->quoteAddressFactory->create();
- $this->dataObjectHelper->populateWithArray(
- $shippingAddress,
- $addressData,
- AddressInterface::class
- );
- $newQuote->setShippingAddress(
- $shippingAddress
- );
- }
-
- $addressData = $this->dataObjectProcessor->buildOutputDataArray(
- $quote->getBillingAddress(),
- AddressInterface::class
- );
- unset($addressData['id']);
- $billingAddress = $this->quoteAddressFactory->create();
- $this->dataObjectHelper->populateWithArray(
- $billingAddress,
- $addressData,
- AddressInterface::class
- );
- $newQuote->setBillingAddress(
- $billingAddress
- );
-
- $newQuote->setTotalsCollectedFlag(false)->collectTotals();
- $this->quoteRepository->save($newQuote);
-
- $this->checkoutSession->replaceQuote($newQuote);
- }
-
- /**
- * Get return url
- *
- * @return string
- */
- private function getReturnUrl(): string
- {
- return $this->configProvider->getBaseUrl() . 'truelayer/checkout/process/';
- }
}
diff --git a/Service/Order/ProcessWebhook.php b/Service/Order/ProcessWebhook.php
index 3a01e08..2810a8d 100644
--- a/Service/Order/ProcessWebhook.php
+++ b/Service/Order/ProcessWebhook.php
@@ -9,6 +9,7 @@
use Exception;
use Magento\Checkout\Model\Session as CheckoutSession;
+use Magento\Framework\Exception\LocalizedException;
use Magento\Quote\Api\CartManagementInterface;
use Magento\Quote\Api\CartRepositoryInterface;
use Magento\Quote\Api\Data\CartInterface;
@@ -29,6 +30,7 @@ class ProcessWebhook
{
public const SUCCESS_MSG = 'Order #%1 successfully captured on TrueLayer';
+
/**
* @var CheckoutSession
*/
@@ -127,11 +129,19 @@ public function execute(string $uuid, string $userId)
);
if (!$quoteId = $transaction->getQuoteId()) {
- $this->logRepository->addDebugLog('webhook', 'no quote id found in transaction');
- return;
+ $message = 'No quote id found in transaction';
+ $this->logRepository->addDebugLog('webhook', $message);
+ throw new LocalizedException(__($message));
}
$quote = $this->quoteRepository->get($quoteId);
+
+ if ($transaction->getAmount() != (int)bcmul((string)$quote->getBaseGrandTotal(), '100')) {
+ $message = 'Quote amount was changed';
+ $this->logRepository->addDebugLog('webhook', $message);
+ throw new LocalizedException(__($message));
+ }
+
$this->checkoutSession->setQuoteId($quoteId);
if (!$this->transactionRepository->isLocked($transaction)) {
@@ -150,6 +160,7 @@ public function execute(string $uuid, string $userId)
}
} catch (Exception $e) {
$this->logRepository->addDebugLog('webhook exception', $e->getMessage());
+ throw new LocalizedException(__($e->getMessage()));
}
}
@@ -176,7 +187,7 @@ private function placeOrder(CartInterface $quote, $uuid, $userId)
$this->sendInvoiceEmail($order);
} catch (Exception $e) {
$this->logRepository->addDebugLog('place order', $e->getMessage());
- return false;
+ throw new LocalizedException(__($e->getMessage()));
}
return $order->getEntityId();
diff --git a/etc/csp_whitelist.xml b/etc/csp_whitelist.xml
new file mode 100644
index 0000000..dca2459
--- /dev/null
+++ b/etc/csp_whitelist.xml
@@ -0,0 +1,10 @@
+
+
+
+
+
+ https://cdn.jsdelivr.net
+
+
+
+
\ No newline at end of file
diff --git a/etc/db_schema.xml b/etc/db_schema.xml
old mode 100644
new mode 100755
index 7cbd0df..6c37177
--- a/etc/db_schema.xml
+++ b/etc/db_schema.xml
@@ -8,6 +8,7 @@
+
diff --git a/view/frontend/layout/checkout_index_index.xml b/view/frontend/layout/checkout_index_index.xml
old mode 100644
new mode 100755
index bbe1509..c153c3c
--- a/view/frontend/layout/checkout_index_index.xml
+++ b/view/frontend/layout/checkout_index_index.xml
@@ -6,9 +6,6 @@
*/
-->
-
-
-
diff --git a/view/frontend/requirejs-config.js b/view/frontend/requirejs-config.js
new file mode 100644
index 0000000..1b7a077
--- /dev/null
+++ b/view/frontend/requirejs-config.js
@@ -0,0 +1,8 @@
+var config = {
+ map: {
+ '*': {
+ 'Magento_SalesRule/js/action/select-payment-method-mixin': 'TrueLayer_Connect/js/action/select-payment-method-mixin'
+ }
+ }
+};
+
diff --git a/view/frontend/web/css/source/_module.less b/view/frontend/web/css/source/_module.less
new file mode 100644
index 0000000..a7be553
--- /dev/null
+++ b/view/frontend/web/css/source/_module.less
@@ -0,0 +1 @@
+@import 'component/_widget.less';
\ No newline at end of file
diff --git a/view/frontend/web/css/source/component/_widget.less b/view/frontend/web/css/source/component/_widget.less
new file mode 100644
index 0000000..6c2aec5
--- /dev/null
+++ b/view/frontend/web/css/source/component/_widget.less
@@ -0,0 +1,63 @@
+//
+// Common
+// _____________________________________________
+
+& when (@media-common = true) {
+
+ // Modify base payment method styles for truelayer method
+ .truelayer-payment-method {
+ .payment-method-title {
+ label {
+ position: relative;
+ padding-left: 32px;
+
+ &::before {
+ content: '';
+
+ position: absolute;
+ top: -6px;
+ left: 0;
+
+ width: 24px;
+ height: 24px;
+ background: url("TrueLayer_Connect::images/bank.svg") no-repeat center center;
+ }
+ }
+ }
+
+ .payment-method-content {
+ iframe {
+ border: 0;
+ }
+
+ .small [name="tl-checkout-widget"],
+ .small .loader {
+ height: 60px;
+ margin-bottom: 18px;
+ }
+
+ .large [name="tl-checkout-widget"],
+ .large .loader {
+ min-height: 160px;
+ margin-bottom: 18px;
+ }
+
+ .widget-button {
+ position: relative;
+ z-index: 10;
+ max-width: 450px;
+ }
+
+ .loader {
+ display: block;
+ background: url("@{baseDir}images/loader-1.gif") no-repeat center center;
+ background-size: 36px;
+ }
+ }
+
+ .message.success,
+ .message.info {
+ margin-bottom: 24px;
+ }
+ }
+}
diff --git a/view/frontend/web/css/styles.css b/view/frontend/web/css/styles.css
deleted file mode 100644
index 3dfed2e..0000000
--- a/view/frontend/web/css/styles.css
+++ /dev/null
@@ -1,17 +0,0 @@
-input#truelayer {
- margin-right: 40px;
-}
-
-input#truelayer:before {
- content: '';
- display: block;
- background-image: url("");
- background-position: 0px 10px;
- background-size: 25px;
- background-repeat: no-repeat;
- width: 60px;
- height: 35px;
- float: left;
- margin-top: -15px;
- margin-left: 25px;
-}
diff --git a/view/frontend/web/images/bank.svg b/view/frontend/web/images/bank.svg
new file mode 100644
index 0000000..8c5ff58
--- /dev/null
+++ b/view/frontend/web/images/bank.svg
@@ -0,0 +1,19 @@
+
+
+
diff --git a/view/frontend/web/js/action/select-payment-method-mixin.js b/view/frontend/web/js/action/select-payment-method-mixin.js
new file mode 100644
index 0000000..c2fb586
--- /dev/null
+++ b/view/frontend/web/js/action/select-payment-method-mixin.js
@@ -0,0 +1,56 @@
+/**
+ * Copyright © Magento, Inc. All rights reserved.
+ * See COPYING.txt for license details.
+ */
+define([
+ 'jquery',
+ 'mage/utils/wrapper',
+ 'Magento_Checkout/js/model/quote',
+ 'Magento_SalesRule/js/model/payment/discount-messages',
+ 'Magento_Checkout/js/action/set-payment-information-extended',
+ 'Magento_Checkout/js/action/get-totals',
+ 'Magento_SalesRule/js/model/coupon'
+], function ($, wrapper, quote, messageContainer, setPaymentInformationExtended, getTotalsAction, coupon) {
+ 'use strict';
+
+ return function (selectPaymentMethodAction) {
+
+ return wrapper.wrap(selectPaymentMethodAction, function (originalSelectPaymentMethodAction, paymentMethod) {
+
+ originalSelectPaymentMethodAction(paymentMethod);
+
+ if (paymentMethod === null) {
+ return;
+ }
+
+ $.when(
+ // Fix: Changed true to false for getting Billing address
+ setPaymentInformationExtended(
+ messageContainer,
+ {
+ method: paymentMethod.method
+ },
+ false
+ )
+ ).done(
+ function () {
+ var deferred = $.Deferred(),
+
+ /**
+ * Update coupon form.
+ */
+ updateCouponCallback = function () {
+ if (quote.totals() && !quote.totals()['coupon_code']) {
+ coupon.setCouponCode('');
+ coupon.setIsApplied(false);
+ }
+ };
+
+ getTotalsAction([], deferred);
+ $.when(deferred).done(updateCouponCallback);
+ }
+ );
+ });
+ };
+
+});
diff --git a/view/frontend/web/js/view/payment/method-renderer/truelayer.js b/view/frontend/web/js/view/payment/method-renderer/truelayer.js
old mode 100644
new mode 100755
index 71b9164..8cc4785
--- a/view/frontend/web/js/view/payment/method-renderer/truelayer.js
+++ b/view/frontend/web/js/view/payment/method-renderer/truelayer.js
@@ -1,113 +1,129 @@
-/**
- * Copyright © TrueLayer Ltd. All rights reserved.
- * See COPYING.txt for license details.
- */
-/*browser:true*/
-/*global define*/
-define(
- [
- 'jquery',
- 'Magento_Checkout/js/view/payment/default',
- 'Magento_Checkout/js/model/error-processor',
- 'Magento_Checkout/js/model/quote',
- 'Magento_Customer/js/model/customer',
- 'Magento_Checkout/js/model/url-builder',
- 'Magento_Checkout/js/model/full-screen-loader',
- 'mage/storage',
- 'Magento_Ui/js/model/messageList',
- 'Magento_Checkout/js/model/payment/additional-validators',
- 'uiRegistry'
- ],
- function ($, Component, errorProcessor, quote, customer, urlBuilder, fullScreenLoader, storage, messageList, additionalValidators, uiRegistry) {
- 'use strict';
-
- var payload = '';
-
- return Component.extend({
- defaults: {
- template: 'TrueLayer_Connect/payment/truelayer'
- },
-
- getCode: function() {
- return 'truelayer';
- },
-
- placeOrder: function (data, event) {
- if (event) {
- event.preventDefault();
- }
+define([
+ 'ko',
+ 'Magento_Checkout/js/view/payment/default',
+ 'Magento_Checkout/js/model/quote',
+ 'Magento_Customer/js/model/customer',
+ 'Magento_Checkout/js/model/step-navigator',
+ 'Magento_Checkout/js/action/set-payment-information',
+ 'mage/translate',
+ 'Magento_Ui/js/model/messageList',
+ 'https://cdn.jsdelivr.net/npm/truelayer-web-sdk/dist/sdk.min.js',
+], function (ko, Component, quote, customer, stepNavigator, setPaymentInformation, $t, messageList, truelayerSdk) {
+ 'use strict';
+
+ return Component.extend({
+ defaults: {
+ template: 'TrueLayer_Connect/payment/truelayer',
+ successMessage: ko.observable(''),
+ infoMessage: ko.observable(''),
+ buttonSize: ko.observable('small'), // 'small' or 'large',
+ isWidgetInit: ko.observable(false),
+ orderResult: null,
+ doneFuncCompletedProtection: ko.observable(0), // Fix: savety against infinite requests
+ },
+
+ initialize() {
+ this._super();
+
+ this.isPaymentStepLoaded();
+ stepNavigator.steps.subscribe(() => this.isPaymentStepLoaded());
+
+ return this;
+ },
- this.isPlaceOrderActionAllowed(false);
- var _this = this;
+ isPaymentStepLoaded() {
+ const steps = stepNavigator.steps();
+ const payment = steps.find((step) => step.code === 'payment');
- if (additionalValidators.validate()) {
- fullScreenLoader.startLoader();
- _this._placeOrder();
+ if (payment && payment.isVisible()) {
+ // Fix: Correcting a duplicate request "set-payment-information"
+ // Trigger prompt on page load when no payment method is selected
+ if (!quote.paymentMethod()) {
+ setPaymentInformation(messageList, { method: this.getCode() }, false);
}
- },
+
+ const timeout = setTimeout(async () => {
+ this.orderResult = await this.orderRequest(customer.isLoggedIn(), quote.getQuoteId());
+ this.initWidget();
- _placeOrder: function () {
- return this.setPaymentInformation().done(function () {
- this.orderRequest(customer.isLoggedIn(), quote.getQuoteId());
- }.bind(this));
- },
+ clearTimeout(timeout);
+ }, 500);
+ }
+ },
- setPaymentInformation: function() {
- var serviceUrl, payload;
+ getCode: () => 'truelayer',
- payload = {
- cartId: quote.getQuoteId(),
- billingAddress: quote.billingAddress(),
- paymentMethod: this.getData()
- };
+ async orderRequest(isLoggedIn, cartId) {
+ try {
+ const response = await fetch(`${window.location.origin}/rest/V1/truelayer/order-request`, {
+ method: 'POST',
+ headers: {
+ 'Accept': 'application/json',
+ 'Content-Type': 'application/json'
+ },
+ body: JSON.stringify({
+ isLoggedIn,
+ cartId,
+ paymentMethod: this.getData(),
+ }),
+ });
+
+ const result = await response.json();
- if (customer.isLoggedIn()) {
- serviceUrl = urlBuilder.createUrl('/carts/mine/set-payment-information', {});
+ if (result[0].success) {
+ return result[0].response;
} else {
- payload.email = quote.guestEmail;
- serviceUrl = urlBuilder.createUrl('/guest-carts/:quoteId/set-payment-information', {
- quoteId: quote.getQuoteId()
- });
+ throw new Error($t('Failed getting order request result.'));
}
+ } catch (error) {
+ messageList.addErrorMessage({ message: error });
+ }
+ },
- return storage.post(
- serviceUrl, JSON.stringify(payload)
- );
- },
-
- orderRequest: function(isLoggedIn, cartId) {
- var url = 'rest/V1/truelayer/order-request';
-
- payload = {
- isLoggedIn: isLoggedIn,
- cartId: cartId,
- paymentMethod: this.getData()
- };
-
- storage.post(
- url,
- JSON.stringify(payload)
- ).done(function (response) {
- if (response[0].success) {
- fullScreenLoader.stopLoader();
- window.location.replace(response[0].payment_page_url);
- } else {
- fullScreenLoader.stopLoader();
- this.addError(response[0].message);
- }
- }.bind(this));
- },
-
- /**
- * Adds error message
- *
- * @param {String} message
- */
- addError: function (message) {
- messageList.addErrorMessage({
- message: message
- });
- },
- });
- }
-);
+ initWidget() {
+ this.isWidgetInit(true);
+ this.doneFuncCompletedProtection(0);
+
+ truelayerSdk.initWebSdk({
+ uiSettings: {
+ size: this.buttonSize(),
+ recommendedPaymentMethod: true,
+ },
+
+ onError: (error) => messageList.addErrorMessage({ message: $t('Widget error: ') + error }),
+ onDone: (response) => this.onDone(response),
+ })
+ .mount(document.getElementById('truelayer-widget-iframe'))
+ .start({
+ paymentId: this.orderResult['payment_id'],
+ resourceToken: this.orderResult['resource_token'],
+ });
+ },
+
+ async onDone(response) {
+ this.doneFuncCompletedProtection(this.doneFuncCompletedProtection() + 1);
+
+ if (this.doneFuncCompletedProtection() === 1) {
+ if (response.resultStatus === 'success') {
+ this.successMessage($t('Payment success: page will reload.'));
+ window.location.replace(`/truelayer/checkout/process/payment_id/${this.orderResult['transaction_id']}`);
+ }
+
+ if (response.resultStatus === 'pending') {
+ this.infoMessage($t('Payment pending: page will reload.'));
+ window.location.replace(`/truelayer/checkout/process/payment_id/${this.orderResult['transaction_id']}`);
+ }
+
+ if (response.resultStatus === 'failed') {
+ this.isWidgetInit(false);
+ this.orderResult = await this.orderRequest(customer.isLoggedIn(), quote.getQuoteId());
+
+ const timeout = setTimeout(() => {
+ this.initWidget();
+ clearTimeout(timeout);
+ }, 1000);
+ }
+ }
+ },
+ });
+});
diff --git a/view/frontend/web/template/payment/truelayer.html b/view/frontend/web/template/payment/truelayer.html
old mode 100644
new mode 100755
index 9caf65e..6d31102
--- a/view/frontend/web/template/payment/truelayer.html
+++ b/view/frontend/web/template/payment/truelayer.html
@@ -4,7 +4,8 @@
* See COPYING.txt for license details.
*/
-->
-
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
-
-
-
-
-
-