You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
283 lines
7.8 KiB
283 lines
7.8 KiB
<?php namespace Wpstudio\Sber\Classes;
|
|
|
|
use OFFLINE\Mall\Classes\Payments\PaymentProvider;
|
|
use OFFLINE\Mall\Classes\Payments\PaymentResult;
|
|
use OFFLINE\Mall\Models\OrderProduct;
|
|
use OFFLINE\Mall\Models\PaymentGatewaySettings;
|
|
use OFFLINE\Mall\Models\OrderState;
|
|
use OFFLINE\Mall\Models\Order;
|
|
use Omnipay\Omnipay;
|
|
use Omnipay\Sberbank\Gateway;
|
|
use Omnipay\Sberbank\Message\AbstractRequest;
|
|
use Omnipay\Sberbank\Message\AbstractResponse;
|
|
use Omnipay\Sberbank\Message\AuthorizeRequest;
|
|
use Omnipay\Sberbank\Message\AuthorizeResponse;
|
|
use Throwable;
|
|
use Session;
|
|
use Lang;
|
|
|
|
|
|
class SberCheckout extends PaymentProvider
|
|
{
|
|
/**
|
|
* The order that is being paid.
|
|
*
|
|
* @var Order
|
|
*/
|
|
public $order;
|
|
/**
|
|
* Data that is needed for the payment.
|
|
* Card numbers, tokens, etc.
|
|
*
|
|
* @var array
|
|
*/
|
|
public $data;
|
|
|
|
/**
|
|
* Return the display name of your payment provider.
|
|
*
|
|
* @return string
|
|
*/
|
|
final public function name(): string
|
|
{
|
|
return Lang::get('wpstudio.sber::lang.settings.sber_checkout');
|
|
}
|
|
|
|
/**
|
|
* Return a unique identifier for this payment provider.
|
|
*
|
|
* @return string
|
|
*/
|
|
final public function identifier(): string
|
|
{
|
|
return 'sber';
|
|
}
|
|
|
|
/**
|
|
* Validate the given input data for this payment.
|
|
*
|
|
* @return bool
|
|
* @throws \October\Rain\Exception\ValidationException
|
|
*/
|
|
final public function validate(): bool
|
|
{
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Process the payment.
|
|
*
|
|
* @param PaymentResult $result
|
|
*
|
|
* @return PaymentResult
|
|
*/
|
|
final public function process(PaymentResult $result): PaymentResult
|
|
{
|
|
$gateway = $this->getGateway();
|
|
|
|
try {
|
|
$request = $gateway->authorize([
|
|
'orderNumber' => $this->order->id,
|
|
'amount' => $this->order->total_in_currency,
|
|
'returnUrl' => $this->returnUrl(),
|
|
'failUrl' => $this->cancelUrl(),
|
|
'description' => Lang::get('wpstudio.sber::lang.messages.order_number').$this->order->order_number,
|
|
]);
|
|
|
|
assert($request instanceof AuthorizeRequest);
|
|
|
|
// hacking private method
|
|
$reflectionMethod = new \ReflectionMethod(AbstractRequest::class, 'setParameter');
|
|
$reflectionMethod->setAccessible(true);
|
|
$reflectionMethod->invoke($request, 'orderBundle', $this->getOrderBundle());
|
|
|
|
$response = $request->send();
|
|
} catch (Throwable $e) {
|
|
return $result->fail([], $e);
|
|
}
|
|
|
|
assert($response instanceof AuthorizeResponse);
|
|
|
|
Session::put('mall.payment.callback', self::class);
|
|
|
|
$this->setOrder($result->order);
|
|
|
|
$result->order->payment_transaction_id = $response->getOrderId();
|
|
|
|
$result->order->save();
|
|
|
|
return $result->redirect($response->getRedirectResponse()->getTargetUrl());
|
|
}
|
|
|
|
/**
|
|
* Gerenate sberbank orderBundle
|
|
* @see https://securepayments.sberbank.ru/wiki/doku.php/integration:api:rest:requests:registerpreauth_cart#orderbundle
|
|
*
|
|
* @return array
|
|
*
|
|
*/
|
|
final public function getOrderBundle(): array
|
|
{
|
|
$orderCreationDateFormatted = $this->order->created_at->format('Y-m-dTH:m:s');
|
|
|
|
$cartItems = $this->getOrderCartItems();
|
|
|
|
if ($cartItems) {
|
|
return [
|
|
'orderCreationDate' => $orderCreationDateFormatted ?? '',
|
|
'cartItems' => $cartItems
|
|
];
|
|
}
|
|
|
|
return [];
|
|
}
|
|
|
|
/**
|
|
* Create order cartitems for order bundle
|
|
*
|
|
* @return array
|
|
*/
|
|
final public function getOrderCartItems(): array
|
|
{
|
|
$cartItems = [];
|
|
|
|
foreach ($this->order->products as $positionId => $product) {
|
|
assert($product instanceof OrderProduct);
|
|
|
|
$cartItems['items'] = [
|
|
'positionId' => $positionId,
|
|
'name' => $product->name,
|
|
'quantity' => [
|
|
'value' => $product->quantity,
|
|
'measure' => ''
|
|
],
|
|
'itemCode' => $product->variant_id ?? $product->product_id,
|
|
'itemPrice' => $product->pricePostTaxes()->float,
|
|
];
|
|
}
|
|
|
|
return $cartItems;
|
|
}
|
|
|
|
/**
|
|
* Y.K. has processed the payment and redirected the user back.
|
|
*
|
|
* @param PaymentResult $result
|
|
*
|
|
* @return PaymentResult
|
|
*/
|
|
final public function complete(PaymentResult $result): PaymentResult
|
|
{
|
|
$this->setOrder($result->order);
|
|
|
|
$gateway = $this->getGateway();
|
|
|
|
try {
|
|
/**
|
|
* It will be similar to calling methods `completeAuthorize()` and `completePurchase()`
|
|
*/
|
|
$response = $gateway->orderStatus(
|
|
[
|
|
'orderId' => $result->order->payment_transaction_id, // gateway order number
|
|
'language' => 'ru'
|
|
]
|
|
)->send();
|
|
} catch (Throwable $e) {
|
|
return $result->fail([], $e);
|
|
}
|
|
|
|
assert($response instanceof AbstractResponse);
|
|
|
|
$data = (array)$response->getData();
|
|
|
|
if ( ! $response->isSuccessful()) {
|
|
return $result->fail($data, $response);
|
|
}
|
|
|
|
return $result->success($data, $response);
|
|
}
|
|
|
|
/**
|
|
* Build the Omnipay Gateway for PayPal.
|
|
*
|
|
* @return \Omnipay\Common\GatewayInterface
|
|
*/
|
|
final protected function getGateway(): Gateway
|
|
{
|
|
$gateway = Omnipay::create('Sberbank');
|
|
|
|
$gateway->setUserName(PaymentGatewaySettings::get('username'));
|
|
$gateway->setPassword(decrypt(PaymentGatewaySettings::get('password')));
|
|
|
|
if (PaymentGatewaySettings::get('sber_test_mode')) {
|
|
$gateway->setTestMode(true);
|
|
}
|
|
|
|
|
|
return $gateway;
|
|
}
|
|
|
|
/**
|
|
* Return any custom backend settings fields.
|
|
*
|
|
* These fields will be rendered in the backend
|
|
* settings page of your provider.
|
|
*
|
|
* @return array
|
|
*/
|
|
final public function settings(): array
|
|
{
|
|
return [
|
|
'sber_test_mode' => [
|
|
'label' => 'wpstudio.sber::lang.settings.sber_test_mode',
|
|
'comment' => 'wpstudio.sber::lang.settings.sber_test_mode_label',
|
|
'span' => 'left',
|
|
'type' => 'switch',
|
|
],
|
|
'username' => [
|
|
'label' => Lang::get('wpstudio.sber::lang.settings.username'),
|
|
'comment' => Lang::get('wpstudio.sber::lang.settings.username_label'),
|
|
'span' => 'left',
|
|
'type' => 'text',
|
|
],
|
|
'password' => [
|
|
'label' => Lang::get('wpstudio.sber::lang.settings.password'),
|
|
'comment' => Lang::get('wpstudio.sber::lang.settings.password_label'),
|
|
'span' => 'left',
|
|
'type' => 'text',
|
|
],
|
|
'setPayedVirtualOrderAsComplete' => [
|
|
'label' => Lang::get('wpstudio.sber::lang.settings.set_payed_virtual_order_as_complete'),
|
|
'span' => 'left',
|
|
'type' => 'checkbox',
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* Setting keys returned from this method are stored encrypted.
|
|
*
|
|
* Use this to store API tokens and other secret data
|
|
* that is needed for this PaymentProvider to work.
|
|
*
|
|
* @return array
|
|
*/
|
|
final public function encryptedSettings(): array
|
|
{
|
|
return ['password'];
|
|
}
|
|
|
|
/**
|
|
* Getting order state id by flag
|
|
*
|
|
* @param $orderStateFlag
|
|
* @return int
|
|
*/
|
|
final protected function getOrderStateId($orderStateFlag): int
|
|
{
|
|
$orderStateModel = OrderState::where('flag', $orderStateFlag)->first();
|
|
|
|
return $orderStateModel->id;
|
|
}
|
|
}
|