diff --git a/README.md b/README.md
index 23ad3e18..cd6e8592 100644
--- a/README.md
+++ b/README.md
@@ -18,7 +18,7 @@ sistemlerinin kullanılabilmesidir.
| EST V3 POS
EstPos altyapının
daha güvenli
(sha512) hash
algoritmasıyla
uygulaması. | -----"----- | -----"----- | -----"----- |
| PayFlex MPI VPOS V4 | Ziraat
Vakıfbank VPOS 7/24
İşbank | NonSecure
3DSecure
Tekrarlanan Ödeme | İptal
İade
Durum sorgulama |
| PayFlex
Common Payment V4
(Ortak Ödeme) | Ziraat
Vakıfbank
İşbank | NonSecure
3DPay
3DHost | İptal
İade |
-| Garanti Virtual POS | Garanti | NonSecure
3DSecure
3DPay
3DHost
Tekrarlanan Ödeme | İptal
İade
Durum sorgulama
Sipariş Tarihçesini sorgulama |
+| Garanti Virtual POS | Garanti | NonSecure
3DSecure
3DPay
3DHost
Tekrarlanan Ödeme | İptal
İade
Durum sorgulama
Sipariş Tarihçesini sorgulama
Geçmiş İşlemleri sorgulama |
| PosNet | YapıKredi | NonSecure
3DSecure
| İptal
İade
Durum sorgulama |
| PosNetV1
(JSON API) | Albaraka Türk | NonSecure
3DSecure | İptal
İade
Durum sorgulama |
| PayFor | Finansbank
Enpara | NonSecure
3DSecure
3DPay
3DHost | İptal
İade
Durum sorgulama
Sipariş Tarihçesini sorgulama
Geçmiş İşlemleri sorgulama |
diff --git a/docs/HISTORY-EXAMPLE.md b/docs/HISTORY-EXAMPLE.md
index a3e5434f..5a1af137 100644
--- a/docs/HISTORY-EXAMPLE.md
+++ b/docs/HISTORY-EXAMPLE.md
@@ -44,7 +44,7 @@ try {
require 'config.php';
-function createHistoryOrder(string $gatewayClass, array $extraData): array
+function createHistoryOrder(string $gatewayClass, array $extraData, string $ip): array
{
$order = [];
$txTime = new \DateTimeImmutable();
@@ -60,6 +60,14 @@ function createHistoryOrder(string $gatewayClass, array $extraData): array
'start_date' => $txTime->modify('-1 day'),
'end_date' => $txTime->modify('+1 day'),
];
+ } elseif (\Mews\Pos\Gateways\GarantiPos::class === $gatewayClass) {
+ $order = [
+ 'ip' => $ip,
+ 'page' => 1, //optional
+ // Başlangıç ve bitiş tarihleri arasında en fazla 30 gün olabilir
+ 'start_date' => $txTime,
+ 'end_date' => $txTime->modify('+1 day'),
+ ];
} elseif (\Mews\Pos\Gateways\AkbankPos::class === $gatewayClass) {
$order = [
// Gün aralığı 1 günden fazla girilemez
@@ -75,7 +83,7 @@ function createHistoryOrder(string $gatewayClass, array $extraData): array
return $order;
}
-$order = createHistoryOrder(get_class($pos), []);
+$order = createHistoryOrder(get_class($pos), [], '127.0.0.1');
$pos->history($order);
$response = $pos->getResponse();
diff --git a/examples/_common-codes/regular/history.php b/examples/_common-codes/regular/history.php
index 4cb080e1..efbee45c 100644
--- a/examples/_common-codes/regular/history.php
+++ b/examples/_common-codes/regular/history.php
@@ -8,7 +8,7 @@
require '../../_templates/_header.php';
-function createHistoryOrder(string $gatewayClass, array $extraData): array
+function createHistoryOrder(string $gatewayClass, array $extraData, string $ip): array
{
$order = [];
$txTime = new \DateTimeImmutable();
@@ -27,6 +27,15 @@ function createHistoryOrder(string $gatewayClass, array $extraData): array
'start_date' => $txTime->modify('-1 day'),
'end_date' => $txTime->modify('+1 day'),
];
+ } elseif (\Mews\Pos\Gateways\GarantiPos::class === $gatewayClass) {
+ $order = [
+ 'ip' => $ip,
+ 'currency' => \Mews\Pos\PosInterface::CURRENCY_USD,
+ 'page' => 1, //optional
+ // Başlangıç ve bitiş tarihleri arasında en fazla 30 gün olabilir
+ 'start_date' => $txTime,
+ 'end_date' => $txTime->modify('+1 day'),
+ ];
} elseif (\Mews\Pos\Gateways\AkbankPos::class === $gatewayClass) {
$order = [
// Gün aralığı 1 günden fazla girilemez
@@ -42,7 +51,7 @@ function createHistoryOrder(string $gatewayClass, array $extraData): array
return $order;
}
-$order = createHistoryOrder(get_class($pos), []);
+$order = createHistoryOrder(get_class($pos), [], $ip);
dump($order);
$transaction = \Mews\Pos\PosInterface::TX_TYPE_HISTORY;
diff --git a/examples/garanti/regular/history.php b/examples/garanti/regular/history.php
new file mode 100644
index 00000000..34a3742d
--- /dev/null
+++ b/examples/garanti/regular/history.php
@@ -0,0 +1,3 @@
+ 'void',
PosInterface::TX_TYPE_REFUND => 'refund',
PosInterface::TX_TYPE_ORDER_HISTORY => 'orderhistoryinq',
+ PosInterface::TX_TYPE_HISTORY => 'orderlistinq',
PosInterface::TX_TYPE_STATUS => 'orderinq',
];
@@ -319,11 +324,45 @@ public function createOrderHistoryRequestData(AbstractPosAccount $posAccount, ar
}
/**
+ * @param GarantiPosAccount $posAccount
* {@inheritDoc}
*/
public function createHistoryRequestData(AbstractPosAccount $posAccount, array $data = []): array
{
- throw new NotImplementedException();
+ $order = $this->prepareHistoryOrder($data);
+
+ $result = [
+ 'Mode' => $this->getMode(),
+ 'Version' => self::API_VERSION,
+ 'Terminal' => $this->getTerminalData($posAccount),
+ 'Customer' => [
+ 'IPAddress' => $order['ip'],
+ ],
+ 'Order' => [
+ 'OrderID' => null,
+ 'GroupID' => null,
+ 'Description' => null,
+ // Başlangıç ve bitiş tarihleri arasında en fazla 30 gün olabilir
+ 'StartDate' => $this->formatRequestDateTime($order['start_date']),
+ 'EndDate' => $this->formatRequestDateTime($order['end_date']),
+ /**
+ * 500 adetten fazla işlem olması durumunda ListPageNum alanında diğer 500 lü grupların görüntülenmesi
+ * için sayfa numarası yazılır.
+ */
+ 'ListPageNum' => $order['page'],
+ ],
+ 'Transaction' => [
+ 'Type' => $this->mapTxType(PosInterface::TX_TYPE_HISTORY),
+ 'Amount' => $this->formatAmount(1), //sabit olarak amount 100 gonderilecek
+ 'CurrencyCode' => $this->mapCurrency(PosInterface::CURRENCY_TRY),
+ 'CardholderPresentCode' => '0',
+ 'MotoInd' => self::MOTO,
+ ],
+ ];
+
+ $result['Terminal']['HashData'] = $this->crypt->createHash($posAccount, $result);
+
+ return $result;
}
@@ -462,6 +501,19 @@ protected function prepareOrderHistoryOrder(array $order): array
return $this->prepareStatusOrder($order);
}
+ /**
+ * @inheritDoc
+ */
+ protected function prepareHistoryOrder(array $data): array
+ {
+ return [
+ 'start_date' => $data['start_date'],
+ 'end_date' => $data['end_date'],
+ 'page' => $data['page'] ?? 1,
+ 'ip' => $data['ip'],
+ ];
+ }
+
/**
* @inheritDoc
*/
@@ -573,4 +625,14 @@ private function createRecurringData(array $recurringData): array
'StartDate' => isset($recurringData['startDate']) ? $recurringData['startDate']->format('Ymd') : '',
];
}
+
+ /**
+ * @param \DateTimeInterface $dateTime
+ *
+ * @return string example 01/12/2010 11:11
+ */
+ private function formatRequestDateTime(\DateTimeInterface $dateTime): string
+ {
+ return $dateTime->format('d/m/Y H:i');
+ }
}
diff --git a/src/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapper.php b/src/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapper.php
index 46979b18..62a89917 100644
--- a/src/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapper.php
+++ b/src/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapper.php
@@ -253,7 +253,7 @@ public function mapStatusResponse(array $rawResponseData): array
}
/** @var array{Response: array} $transaction */
- $transaction = $rawResponseData['Transaction'];
+ $transaction = $rawResponseData['Transaction'];
/** @var array $orderInqResult */
$orderInqResult = $rawResponseData['Order']['OrderInqResult'];
$defaultResponse = $this->getDefaultStatusResponse($rawResponseData);
@@ -333,7 +333,36 @@ public function mapOrderHistoryResponse(array $rawResponseData): array
*/
public function mapHistoryResponse(array $rawResponseData): array
{
- throw new NotImplementedException();
+ $rawResponseData = $this->emptyStringsToNull($rawResponseData);
+ $procReturnCode = $this->getProcReturnCode($rawResponseData);
+ $status = self::TX_DECLINED;
+ if (self::PROCEDURE_SUCCESS_CODE === $procReturnCode) {
+ $status = self::TX_APPROVED;
+ }
+
+ $mappedTransactions = [];
+ if (self::TX_APPROVED === $status) {
+ $rawTransactions = $rawResponseData['Order']['OrderListInqResult']['OrderTxnList']['OrderTxn'];
+ if (\count($rawTransactions) !== \count($rawTransactions, COUNT_RECURSIVE)) {
+ foreach ($rawTransactions as $transaction) {
+ $mappedTransaction = $this->mapSingleHistoryTransaction($transaction);
+ $mappedTransactions[] = $mappedTransaction;
+ }
+ } else {
+ $mappedTransactions[] = $this->mapSingleHistoryTransaction($rawTransactions);
+ }
+ }
+
+ return [
+ 'proc_return_code' => $procReturnCode,
+ 'error_code' => self::TX_DECLINED === $status ? $procReturnCode : null,
+ 'error_message' => self::TX_DECLINED === $status ? $rawResponseData['Transaction']['Response']['ErrorMsg'] : null,
+ 'status' => $status,
+ 'status_detail' => $this->getStatusDetail($procReturnCode),
+ 'trans_count' => \count($mappedTransactions),
+ 'transactions' => $mappedTransactions,
+ 'all' => $rawResponseData,
+ ];
}
/**
@@ -352,6 +381,23 @@ public function extractMdStatus(array $raw3DAuthResponseData): ?string
return $raw3DAuthResponseData['mdstatus'] ?? null;
}
+ /**
+ * @inheritDoc
+ */
+ public function mapTxType($txType): ?string
+ {
+ $historyResponseTxTypes = [
+ 'Satis' => PosInterface::TX_TYPE_PAY_AUTH,
+ 'On Otorizasyon' => PosInterface::TX_TYPE_PAY_PRE_AUTH,
+ 'On Otorizasyon Kapama' => PosInterface::TX_TYPE_PAY_POST_AUTH,
+ 'Iade' => PosInterface::TX_TYPE_REFUND,
+ 'Iptal' => PosInterface::TX_TYPE_CANCEL,
+ // ... Odul Sorgulama
+ ];
+
+ return $historyResponseTxTypes[$txType] ?? parent::mapTxType($txType);
+ }
+
/**
* 100001 => 1000.01
* @param string $amount
@@ -458,6 +504,40 @@ protected function getProcReturnCode(array $response): ?string
return $response['Transaction']['Response']['Code'] ?? null;
}
+ /**
+ * @inheritDoc
+ */
+ protected function mapCurrency(string $currency): string
+ {
+ $historyResponseCurrencyMapping = [
+ 'TL' => PosInterface::CURRENCY_TRY,
+ 'USD' => PosInterface::CURRENCY_USD,
+ 'EUR' => PosInterface::CURRENCY_EUR,
+ 'RUB' => PosInterface::CURRENCY_RUB,
+ 'JPY' => PosInterface::CURRENCY_JPY,
+ 'GBP' => PosInterface::CURRENCY_GBP,
+ ];
+
+ return $historyResponseCurrencyMapping[$currency] ?? parent::mapCurrency($currency);
+ }
+
+ /**
+ * @inheritDoc
+ */
+ protected function mapInstallment(?string $installment): int
+ {
+ if (null === $installment) {
+ return 0;
+ }
+
+ // history response
+ if ('Pesin' === $installment || '1' === $installment) {
+ return 0;
+ }
+
+ return parent::mapInstallment($installment);
+ }
+
/**
* @param array $rawTx
*
@@ -494,4 +574,97 @@ private function mapSingleOrderHistoryTransaction(array $rawTx): array
return $defaultResponse;
}
+
+ /**
+ * @param array $rawTx
+ *
+ * @return array
+ */
+ private function mapSingleHistoryTransaction(array $rawTx): array
+ {
+ $procReturnCode = $rawTx['ResponseCode'];
+ $status = self::TX_DECLINED;
+ if (self::PROCEDURE_SUCCESS_CODE === $procReturnCode) {
+ $status = self::TX_APPROVED;
+ }
+
+ $defaultResponse = $this->getDefaultOrderHistoryTxResponse();
+ $defaultResponse['auth_code'] = $rawTx['AuthCode'] ?? null;
+ $defaultResponse['ref_ret_num'] = $rawTx['RetrefNum'] ?? null;
+ $defaultResponse['order_id'] = $rawTx['OrderID'];
+ $defaultResponse['batch_num'] = $rawTx['BatchNum'];
+ $defaultResponse['proc_return_code'] = $procReturnCode;
+ $defaultResponse['transaction_type'] = null !== $rawTx['TrxType'] ? $this->mapTxType($rawTx['TrxType']) : null;
+ $defaultResponse['order_status'] = null !== $rawTx['Status'] ? $this->mapHistoryOrderStatus($rawTx['Status'], $defaultResponse['transaction_type']) : null;
+ $defaultResponse['status'] = $status;
+ $defaultResponse['status_detail'] = $this->getStatusDetail($procReturnCode);
+ $defaultResponse['error_code'] = self::TX_APPROVED === $status ? null : $procReturnCode;
+ $defaultResponse['error_message'] = self::TX_APPROVED === $status ? null : $rawTx['SysErrMsg'];
+
+ // 3D Secure => 3D
+ // 3D Pay => 3D
+ // NonSecure => ''
+ $defaultResponse['payment_model'] = '3D' === $rawTx['SafeType'] ? PosInterface::MODEL_3D_SECURE : PosInterface::MODEL_NON_SECURE;
+ $defaultResponse['transaction_time'] = null !== $rawTx['LastTrxDate'] ? new \DateTimeImmutable($rawTx['LastTrxDate']) : null;
+ if (self::TX_APPROVED === $status) {
+ $defaultResponse['masked_number'] = $rawTx['CardNumberMasked'];
+ $defaultResponse['installment_count'] = $this->mapInstallment($rawTx['InstallmentCnt']);
+ $defaultResponse['currency'] = null !== $rawTx['CurrencyCode'] ? $this->mapCurrency($rawTx['CurrencyCode']) : null;
+ $defaultResponse['first_amount'] = null !== $rawTx['AuthAmount'] ? $this->formatAmount($rawTx['AuthAmount']) : null;
+ if ($defaultResponse['order_status'] === PosInterface::PAYMENT_STATUS_PAYMENT_COMPLETED) {
+ $defaultResponse['capture_amount'] = $defaultResponse['first_amount'];
+ $defaultResponse['capture'] = $defaultResponse['first_amount'] > 0 ? $defaultResponse['capture_amount'] === $defaultResponse['first_amount'] : null;
+ $defaultResponse['capture_time'] = $defaultResponse['transaction_time'];
+ }
+ }
+
+ return $defaultResponse;
+ }
+
+ /**
+ * todo anlasilmayan durumlar:
+ * - "Status" => "Iptal", "TrxType" => "Satis"
+ * - "Status" => "Iptal", "TrxType" => "On Otorizasyon"
+ * - "Status" => "Iptal", "TrxType" => "Iptal"
+ * - "Status" => "Iptal", "TrxType" => "Iade"
+ *
+ * @param string $txStatus
+ * @param PosInterface::TX_TYPE_*|null $txType
+ *
+ * @return string|null
+ */
+ private function mapHistoryOrderStatus(string $txStatus, ?string $txType): ?string
+ {
+ if (null === $txType) {
+ return null;
+ }
+ // txStatus possible values:
+ // Basarili
+ // Basarisiz
+ // Iptal
+ // Onaylandi
+
+ if ('Basarili' === $txStatus || 'Onaylandi' === $txStatus) {
+ if (PosInterface::TX_TYPE_CANCEL === $txType) {
+ return PosInterface::PAYMENT_STATUS_CANCELED;
+ }
+ if (PosInterface::TX_TYPE_REFUND === $txType) {
+ // todo how can we decide if order is partially or fully refunded?
+ return PosInterface::PAYMENT_STATUS_FULLY_REFUNDED;
+ }
+ if (PosInterface::TX_TYPE_PAY_AUTH === $txType || PosInterface::TX_TYPE_PAY_POST_AUTH === $txType) {
+ return PosInterface::PAYMENT_STATUS_PAYMENT_COMPLETED;
+ }
+ if (PosInterface::TX_TYPE_PAY_PRE_AUTH === $txType) {
+ return PosInterface::PAYMENT_STATUS_PRE_AUTH_COMPLETED;
+ }
+
+ return null;
+ }
+ if ('Iptal' === $txStatus) {
+ return null;
+ }
+
+ return PosInterface::PAYMENT_STATUS_ERROR;
+ }
}
diff --git a/src/Gateways/GarantiPos.php b/src/Gateways/GarantiPos.php
index 550ba002..b54fc28a 100644
--- a/src/Gateways/GarantiPos.php
+++ b/src/Gateways/GarantiPos.php
@@ -14,7 +14,6 @@
use Mews\Pos\Event\RequestDataPreparedEvent;
use Mews\Pos\Exceptions\HashMismatchException;
use Mews\Pos\Exceptions\UnsupportedPaymentModelException;
-use Mews\Pos\Exceptions\UnsupportedTransactionTypeException;
use Mews\Pos\PosInterface;
use Symfony\Component\HttpFoundation\Request;
@@ -48,7 +47,7 @@ class GarantiPos extends AbstractGateway
PosInterface::TX_TYPE_CANCEL => true,
PosInterface::TX_TYPE_REFUND => true,
PosInterface::TX_TYPE_ORDER_HISTORY => true,
- PosInterface::TX_TYPE_HISTORY => false,
+ PosInterface::TX_TYPE_HISTORY => true,
];
@@ -138,14 +137,6 @@ public function make3DHostPayment(Request $request, array $order, string $txType
throw new UnsupportedPaymentModelException();
}
- /**
- * @inheritDoc
- */
- public function history(array $data): PosInterface
- {
- throw new UnsupportedTransactionTypeException();
- }
-
/**
* @inheritDoc
*
diff --git a/tests/Functional/AkbankPosTest.php b/tests/Functional/AkbankPosTest.php
index 2fd8bbe7..667f75e3 100644
--- a/tests/Functional/AkbankPosTest.php
+++ b/tests/Functional/AkbankPosTest.php
@@ -146,7 +146,7 @@ function (RequestDataPreparedEvent $requestDataPreparedEvent) use (&$eventIsThro
public function testHistorySuccess(): void
{
- $historyOrder = $this->createHistoryOrder(\get_class($this->pos), []);
+ $historyOrder = $this->createHistoryOrder(\get_class($this->pos), [], '127.0.0.1');
$eventIsThrown = false;
$this->eventDispatcher->addListener(
@@ -389,7 +389,7 @@ function (RequestDataPreparedEvent $requestDataPreparedEvent) use (&$eventIsThro
*/
public function testRecurringHistorySuccess(): void
{
- $historyOrder = $this->createHistoryOrder(\get_class($this->pos), []);
+ $historyOrder = $this->createHistoryOrder(\get_class($this->pos), [], '127.0.0.1');
$eventIsThrown = false;
$this->eventDispatcher->addListener(
diff --git a/tests/Functional/GarantiPosTest.php b/tests/Functional/GarantiPosTest.php
index 65f6c24a..1c5a5e5c 100644
--- a/tests/Functional/GarantiPosTest.php
+++ b/tests/Functional/GarantiPosTest.php
@@ -260,4 +260,28 @@ function (RequestDataPreparedEvent $requestDataPreparedEvent) use (&$eventIsThro
$this->assertNotEmpty($response);
$this->assertTrue($eventIsThrown);
}
+
+ /**
+ * @depends testStatusSuccess
+ */
+ public function testHistorySuccess(): void
+ {
+ $historyOrder = $this->createHistoryOrder(\get_class($this->pos), [], '127.0.0.1');
+
+ $eventIsThrown = false;
+ $this->eventDispatcher->addListener(
+ RequestDataPreparedEvent::class,
+ function (RequestDataPreparedEvent $requestDataPreparedEvent) use (&$eventIsThrown): void {
+ $eventIsThrown = true;
+ $this->assertSame(PosInterface::TX_TYPE_HISTORY, $requestDataPreparedEvent->getTxType());
+ $this->assertCount(6, $requestDataPreparedEvent->getRequestData());
+ });
+
+ $this->pos->history($historyOrder);
+
+ $response = $this->pos->getResponse();
+ $this->assertIsArray($response);
+ $this->assertTrue($eventIsThrown);
+ $this->assertNotEmpty($response['transactions']);
+ }
}
diff --git a/tests/Functional/PayForPosTest.php b/tests/Functional/PayForPosTest.php
index e72d60e8..75f157e9 100644
--- a/tests/Functional/PayForPosTest.php
+++ b/tests/Functional/PayForPosTest.php
@@ -171,7 +171,7 @@ function (RequestDataPreparedEvent $requestDataPreparedEvent) use (&$eventIsThro
public function testHistorySuccess(): void
{
- $historyOrder = $this->createHistoryOrder(\get_class($this->pos), []);
+ $historyOrder = $this->createHistoryOrder(\get_class($this->pos), [], '127.0.0.1');
$eventIsThrown = false;
$this->eventDispatcher->addListener(
diff --git a/tests/Functional/PaymentTestTrait.php b/tests/Functional/PaymentTestTrait.php
index 60144077..9119da54 100644
--- a/tests/Functional/PaymentTestTrait.php
+++ b/tests/Functional/PaymentTestTrait.php
@@ -248,18 +248,17 @@ private function createOrderHistoryOrder(string $gatewayClass, array $lastRespon
return $order;
}
- private function createHistoryOrder(string $gatewayClass, array $extraData): array
+ private function createHistoryOrder(string $gatewayClass, array $extraData, string $ip): array
{
+ $txTime = new \DateTimeImmutable();
if (PayForPos::class === $gatewayClass) {
return [
// odeme tarihi
- 'transaction_date' => $extraData['transaction_date'] ?? new \DateTimeImmutable(),
+ 'transaction_date' => $extraData['transaction_date'] ?? $txTime,
];
}
if (\Mews\Pos\Gateways\VakifKatilimPos::class === $gatewayClass) {
- $txTime = new \DateTimeImmutable();
-
return [
'page' => 1,
'page_size' => 20,
@@ -271,9 +270,17 @@ private function createHistoryOrder(string $gatewayClass, array $extraData): arr
];
}
- if (\Mews\Pos\Gateways\AkbankPos::class === $gatewayClass) {
- $txTime = new \DateTimeImmutable();
+ if (\Mews\Pos\Gateways\GarantiPos::class === $gatewayClass) {
+ return [
+ 'ip' => $ip,
+ 'page' => 1,
+ // Başlangıç ve bitiş tarihleri arasında en fazla 30 gün olabilir
+ 'start_date' => $txTime->modify('-1 day'),
+ 'end_date' => $txTime->modify('+1 day'),
+ ];
+ }
+ if (\Mews\Pos\Gateways\AkbankPos::class === $gatewayClass) {
return [
// Gün aralığı 1 günden fazla girilemez
'start_date' => $txTime->modify('-23 hour'),
diff --git a/tests/Unit/DataMapper/RequestDataMapper/GarantiPosRequestDataMapperTest.php b/tests/Unit/DataMapper/RequestDataMapper/GarantiPosRequestDataMapperTest.php
index 21cae2a2..7d5c2f11 100644
--- a/tests/Unit/DataMapper/RequestDataMapper/GarantiPosRequestDataMapperTest.php
+++ b/tests/Unit/DataMapper/RequestDataMapper/GarantiPosRequestDataMapperTest.php
@@ -190,6 +190,19 @@ public function testCreateOrderHistoryRequestData(array $order, array $expectedD
$this->assertEquals($expectedData, $actual);
}
+ /**
+ * @dataProvider historyRequestDataProvider
+ */
+ public function testCreateHistoryRequestData(array $data, array $expectedData): void
+ {
+ $actualData = $this->requestDataMapper->createHistoryRequestData($this->account, $data);
+
+ \ksort($expectedData);
+ \ksort($actualData);
+
+ $this->assertSame($expectedData, $actualData);
+ }
+
/**
* @dataProvider create3DPaymentRequestDataDataProvider
*/
@@ -491,8 +504,8 @@ public static function orderHistoryRequestDataProvider(): array
return [
[
'order' => [
- 'id' => 'order222',
- 'ip' => '156.155.154.153',
+ 'id' => 'order222',
+ 'ip' => '156.155.154.153',
'installment' => 0,
],
'expected' => [
@@ -524,6 +537,51 @@ public static function orderHistoryRequestDataProvider(): array
];
}
+ public static function historyRequestDataProvider(): array
+ {
+ return [
+ [
+ 'data' => [
+ 'start_date' => new \DateTime('2022-05-18 00:00:00'),
+ 'end_date' => new \DateTime('2022-05-18 23:59:59'),
+ 'ip' => '127.0.0.1',
+ ],
+ 'expected' => [
+ 'Customer' => [
+ 'IPAddress' => '127.0.0.1',
+ ],
+
+ 'Mode' => 'TEST',
+ 'Order' => [
+ 'OrderID' => null,
+ 'GroupID' => null,
+ 'Description' => null,
+ 'StartDate' => '18/05/2022 00:00',
+ 'EndDate' => '18/05/2022 23:59',
+ 'ListPageNum' => 1,
+ ],
+
+ 'Terminal' => [
+ 'ProvUserID' => 'PROVAUT',
+ 'UserID' => 'PROVAUT',
+ 'HashData' => '9B53A55199EBAD2F486089FD7310C4BC0C61A99FC37EF61F6BBAE67FA17E47641540B203E83C9F2E64DB64B971FE6FF604274316F6D010426D6AA91BE1D924E6',
+ 'ID' => '30691298',
+ 'MerchantID' => '7000679',
+ ],
+
+ 'Transaction' => [
+ 'Type' => 'orderlistinq',
+ 'Amount' => 100,
+ 'CurrencyCode' => '949',
+ 'CardholderPresentCode' => '0',
+ 'MotoInd' => 'N',
+ ],
+ 'Version' => '512',
+ ],
+ ],
+ ];
+ }
+
public static function create3DPaymentRequestDataDataProvider(): \Generator
{
$account = AccountFactory::createGarantiPosAccount(
diff --git a/tests/Unit/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapperTest.php b/tests/Unit/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapperTest.php
index 5ceca6d8..d33877b4 100644
--- a/tests/Unit/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapperTest.php
+++ b/tests/Unit/DataMapper/ResponseDataMapper/GarantiPosResponseDataMapperTest.php
@@ -7,6 +7,7 @@
use Mews\Pos\DataMapper\RequestDataMapper\GarantiPosRequestDataMapper;
use Mews\Pos\DataMapper\ResponseDataMapper\GarantiPosResponseDataMapper;
+use Mews\Pos\Exceptions\NotImplementedException;
use Mews\Pos\Factory\CryptFactory;
use Mews\Pos\Gateways\GarantiPos;
use Mews\Pos\PosInterface;
@@ -97,6 +98,12 @@ public function testMap3DPaymentData(array $order, string $txType, array $threeD
$this->assertSame($expectedData, $actualData);
}
+ public function testMap3DHostResponseData(): void
+ {
+ $this->expectException(NotImplementedException::class);
+ $this->responseDataMapper->map3DHostResponseData([], PosInterface::TX_TYPE_PAY_AUTH, []);
+ }
+
/**
* @dataProvider threeDPayPaymentDataProvider
*/
@@ -164,6 +171,44 @@ public function testOrderMapHistoryResponse(array $responseData, array $expected
$this->assertCount($actualData['trans_count'], $actualData['transactions']);
+ $this->assertArrayHasKey('all', $actualData);
+ $this->assertIsArray($actualData['all']);
+ $this->assertNotEmpty($actualData['all']);
+ unset($actualData['all']);
+ $this->assertSame($expectedData, $actualData);
+ }
+
+ /**
+ * @dataProvider historyTestDataProvider
+ */
+ public function testMapHistoryResponse(array $responseData, array $expectedData): void
+ {
+ $actualData = $this->responseDataMapper->mapHistoryResponse($responseData);
+
+ if (count($actualData['transactions']) > 1
+ && null !== $actualData['transactions'][0]['transaction_time']
+ && null !== $actualData['transactions'][1]['transaction_time']
+ ) {
+ $this->assertGreaterThan(
+ $actualData['transactions'][0]['transaction_time'],
+ $actualData['transactions'][1]['transaction_time']
+ );
+ }
+
+ $this->assertCount($actualData['trans_count'], $actualData['transactions']);
+
+ foreach (array_keys($actualData['transactions']) as $key) {
+ $this->assertEquals($expectedData['transactions'][$key]['transaction_time'], $actualData['transactions'][$key]['transaction_time'], 'tx: '.$key);
+ $this->assertEquals($expectedData['transactions'][$key]['capture_time'], $actualData['transactions'][$key]['capture_time'], 'tx: '.$key);
+ unset($actualData['transactions'][$key]['transaction_time'], $expectedData['transactions'][$key]['transaction_time']);
+ unset($actualData['transactions'][$key]['capture_time'], $expectedData['transactions'][$key]['capture_time']);
+ \ksort($actualData['transactions'][$key]);
+ \ksort($expectedData['transactions'][$key]);
+ }
+
+ $this->assertArrayHasKey('all', $actualData);
+ $this->assertIsArray($actualData['all']);
+ $this->assertNotEmpty($actualData['all']);
unset($actualData['all']);
$this->assertSame($expectedData, $actualData);
}
@@ -1732,4 +1777,203 @@ public function orderHistoryTestDataProvider(): array
],
];
}
+
+ public static function historyTestDataProvider(): \Generator
+ {
+ $dateRangeHistoryExpected = \json_decode(\file_get_contents(__DIR__.'/../../test_data/garanti/history/daily_range_history_expected.json'), true);
+
+ foreach ($dateRangeHistoryExpected['transactions'] as &$item) {
+ if (null !== $item['transaction_time']) {
+ $item['transaction_time'] = new \DateTimeImmutable(
+ $item['transaction_time']['date'],
+ new \DateTimeZone($item['transaction_time']['timezone'])
+ );
+ }
+ if (null !== $item['capture_time']) {
+ $item['capture_time'] = new \DateTimeImmutable(
+ $item['capture_time']['date'],
+ new \DateTimeZone($item['capture_time']['timezone'])
+ );
+ }
+ }
+
+ yield 'success_data_range_history' => [
+ 'responseData' => \json_decode(\file_get_contents(__DIR__.'/../../test_data/garanti/history/date_range_history.json'), true),
+ //'responseData' => \json_decode(\file_get_contents(__DIR__.'/../../../../var/garanti-last-2-year-history.json'), true),
+ 'expectedData' => $dateRangeHistoryExpected,
+ ];
+ yield 'success_single_transaction' => [
+ 'responseData' => [
+ 'Mode' => '',
+ 'Terminal' => [
+ 'ProvUserID' => 'PROVAUT',
+ 'UserID' => 'PROVAUT',
+ 'ID' => '30691298',
+ 'MerchantID' => '7000679',
+ ],
+ 'Customer' => [
+ 'IPAddress' => '172.26.0.1',
+ 'EmailAddress' => '',
+ ],
+ 'Order' => [
+ 'OrderID' => '',
+ 'GroupID' => '',
+ 'OrderListInqResult' => [
+ 'OrderTxnList' => [
+ 'TotalTxnCount' => '1',
+ 'TotalPageCount' => '1',
+ 'ActPageNum' => '1',
+ 'OrderTxn' => [
+ 'Id' => '1',
+ 'LastTrxDate' => '2024-06-03 16:06:29',
+ 'TrxType' => 'Satis',
+ 'OrderID' => '202406036C78',
+ 'Name' => '',
+ 'CardNumberMasked' => '42822090****8015',
+ 'ExpireDate' => '0830',
+ 'BankBin' => '42822090',
+ 'BatchNum' => '576200',
+ 'AuthCode' => '304919',
+ 'RetrefNum' => '415501677066',
+ 'OrigRetrefNum' => '',
+ 'InstallmentCnt' => 'Pesin',
+ 'Status' => 'Basarili',
+ 'AuthAmount' => '1001',
+ 'CurrencyCode' => 'TL',
+ 'RemainingBNSAmount' => '0',
+ 'UsedFBBAmount' => '0',
+ 'UsedChequeType' => '',
+ 'UsedChequeCount' => '0',
+ 'UsedChequeAmount' => '0',
+ 'SafeType' => '',
+ 'Comment1' => '',
+ 'Comment2' => '',
+ 'Comment3' => '',
+ 'UserId' => 'PROVAUT',
+ 'Settlement' => 'N',
+ 'EmailAddress' => '',
+ 'RecurringTotalPaymentNum' => '0',
+ 'RecurringLastPaymentNum' => '0',
+ 'RecurringTxnAmount' => '0',
+ 'ResponseCode' => '00',
+ 'SysErrMsg' => '',
+ ],
+ ],
+ ],
+ ],
+ 'Transaction' => [
+ 'Response' => [
+ 'Source' => 'GVPS',
+ 'Code' => '00',
+ 'ReasonCode' => '',
+ 'Message' => 'Approved',
+ 'ErrorMsg' => '',
+ 'SysErrMsg' => '',
+ ],
+ 'RetrefNum' => '',
+ 'AuthCode' => '',
+ 'BatchNum' => '',
+ 'SequenceNum' => '',
+ 'ProvDate' => '20240603 16:07:07',
+ 'CardNumberMasked' => '',
+ 'CardHolderName' => '',
+ 'CardType' => '',
+ 'HashData' => 'C1DD90277E3CE36D6226FF02E59D95999D78793CF8942860600BA2800A63CB991A518C1DECA1609C99DA8F8995CBB78A54E7D34F337A8BFF60D10B6DB47C8750',
+ 'HostMsgList' => '',
+ 'RewardInqResult' => [
+ 'RewardList' => '',
+ 'ChequeList' => '',
+ ],
+ ],
+ ],
+ 'expectedData' => [
+ 'proc_return_code' => '00',
+ 'error_code' => null,
+ 'error_message' => null,
+ 'status' => 'approved',
+ 'status_detail' => 'approved',
+ 'trans_count' => 1,
+ 'transactions' => [
+ [
+ 'auth_code' => '304919',
+ 'proc_return_code' => '00',
+ 'transaction_id' => null,
+ 'transaction_time' => new \DateTimeImmutable('2024-06-03 16:06:29'),
+ 'capture_time' => new \DateTimeImmutable('2024-06-03 16:06:29'),
+ 'error_message' => null,
+ 'ref_ret_num' => '415501677066',
+ 'order_status' => 'PAYMENT_COMPLETED',
+ 'transaction_type' => 'pay',
+ 'first_amount' => 10.01,
+ 'capture_amount' => 10.01,
+ 'status' => 'approved',
+ 'error_code' => null,
+ 'status_detail' => 'approved',
+ 'capture' => true,
+ 'currency' => 'TRY',
+ 'masked_number' => '42822090****8015',
+ 'order_id' => '202406036C78',
+ 'batch_num' => '576200',
+ 'payment_model' => 'regular',
+ 'installment_count' => 0,
+ ],
+ ],
+ ],
+ ];
+ yield 'fail_invalid_fields' => [
+ 'responseData' => [
+ 'Mode' => '',
+ 'Terminal' => [
+ 'ProvUserID' => 'PROVAUT',
+ 'UserID' => 'PROVAUT',
+ 'ID' => '30691298',
+ 'MerchantID' => '7000679',
+ ],
+ 'Customer' => [
+ 'IPAddress' => '',
+ 'EmailAddress' => '',
+ ],
+ 'Order' => [
+ 'OrderID' => '',
+ 'GroupID' => '',
+ 'OrderListInqResult' => [
+ 'OrderTxnList' => '',
+ ],
+ ],
+ 'Transaction' => [
+ 'Response' => [
+ 'Source' => 'GVPS',
+ 'Code' => '92',
+ 'ReasonCode' => '0002',
+ 'Message' => 'Declined',
+ 'ErrorMsg' => 'Giriş yaptığınız işlem tipi için zorunlu alanları kontrol ediniz',
+ 'SysErrMsg' => 'CustomerIPAddress field must contain value because of the Mandatory Rule:null',
+ ],
+ 'RetrefNum' => '',
+ 'AuthCode' => '',
+ 'BatchNum' => '',
+ 'SequenceNum' => '',
+ 'ProvDate' => '20240530 12:53:46',
+ 'CardNumberMasked' => '',
+ 'CardHolderName' => '',
+ 'CardType' => '',
+ 'HashData' => '09852B466F45FE00769BE0E40F028FFF7B560CCA5871F8E562B910C2E9CEF9972A8C7F8655D67D1E31B24E81BF57F5B35F8446A94591256DCFEB92D551FEC858',
+ 'HostMsgList' => '',
+ 'RewardInqResult' => [
+ 'RewardList' => '',
+ 'ChequeList' => '',
+ ],
+ ],
+ ],
+ 'expectedData' => [
+ 'proc_return_code' => '92',
+ 'error_code' => '92',
+ 'error_message' => 'Giriş yaptığınız işlem tipi için zorunlu alanları kontrol ediniz',
+ 'status' => 'declined',
+ 'status_detail' => 'invalid_transaction',
+ 'trans_count' => 0,
+ 'transactions' => [],
+ ],
+ ];
+ }
}
diff --git a/tests/Unit/Gateways/GarantiPosTest.php b/tests/Unit/Gateways/GarantiPosTest.php
index c29803fa..a47b9671 100644
--- a/tests/Unit/Gateways/GarantiPosTest.php
+++ b/tests/Unit/Gateways/GarantiPosTest.php
@@ -13,7 +13,6 @@
use Mews\Pos\Entity\Card\CreditCardInterface;
use Mews\Pos\Event\RequestDataPreparedEvent;
use Mews\Pos\Exceptions\UnsupportedPaymentModelException;
-use Mews\Pos\Exceptions\UnsupportedTransactionTypeException;
use Mews\Pos\Factory\AccountFactory;
use Mews\Pos\Factory\CreditCardFactory;
use Mews\Pos\Gateways\GarantiPos;
@@ -449,10 +448,38 @@ public function testRefundRequest(array $order, string $apiUrl): void
$this->pos->refund($order);
}
- public function testHistoryRequest(): void
+ /**
+ * @dataProvider historyRequestDataProvider
+ */
+ public function testHistoryRequest(array $order, string $apiUrl): void
{
- $this->expectException(UnsupportedTransactionTypeException::class);
- $this->pos->history([]);
+ $account = $this->pos->getAccount();
+ $txType = PosInterface::TX_TYPE_HISTORY;
+ $requestData = ['createHistoryRequestData'];
+
+ $this->requestMapperMock->expects(self::once())
+ ->method('createHistoryRequestData')
+ ->with($account, $order)
+ ->willReturn($requestData);
+
+ $decodedResponse = ['decodedData'];
+ $this->configureClientResponse(
+ $txType,
+ $apiUrl,
+ $requestData,
+ 'request-body',
+ 'response-body',
+ $decodedResponse,
+ $order,
+ PosInterface::MODEL_NON_SECURE
+ );
+
+ $this->responseMapperMock->expects(self::once())
+ ->method('mapHistoryResponse')
+ ->with($decodedResponse)
+ ->willReturn(['result']);
+
+ $this->pos->history($order);
}
/**
@@ -632,4 +659,18 @@ private function configureClientResponse(
&& $order === $dispatchedEvent->getOrder()
&& $paymentModel === $dispatchedEvent->getPaymentModel()));
}
+
+ public static function historyRequestDataProvider(): array
+ {
+ return [
+ [
+ 'order' => [
+ 'ip' => '127.0.0.1',
+ 'start_date' => new \DateTimeImmutable(),
+ 'end_date' => new \DateTimeImmutable(),
+ ],
+ 'api_url' => 'https://sanalposprovtest.garantibbva.com.tr/VPServlet',
+ ],
+ ];
+ }
}
diff --git a/tests/Unit/test_data/garanti/history/daily_range_history_expected.json b/tests/Unit/test_data/garanti/history/daily_range_history_expected.json
new file mode 100644
index 00000000..aa52d2c7
--- /dev/null
+++ b/tests/Unit/test_data/garanti/history/daily_range_history_expected.json
@@ -0,0 +1,155 @@
+{
+ "proc_return_code": "00",
+ "error_code": null,
+ "error_message": null,
+ "status": "approved",
+ "status_detail": "approved",
+ "trans_count": 5,
+ "transactions": [
+ {
+ "auth_code": "304919",
+ "proc_return_code": "00",
+ "transaction_id": null,
+ "transaction_time": {
+ "date": "2024-06-03 14:04:19.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "capture_time": {
+ "date": "2024-06-03 14:04:19.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "error_message": null,
+ "ref_ret_num": "415501676495",
+ "order_status": "PAYMENT_COMPLETED",
+ "transaction_type": "pay",
+ "first_amount": 10.01,
+ "capture_amount": 10.01,
+ "status": "approved",
+ "error_code": null,
+ "status_detail": "approved",
+ "capture": true,
+ "currency": "TRY",
+ "masked_number": "42822090****8015",
+ "order_id": "202406033030",
+ "batch_num": "576200",
+ "payment_model": "regular",
+ "installment_count": 0
+ },
+ {
+ "auth_code": null,
+ "proc_return_code": "05",
+ "transaction_id": null,
+ "transaction_time": {
+ "date": "2024-06-03 14:04:29.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "capture_time": null,
+ "error_message": "RPC-05 condition was raised",
+ "ref_ret_num": "415501676496",
+ "order_status": "ERROR",
+ "transaction_type": "refund",
+ "first_amount": null,
+ "capture_amount": null,
+ "status": "declined",
+ "error_code": "05",
+ "status_detail": "reject",
+ "capture": null,
+ "currency": null,
+ "masked_number": null,
+ "order_id": "202406033030",
+ "batch_num": "576200",
+ "payment_model": "regular"
+ },
+ {
+ "auth_code": "304919",
+ "proc_return_code": "00",
+ "transaction_id": null,
+ "transaction_time": {
+ "date": "2024-06-03 14:20:59.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "capture_time": {
+ "date": "2024-06-03 14:20:59.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "error_message": null,
+ "ref_ret_num": "415501676585",
+ "order_status": "PAYMENT_COMPLETED",
+ "transaction_type": "pay",
+ "first_amount": 4903.17,
+ "capture_amount": 4903.17,
+ "status": "approved",
+ "error_code": null,
+ "status_detail": "approved",
+ "capture": true,
+ "currency": "TRY",
+ "masked_number": "42822090****8015",
+ "order_id": "__lwyvqzf9_____65848",
+ "batch_num": "576200",
+ "payment_model": "3d",
+ "installment_count": 0
+ },
+ {
+ "auth_code": "304919",
+ "proc_return_code": "00",
+ "transaction_id": null,
+ "transaction_time": {
+ "date": "2024-06-03 14:50:07.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "capture_time": {
+ "date": "2024-06-03 14:50:07.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "error_message": null,
+ "ref_ret_num": "415501676711",
+ "order_status": "PAYMENT_COMPLETED",
+ "transaction_type": "pay",
+ "first_amount": 276.0,
+ "capture_amount": 276.0,
+ "status": "approved",
+ "error_code": null,
+ "status_detail": "approved",
+ "capture": true,
+ "currency": "TRY",
+ "masked_number": "42822090****8015",
+ "order_id": "00000093877E202406031450",
+ "batch_num": "576200",
+ "payment_model": "3d",
+ "installment_count": 0
+ },
+ {
+ "auth_code": null,
+ "proc_return_code": "14",
+ "transaction_id": null,
+ "transaction_time": {
+ "date": "2024-06-03 15:09:57.000000",
+ "timezone_type": 3,
+ "timezone": "UTC"
+ },
+ "capture_time": null,
+ "error_message": "HATALI KART NUMARASI.",
+ "ref_ret_num": "415501676775",
+ "order_status": "ERROR",
+ "transaction_type": "pre",
+ "first_amount": null,
+ "capture_amount": null,
+ "status": "declined",
+ "error_code": "14",
+ "status_detail": null,
+ "capture": null,
+ "currency": null,
+ "masked_number": null,
+ "order_id": "202406038F95",
+ "batch_num": "576200",
+ "payment_model": "regular"
+ }
+ ]
+}
diff --git a/tests/Unit/test_data/garanti/history/date_range_history.json b/tests/Unit/test_data/garanti/history/date_range_history.json
new file mode 100644
index 00000000..fbfce410
--- /dev/null
+++ b/tests/Unit/test_data/garanti/history/date_range_history.json
@@ -0,0 +1,225 @@
+{
+ "Mode": "",
+ "Terminal": {
+ "ProvUserID": "PROVAUT",
+ "UserID": "PROVAUT",
+ "ID": "30691298",
+ "MerchantID": "7000679"
+ },
+ "Customer": {
+ "IPAddress": "172.26.0.1",
+ "EmailAddress": ""
+ },
+ "Order": {
+ "OrderID": "",
+ "GroupID": "",
+ "OrderListInqResult": {
+ "OrderTxnList": {
+ "TotalTxnCount": "5",
+ "TotalPageCount": "1",
+ "ActPageNum": "1",
+ "OrderTxn": [
+ {
+ "Id": "1",
+ "LastTrxDate": "2024-06-03 14:04:19",
+ "TrxType": "Satis",
+ "OrderID": "202406033030",
+ "Name": "",
+ "CardNumberMasked": "42822090****8015",
+ "ExpireDate": "0830",
+ "BankBin": "42822090",
+ "BatchNum": "576200",
+ "AuthCode": "304919",
+ "RetrefNum": "415501676495",
+ "OrigRetrefNum": "",
+ "InstallmentCnt": "Pesin",
+ "Status": "Basarili",
+ "AuthAmount": "1001",
+ "CurrencyCode": "TL",
+ "RemainingBNSAmount": "0",
+ "UsedFBBAmount": "0",
+ "UsedChequeType": "",
+ "UsedChequeCount": "0",
+ "UsedChequeAmount": "0",
+ "SafeType": "",
+ "Comment1": "",
+ "Comment2": "",
+ "Comment3": "",
+ "UserId": "PROVAUT",
+ "Settlement": "N",
+ "EmailAddress": "",
+ "RecurringTotalPaymentNum": "0",
+ "RecurringLastPaymentNum": "0",
+ "RecurringTxnAmount": "0",
+ "ResponseCode": "00",
+ "SysErrMsg": ""
+ },
+ {
+ "Id": "2",
+ "LastTrxDate": "2024-06-03 14:04:29",
+ "TrxType": "Iade",
+ "OrderID": "202406033030",
+ "Name": "",
+ "CardNumberMasked": "42822090****8015",
+ "ExpireDate": "0830",
+ "BankBin": "42822090",
+ "BatchNum": "576200",
+ "AuthCode": "",
+ "RetrefNum": "415501676496",
+ "OrigRetrefNum": "415501676495",
+ "InstallmentCnt": "Pesin",
+ "Status": "Basarisiz",
+ "AuthAmount": "801",
+ "CurrencyCode": "TL",
+ "RemainingBNSAmount": "0",
+ "UsedFBBAmount": "0",
+ "UsedChequeType": "",
+ "UsedChequeCount": "0",
+ "UsedChequeAmount": "0",
+ "SafeType": "",
+ "Comment1": "",
+ "Comment2": "",
+ "Comment3": "",
+ "UserId": "PROVRFN",
+ "Settlement": "E",
+ "EmailAddress": "",
+ "RecurringTotalPaymentNum": "0",
+ "RecurringLastPaymentNum": "0",
+ "RecurringTxnAmount": "0",
+ "ResponseCode": "05",
+ "SysErrMsg": "RPC-05 condition was raised"
+ },
+ {
+ "Id": "3",
+ "LastTrxDate": "2024-06-03 14:20:59",
+ "TrxType": "Satis",
+ "OrderID": "__lwyvqzf9_____65848",
+ "Name": "",
+ "CardNumberMasked": "42822090****8015",
+ "ExpireDate": "0327",
+ "BankBin": "42822090",
+ "BatchNum": "576200",
+ "AuthCode": "304919",
+ "RetrefNum": "415501676585",
+ "OrigRetrefNum": "",
+ "InstallmentCnt": "Pesin",
+ "Status": "Basarili",
+ "AuthAmount": "490317",
+ "CurrencyCode": "TL",
+ "RemainingBNSAmount": "0",
+ "UsedFBBAmount": "0",
+ "UsedChequeType": "",
+ "UsedChequeCount": "0",
+ "UsedChequeAmount": "0",
+ "SafeType": "3D",
+ "Comment1": "",
+ "Comment2": "",
+ "Comment3": "",
+ "UserId": "GARANTI",
+ "Settlement": "N",
+ "EmailAddress": "yigit.ozerdem@etstur.com",
+ "RecurringTotalPaymentNum": "0",
+ "RecurringLastPaymentNum": "0",
+ "RecurringTxnAmount": "0",
+ "ResponseCode": "00",
+ "SysErrMsg": ""
+ },
+ {
+ "Id": "4",
+ "LastTrxDate": "2024-06-03 14:50:07",
+ "TrxType": "Satis",
+ "OrderID": "00000093877E202406031450",
+ "Name": "",
+ "CardNumberMasked": "42822090****8015",
+ "ExpireDate": "0926",
+ "BankBin": "42822090",
+ "BatchNum": "576200",
+ "AuthCode": "304919",
+ "RetrefNum": "415501676711",
+ "OrigRetrefNum": "",
+ "InstallmentCnt": "Pesin",
+ "Status": "Basarili",
+ "AuthAmount": "27600",
+ "CurrencyCode": "TL",
+ "RemainingBNSAmount": "0",
+ "UsedFBBAmount": "0",
+ "UsedChequeType": "",
+ "UsedChequeCount": "0",
+ "UsedChequeAmount": "0",
+ "SafeType": "3D",
+ "Comment1": "",
+ "Comment2": "",
+ "Comment3": "",
+ "UserId": "KKB_USER",
+ "Settlement": "N",
+ "EmailAddress": "ozlemtopuz@kkb.com.tr",
+ "RecurringTotalPaymentNum": "0",
+ "RecurringLastPaymentNum": "0",
+ "RecurringTxnAmount": "0",
+ "ResponseCode": "00",
+ "SysErrMsg": ""
+ },
+ {
+ "Id": "5",
+ "LastTrxDate": "2024-06-03 15:09:57",
+ "TrxType": "On Otorizasyon",
+ "OrderID": "202406038F95",
+ "Name": "",
+ "CardNumberMasked": "42822090****8015",
+ "ExpireDate": "0830",
+ "BankBin": "42822090",
+ "BatchNum": "576200",
+ "AuthCode": "",
+ "RetrefNum": "415501676775",
+ "OrigRetrefNum": "",
+ "InstallmentCnt": "Pesin",
+ "Status": "Basarisiz",
+ "AuthAmount": "1001",
+ "CurrencyCode": "TL",
+ "RemainingBNSAmount": "0",
+ "UsedFBBAmount": "0",
+ "UsedChequeType": "",
+ "UsedChequeCount": "0",
+ "UsedChequeAmount": "0",
+ "SafeType": "",
+ "Comment1": "",
+ "Comment2": "",
+ "Comment3": "",
+ "UserId": "PROVAUT",
+ "Settlement": "E",
+ "EmailAddress": "",
+ "RecurringTotalPaymentNum": "0",
+ "RecurringLastPaymentNum": "0",
+ "RecurringTxnAmount": "0",
+ "ResponseCode": "14",
+ "SysErrMsg": "HATALI KART NUMARASI."
+ }
+ ]
+ }
+ }
+ },
+ "Transaction": {
+ "Response": {
+ "Source": "GVPS",
+ "Code": "00",
+ "ReasonCode": "",
+ "Message": "Approved",
+ "ErrorMsg": "",
+ "SysErrMsg": ""
+ },
+ "RetrefNum": "",
+ "AuthCode": "",
+ "BatchNum": "",
+ "SequenceNum": "",
+ "ProvDate": "20240603 15:12:44",
+ "CardNumberMasked": "",
+ "CardHolderName": "",
+ "CardType": "",
+ "HashData": "8143B6233C873EC7C38AA560C98F34C646CB1F375B84C6760D6EDEFD914F396042C88625F5F5BA205C64C0C7D4C2ACD8F478EEE730AFA159930B5D479BEEC1F5",
+ "HostMsgList": "",
+ "RewardInqResult": {
+ "RewardList": "",
+ "ChequeList": ""
+ }
+ }
+}