Skip to content

Commit

Permalink
Merge pull request #13 from academe/issue12
Browse files Browse the repository at this point in the history
See Issue #12 

Changes to `completeAuthorize()` and `completePurchase()` to ensure the results of the transaction the user returns with is the transaction the application was expecting. The authorization will not show as successful unless the `transactionId` matches.
  • Loading branch information
judgej authored Feb 21, 2019
2 parents 47907bf + 9bccbf5 commit 7453e74
Show file tree
Hide file tree
Showing 10 changed files with 695 additions and 361 deletions.
111 changes: 63 additions & 48 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -221,7 +221,7 @@ $request = $gateway->purchase([
'items' => $items, // array or ItemBag of Omnipay\Common\Item or Omnipay\Wirecard\Extend\Item objects
//
// These three URLs are required to the gateway, but will be defaulted to the
// returnUrl where they are not set.
// returnUrl if they remain not set.
'returnUrl' => 'https://example.com/complete',
//'cancelUrl' => 'https://example.com/complete?status=cancel', // User cancelled
//'failureUrl' => 'https://example.com/complete?status=failure', // Failed to authorise
Expand Down Expand Up @@ -283,11 +283,11 @@ You will need the original transaction reference from the `completeAuthorize`
response or the `acceptNotification` server response:

```php
$transactionReference = $complete_response->getTransactionReference();
$transactionReference = $completeResponse->getTransactionReference();

// or

$transactionReference = $server_response->getTransactionReference();
$transactionReference = $serverResponse->getTransactionReference();
```

Then send the request for the original full amount:
Expand Down Expand Up @@ -328,68 +328,83 @@ This is set up and used exactly the same as for `capture`.

### Complete Purchase/Authorize

This payment method will send the user off to the Wirecard site to authorise
a payment. The user will return with the result of the transaction, which
is parsed by the `completePurchase` object.

$complete_purchase_request = $gateway->completePurchase();

The message will be signed to check for alteration en route, so the gateway
needs to be given the `secret` when instantiating it.

Here `$complete_purchase_request` will contain all the data needed to parse the
result.

// Checks the message is correctly signed.
$complete_purchase_request->isValid();

// Checks if the authorisation was successful.
// If the fingerprint signing fails, then this will return false.
$complete_purchase_request->isSuccessful();

// Get the success or failure message.
// Some messages are generated by the gateway, and some are filled
// in by this driver.
$complete_purchase_request->getMessage();

// Checks if the authorisation was cancelled by the user.
$complete_purchase_request->isCancelled();

// Get the raw data.
$complete_purchase_request->getData();

// Get the transaction ID (generated by the merchant site).
$complete_purchase_request->getTransactonId();

// Get the transaction reference (generated by the gateway).
$complete_purchase_request->getTransactonReference();

The merchant site will normally `send()` the `$complete_purchase_request`
to get the final response object. In this case, you will just get the same
object back - it acts as both request and response.
The *Wirecard Checkout Page* payment method will send the user off to the
Wirecard site to authorise a payment.
The user will return with the result of the transaction, which
is parsed by the `completePurchase` or `completeAuthorize` object.

$completePurchaseRequest = $gateway->completePurchase([
'transactionId' => $origionalTransactionId,
]);

The `$origionalTransactionId` is the merchant site transaction ID provided in
the original `$gateway->purchase([...])` request, and is mandatory.
It is set here to ensure the application is responding to the correct
transaction result.

The message is signed by the gateway to check for tampering en route,
so the gateway needs to be given the `secret` when instantiating it.

Here $completePurchaseRequest will contain all the data needed to parse
the result.

```php
$complete_purchase_response = $complete_purchase_request->send();
// $complete_purchase_response == $complete_purchase_request // true
// The request must be used to generate a response with the final results.

$completePurchaseResponse = $completePurchaseRequest->send();

// Checks if the authorisation was successful and the message is valid.
// If the fingerprint signing fails, then this will return `false`.
// If the delivered `transactionId` differs from the expected `transactionId`
// then this will also return `false`

$completePurchaseResponse->isSuccessful();

// Get the success or failure message.
// Some messages are generated by the gateway, and some are filled
// in by this driver.

$completePurchaseResponse->getMessage();

// Checks if the authorisation was cancelled by the user.

$completePurchaseResponse->isCancelled();

// Get the raw data.

$completePurchaseResponse->getData();

// Get the transaction ID (generated by the merchant site).

$completePurchaseResponse->getTransactonId();

// Get the transaction reference (generated by the gateway).

$completePurchaseResponse->getTransactonReference();

// Just confirms if the message signature is valid.

$completePurchaseResponse->isValid();
```

### Page Recur Authorize/Purchase Request

A new authorisation or purchase can be created from an existing order.

```php
// or $gateway->recurAuthorize([...])
$request = $gateway->recurPurchase([
// Also `$gateway->recurAuthorize([...])`

$recurRequest = $gateway->recurPurchase([
'amount' => 3.10,
'currency' => 'GBP',
'description' => 'A recurring payment',
'sourceOrderNumber' => $originalTransactionReference,
]);

$response = $request->send();
$recurResponse = $recurRequest->send();

// The order reference is needed to capture the payment if just authorizing.
$new_order_number = $response->getOrderReference();
$newOrderNumber = $recurResponse->getOrderReference();
```

This is a backend operation, though takes many parameters that are otherwise
Expand Down
4 changes: 2 additions & 2 deletions src/CheckoutPageGateway.php
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,7 @@ public function recurPurchase(array $parameters = array())
public function completeAuthorize(array $parameters = array())
{
return $this->createRequest(
'\Omnipay\Wirecard\Message\Checkout\Page\Complete',
'\Omnipay\Wirecard\Message\Checkout\Page\CompleteRequest',
$parameters
);
}
Expand All @@ -97,7 +97,7 @@ public function completeAuthorize(array $parameters = array())
public function completePurchase(array $parameters = array())
{
return $this->createRequest(
'\Omnipay\Wirecard\Message\Checkout\Page\Complete',
'\Omnipay\Wirecard\Message\Checkout\Page\CompleteRequest',
$parameters
);
}
Expand Down
2 changes: 1 addition & 1 deletion src/Message/Checkout/AbstractRequest.php
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ abstract class AbstractRequest extends MessageAbstractRequest
const AUTO_DEPOSIT_NO = 'no';

// The name of the custom field the transaction ID will go into.
const CUSTOM_FIELD_NAME_TRANSACTION_ID = 'omnipay_transactionId';
const CUSTOM_FIELD_NAME_TRANSACTION_ID = 'omnipayTransactionId';

/**
* The Payment Type will default to SELECT if not set.
Expand Down
73 changes: 73 additions & 0 deletions src/Message/Checkout/Page/CompleteRequest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
<?php

namespace Omnipay\Wirecard\Message\Checkout\Page;

/**
* Complete a Wirecard Checkout Page purchase transaction on the
* user returning to the merchant shop.
* Experimentally, this one class covers both the request and the response,
* since not further requests back to the gateway are needed.
* The advantage of doing this is that all the results needed are in the
* initial request object. A merchant site can still send() that message
* and get the same message back.
* It should be possible to extend this as the notification handler too. If
* so, then the HasFingerprintTrait becomes redundant.
*/

use Omnipay\Common\Message\ResponseInterface as OmnipayResponseInterface;
use Omnipay\Wirecard\Message\Checkout\AbstractRequest;
use Omnipay\Wirecard\Message\AbstractResponse;
use Omnipay\Wirecard\Message\HasDataTrait;

class CompleteRequest extends AbstractRequest
{
use HasDataTrait;

/**
* @inheric
*/
public function getData()
{
return $this->httpRequest->request->all();
}

/**
* Create a new Response message given the raw data in the response.
*/
protected function createResponse($data)
{
$this->validate('secret', 'transactionId');

$this->response = new CompleteResponse($this, $this->getData());

// Set the original transactionId and the secret, both for
// validating the response.

//$this->response->setOriginalTransactionId($this->getTransactionId());
//$this->response->setSecret($this->getSecret());

return $this->response;
}

/**
* The secret for hashing.
*/
public function setSecret($value)
{
return $this->setParameter('secret', $value);
}

public function getSecret()
{
return $this->getParameter('secret');
}

/**
* The transaction ID supplied by the gateway, in case the
* application needs to look it up before calling send().
*/
public function getOriginalTransactionId()
{
return $this->getDataValue(AbstractRequest::CUSTOM_FIELD_NAME_TRANSACTION_ID);
}
}
90 changes: 90 additions & 0 deletions src/Message/Checkout/Page/CompleteResponse.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
<?php

namespace Omnipay\Wirecard\Message\Checkout\Page;

/**
* Complete a Wirecard Checkout Page purchase transaction on the
* user returning to the merchant shop.
* Experimentally, this one class covers both the request and the response,
* since not further requests back to the gateway are needed.
* The advantage of doing this is that all the results needed are in the
* initial request object. A merchant site can still send() that message
* and get the same message back.
* It should be possible to extend this as the notification handler too. If
* so, then the HasFingerprintTrait becomes redundant.
*/

use Omnipay\Common\Message\AbstractResponse as OmnipayAbstractResponse;
use Omnipay\Wirecard\Message\Checkout\AbstractRequest;
use Omnipay\Wirecard\Message\AbstractResponse;
use Omnipay\Wirecard\Message\HasDataTrait;
use Omnipay\Wirecard\Message\HandlesNotificationTrait;

class CompleteResponse extends OmnipayAbstractResponse
{
use HasDataTrait;
use HandlesNotificationTrait;

protected $transactionIdCheckEnabled = true;

protected $originalTransactionId;
protected $secret;

public function isRedirect()
{
return false;
}

/**
* The secret for hashing to check the signature.
* Overrides the secret in the request.
*
* @return $this
*/
public function setSecret($value)
{
$this->secret = $value;
return $this;
}

/**
* Use the override secret if set, otherwise attempt to get the
* secret from the request, if the request supports the secret.
*
* @return string|null
*/
public function getSecret()
{
if ($this->secret) {
return $this->secret;
}

if (method_exists($this->getRequest(), 'getSecret')) {
return $this->getRequest()->getSecret();
}
}

/**
* The original transactionId that we are expecting to get back.
*/
public function setOriginalTransactionId($value)
{
$this->originalTransactionId = $value;
return $this;
}

public function getOriginalTransactionId()
{
return $this->originalTransactionId
?? $this->getRequest()->getTransactionId();
}

/**
* @inherit
*/
public function isExpectedTransactionId()
{
return $this->getTransactionId()
&& $this->getTransactionId() === $this->getOriginalTransactionId();
}
}
Loading

0 comments on commit 7453e74

Please sign in to comment.