simpay / ecommerce
PHP SDK for integrating SimPay payments into e-commerce platforms
Requires
- php: >=8.1
- ext-curl: *
- ext-json: *
Requires (Dev)
- phpstan/phpstan: ^1.10
- phpunit/phpunit: ^10.0
README
Wspólne SDK do obsługi płatności SimPay we wszystkich naszych pluginach e-commerce.
- Walidacja IPN — sprawdzenie User-Agent, wymaganych pól, podpisu SHA-256, service_id
- Walidacja IP — pobranie listy IP SimPay z API, cache, porównanie
- Porównywanie kwot — float-safe porównanie czy zapłacono wystarczająco
- Klient API — tworzenie transakcji, refundy, BLIK, kanały płatności
- Builder transakcji — budowanie payloadu z walidacją pól
Instalacja
W composer.json pluginu:
{
"repositories": [
{ "type": "path", "url": "../simpay-ecommerce" }
],
"require": {
"simpay/ecommerce": "*"
}
}
Struktura
src/
├── SimPay.php # Główna fasada — punkt wejścia
├── SimPayClient.php # Klient HTTP do API (transakcje, refundy, BLIK, OneClick, kanały, IP)
├── Configuration.php # Credentials + metadata platformy
├── BlikAlias.php # DTO aliasu BLIK OneClick
├── IpnValidator.php # Walidacja webhooków (wersja UA, pola, podpis, service_id)
├── IpnPayload.php # DTO wyniku walidacji IPN (+ helpery BLIK alias)
├── IpAllowlistService.php # Walidacja IP z opcjonalnym cache
├── TransactionBuilder.php # Fluent builder payloadu transakcji
├── AmountVerifier.php # Porównywanie kwot (float-safe)
├── PaymentStatus.php # Stałe statusów (transaction_paid, alias_active, etc.)
├── CacheInterface.php # Interfejs cache — implementuj per-platforma
├── Http/
│ ├── HttpClientInterface.php
│ ├── HttpResponse.php
│ └── CurlHttpClient.php # Domyślna implementacja
└── Exception/
├── SimPayException.php
├── ApiException.php
├── AliasNotUniqueException.php # Błąd wieloznaczności aliasu BLIK
├── HttpException.php
├── IpnException.php
└── IpNotAllowedException.php
Użycie
Inicjalizacja
use SimPay\SDK\SimPay; $simpay = new SimPay( bearerToken: $config->get('simpay_bearer'), serviceId: $config->get('simpay_service_id'), signatureKey: $config->get('simpay_signature_key'), platform: 'opencart', platformVersion: VERSION );
Tworzenie transakcji
$payload = $simpay->transaction() ->setAmount((float) $order['total'], $order['currency_code']) ->setDescription('Zamówienie #' . $order['order_id']) ->setControl((string) $order['order_id']) ->setCustomer($name, $email, $ip, $countryIso) ->setReturnUrls($successUrl, $failureUrl) ->setBilling($firstName, $lastName, $street, $building, $city, $postCode, $countryIso) ->toArray(); $response = $simpay->client()->createTransaction($payload); $redirectUrl = $response['data']['redirectUrl'];
Obsługa IPN (webhook)
use SimPay\SDK\SimPay; use SimPay\SDK\AmountVerifier; use SimPay\SDK\PaymentStatus; $payload = json_decode(file_get_contents('php://input'), true); try { $ipn = $simpay->handleIpn( payload: $payload, userAgent: $_SERVER['HTTP_USER_AGENT'] ?? null, remoteIp: $_SERVER['REMOTE_ADDR'] ?? null // null = pomiń sprawdzenie IP ); } catch (\SimPay\SDK\Exception\IpNotAllowedException $e) { http_response_code(403); exit($e->getMessage()); } catch (\SimPay\SDK\Exception\IpnException $e) { http_response_code(400); exit($e->getMessage()); } // Rozróżnianie eventów if ($ipn->isTransactionEvent()) { if (!$ipn->isPaid()) { exit('OK'); // jeszcze nie opłacone, potwierdź odbiór } $orderId = $ipn->getControl(); $paidAmount = $ipn->getEffectiveAmount($orderCurrency); if (!AmountVerifier::isAmountSufficient($orderTotal, $paidAmount)) { http_response_code(400); exit('Amount mismatch'); } // Aktualizuj status zamówienia } if ($ipn->isRefundEvent()) { // Obsłuż refund $refundId = $ipn->data['id']; $status = $ipn->getStatus(); } exit('OK');
Refundy
$simpay->client()->createRefund($transactionId); // pełny $simpay->client()->createRefund($transactionId, 50.00); // częściowy
BLIK Level 0
// Prosta płatność BLIK Level 0 (bez rejestracji aliasu) $simpay->client()->sendBlikLevel0($transactionId, '123456');
BLIK Level 0 + OneClick (Płatności bez kodu)
Krok 1: Pierwsza płatność z rejestracją aliasu
Podczas pierwszej płatności klient wpisuje 6-cyfrowy kod BLIK, a Ty rejestrujesz alias, aby umożliwić przyszłe płatności bez kodu:
use SimPay\SDK\BlikAlias; $alias = BlikAlias::register( label: 'Nazwa Twojego Sklepu', // Zgodna z certyfikacją BLIK value: 'unikalne_id_klienta_123' // Twoje wewnętrzne ID klienta ); $simpay->client()->sendBlikLevel0($transactionId, '123456', $alias);
Krok 2: Obsługa IPN aliasu (blik:alias_status_changed)
Po płatności aplikacja bankowa zapyta klienta o zapis sklepu jako zaufanego.
Nasłuchuj zdarzenia IPN blik:alias_status_changed:
$ipn = $simpay->handleIpn($payload, $userAgent, $remoteIp); if ($ipn->isBlikAliasEvent() && $ipn->isAliasActive()) { $aliasUuid = $ipn->getAliasId(); // UUID aliasu SimPay // Zapisz $aliasUuid w bazie danych przy koncie klienta $db->saveAliasUuid($userId, $aliasUuid); }
Krok 3: Płatność OneClick (bez kodu BLIK)
Metoda A — przez UUID SimPay (rekomendowana, zawsze jednoznaczna):
$alias = BlikAlias::fromUuid( uuid: $savedAliasUuid, // UUID z kroku 2 label: 'Nazwa Twojego Sklepu' ); $simpay->client()->sendBlikOneClick($transactionId, $alias); // HTTP 204 — klient dostaje push w aplikacji bankowej
Metoda B — przez własne ID klienta (value + type):
$alias = BlikAlias::fromValue( value: 'unikalne_id_klienta_123', label: 'Nazwa Twojego Sklepu' ); $simpay->client()->sendBlikOneClick($transactionId, $alias);
Obsługa błędu ALIAS_NOT_UNIQUE
Gdy klient ma kilka aplikacji bankowych podpiętych pod to samo ID (tylko Metoda B):
use SimPay\SDK\Exception\AliasNotUniqueException; try { $simpay->client()->sendBlikOneClick($transactionId, $alias); } catch (AliasNotUniqueException $e) { // Wyświetl klientowi wybór banku $alternatives = $e->getAlternatives(); // Każda alternatywa: ['label' => 'PKO BP', 'blik_id' => 953824, ...] // Po wyborze klienta — ponów z blik_id: $selectedBlikId = $alternatives[$userChoice]['blik_id']; $resolvedAlias = $alias->withBlikId($selectedBlikId); $simpay->client()->sendBlikOneClick($transactionId, $resolvedAlias); }
Kanały płatności
$channels = $simpay->client()->getChannels();
Adapter HTTP (opcjonalny)
Domyślnie SDK używa cURL. Jeśli platforma wymaga innego klienta HTTP:
use SimPay\SDK\Http\HttpClientInterface; use SimPay\SDK\Http\HttpResponse; class WordPressHttpClient implements HttpClientInterface { public function request(string $method, string $url, array $headers = [], ?array $body = null, int $timeout = 30): HttpResponse { $response = wp_remote_request($url, [ 'method' => $method, 'headers' => $headers, 'timeout' => $timeout, 'body' => $body !== null ? json_encode($body) : null, ]); if (is_wp_error($response)) { throw new \SimPay\SDK\Exception\HttpException($response->get_error_message()); } return new HttpResponse( wp_remote_retrieve_response_code($response), wp_remote_retrieve_body($response) ); } } $simpay = new SimPay($bearer, $serviceId, $signatureKey, 'wordpress', $version, new WordPressHttpClient());
Adapter Cache (opcjonalny)
Przyspiesza walidację IP — zamiast odpytywać API przy każdym IPN:
use SimPay\SDK\CacheInterface; class PrestaShopCache implements CacheInterface { public function get(string $key): mixed { $val = \Cache::getInstance()->get($key); return $val !== false ? $val : null; } public function set(string $key, mixed $value, int $ttl = 3600): void { \Cache::getInstance()->set($key, $value, $ttl); } public function delete(string $key): void { \Cache::getInstance()->delete($key); } } $simpay = new SimPay($bearer, $serviceId, $signatureKey, 'prestashop', _PS_VERSION_, null, new PrestaShopCache());
Stałe statusów
use SimPay\SDK\PaymentStatus; PaymentStatus::isPaid($status); // transaction_paid | transaction_confirmed PaymentStatus::isFinal($status); // paid, confirmed, canceled, expired, failure, fraud, refunded PaymentStatus::isFailed($status); // canceled, expired, failure, fraud // Stałe do użycia w match/switch: PaymentStatus::TRANSACTION_PAID PaymentStatus::TRANSACTION_CANCELED PaymentStatus::TRANSACTION_EXPIRED PaymentStatus::TRANSACTION_FAILURE PaymentStatus::TRANSACTION_FRAUD PaymentStatus::TRANSACTION_REFUNDED // BLIK OneClick: PaymentStatus::ALIAS_ACTIVE PaymentStatus::ALIAS_INACTIVE PaymentStatus::ALIAS_PENDING // Typy eventów IPN: PaymentStatus::TYPE_TRANSACTION_STATUS // 'transaction:status_changed' PaymentStatus::TYPE_REFUND_STATUS // 'transaction_refund:status_changed' PaymentStatus::TYPE_BLIK_ALIAS_STATUS // 'blik:alias_status_changed' PaymentStatus::TYPE_BLIK_CODE_STATUS // 'transaction_blik_level0:code_status_changed'