HEX
Server: nginx/1.27.1
System: Linux in-4 5.15.0-131-generic #141-Ubuntu SMP Fri Jan 10 21:18:28 UTC 2025 x86_64
User: ilikadirect (1186)
PHP: 7.4.33
Disabled: exec,passthru,shell_exec,system,proc_open,popen,parse_ini_file,show_source
Upload Files
File: /storage/v6964/school/public_html/school/application/libraries/merchant/merchant_ideal.php
<?php if ( ! defined('BASEPATH')) exit('No direct script access allowed');

/*
 * CI-Merchant Library
 *
 * Copyright (c) 2011-2012 Adrian Macneil
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */

/**
 * Merchant iDEAL Class
 *
 * Payment processing using iDEAL
 * Documentation: Extremely hard to find!
 */
class Merchant_ideal extends Merchant_driver
{
	const LANGUAGE = 'nl';
	const EXPIRATION_PERIOD = 'PT10M';

	public function default_settings()
	{
		return array(
			'acquirer_url' => '',
			'merchant_id' => '',
			'public_key_path' => '',
			'private_key_path' => '',
			'private_key_password' => '',
		);
	}

	public function issuers()
	{
		$request = $this->_new_request('DirectoryReq');

		$message = (string)$request->createDateTimeStamp;
		$message .= (string)$request->Merchant->merchantID;
		$message .= (string)$request->Merchant->subID;
		$request->Merchant->tokenCode = $this->_generate_signature($message);

		$response = $this->_post_ideal_request($request);
		return $response;
	}

	public function purchase()
	{
		$request = $this->_build_purchase();
		$response = $this->_post_ideal_request($request);
		$this->redirect((string)$response->Issuer->issuerAuthenticationURL);
	}

	public function purchase_return()
	{
		$request = $this->_build_purchase_return();
		$response = $this->_post_ideal_request($request);
		return new Merchant_ideal_response($response);
	}

	protected function _build_purchase()
	{
		$this->require_params('issuer', 'return_url');

		$request = $this->_new_request('AcquirerTrxReq');

		$request->Issuer->issuerID = $this->param('issuer');
		$request->Merchant->merchantReturnURL = $this->param('return_url');

		$request->Transaction->purchaseID = $this->param('transaction_id');
		$request->Transaction->amount = $this->amount_cents();
		$request->Transaction->currency = $this->param('currency');
		$request->Transaction->expirationPeriod = self::EXPIRATION_PERIOD;
		$request->Transaction->language = self::LANGUAGE;
		$request->Transaction->description = $this->param('description');
		$request->Transaction->entranceCode = $this->param('transaction_hash');

		// signature fields
		$message = (string)$request->createDateTimeStamp;
		$message .= (string)$request->Issuer->issuerID;
		$message .= (string)$request->Merchant->merchantID;
		$message .= (string)$request->Merchant->subID;
		$message .= (string)$request->Merchant->merchantReturnURL;
		$message .= (string)$request->Transaction->purchaseID;
		$message .= (string)$request->Transaction->amount;
		$message .= (string)$request->Transaction->currency;
		$message .= (string)$request->Transaction->language;
		$message .= (string)$request->Transaction->description;
		$message .= (string)$request->Transaction->entranceCode;
		$request->Merchant->tokenCode = $this->_generate_signature($message);

		return $request;
	}

	protected function _build_purchase_return()
	{
		$trxid = (string)$this->CI->input->get('trxid');
		if (empty($trxid))
		{
			throw new Merchant_exception(lang('merchant_invalid_response'));
		}

		$request = $this->_new_request('AcquirerStatusReq');
		$request->Transaction->transactionID = $trxid;

		// signature fields
		$message = (string)$request->createDateTimeStamp;
		$message .= (string)$request->Merchant->merchantID;
		$message .= (string)$request->Merchant->subID;
		$message .= (string)$request->Transaction->transactionID;
		$request->Merchant->tokenCode = $this->_generate_signature($message);

		return $request;
	}

	protected function _new_request($action)
	{
		$request = new SimpleXMLElement('<?xml version="1.0" encoding="UTF-8"?>'."<$action></$action>");
		$request->addAttribute('xmlns', 'http://www.idealdesk.com/Message');
		$request->addAttribute('version', '1.1.0');

		$request->createDateTimeStamp = gmdate('Y-m-d\TH:i:s.000\Z');

		$request->Merchant->merchantID = $this->setting('merchant_id');
		$request->Merchant->subID = 0;
		$request->Merchant->authentication = 'SHA1_RSA';
		$request->Merchant->token = $this->_public_key_digest();

		return $request;
	}

	protected function _public_key_digest()
	{
		$cert = '';
		if (openssl_x509_export('file://'.$this->setting('public_key_path'), $cert))
		{
			$cert = str_replace(array('-----BEGIN CERTIFICATE-----', '-----END CERTIFICATE-----'), '', $cert);
			return strtoupper(sha1(base64_decode($cert)));
		}

		throw new Merchant_exception("Missing or invalid public key!");
	}

	protected function _generate_signature($message)
	{
		$message = preg_replace('/[\s]+/', '', $message);

		if ($key = openssl_get_privatekey('file://'.$this->setting('private_key_path'), $this->setting('private_key_password')))
		{
			$signature = '';
			openssl_sign($message, $signature, $key);
			openssl_free_key($key);
			return base64_encode($signature);
		}

		throw new Merchant_exception("Missing or invalid private key!");
	}

	protected function _post_ideal_request($request)
	{
		$response = $this->post_request($this->setting('acquirer_url'), $request->asXML());
		$xml = simplexml_load_string($response);

		if (isset($xml->Error))
		{
			$errorCode = (string)$xml->Error->errorCode;
			$errorMessage = (string)$xml->Error->errorMessage;
			$errorDetail = (string)$xml->Error->errorDetail;
			throw new Merchant_exception("$errorMessage ($errorCode: $errorDetail)");
		}

		return $xml;
	}
}

class Merchant_ideal_response extends Merchant_response
{
	public function __construct($response)
	{
		$this->_reference = (string)$response->Transaction->transactionID;

		if ((string)$response->Transaction->status == 'Success')
		{
			$this->_status = self::COMPLETE;
		}
		else
		{
			$this->_status = self::FAILED;
		}
	}
}

/* End of file ./libraries/merchant/drivers/merchant_ideal.php */