Merge pull request #425 from woocommerce/pcp-409-fix-renewal

Update renewal handler
This commit is contained in:
Emili Castells 2022-01-11 10:07:45 +01:00 committed by GitHub
commit 44bee1cf6b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 155 additions and 136 deletions

View file

@ -23,12 +23,14 @@ return array(
$endpoint = $container->get( 'api.endpoint.order' ); $endpoint = $container->get( 'api.endpoint.order' );
$purchase_unit_factory = $container->get( 'api.factory.purchase-unit' ); $purchase_unit_factory = $container->get( 'api.factory.purchase-unit' );
$payer_factory = $container->get( 'api.factory.payer' ); $payer_factory = $container->get( 'api.factory.payer' );
$environment = $container->get( 'onboarding.environment' );
return new RenewalHandler( return new RenewalHandler(
$logger, $logger,
$repository, $repository,
$endpoint, $endpoint,
$purchase_unit_factory, $purchase_unit_factory,
$payer_factory $payer_factory,
$environment
); );
}, },
'subscription.repository.payment-token' => static function ( ContainerInterface $container ): PaymentTokenRepository { 'subscription.repository.payment-token' => static function ( ContainerInterface $container ): PaymentTokenRepository {

View file

@ -10,21 +10,26 @@ declare(strict_types=1);
namespace WooCommerce\PayPalCommerce\Subscription; namespace WooCommerce\PayPalCommerce\Subscription;
use WooCommerce\PayPalCommerce\ApiClient\Endpoint\OrderEndpoint; use WooCommerce\PayPalCommerce\ApiClient\Endpoint\OrderEndpoint;
use WooCommerce\PayPalCommerce\ApiClient\Entity\Order;
use WooCommerce\PayPalCommerce\ApiClient\Entity\OrderStatus;
use WooCommerce\PayPalCommerce\ApiClient\Entity\PaymentToken; use WooCommerce\PayPalCommerce\ApiClient\Entity\PaymentToken;
use WooCommerce\PayPalCommerce\ApiClient\Factory\PayerFactory; use WooCommerce\PayPalCommerce\ApiClient\Factory\PayerFactory;
use WooCommerce\PayPalCommerce\ApiClient\Factory\PurchaseUnitFactory; use WooCommerce\PayPalCommerce\ApiClient\Factory\PurchaseUnitFactory;
use WooCommerce\PayPalCommerce\Onboarding\Environment;
use WooCommerce\PayPalCommerce\Vaulting\PaymentTokenRepository; use WooCommerce\PayPalCommerce\Vaulting\PaymentTokenRepository;
use WooCommerce\PayPalCommerce\WcGateway\Gateway\PayPalGateway;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use WooCommerce\PayPalCommerce\WcGateway\Processor\AuthorizedPaymentsProcessor; use WooCommerce\PayPalCommerce\WcGateway\Processor\AuthorizedPaymentsProcessor;
use WooCommerce\PayPalCommerce\WcGateway\Processor\OrderMetaTrait;
use WooCommerce\PayPalCommerce\WcGateway\Processor\PaymentsStatusHandlingTrait;
use WooCommerce\PayPalCommerce\WcGateway\Processor\TransactionIdHandlingTrait;
/** /**
* Class RenewalHandler * Class RenewalHandler
*/ */
class RenewalHandler { class RenewalHandler {
use OrderMetaTrait;
use TransactionIdHandlingTrait;
use PaymentsStatusHandlingTrait;
/** /**
* The logger. * The logger.
* *
@ -60,6 +65,13 @@ class RenewalHandler {
*/ */
private $payer_factory; private $payer_factory;
/**
* The environment.
*
* @var Environment
*/
protected $environment;
/** /**
* RenewalHandler constructor. * RenewalHandler constructor.
* *
@ -68,13 +80,15 @@ class RenewalHandler {
* @param OrderEndpoint $order_endpoint The order endpoint. * @param OrderEndpoint $order_endpoint The order endpoint.
* @param PurchaseUnitFactory $purchase_unit_factory The purchase unit factory. * @param PurchaseUnitFactory $purchase_unit_factory The purchase unit factory.
* @param PayerFactory $payer_factory The payer factory. * @param PayerFactory $payer_factory The payer factory.
* @param Environment $environment The environment.
*/ */
public function __construct( public function __construct(
LoggerInterface $logger, LoggerInterface $logger,
PaymentTokenRepository $repository, PaymentTokenRepository $repository,
OrderEndpoint $order_endpoint, OrderEndpoint $order_endpoint,
PurchaseUnitFactory $purchase_unit_factory, PurchaseUnitFactory $purchase_unit_factory,
PayerFactory $payer_factory PayerFactory $payer_factory,
Environment $environment
) { ) {
$this->logger = $logger; $this->logger = $logger;
@ -82,6 +96,7 @@ class RenewalHandler {
$this->order_endpoint = $order_endpoint; $this->order_endpoint = $order_endpoint;
$this->purchase_unit_factory = $purchase_unit_factory; $this->purchase_unit_factory = $purchase_unit_factory;
$this->payer_factory = $payer_factory; $this->payer_factory = $payer_factory;
$this->environment = $environment;
} }
/** /**
@ -90,52 +105,23 @@ class RenewalHandler {
* @param \WC_Order $wc_order The WooCommerce order. * @param \WC_Order $wc_order The WooCommerce order.
*/ */
public function renew( \WC_Order $wc_order ) { public function renew( \WC_Order $wc_order ) {
$this->logger->log(
'info',
sprintf(
// translators: %d is the id of the order.
__( 'Start moneytransfer for order %d', 'woocommerce-paypal-payments' ),
(int) $wc_order->get_id()
),
array(
'order' => $wc_order,
)
);
try { try {
$this->process_order( $wc_order ); $this->process_order( $wc_order );
} catch ( \Exception $error ) { } catch ( \Exception $error ) {
$this->logger->log( $this->logger->error(
'error',
sprintf( sprintf(
// translators: %1$d is the order number, %2$s the error message. 'An error occurred while trying to renew the subscription for order %1$d: %2$s',
__( $wc_order->get_id(),
'An error occured while trying to renew the subscription for order %1$d: %2$s',
'woocommerce-paypal-payments'
),
(int) $wc_order->get_id(),
$error->getMessage() $error->getMessage()
),
array(
'order' => $wc_order,
) )
); );
return; return;
} }
$this->logger->log( $this->logger->info(
'info',
sprintf( sprintf(
// translators: %d is the order number. 'Renewal for order %d is completed.',
__( $wc_order->get_id()
'Moneytransfer for order %d is completed.',
'woocommerce-paypal-payments'
),
(int) $wc_order->get_id()
),
array(
'order' => $wc_order,
) )
); );
} }
@ -164,7 +150,19 @@ class RenewalHandler {
$token $token
); );
$this->capture_order( $order, $wc_order ); $this->add_paypal_meta( $wc_order, $order, $this->environment );
if ( $order->intent() === 'AUTHORIZE' ) {
$order = $this->order_endpoint->authorize( $order );
$wc_order->update_meta_data( AuthorizedPaymentsProcessor::CAPTURED_META_KEY, 'false' );
}
$transaction_id = $this->get_paypal_order_transaction_id( $order );
if ( $transaction_id ) {
$this->update_transaction_id( $transaction_id, $wc_order );
}
$this->handle_new_order_status( $order, $wc_order );
} }
/** /**
@ -185,12 +183,8 @@ class RenewalHandler {
if ( ! $tokens ) { if ( ! $tokens ) {
$error_message = sprintf( $error_message = sprintf(
// translators: %d is the customer id.
__(
'Payment failed. No payment tokens found for customer %d.', 'Payment failed. No payment tokens found for customer %d.',
'woocommerce-paypal-payments' $customer->get_id()
),
(int) $customer->get_id()
); );
$wc_order->update_status( $wc_order->update_status(
@ -198,14 +192,7 @@ class RenewalHandler {
$error_message $error_message
); );
$this->logger->log( $this->logger->error( $error_message );
'error',
$error_message,
array(
'customer' => $customer,
'order' => $wc_order,
)
);
} }
$subscription = function_exists( 'wcs_get_subscription' ) ? wcs_get_subscription( $wc_order->get_meta( '_subscription_renewal' ) ) : null; $subscription = function_exists( 'wcs_get_subscription' ) ? wcs_get_subscription( $wc_order->get_meta( '_subscription_renewal' ) ) : null;
@ -223,25 +210,4 @@ class RenewalHandler {
return current( $tokens ); return current( $tokens );
} }
/**
* If the PayPal order is captured/authorized the WooCommerce order gets updated accordingly.
*
* @param Order $order The PayPal order.
* @param \WC_Order $wc_order The related WooCommerce order.
*/
private function capture_order( Order $order, \WC_Order $wc_order ) {
if ( $order->intent() === 'CAPTURE' && $order->status()->is( OrderStatus::COMPLETED ) ) {
$wc_order->update_status(
'processing',
__( 'Payment received.', 'woocommerce-paypal-payments' )
);
}
if ( $order->intent() === 'AUTHORIZE' ) {
$this->order_endpoint->authorize( $order );
$wc_order->update_meta_data( AuthorizedPaymentsProcessor::CAPTURED_META_KEY, 'false' );
}
}
} }

View file

@ -3,18 +3,25 @@ declare(strict_types=1);
namespace WooCommerce\PayPalCommerce\Subscription; namespace WooCommerce\PayPalCommerce\Subscription;
use Dhii\Container\Dictionary;
use Exception;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use WooCommerce\PayPalCommerce\ApiClient\Endpoint\OrderEndpoint; use WooCommerce\PayPalCommerce\ApiClient\Endpoint\OrderEndpoint;
use WooCommerce\PayPalCommerce\ApiClient\Entity\Capture;
use WooCommerce\PayPalCommerce\ApiClient\Entity\CaptureStatus;
use WooCommerce\PayPalCommerce\ApiClient\Entity\Order; use WooCommerce\PayPalCommerce\ApiClient\Entity\Order;
use WooCommerce\PayPalCommerce\ApiClient\Entity\Payer; use WooCommerce\PayPalCommerce\ApiClient\Entity\Payer;
use WooCommerce\PayPalCommerce\ApiClient\Entity\Payments;
use WooCommerce\PayPalCommerce\ApiClient\Entity\PaymentToken; use WooCommerce\PayPalCommerce\ApiClient\Entity\PaymentToken;
use WooCommerce\PayPalCommerce\ApiClient\Entity\PurchaseUnit; use WooCommerce\PayPalCommerce\ApiClient\Entity\PurchaseUnit;
use WooCommerce\PayPalCommerce\ApiClient\Factory\PayerFactory; use WooCommerce\PayPalCommerce\ApiClient\Factory\PayerFactory;
use WooCommerce\PayPalCommerce\ApiClient\Factory\PurchaseUnitFactory; use WooCommerce\PayPalCommerce\ApiClient\Factory\PurchaseUnitFactory;
use WooCommerce\PayPalCommerce\Onboarding\Environment;
use WooCommerce\PayPalCommerce\TestCase; use WooCommerce\PayPalCommerce\TestCase;
use Mockery; use Mockery;
use Mockery\Adapter\Phpunit\MockeryPHPUnitIntegration; use Mockery\Adapter\Phpunit\MockeryPHPUnitIntegration;
use WooCommerce\PayPalCommerce\Vaulting\PaymentTokenRepository; use WooCommerce\PayPalCommerce\Vaulting\PaymentTokenRepository;
use WooCommerce\PayPalCommerce\WcGateway\Gateway\PayPalGateway;
class RenewalHandlerTest extends TestCase class RenewalHandlerTest extends TestCase
{ {
@ -25,6 +32,7 @@ class RenewalHandlerTest extends TestCase
private $orderEndpoint; private $orderEndpoint;
private $purchaseUnitFactory; private $purchaseUnitFactory;
private $payerFactory; private $payerFactory;
private $environment;
private $sut; private $sut;
public function setUp(): void public function setUp(): void
@ -36,13 +44,20 @@ class RenewalHandlerTest extends TestCase
$this->orderEndpoint = Mockery::mock(OrderEndpoint::class); $this->orderEndpoint = Mockery::mock(OrderEndpoint::class);
$this->purchaseUnitFactory = Mockery::mock(PurchaseUnitFactory::class); $this->purchaseUnitFactory = Mockery::mock(PurchaseUnitFactory::class);
$this->payerFactory = Mockery::mock(PayerFactory::class); $this->payerFactory = Mockery::mock(PayerFactory::class);
$this->environment = new Environment(new Dictionary([]));
$this->logger->shouldReceive('error')->andReturnUsing(function ($msg) {
throw new Exception($msg);
});
$this->logger->shouldReceive('info');
$this->sut = new RenewalHandler( $this->sut = new RenewalHandler(
$this->logger, $this->logger,
$this->repository, $this->repository,
$this->orderEndpoint, $this->orderEndpoint,
$this->purchaseUnitFactory, $this->purchaseUnitFactory,
$this->payerFactory $this->payerFactory,
$this->environment
); );
} }
@ -52,24 +67,64 @@ class RenewalHandlerTest extends TestCase
*/ */
public function testRenewProcessOrder() public function testRenewProcessOrder()
{ {
$transactionId = 'ABC123';
$wcOrder = Mockery::mock(\WC_Order::class); $wcOrder = Mockery::mock(\WC_Order::class);
$customer = Mockery::mock('overload:WC_Customer'); $customer = Mockery::mock('overload:WC_Customer');
$token = Mockery::mock(PaymentToken::class); $token = Mockery::mock(PaymentToken::class);
$purchaseUnit = Mockery::mock(PurchaseUnit::class);
$payer = Mockery::mock(Payer::class); $payer = Mockery::mock(Payer::class);
$order = Mockery::mock(Order::class); $order = Mockery::mock(Order::class);
$this->logger->shouldReceive('log'); $capture = Mockery::mock(Capture::class);
$capture->expects('id')
->andReturn($transactionId);
$capture->expects('status')
->andReturn(new CaptureStatus(CaptureStatus::COMPLETED));
$payments = Mockery::mock(Payments::class);
$payments->shouldReceive('captures')
->andReturn([$capture]);
$purchaseUnit = Mockery::mock(PurchaseUnit::class);
$purchaseUnit->shouldReceive('payments')
->andReturn($payments);
$order
->shouldReceive('id')
->andReturn('101');
$order->shouldReceive('intent')
->andReturn('CAPTURE');
$order->shouldReceive('status->is')
->andReturn(true);
$order
->shouldReceive('purchase_units')
->andReturn([$purchaseUnit]);
$wcOrder $wcOrder
->shouldReceive('get_id') ->shouldReceive('get_id')
->andReturn(1); ->andReturn(1);
$wcOrder $wcOrder
->shouldReceive('get_customer_id') ->shouldReceive('get_customer_id')
->andReturn(1); ->andReturn(2);
$this->repository->shouldReceive('for_user_id') $wcOrder
->andReturn($token); ->expects('update_meta_data')
->with(PayPalGateway::ORDER_ID_META_KEY, '101');
$wcOrder
->expects('update_meta_data')
->with(PayPalGateway::INTENT_META_KEY, 'CAPTURE');
$wcOrder
->expects('update_meta_data')
->with(PayPalGateway::ORDER_PAYMENT_MODE_META_KEY, 'live');
$wcOrder
->expects('payment_complete');
$wcOrder
->expects('set_transaction_id');
$this->repository->shouldReceive('all_for_user_id')
->andReturn([$token]);
$customer->shouldReceive('get_id') $customer->shouldReceive('get_id')
->andReturn(1); ->andReturn(1);
$this->purchaseUnitFactory->shouldReceive('from_wc_order') $this->purchaseUnitFactory->shouldReceive('from_wc_order')
->andReturn($purchaseUnit); ->andReturn($purchaseUnit);
$this->payerFactory->shouldReceive('from_customer') $this->payerFactory->shouldReceive('from_customer')
@ -79,10 +134,6 @@ class RenewalHandlerTest extends TestCase
->with([$purchaseUnit], $payer, $token) ->with([$purchaseUnit], $payer, $token)
->andReturn($order); ->andReturn($order);
$order->shouldReceive('intent')
->andReturn('CAPTURE');
$order->shouldReceive('status->is')
->andReturn(true);
$wcOrder->shouldReceive('update_status'); $wcOrder->shouldReceive('update_status');
$this->sut->renew($wcOrder); $this->sut->renew($wcOrder);