TP3.2.3 接入银联支付
项目接入银联支付的过程, 在此记录下,希望能帮助开发盆友平坑。
银联SKD链接:https://open.unionpay.com/ajweb/product/newProDetail?proId=1&cataId=14
首先我们先下载官方提供的SDK ,下载好了解压选择版本 ,里面有PHP java .net 这里我们自然是选择PHP版本的,接入前我们先配环境,它要我们PHP
的版本 在5.3以上,并且需开启环境的curl、openssl功能。
然后就是它提供的测试证书了,默认在window系统是放在D:/certs ,意思是在你的电脑的D创建一个名为certs 的文件夹,然后将4个证书放进去,测试的名为
acp_test_enc.cer acp_test_middle.cer acp_test_root.cer acp_test_sign.pfx ,在创建一个名为logs文件夹D:/logs/ 放支付生成的日志文件,linux中请
修改成Linux中的路径。
如图:
你可以在assets文件夹中找到你要的证书,测试环境官方提供四个证书,生产环境官方提供三个,还个签名证书就是后缀为.pfx 的需要你去你的银联那里申请
下载对应的你还需要可以需要解签名证书的密码 和商户号 , 这些后面都会提到的。
在然后我们将名为SDK文件夹中的六个文件放到我们项目放到 ThinkPHP\Library\Vendor\Yunpay 的文件夹中
写逻辑代码前你还得前配置好你的acp_sdk.ini 文件 ,文件里面都写的很清楚,这里就不一一解释了,这里是我的支付代码
//银联充值操作
public function pay()
{
header ( 'Content-type:text/html;charset=utf-8' );
Vendor('Yunpay.acp_service'); //前台通知地址
$frontUrl = "http://".I("server.HTTP_HOST")."/Assets/rechargedetail";
//后台通知地址
$backUrl = "http://".I("server.HTTP_HOST");
$params = array(
//以下信息非特殊情况不需要改动
'version' => \com\unionpay\acp\sdk\SDKConfig::getSDKConfig()->version, //版本号
'encoding' => 'utf-8', //编码方式
'txnType' => '01', //交易类型
'txnSubType' => '01', //交易子类
'bizType' => '000201', //业务类型
'frontUrl' => $frontUrl, //前台通知地址
'backUrl' => $backUrl, //后台通知地址
'signMethod' => \com\unionpay\acp\sdk\SDKConfig::getSDKConfig()->signMethod, //签名方法
'channelType' => '08', //渠道类型,07-PC,08-手机
'accessType' => '0', //接入类型
'currencyCode' => '156',
// 超过超时时间调查询接口应答origRespCode不是A6或者00的就可以判断为失败。
'payTimeout' => date('YmdHis', strtotime('+15 minutes')) //订单发送时间
);
$txnAmt = I('post.txnAmt'); //交易金额
$orderId = I('post.orderId'); //商户订单号 //加入商户参数
$params['txnAmt'] = $txnAmt*100;
$params['merId'] = C('Yunpay.merId'); //商户号
$params['orderId'] = $orderId;
$params['txnTime'] = date('YmdHis'); //商品描述,可空
$body = trim(I('post.WIDbody'));
$ud = session('users.uid');
$data = array(
'uid'=>$ud, //用户id
'win_code'=>$orderId, //商户订单号
'winsubject'=>I('post.WIDsubject'), //订单名称
'wintotal_amount'=>$txnAmt, //付款金额
'winbody'=>I('post.WIDbody'), //商品描述
'state'=>'yl', //支付方式
'status'=>'0', //是否支付
'ordertime'=>time() //交易时间
);
M("pay_record")->add($data); // 保存交易信息
\com\unionpay\acp\sdk\AcpService::sign ( $params );
$uri = \com\unionpay\acp\sdk\SDKConfig::getSDKConfig()->frontTransUrl;
$html_form = \com\unionpay\acp\sdk\AcpService::createAutoFormHtml( $params, $uri );
echo $html_form; }
这里我曾试过删除里面的命名空间用 new的方式去写,这样可以简洁代码,但是后面异步的时候报错,客服说不能删除命名空间,这样会导致方法名重复,所以还是乖乖的用demo
中提供的方法。
异步方法--------
//银联充值异步
public function xxx()
{
Vendor('Yunpay.acp_service');
$logger = \com\unionpay\acp\sdk\LogUtil::getLogger();
$logger->LogInfo("receive back notify: " . \com\unionpay\acp\sdk\createLinkString ( $_POST, false, true ));
if (isset ( $_POST ['signature'] )) {
// echo \com\unionpay\acp\sdk\AcpService::validate ( $_POST ) ? '验签成功' : '验签失败';
$respCode = I('post.respCode');
$orderId = I('post.orderId'); // 商户订单号
$total_amount = I('post.settleAmt'); //订单金额
$trade_no = I('post.queryId'); // queryId 银联唯一标识一笔交易 //判断respCode=00、A6后,对涉及资金类的交易,请再发起查询接口查询,确定交易成功后更新数据库。
if( $respCode=='00' ){
$this->unionpay($orderId,$total_amount,$trade_no);
}else{
$res = $this->confirmpay($orderId,'1');
if( $res == 'Successful' ){
$this->unionpay($orderId,$total_amount,$trade_no);
} else {
echo '交易失败';
}
} } else {
echo '签名为空';
} } //银联充值
public function unionpay($orderId,$total_amount,$trade_no)
{
$per = M("pay_record")->where('win_code='.$orderId)->find(); //查找该订单
if( $per['status']=='1' ){
echo '已充值';
return;
}
$Pay = M("pay");
// 在Pay模型中启动事务
$Pay->startTrans();
// 进行相关的业务逻辑操作
$res = $Pay->where('uid='.$per['uid'])->setInc('money',$total_amount/100);
//数据组合
$data = array(
'alipay_number'=>$trade_no, //银联唯一标识
'status'=>'1', //交易状态
'paytime'=>time() //交易时间
);
M("pay_record")->where('win_code='.$orderId)->save($data); // 修改交易信息 if (!empty($res)){
// 提交事务
$Pay->commit();
}else{
// 事务回滚
$Pay->rollback();
}
} //确定是否充值操作
public function confirmpay($orderId,$L)
{
header ( 'Content-type:text/html;charset=utf-8' );
Vendor('Yunpay.acp_service');
$params = array(
//以下信息非特殊情况不需要改动
'version' => \com\unionpay\acp\sdk\SDKConfig::getSDKConfig()->version, //版本号
'encoding' => 'utf-8', //编码方式
'signMethod' => \com\unionpay\acp\sdk\SDKConfig::getSDKConfig()->signMethod, //签名方法
'txnType' => '00', //交易类型
'txnSubType' => '00', //交易子类
'bizType' => '000000', //业务类型
'accessType' => '0', //接入类型
'channelType' => '07', //渠道类型
);
if($L == '0'){
$time = M("order_pay")->where('win_code = "'.$orderId.'"')->find()['addtime'];
}else{
$time = M("pay_record")->where('win_code = "'.$orderId.'"')->find()['ordertime'];
}
$params['merId'] = C('Yunpay.merId'); //商户号
$params['orderId'] = $orderId; //交易的订单号
$params['txnTime'] = date('YmdHis',$time); //订单发送时间 \com\unionpay\acp\sdk\AcpService::sign ( $params ); // 签名
$url = \com\unionpay\acp\sdk\SDKConfig::getSDKConfig()->singleQueryUrl; $result_arr = \com\unionpay\acp\sdk\AcpService::post ( $params, $url);
if(count($result_arr)<=0) { //没收到200应答的情况
return 'No200';
}
if (!\com\unionpay\acp\sdk\AcpService::validate ($result_arr) ){
return "应答报文验签失败";
}
if ($result_arr["respCode"] == "00"){
if ($result_arr["origRespCode"] == "00"){
//交易成功
//TODO
return "Successful";
} else if ($result_arr["origRespCode"] == "03"
|| $result_arr["origRespCode"] == "04"
|| $result_arr["origRespCode"] == "05"){
//后续需发起交易状态查询交易确定交易状态
//TODO
return "交易处理中,请稍微查询";
} else {
//其他应答码做以失败处理
//TODO
return "交易失败:" . $result_arr["origRespMsg"];
}
} else if ($result_arr["respCode"] == "03"
|| $result_arr["respCode"] == "04"
|| $result_arr["respCode"] == "05" ){
//后续需发起交易状态查询交易确定交易状态
//TODO
return "处理超时,请稍微查询";
} else {
//其他应答码做以失败处理
//TODO
return "失败:" . $result_arr["respMsg"];
} }
这里第一个方法xxx 中的respCode等于00 就是支付成功 ,如果没有需要根据你生成的订单号在次查询在结果。这里客服说这种失败不好模拟,就不说了,但是这操作方法还是
建议写下,以防万一 。
最后说明下几个参数 queryId 银联唯一标识,需要保存, 还有银联支付是按 '分' 做单位的 所以支付跳转前 假如是1元,你得乘以100,它才可以识别为1元,要不然就是0.01元
然后异步到你的时候,如果你是元的单位在除于100,如果是分就不用了。
搞定收工 , 祝大家早日成为大牛