php 对接国外支付 ipay88支付
ipay88支付
近期接了一个国外的项目,客户指定要这种支付,就搞搞呗,其实流程和思路都是差不多的,往下看
他的流程其实非常简单 下面的流程仔细看看,看懂了就会了
1 首先我们需要先获取下单所需要的参数(这个需要去ipay88官网去申请),
就是下面main.php里面的一些参数merchantCode,merchantKey
2 参数完整之后我们要做的就是验证签名了(下面有我的验证签名的方式,支付类文件,拿着用就好)
3 当我们在页面实际付款的时候,我们需要使用一个form表单 post方式执行我们的操作,
我们需要将签名所需要的参数都以form表单的形式提交到ipay88给我们提供的地址,会自动跳转到支付页(成功唤醒就代表对接成功了)
4 注意form表单提交的时候,参数一定要完整
5 测试的时候,ipay88只支持信用卡支付(visa,master),并且支付金额只能为1元
yii-iPay88 API将帮助您为应用程序实现iPay88付款网关。此API中包含以下操作
- 正常付款
- 定期付款
- 定期付款终止
- 后端通知
认证方式
yii-iPay88 API使用十六进制和base64编码技术来创建唯一签名。然后,此签名将用于客户端的每个请求以及iPay88服务器发送的每个响应。
基本上,此签名基于商户代码,商户密钥,付款金额,货币代码,refno。等。所有API请求都必须通过HTTPS进行。
用法
main.php
'components' => array(
....
'ipay' => array(
'class'=>'Ipay',
'merchantCode'=>'<<merchantCode>>',
'merchantKey'=>'<<merchantKey>>',
'currencyCode'=>'MYR', // length 5
'responseUrl'=>'http://<<hostname>>/ipay/response',
'backendUrl'=>'http://<<hostname>>/ipaybackend/response',
'requeryUrl'=>'https://www.mobile88.com/epayment/enquiry.asp',
'paymentUrl'=>'https://www.mobile88.com/epayment/entry.asp',
'recurringUrlSubscription'=>'https://www.ipay88.com/recurringpayment/webservice/RecurringPayment.asmx/Subscription',
'recurringUrlTermination'=>'https://www.ipay88.com/recurringpayment/webservice/RecurringPayment.asmx/Termination' ),
....
)
一个虚拟控制器,演示此组件用于iPay88正常付款的用法并获得该响应,这个也就是你传给支付类文件的参数方法
IpayController.php
class IpayController extends Controller { const TRANSACTION_TYPE_PAYMENT = 'payment';
const TRANSACTION_TYPE_RECURRING_SUBSCRIPTION = 'recurring_subscription';
const TRANSACTION_TYPE_RECURRING_TERMINATION = 'recurring_termination'; /*
* iPay88 normal payment Method
*/
public function actionPayment() { // Unique merchant transaction number / Order ID (Retry for same RefNo only valid for 30 mins). (length 20)
$paymentParams['RefNo'] = 'TEST123'; // (Optional) (int)
$paymentParams['PaymentId'] = ''; // Payment amount with two decimals.
$paymentParams['Amount'] = '1.00'; // Product description. (length 100)
$paymentParams['ProdDesc'] = 'This is a test product'; // Customer name. (length 100)
$paymentParams['UserName'] = 'Abc'; // Customer email. (length 100)
$paymentParams['UserEmail'] = 'abc@xyz.com'; // Customer contact. (length 20)
$paymentParams['UserContact'] = '*************'; // (Optional) Merchant remarks. (length 100)
$paymentParams['Remark'] = 'Here is the description'; $paymentFields = Yii::app()->ipay->getPaymentFields($paymentParams, self::TRANSACTION_TYPE_PAYMENT);
$transactionUrl = Yii::app()->ipay->getTransactionUrl(self::TRANSACTION_TYPE_PAYMENT);
$this->render('Payment', array(
'paymentFields' => $paymentFields,
'transactionUrl' => $transactionUrl
));
} }
/Ipay.php 这个就是你的支付类文件了,用的时候改一下里面的参数就可以,因为这个是基于yii框架的,你可以改一下里面的有些参数,或者删除就好
class Ipay extends CApplicationComponent { /**
* Normal iPay88 payment method
*/
const TRANSACTION_TYPE_PAYMENT = 'payment'; /**
* Normal iPay88 recurring payment subscription
*/
const TRANSACTION_TYPE_RECURRING_SUBSCRIPTION = 'recurring_subscription'; /**
* Normal iPay88 recurring payment termination
*/
const TRANSACTION_TYPE_RECURRING_TERMINATION = 'recurring_termination'; /**
* Merchant code assigned by iPay88
*/
public $merchantCode; /**
* Merchant Key assigned by iPay88
*/
public $merchantKey; /**
* Currency Code max length 5
*/
public $currencyCode; /**
* Merchant code assigned by iPay88
*/
public $responseUrl; /*
* Response Url or Return Url after payment
*/
public $paymentUrl; /*
* Backend Url or Notify Url after payment (Send response by iPay88 server)
*/
public $backendUrl; /*
* Requery from iPay88 server regarding bill details
*/
public $requeryUrl; /*
* ipay88 Recurring Payment Url
*/
public $recurringUrlSubscription; /*
* ipay88 Recurring Payment Termination Url
*/
public $recurringUrlTermination; /*
* Details to be sent to IPay88 for payment request.
*/
private $paymentRequest = array(
'MerchantCode', // Merchant code assigned by iPay88. (length 20)
'PaymentId', // (Optional) (int)
'RefNo', // Unique merchant transaction number / Order ID (Retry for same RefNo only valid for 30 mins). (length 20)
'Amount', // Payment amount with two decimals.
'Currency', // (length 5)
'ProdDesc', // Product description. (length 100)
'UserName', // Customer name. (length 100)
'UserEmail', // Customer email. (length 100)
'UserContact', // Customer contact. (length 20)
'Remark', // (Optional) Merchant remarks. (length 100)
'Lang', // (Optional) Encoding type:- ISO-8859-1 (English), UTF-8 (Unicode), GB2312 (Chinese Simplified), GD18030 (Chinese Simplified), BIG5 (Chinese Traditional)
'Signature',
'ResponseURL',
'BackendURL',
); /*
* Details to be sent to iPay88 for recurring subscription payment request.
*/
private $recurringSubscriptionRequest = array(
'MerchantCode', // Merchant code assigned by iPay88. (length 20)
'RefNo', // Unique merchant transaction number / Order ID. (length 20)
'FirstPaymentDate', // (ddmmyyyy)
'Currency', // MYR only. (length 5)
'Amount', // Payment amount with two decimals.
'NumberOfPayments', // (int)
'Frequency', // Frequency type; 1 - Monthly, 2 - Quarterly, 3 - Half-Yearly, 4 - Yearly. (int)
'Desc', // Product description. (length 100)
'CC_Name', // Name printed on credit card. (length 100)
'CC_PAN', // 16-digit credit card number (Visa/Mastercard). (length 16)
'CC_CVC', // 3-digit verification code behind credit card. (length 3)
'CC_ExpiryDate', // Credit card expiry date. (mmyyyy)
'CC_Country', // Credit card issuing country. (length 100)
'CC_Bank', // Credit card issuing bank. (length 100)
'CC_Ic', // Credit card holder IC / Passport number. (length 50)
'CC_Email', // Credit card holder email address. (length 255)
'CC_Phone', // Credit card phone number. (length 100)
'CC_Remark', // (Optional) Remarks. (varchar 100)
'P_Name', // Subscriber name as printed in IC / Passport. (length 100)
'P_Email', // Subscriber email address. (length 255)
'P_Phone', // Subscriber phone number. (length 100)
'P_Addrl1', // Subscriber address line 1. (length 100)
'P_Addrl2', // (Optional) Subscriber address line 2. (length 100)
'P_City', // Subscriber city. (length 100)
'P_State', // Subscriber state. (length 100)
'P_Zip', // Subscriber zip code. (length 100)
'P_Country', // Subscriber country. (varchar 100)
'BackendURL', // Payment backend response page. (length 255)
'Signature', // SHA1 signature. (length 100)
); /*
* Get required payment fields
*/
public function getPaymentFields($reqParams = null, $paymentType) {
$retnParams = array();
try {
if (isset($reqParams) && (count($reqParams) > )) { if (isset($paymentType) && $paymentType != "") {
$paymentType = strtolower(trim($paymentType));
switch ($paymentType) {
case 'payment':
$retnParams = $this->__getPaymentField($reqParams, $paymentType);
break;
case 'recurring_subscription':
$retnParams = $this->__getRecurringSubscriptionField($reqParams, $paymentType);
break;
case 'recurring_termination':
$retnParams = $this->__getRecurringTerminationField($reqParams, $paymentType);
break;
}
} else {
throw new Exception("Ipay: Payment method missing");
}
} else {
throw new Exception("Ipay: Required Parameters missing");
}
} catch (Exception $e) {
Yii::log($e->getMessage(), CLogger::LEVEL_ERROR);
}
return $retnParams;
} /*
* Code for hex2bin
*/
public function _hex2bin($hexSource) {
$bin = '';
for ($i = ; $i < strlen($hexSource); $i = $i + ) {
$bin .= chr(hexdec(substr($hexSource, $i, )));
}
return $bin;
} /*
* Get payment fields for normal payment fields
*/
public function __getPaymentField($reqParams, $paymentType) {
$retnParams = array();
foreach ($this->paymentRequest as $pymtKey) {
if (isset($reqParams[$pymtKey])) {
$retnParams[$pymtKey] = $reqParams[$pymtKey];
} else { switch ($pymtKey) {
case 'MerchantCode':
$retnParams[$pymtKey] = $this->merchantCode;
break;
case 'Currency':
$retnParams[$pymtKey] = $this->currencyCode;
break;
case 'Lang':
$retnParams[$pymtKey] = 'UTF-8'; //(Optional) Encoding type:- ISO-8859-1 (English), UTF-8 (Unicode), GB2312 (Chinese Simplified), GD18030 (Chinese Simplified), BIG5 (Chinese Traditional)
break;
case 'Signature':
$retnParams[$pymtKey] = $this->__createSignature($retnParams, $paymentType); // SHA1 signature.
break;
case 'ResponseURL':
$retnParams[$pymtKey] = $this->responseUrl; // (Optional) Payment response page.
break;
case 'BackendURL':
$retnParams[$pymtKey] = $this->backendUrl; // (Optional) BackendURL but should security purpose
break;
}
}
} return $retnParams;
} /*
* Get payment fields for recurring payment
*/
public function __getRecurringSubscriptionField($reqParams, $paymentType) {
$retnParams = array();
foreach ($this->recurringSubscriptionRequest as $pymtKey) {
if (isset($reqParams[$pymtKey])) {
$retnParams[$pymtKey] = $reqParams[$pymtKey];
} else { switch ($pymtKey) {
case 'MerchantCode':
$retnParams[$pymtKey] = $this->merchantCode;
break;
case 'Currency':
$retnParams[$pymtKey] = $this->currencyCode;
break;
case 'Lang':
$retnParams[$pymtKey] = 'UTF-8'; //(Optional) Encoding type:- ISO-8859-1 (English), UTF-8 (Unicode), GB2312 (Chinese Simplified), GD18030 (Chinese Simplified), BIG5 (Chinese Traditional)
break;
case 'Signature':
$retnParams[$pymtKey] = $this->__createSignature($retnParams, $paymentType); // SHA1 signature.
break;
case 'ResponseURL':
$retnParams[$pymtKey] = $this->responseUrl; // (Optional) Payment response page.
break;
case 'BackendURL':
$retnParams[$pymtKey] = $this->backendUrl; // (Optional) BackendURL but should security purpose
break;
}
}
} return $retnParams;
} /*
* Get payment fields for recurring payment termination
*/
public function __getRecurringTerminationField($reqParams, $paymentType) {
$retnParams = array();
foreach ($this->recurringSubscriptionRequest as $pymtKey) {
if (isset($reqParams[$pymtKey])) {
$retnParams[$pymtKey] = $reqParams[$pymtKey];
} else { switch ($pymtKey) {
case 'MerchantCode':
$retnParams[$pymtKey] = $this->merchantCode;
break;
}
}
} return $retnParams;
} /*
* Create signature for payment
*/
public function __createSignature($signatureParams, $paymentType) {
//echo "<pre>";
//print_r($signatureParams);
$signature = '';
if (isset($signatureParams)) {
$_signatureParams = array();
if ($paymentType == self::TRANSACTION_TYPE_PAYMENT) {
$_signatureParams = array('MerchantCode', 'RefNo', 'Amount', 'Currency');
} else if ($paymentType == self::TRANSACTION_TYPE_RECURRING_SUBSCRIPTION) {
$_signatureParams = array('MerchantCode', 'RefNo', 'FirstPaymentDate', 'Currency', 'Amount', 'NumberOfPayments', 'Frequency', 'CC_PAN');
} else if ($paymentType == self::TRANSACTION_TYPE_RECURRING_TERMINATION) {
$_signatureParams = array('MerchantCode', 'RefNo');
} foreach ($_signatureParams as $val) {
if (!isset($signatureParams[$val])) {
throw new Exception("Ipay: Missing required parameters for signature.");
return false;
}
}
} // Make sure the order is correct.
if ($paymentType == self::TRANSACTION_TYPE_PAYMENT) {
$signature .= $this->merchantKey;
$signature .= $signatureParams['MerchantCode'];
$signature .= $signatureParams['PaymentId'];
$signature .= $signatureParams['RefNo'];
$signature .= preg_replace("/[^\d]+/", "", $signatureParams['Amount']);
$signature .= $signatureParams['Currency'];
} else if ($paymentType == self::TRANSACTION_TYPE_RECURRING_SUBSCRIPTION) {
$signature .= $signatureParams['MerchantCode'];
$signature .= $this->merchantKey;
$signature .= $signatureParams['RefNo'];
$signature .= $signatureParams['FirstPaymentDate'];
$signature .= $signatureParams['Currency'];
$signature .= $signatureParams['Amount'];
$signature .= $signatureParams['NumberOfPayments'];
$signature .= $signatureParams['Frequency'];
$signature .= $signatureParams['CC_PAN'];
} else if ($paymentType == self::TRANSACTION_TYPE_RECURRING_TERMINATION) {
$signature .= $signatureParams['MerchantCode'];
$signature .= $this->merchantKey;
$signature .= $signatureParams['RefNo'];
} // Hash the signature.
return $signature = base64_encode($this->_hex2bin(sha1($signature)));
} /*
* Get url for respective payment redirection url
*/
public function getTransactionUrl($paymentType) {
if ($paymentType == self::TRANSACTION_TYPE_PAYMENT) {
return $this->paymentUrl;
} else if ($paymentType == self::TRANSACTION_TYPE_RECURRING_SUBSCRIPTION) {
return $this->recurringUrlSubscription;
} else if ($paymentType == self::TRANSACTION_TYPE_RECURRING_TERMINATION) {
return $this->recurringUrlTermination;
}
} /*
* iPay88 payment signature validation
*/
public function checkiPay88Signature($reqParams) {
$status = 'fail';
try {
if (isset($reqParams) && count($reqParams) > ) {
$orginalKey = $this->merchantKey . $this->merchantCode;
if (isset($reqParams['RefNo'])) {
$orginalKey .=$reqParams['RefNo'];
} if (isset($reqParams['Amount'])) {
$orginalKey .=preg_replace("/[^\d]+/", "", $reqParams['Amount']);
}
$orginalKey .= $this->currencyCode;
if (isset($reqParams['Status'])) {
$orginalKey .=$reqParams['Status'];
} $orginalKeyGen = base64_encode($this->_hex2bin(sha1($orginalKey)));
$returnKey = $this->merchantKey;
if (isset($reqParams['MerchantCode'])) {
$returnKey .=$reqParams['MerchantCode'];
} if (isset($reqParams['RefNo'])) {
$returnKey .=$reqParams['RefNo'];
}
if (isset($reqParams['Amount'])) {
$returnKey .=preg_replace("/[^\d]+/", "", $reqParams['Amount']);
}
if (isset($reqParams['Currency'])) {
$returnKey .=$reqParams['Currency'];
}
if (isset($reqParams['Status'])) {
$returnKey .=$reqParams['Status'];
} $returnKeyGen = base64_encode($this->_hex2bin(sha1($returnKey)));
if ($orginalKeyGen === $returnKeyGen) {
$status = 'success';
}
} else {
throw new Exception("Ipay::checkiPay88Signature: Params missing");
}
} catch (exception $e) {
Yii::log($e->getMessage(), CLogger::LEVEL_ERROR);
} return $status;
} /*
* Curl hit to get bill deyails
*/
public function requeryPayment($rawPostData) {
try {
$result = '';
if (is_callable('curl_init')) {
if (isset($rawPostData) && $rawPostData != "") {
$ch = curl_init();
$url = $this->requeryUrl . '?' . $rawPostData;
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
$result = curl_exec($ch);
curl_close($ch);
} else {
throw new Exception("Ipay::requeryPayment: No request string");
}
} else {
throw new Exception("Ipay::requeryPayment: Curl not enabled");
}
} catch (exception $e) {
Yii::log($e->getMessage(), CLogger::LEVEL_ERROR);
} return $result;
} }
views / ipay / payment.php 这个相当于你的回调地址了,下单成功后跳转到这里执行你想要的操作
<?php $rawPostData = file_get_contents('php://input');
$resultData = array();
if (strlen($rawPostData) > ) {
$rawPostArray = explode('&', $rawPostData);
foreach ($rawPostArray as $keyval) {
$keyval = explode('=', $keyval);
if (count($keyval) == )
$resultData[$keyval[]] = urldecode($keyval[]);
}
}
这个就是模拟的form表单了,参照这个提交就没问题了 <FORM method="post" name="ePayment"
action="https://payment.ipay88.com.my/ePayment/entry.asp">
<INPUT type="hidden" name="MerchantCode" value="M00003">
<INPUT type="hidden" name="PaymentId" value="2">
<INPUT type="hidden" name="RefNo" value="A00000001">
<INPUT type="hidden" name="Amount" value="1.00">
<INPUT type="hidden" name="Currency" value="MYR">
<INPUT type="hidden" name="ProdDesc" value="Photo Print">
<INPUT type="hidden" name="UserName" value="John Tan">
<INPUT type="hidden" name="UserEmail" value="john@hotmail.com">
<INPUT type="hidden" name="UserContact" value="0126500100">
<INPUT type="hidden" name="Remark" value="gfdfgd">
<INPUT type="hidden" name="Lang" value="UTF-8">
<INPUT type="hidden" name="SignatureType" value="SHA256">
<INPUT type="hidden" name="Signature"
value="b81af9c4048b0f6c447129f0f5c0eec8d67cbe19eec26f2cdaba5df4f4dc5a28">
<INPUT type="hidden" name="ResponseURL"
value="http://gx.oeob.net/mobile/respons_ipay.php">
<INPUT type="hidden" name="BackendURL"
value="http://gx.oeob.net/mobile/respons_ipay.php">
<INPUT type="submit" value="Proceed with Payment" name="Submit">
</FORM>
如果有不明白的地方,评论给我,或者发我邮箱2653293344@qq.com
php 对接国外支付 ipay88支付的更多相关文章
- php对接微信小程序支付
前言:这里我就假装你已经注册了微信小程序,并且基本的配置都已经好了.注: 个人注册小程序不支持微信支付,所以我还是假装你是企业或者个体工商户的微信小程序,其他的商户号注册,二者绑定,授权,支付开通,就 ...
- 支付宝支付-APP支付服务端详解
支付宝APP支付服务端详解 前面接了微信支付,相比微信支付,支付宝APP支付提供了支付分装类,下面将实现支付宝APP支付.订单查询.支付结果异步通知.APP支付申请参数说明,以及服务端返回APP端发起 ...
- php大力力 [051节] 支付宝支付.申请支付资质,等待审核中
https://beecloud.cn/doc/payapply/?index=6 支付宝支付申请支付资质 一.注册支付宝用户 在支付宝官网注册成为用户 二.签约对应支付产品 应用集成支付宝支付,需要 ...
- 支付宝支付-常用支付API详解(查询、退款、提现等)
所有的接口支持沙盒环境的测试 1.前言 前面几篇文件详细介绍了 支付宝提现.扫码支付.条码支付.Wap支付.App支付 支付宝支付-提现到个人支付宝 支付宝支付-扫码支付 支付宝支付-刷卡支付(条码支 ...
- Android支付——支付宝支付总结
摘要:分享牛系列.分享牛转载.第三方支付,java第三方支付.android第三方支付. 原文地址:http://blog.csdn.net/zwl5670/article/details/51219 ...
- Python实现微信刷卡支付(条码支付)MicroPay
转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/7686765.html 一:资料阅读 场景介绍:https://pay.weixin.qq.com/wiki/d ...
- 微信JSAPI 公众号支付 H5支付以及APP支付 WEBAPI接口开发测试
统一下单入口 调用该方法入口: public void WxPayAPI() { //string PayPrice ="99.9"; ////订单号 //string Payor ...
- 支付宝支付-常用支付API详解(查询、退款、提现等)-转
所有的接口支持沙盒环境的测试 1.前言 前面几篇文件详细介绍了 支付宝提现.扫码支付.条码支付.Wap支付.App支付 支付宝支付-提现到个人支付宝 支付宝支付-扫码支付 支付宝支付-刷卡支付(条码支 ...
- Android 微信支付&支付宝支付
由于项目需求,加入这2个功能记录一些需要注意的地方 一.微信支付 微信支付在2016年4月份左右稍微调整了一下支付过程,但是文档却没怎么更新,这也是百度上为什么那么多开发者都说微信是个大坑. 身为一个 ...
随机推荐
- Centos7 安装Python3.7
如果电脑自带的python2.7 先卸载 1.强制删除已安装python及其关联 rpm -qa|grep python|xargs rpm -ev --allmatches --nodeps 2.删 ...
- C++中static关键字的用法
运行一个完整的程序.我们可将整个存储区分为四块: (1)栈区:就比如局部变量,对应的函数参数等这些,调用完之后相应的内存会自己释放掉,很让人省心. (2)堆区:堆来堆去的.得要人动手.所以得我们自己手 ...
- #《Essential C++》读书笔记# 第五章 面向对象编程风格
基础知识 继承机制定义了父子(parent/child)关系.父类(parent)定义了所有子类(children)共通的共有接口(public interface)和私有实现(private imp ...
- 如何在Idea中使用Git将项目代码上传到码云
参考链接:https://blog.csdn.net/zzybbh/article/details/88172140
- PHP0012:PHP操作文件目录
WIN下文件夹的只读权限是0555
- css3基本选择器+属性选择器+动态伪类+UI状态伪类+结构类
后代选择器 祖先元素 后代元素{ } 子元素选择器(直接子元素选择器) 父元素>子元素{ } 兄弟选择器 元素+兄弟元素(紧邻该元素之后的下一个兄弟元素) 所有兄弟元素选择器 元素~兄弟元素(该 ...
- docker - apt-get更换国内源解决Dockerfile构建速度过慢
背景 使用ubuntu镜像一般apt-get源地址都是在国外导致在构建时因为源地址问题导致下载速度极其得慢 在构建中应事先修改apt-get源地址来避免因下载速度过慢导致的构建缓慢问题 方案 在Doc ...
- python全栈学习 day02
pycharm 安装设置: 按照百度百科或者官网介绍下载,安装. 激活步骤 1:改host 2.输入激活信息,注意有效期. python 逻辑运算符://返回的均为bool值 与 and A and ...
- sklearn.metrics中的评估方法
https://www.cnblogs.com/mindy-snail/p/12445973.html 1.confusion_matrix 利用混淆矩阵进行评估 混淆矩阵说白了就是一张表格- 所有正 ...
- gulp常用插件之gulp-babel使用
更多gulp常用插件使用请访问:gulp常用插件汇总 gulp-babel这是Babel的Gulp插件. 此自述文件适用于gulp-babel v8 + Babel v7检查7.x分支以了解使用Bab ...