diff --git a/UPGRADE.md b/UPGRADE.md
index 76d821d5..4aa87617 100644
--- a/UPGRADE.md
+++ b/UPGRADE.md
@@ -1,3 +1,15 @@
+### UPGRADE FROM 1.3.0 to 1.3.1
+
+1. `sylius_paypal_plugin_pay_with_paypal_form` route now operates on both payment ID and order token. URl then changed from
+ `/pay-with-paypal/{id}` to `/pay-with-paypal/{orderToken}/{paymentId}`. If you use this route anywhere in your application, you
+ need to change the URL attributes
+
+### UPGRADE FROM 1.2.3 to 1.2.4
+
+1. `sylius_paypal_plugin_pay_with_paypal_form` route now operates on both payment ID and order token. URl then changed from
+ `/pay-with-paypal/{id}` to `/pay-with-paypal/{orderToken}/{paymentId}`. If you use this route anywhere in your application, you
+ need to change the URL attributes
+
### UPGRADE FROM 1.0.X TO 1.1.0
1. Upgrade your application to [Sylius 1.8](https://github.com/Sylius/Sylius/blob/master/UPGRADE-1.8.md).
diff --git a/phpstan.neon b/phpstan.neon
index 46e93df9..71b3ff4c 100644
--- a/phpstan.neon
+++ b/phpstan.neon
@@ -14,3 +14,6 @@ parameters:
# Test dependencies
- 'tests/Application/app/**.php'
- 'tests/Application/src/**.php'
+
+ ignoreErrors:
+ - '/Call to an undefined method Sylius\\Component\\Core\\Repository\\PaymentRepositoryInterface\:\:createQueryBuilder\(\)\./'
diff --git a/psalm.xml b/psalm.xml
index f38d221b..b117a394 100644
--- a/psalm.xml
+++ b/psalm.xml
@@ -12,6 +12,7 @@
+
diff --git a/spec/Payum/Action/ResolveNextRouteActionSpec.php b/spec/Payum/Action/ResolveNextRouteActionSpec.php
index a5a2362d..99a8ec61 100644
--- a/spec/Payum/Action/ResolveNextRouteActionSpec.php
+++ b/spec/Payum/Action/ResolveNextRouteActionSpec.php
@@ -17,6 +17,7 @@ final class ResolveNextRouteActionSpec extends ObjectBehavior
function it_executes_resolve_next_route_request_with_processing_payment(
ResolveNextRoute $request,
PaymentInterface $payment,
+ OrderInterface $order,
PaymentMethodInterface $paymentMethod,
GatewayConfigInterface $gatewayConfig
): void {
@@ -28,8 +29,11 @@ function it_executes_resolve_next_route_request_with_processing_payment(
$paymentMethod->getGatewayConfig()->willReturn($gatewayConfig);
$gatewayConfig->getFactoryName()->willReturn('sylius.pay_pal');
+ $payment->getOrder()->willReturn($order);
+ $order->getTokenValue()->willReturn('123!@#asd');
+
$request->setRouteName('sylius_paypal_plugin_pay_with_paypal_form')->shouldBeCalled();
- $request->setRouteParameters(['id' => 12])->shouldBeCalled();
+ $request->setRouteParameters(['orderToken' => '123!@#asd', 'paymentId' => 12])->shouldBeCalled();
$this->execute($request);
}
@@ -37,10 +41,12 @@ function it_executes_resolve_next_route_request_with_processing_payment(
function it_executes_resolve_next_route_request_with_completed_payment(
ResolveNextRoute $request,
PaymentInterface $payment,
+ OrderInterface $order,
PaymentMethodInterface $paymentMethod,
GatewayConfigInterface $gatewayConfig
): void {
$request->getFirstModel()->willReturn($payment);
+ $payment->getOrder()->willReturn($order);
$payment->getState()->willReturn(PaymentInterface::STATE_COMPLETED);
$payment->getMethod()->willReturn($paymentMethod);
diff --git a/src/Controller/PayWithPayPalFormAction.php b/src/Controller/PayWithPayPalFormAction.php
index 8e2ace2c..5f810629 100644
--- a/src/Controller/PayWithPayPalFormAction.php
+++ b/src/Controller/PayWithPayPalFormAction.php
@@ -49,8 +49,11 @@ public function __construct(
public function __invoke(Request $request): Response
{
+ $paymentId = (string) $request->attributes->get('paymentId');
+ $orderToken = (string) $request->attributes->get('orderToken');
+
/** @var PaymentInterface $payment */
- $payment = $this->paymentRepository->find($request->attributes->get('id'));
+ $payment = $this->findOneByPaymentIdOrderToken($paymentId, $orderToken);
/** @var PaymentMethodInterface $paymentMethod */
$paymentMethod = $payment->getMethod();
@@ -79,4 +82,22 @@ public function __invoke(Request $request): Response
'partner_attribution_id' => $partnerAttributionId,
]));
}
+
+ /**
+ * Need to be used due to support for Sylius 1.8.
+ * After dropping it, we can switch to Sylius\Component\Core\Repository\PaymentRepositoryInterface::findOneByOrderToken
+ */
+ private function findOneByPaymentIdOrderToken(string $paymentId, string $orderToken): ?PaymentInterface
+ {
+ return $this->paymentRepository
+ ->createQueryBuilder('p')
+ ->innerJoin('p.order', 'o')
+ ->andWhere('p.id = :paymentId')
+ ->andWhere('o.tokenValue = :orderToken')
+ ->setParameter('paymentId', $paymentId)
+ ->setParameter('orderToken', $orderToken)
+ ->getQuery()
+ ->getOneOrNullResult()
+ ;
+ }
}
diff --git a/src/Payum/Action/ResolveNextRouteAction.php b/src/Payum/Action/ResolveNextRouteAction.php
index a7ed2c72..30f6d80a 100644
--- a/src/Payum/Action/ResolveNextRouteAction.php
+++ b/src/Payum/Action/ResolveNextRouteAction.php
@@ -19,9 +19,14 @@ public function execute($request): void
/** @var PaymentInterface $payment */
$payment = $request->getFirstModel();
+ /** @var OrderInterface $order */
+ $order = $payment->getOrder();
+
if ($payment->getState() === PaymentInterface::STATE_NEW) {
$request->setRouteName('sylius_paypal_plugin_pay_with_paypal_form');
- $request->setRouteParameters(['id' => $payment->getId()]);
+ $request->setRouteParameters(
+ ['orderToken' => $order->getTokenValue(), 'paymentId' => $payment->getId()]
+ );
return;
}
@@ -32,9 +37,6 @@ public function execute($request): void
return;
}
- /** @var OrderInterface $order */
- $order = $payment->getOrder();
-
$request->setRouteName('sylius_shop_order_show');
$request->setRouteParameters(['tokenValue' => $order->getTokenValue()]);
}
diff --git a/src/Resources/config/shop_routing.yaml b/src/Resources/config/shop_routing.yaml
index 835c8ed1..7310c157 100644
--- a/src/Resources/config/shop_routing.yaml
+++ b/src/Resources/config/shop_routing.yaml
@@ -1,5 +1,5 @@
sylius_paypal_plugin_pay_with_paypal_form:
- path: /pay-with-paypal/{id}
+ path: /pay-with-paypal/{orderToken}/{paymentId}
methods: [GET]
defaults:
_controller: Sylius\PayPalPlugin\Controller\PayWithPayPalFormAction
diff --git a/src/Resources/views/payWithPaypal.html.twig b/src/Resources/views/payWithPaypal.html.twig
index ef09d17a..e164e4d1 100644
--- a/src/Resources/views/payWithPaypal.html.twig
+++ b/src/Resources/views/payWithPaypal.html.twig
@@ -329,6 +329,8 @@
});
if (paypal.HostedFields.isEligible() === true) {
+ let processingOrderId;
+
paypal.HostedFields.render({
createOrder: function(data, actions) {
document.querySelector('#paypal-payment-container').classList.add('loading');
@@ -339,6 +341,8 @@
}).then(function(res) {
return res.json();
}).then(function(data) {
+ processingOrderId = data.orderID;
+
return data.orderID;
});
},
@@ -419,6 +423,7 @@
if (formValid) {
hostedFields.submit({
+ contingencies: ['SCA_ALWAYS'],
cardholderName: document.getElementById('card-holder-name').value,
billingAddress: {
streetAddress: document.getElementById('card-billing-address-street').value,
@@ -428,20 +433,35 @@
countryCodeAlpha2: document.getElementById('card-billing-address-country').value
}
}).then(payload => {
- return fetch(completePayPalOrderUrl, {
- method: 'post'
- }).then(function(res) {
- return res.json();
+ if (payload.authenticationReason == 'SUCCESSFUL' && payload.authenticationStatus == 'YES') {
+ return fetch(completePayPalOrderUrl, {
+ method: 'post'
+ }).then(function(res) {
+ return res.json();
+ }).then(function(data) {
+ if (data.status == 'processing') {
+ return fetch(cancelPayPalPaymentUrl, {
+ method: 'post',
+ headers: { 'content-type': 'application/json' },
+ body: JSON.stringify({ payPalOrderId: data.orderID })
+ }).then(window.location.reload());
+ }
+
+ window.location.href = data.return_url;
+ });
+ }
+
+
+ return fetch(errorPayPalPaymentUrl, {
+ method: 'post',
+ headers: { 'content-type': 'application/json' },
+ body: JSON.stringify('Invalid 3D Secure authentication.')
}).then(function(data) {
- if (data.status == 'processing') {
- return fetch(cancelPayPalPaymentUrl, {
- method: 'post',
- headers: { 'content-type': 'application/json' },
- body: JSON.stringify({ payPalOrderId: data.orderID })
- }).then(window.location.reload());
- }
-
- window.location.href = data.return_url;
+ return fetch(cancelPayPalPaymentUrl, {
+ method: 'post',
+ headers: { 'content-type': 'application/json' },
+ body: JSON.stringify({ payPalOrderId: processingOrderId })
+ }).then(window.location.reload());
});
});
} else {