| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  | <?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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-13 07:59:08 +03:00
										 |  |  |             $request->setOrderBundle($this->getOrderBundle()); | 
					
						
							| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |             $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 = []; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-13 07:59:08 +03:00
										 |  |  |         $items = []; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  |         foreach ($this->order->products as $positionId => $product) { | 
					
						
							|  |  |  |             assert($product instanceof OrderProduct); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-13 07:59:08 +03:00
										 |  |  |             $items[] = [ | 
					
						
							|  |  |  |                 'positionId' => $positionId + 1, | 
					
						
							| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  |                 'name' => $product->name, | 
					
						
							|  |  |  |                 'quantity' => [ | 
					
						
							|  |  |  |                     'value' => $product->quantity, | 
					
						
							| 
									
										
										
										
											2021-07-13 07:59:08 +03:00
										 |  |  |                     'measure' => 'шт.' | 
					
						
							| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  |                 ], | 
					
						
							|  |  |  |                 'itemCode' => $product->variant_id ?? $product->product_id, | 
					
						
							| 
									
										
										
										
											2021-07-13 07:59:08 +03:00
										 |  |  |                 'itemPrice' => $product->pricePostTaxes()->integer, | 
					
						
							| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  |             ]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-13 07:59:08 +03:00
										 |  |  |         $cartItems['items'] = $items; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  |         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( | 
					
						
							|  |  |  |                 [ | 
					
						
							| 
									
										
										
										
											2021-06-24 21:43:21 +03:00
										 |  |  |                     'orderId' => $result->order->payment_transaction_id, // gateway order number
 | 
					
						
							| 
									
										
										
										
											2021-06-18 12:15:18 +03:00
										 |  |  |                     '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', | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |