diff --git a/.editorConfig b/.editorConfig new file mode 100644 index 0000000..cd8eb86 --- /dev/null +++ b/.editorConfig @@ -0,0 +1,15 @@ +; This file is for unifying the coding style for different editors and IDEs. +; More information at http://editorconfig.org + +root = true + +[*] +charset = utf-8 +indent_size = 4 +indent_style = space +end_of_line = lf +insert_final_newline = true +trim_trailing_whitespace = true + +[*.md] +trim_trailing_whitespace = false diff --git a/.travis.yml b/.travis.yml index 7dcfa7a..eb13864 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,7 +1,6 @@ language: php php: - - 5.3 - 5.4 - 5.5 - 5.6 diff --git a/src/Message/AbstractRequest.php b/src/Message/AbstractRequest.php index ffcb047..75fd802 100644 --- a/src/Message/AbstractRequest.php +++ b/src/Message/AbstractRequest.php @@ -16,6 +16,7 @@ abstract class AbstractRequest extends \Omnipay\Common\Message\AbstractRequest { protected $liveEndpoint = 'https://api.ewaypayments.com'; protected $testEndpoint = 'https://api.sandbox.ewaypayments.com'; + protected $action; public function getApiKey() { @@ -84,6 +85,22 @@ public function setInvoiceReference($value) return $this->setParameter('invoiceReference', $value); } + /** + * @return string|NULL + */ + public function getAction() + { + return $this->action; + } + + /** + * @param string $action + */ + public function setAction($action) + { + $this->action = $action; + } + protected function getBaseData() { $data = array(); diff --git a/src/Message/RapidCreateCardRequest.php b/src/Message/RapidCreateCardRequest.php new file mode 100644 index 0000000..bbd7095 --- /dev/null +++ b/src/Message/RapidCreateCardRequest.php @@ -0,0 +1,45 @@ +validate('returnUrl'); + + $data = $this->getBaseData(); + + $data['TransactionType'] = 'Purchase'; + $data['RedirectUrl'] = $this->getReturnUrl(); + + // Shared page parameters (optional) + $data['CancelUrl'] = $this->getCancelUrl(); + + $data['Payment'] = array(); + + if ($this->getAction() === 'Purchase') { + $data['Payment']['TotalAmount'] = (int) $this->getAmountInteger(); + $data['Payment']['InvoiceNumber'] = $this->getTransactionId(); + $data['Payment']['InvoiceDescription'] = $this->getDescription(); + $data['Payment']['CurrencyCode'] = $this->getCurrency(); + $data['Payment']['InvoiceReference'] = $this->getInvoiceReference(); + $data['Method'] = 'TokenPayment'; + } else { + $data['Method'] = 'CreateTokenCustomer'; + $data['Payment']['TotalAmount'] = 0; + } + + return $data; + } +} diff --git a/src/Message/RapidDirectAbstractRequest.php b/src/Message/RapidDirectAbstractRequest.php index e7c7b50..e29e89b 100644 --- a/src/Message/RapidDirectAbstractRequest.php +++ b/src/Message/RapidDirectAbstractRequest.php @@ -62,8 +62,12 @@ protected function getBaseData() if ($this->getCard()) { $data['Customer']['CardDetails'] = array(); $data['Customer']['CardDetails']['Name'] = $this->getCard()->getName(); - $data['Customer']['CardDetails']['ExpiryMonth'] = $this->getCard()->getExpiryDate('m'); - $data['Customer']['CardDetails']['ExpiryYear'] = $this->getCard()->getExpiryDate('y'); + + if ($this->getCard()->getExpiryYear() && $this->getCard()->getExpiryMonth()) { + // Expiry date not required if token present + $data['Customer']['CardDetails']['ExpiryMonth'] = $this->getCard()->getExpiryDate('m'); + $data['Customer']['CardDetails']['ExpiryYear'] = $this->getCard()->getExpiryDate('y'); + } $data['Customer']['CardDetails']['CVN'] = $this->getCard()->getCvv(); if ($this->getEncryptedCardNumber()) { diff --git a/src/Message/RapidDirectCreateCardRequest.php b/src/Message/RapidDirectCreateCardRequest.php index 7894242..26b66f1 100644 --- a/src/Message/RapidDirectCreateCardRequest.php +++ b/src/Message/RapidDirectCreateCardRequest.php @@ -5,6 +5,9 @@ namespace Omnipay\Eway\Message; +use Omnipay\Eway\RapidDirectGateway; +use Omnipay\Omnipay; + /** * eWAY Rapid Direct Create Card Request * @@ -78,6 +81,25 @@ public function sendData($data) ->setAuth($this->getApiKey(), $this->getPassword()) ->send(); - return $this->response = new RapidDirectCreateCardResponse($this, $httpResponse->json()); + $this->response = new RapidDirectCreateCardResponse($this, $httpResponse->json()); + + if ($this->getAction() === 'Purchase' && $this->response->isSuccessful()) { + /** @var RapidDirectGateway $purchaseGateway */ + $purchaseGateway = Omnipay::create('Eway_RapidDirect'); + $purchaseGateway->setApiKey($this->getApiKey()); + $purchaseGateway->setPassword($this->getPassword()); + $purchaseGateway->setTestMode($this->getTestMode()); + $purchaseResponse = $purchaseGateway->purchase(array( + 'amount' => $this->getAmount(), + 'currency' => $this->getCurrency(), + 'description' => $this->getDescription(), + 'transactionId' => $this->getTransactionId(), + 'card' => $this->getCard(), + 'cardReference' => $this->response->getCardReference(), + ))->send(); + $this->response->setPurchaseResponse($purchaseResponse); + } + + return $this->response; } } diff --git a/src/Message/RapidDirectCreateCardResponse.php b/src/Message/RapidDirectCreateCardResponse.php index 2648620..5441049 100644 --- a/src/Message/RapidDirectCreateCardResponse.php +++ b/src/Message/RapidDirectCreateCardResponse.php @@ -2,20 +2,47 @@ /** * eWAY Rapid Direct Create Card Response */ - + namespace Omnipay\Eway\Message; +use Omnipay\SecurePay\Message\DirectPostCompletePurchaseResponse; + /** * eWAY Rapid Direct Create Card Response - * - * This is the response class for Rapid Direct when creating + * + * This is the response class for Rapid Direct when creating * or updating a card * */ class RapidDirectCreateCardResponse extends RapidResponse { + /** + * @var DirectPostCompletePurchaseResponse + */ + protected $purchaseResponse; + + /** + * @return DirectPostCompletePurchaseResponse + */ + public function getPurchaseResponse() + { + return $this->purchaseResponse; + } + + /** + * @param DirectPostCompletePurchaseResponse $purchaseResponse + */ + public function setPurchaseResponse($purchaseResponse) + { + $this->purchaseResponse = $purchaseResponse; + } + public function isSuccessful() { - return $this->data['ResponseMessage'] == 'A2000'; + if (!$this->getPurchaseResponse()) { + return $this->data['ResponseMessage'] == 'A2000'; + } else { + return ($this->data['ResponseMessage'] == 'A2000' && $this->purchaseResponse->isSuccessful()); + } } } diff --git a/src/Message/RapidDirectPurchaseRequest.php b/src/Message/RapidDirectPurchaseRequest.php index b9a187c..ed8003d 100644 --- a/src/Message/RapidDirectPurchaseRequest.php +++ b/src/Message/RapidDirectPurchaseRequest.php @@ -82,6 +82,11 @@ public function getData() $data['Payment']['CurrencyCode'] = $this->getCurrency(); $data['Payment']['InvoiceReference'] = $this->getInvoiceReference(); + if (empty($data['Customer']['CardDetails']['CVN']) && $this->getCardReference()) { + // We have a token and card is not present so treat as MOTO. + $data['TransactionType'] = 'MOTO'; + } + if ($this->getCardReference()) { $data['Method'] = 'TokenPayment'; } else { diff --git a/src/Message/RapidResponse.php b/src/Message/RapidResponse.php index a3b2ccc..67ad353 100644 --- a/src/Message/RapidResponse.php +++ b/src/Message/RapidResponse.php @@ -39,6 +39,16 @@ public function getRedirectData() } } + /** + * Is the response a transparent redirect? + * + * @return boolean + */ + public function isTransparentRedirect() + { + return true; + } + /** * Get a card reference (eWAY Token), for createCard requests. * @@ -49,12 +59,32 @@ public function getCardReference() if (isset($this->data['Customer']['TokenCustomerID'])) { return $this->data['Customer']['TokenCustomerID']; } + if (isset($this->data['TokenCustomerID'])) { + // This format appears when creating a card and making a concurrent + // payment using Shared or Transparent redirect methods. + return $this->data['TokenCustomerID']; + } return null; } + /** + * Get the transaction ID as generated by the merchant website. + * + * @return string + */ + public function getTransactionId() + { + return $this->data['InvoiceNumber']; + } + /** * Get InvoiceNumber - merchant reference for a transaction + * + * @deprecated Omnipay standard interface is to return this when getTransactionId + * is called + * + * @see https://github.com/thephpleague/omnipay#successful-response */ public function getInvoiceNumber() { diff --git a/src/Message/RapidSharedCreateCardRequest.php b/src/Message/RapidSharedCreateCardRequest.php index 8c097ee..6c4fdd0 100644 --- a/src/Message/RapidSharedCreateCardRequest.php +++ b/src/Message/RapidSharedCreateCardRequest.php @@ -14,12 +14,30 @@ */ class RapidSharedCreateCardRequest extends RapidSharedPurchaseRequest { + protected $action; + + /** + * @return string|NULL + */ + public function getAction() + { + return $this->action; + } + + /** + * @param string $action + */ + public function setAction($action) + { + $this->action = $action; + } + public function getData() { $this->validate('returnUrl'); $data = $this->getBaseData(); - $data['Method'] = 'CreateTokenCustomer'; + $data['TransactionType'] = 'Purchase'; $data['RedirectUrl'] = $this->getReturnUrl(); @@ -32,7 +50,18 @@ public function getData() $data['CustomView'] = $this->getCustomView(); $data['Payment'] = array(); - $data['Payment']['TotalAmount'] = 0; + + if ($this->getAction() === 'Purchase') { + $data['Payment']['TotalAmount'] = (int) $this->getAmountInteger(); + $data['Payment']['InvoiceNumber'] = $this->getTransactionId(); + $data['Payment']['InvoiceDescription'] = $this->getDescription(); + $data['Payment']['CurrencyCode'] = $this->getCurrency(); + $data['Payment']['InvoiceReference'] = $this->getInvoiceReference(); + $data['Method'] = 'TokenPayment'; + } else { + $data['Method'] = 'CreateTokenCustomer'; + $data['Payment']['TotalAmount'] = 0; + } return $data; } diff --git a/src/RapidGateway.php b/src/RapidGateway.php index 48cff5d..92625ae 100644 --- a/src/RapidGateway.php +++ b/src/RapidGateway.php @@ -6,6 +6,7 @@ namespace Omnipay\Eway; use Omnipay\Common\AbstractGateway; +use Omnipay\Omnipay; /** * eWAY Rapid Transparent Redirect Gateway @@ -64,6 +65,13 @@ public function setPassword($value) public function purchase(array $parameters = array()) { + if (!empty($parameters['cardTransactionType']) && $parameters['cardTransactionType'] === 'continuous') { + $gateway = Omnipay::create('Eway_RapidDirect'); + $gateway->setApiKey($this->getApiKey()); + $gateway->setPassword($this->getPassword()); + $gateway->setTestMode($this->getTestMode()); + return $gateway->createRequest('\Omnipay\Eway\Message\RapidDirectPurchaseRequest', $parameters); + } return $this->createRequest('\Omnipay\Eway\Message\RapidPurchaseRequest', $parameters); } @@ -76,4 +84,9 @@ public function refund(array $parameters = array()) { return $this->createRequest('\Omnipay\Eway\Message\RefundRequest', $parameters); } + + public function createCard(array $parameters = array()) + { + return $this->createRequest('\Omnipay\Eway\Message\RapidCreateCardRequest', $parameters); + } } diff --git a/src/RapidSharedGateway.php b/src/RapidSharedGateway.php index 5f5696a..882549f 100644 --- a/src/RapidSharedGateway.php +++ b/src/RapidSharedGateway.php @@ -6,6 +6,7 @@ namespace Omnipay\Eway; use Omnipay\Common\AbstractGateway; +use Omnipay\Omnipay; /** * eWAY Rapid Responsive Shared Page Gateway @@ -61,6 +62,13 @@ public function setPassword($value) public function purchase(array $parameters = array()) { + if (!empty($parameters['cardTransactionType']) && $parameters['cardTransactionType'] === 'continuous') { + $gateway = Omnipay::create('Eway_RapidDirect'); + $gateway->setApiKey($this->getApiKey()); + $gateway->setPassword($this->getPassword()); + $gateway->setTestMode($this->getTestMode()); + return $gateway->createRequest('\Omnipay\Eway\Message\RapidDirectPurchaseRequest', $parameters); + } return $this->createRequest('\Omnipay\Eway\Message\RapidSharedPurchaseRequest', $parameters); } diff --git a/tests/Message/RapidDirectCreateCardRequestTest.php b/tests/Message/RapidDirectCreateCardRequestTest.php index df280b2..e80b7e4 100644 --- a/tests/Message/RapidDirectCreateCardRequestTest.php +++ b/tests/Message/RapidDirectCreateCardRequestTest.php @@ -69,7 +69,38 @@ public function testGetData() $this->assertSame('4111111111111111', $data['Customer']['CardDetails']['Number']); $this->assertSame('12', $data['Customer']['CardDetails']['ExpiryMonth']); } - + + /** + * Test that expiry is optional and not mis-set if token is present. + */ + public function testGetDataNoExpiryTokenPresent() + { + $this->request->initialize(array( + 'apiKey' => 'my api key', + 'password' => 'secret', + 'transactionId' => '999', + 'description' => 'new car', + 'currency' => 'AUD', + 'invoiceReference' => 'INV-123', + 'card' => array( + 'title' => 'Mr.', + 'firstName' => 'John', + 'lastName' => 'Smith', + 'shippingFirstName' => 'Bob', + 'shippingLastName' => 'Mann', + 'shippingAddress1' => 'Level 1', + 'shippingAddress2' => '123 Test Lane', + 'shippingState' => 'NSW', + 'shippingCountry' => 'AU', + 'cardReference' => 'myRef', + ), + )); + + $data = $this->request->getData(); + $this->assertTrue(!isset($data['Customer']['CardDetails']['ExpiryMonth'])); + $this->assertTrue(!isset($data['Customer']['CardDetails']['ExpiryYear'])); + } + public function testSendSuccess() { $this->setMockHttpResponse('RapidDirectCreateCardRequestSuccess.txt');