首页 / 网页编程 / PHP / php银联网页支付实现方法
本文实例讲述了php银联网页支付实现方法。分享给大家供大家参考。具体分析如下:
这里介绍的银联WAP支付功能,仅限消费功能。
1. PHP代码如下:
复制代码 代码如下:<?php
namespace commonservices;
class UnionPay
{
/**
* 支付配置
* @var array
*/
public $config = [];
/**
* 支付参数,提交到银联对应接口的所有参数
* @var array
*/
public $params = [];
/**
* 自动提交表单模板
* @var string
*/
private $formTemplate = <<<"HTML"
<!DOCTYPE HTML>
<html>
<head>
<meta charset="utf-8">
<title>支付</title>
</head>
<body>
<div style="text-align:center">跳转中...</div>
<form id="pay_form" name="pay_form" action="%s" method="post">
%s
</form>
<script type="text/javascript">
document.onreadystatechange = function(){
if(document.readyState == "complete") {
document.pay_form.submit();
}
};
</script>
</body>
</html>
HTML;
/**
* 构建自动提交HTML表单
* @return string
*/
public function createPostForm()
{
$this->params["signature"] = $this->sign();
$input = "";
foreach($this->params as $key => $item) {
$input .= " <input type="hidden" name="{$key}" value="{$item}">
";
}
return sprintf($this->formTemplate, $this->config["frontUrl"], $input);
}
/**
* 验证签名
* 验签规则:
* 除signature域之外的所有项目都必须参加验签
* 根据key值按照字典排序,然后用&拼接key=value形式待验签字符串;
* 然后对待验签字符串使用sha1算法做摘要;
* 用银联公钥对摘要和签名信息做验签操作
*
* @throws Exception
* @return bool
*/
public function verifySign()
{
$publicKey = $this->getVerifyPublicKey();
$verifyArr = $this->filterBeforSign();
ksort($verifyArr);
$verifyStr = $this->arrayToString($verifyArr);
$verifySha1 = sha1($verifyStr);
$signature = base64_decode($this->params["signature"]);
$result = openssl_verify($verifySha1, $signature, $publicKey);
if($result === -1) {
throw new Exception("Verify Error:".openssl_error_string());
}
return $result === 1 ? true : false;
}
/**
* 取签名证书ID(SN)
* @return string
*/
public function getSignCertId()
{
return $this->getCertIdPfx($this->config["signCertPath"]);
}
/**
* 签名数据
* 签名规则:
* 除signature域之外的所有项目都必须参加签名
* 根据key值按照字典排序,然后用&拼接key=value形式待签名字符串;
* 然后对待签名字符串使用sha1算法做摘要;
* 用银联颁发的私钥对摘要做RSA签名操作
* 签名结果用base64编码后放在signature域
*
* @throws InvalidArgumentException
* @return multitype|string
*/
private function sign() {
$signData = $this->filterBeforSign();
ksort($signData);
$signQueryString = $this->arrayToString($signData);
if($this->params["signMethod"] == 01) {
//签名之前先用sha1处理
//echo $signQueryString;exit;
$datasha1 = sha1($signQueryString);
$signed = $this->rsaSign($datasha1);
} else {
throw new InvalidArgumentException("Nonsupport Sign Method");
}
return $signed;
}
/**
* 数组转换成字符串
* @param array $arr
* @return string
*/
private function arrayToString($arr)
{
$str = "";
foreach($arr as $key => $value) {
$str .= $key."=".$value."&";
}
return substr($str, 0, strlen($str) - 1);
}
/**
* 过滤待签名数据
* signature域不参加签名
*
* @return array
*/
private function filterBeforSign()
{
$tmp = $this->params;
unset($tmp["signature"]);
return $tmp;
}
/**
* RSA签名数据,并base64编码
* @param string $data 待签名数据
* @return mixed
*/
private function rsaSign($data)
{
$privatekey = $this->getSignPrivateKey();
$result = openssl_sign($data, $signature, $privatekey);
if($result) {
return base64_encode($signature);
}
return false;
}
/**
* 取.pfx格式证书ID(SN)
* @return string
*/
private function getCertIdPfx($path)
{
$pkcs12certdata = file_get_contents($path);
openssl_pkcs12_read($pkcs12certdata, $certs, $this->config["signCertPwd"]);
$x509data = $certs["cert"];
openssl_x509_read($x509data);
$certdata = openssl_x509_parse($x509data);
return $certdata["serialNumber"];
}
/**
* 取.cer格式证书ID(SN)
* @return string
*/
private function getCertIdCer($path)
{
$x509data = file_get_contents($path);
openssl_x509_read($x509data);
$certdata = openssl_x509_parse($x509data);
return $certdata["serialNumber"];
}
/**
* 取签名证书私钥
* @return resource
*/
private function getSignPrivateKey()
{
$pkcs12 = file_get_contents($this->config["signCertPath"]);
openssl_pkcs12_read($pkcs12, $certs, $this->config["signCertPwd"]);
return $certs["pkey"];
}
/**
* 取验证签名证书
* @throws InvalidArgumentException
* @return string
*/
private function getVerifyPublicKey()
{
//先判断配置的验签证书是否银联返回指定的证书是否一致
if($this->getCertIdCer($this->config["verifyCertPath"]) != $this->params["certId"]) {
throw new InvalidArgumentException("Verify sign cert is incorrect");
}
return file_get_contents($this->config["verifyCertPath"]);
}
}
2. 配置示例
复制代码 代码如下://银联支付设置
"unionpay" => [
//测试环境参数
"frontUrl" => "https://101.231.204.80:5000/gateway/api/frontTransReq.do", //前台交易请求地址
//"singleQueryUrl" => "https://101.231.204.80:5000/gateway/api/queryTrans.do", //单笔查询请求地址
"signCertPath" => __DIR__."/../keys/unionpay/test/sign/700000000000001_acp.pfx", //签名证书路径
"signCertPwd" => "000000", //签名证书密码
"verifyCertPath" => __DIR__."/../keys/unionpay/test/verify/verify_sign_acp.cer", //验签证书路径
"merId" => "xxxxxxx",
//正式环境参数
//"frontUrl" => "https://101.231.204.80:5000/gateway/api/frontTransReq.do", //前台交易请求地址
//"singleQueryUrl" => "https://101.231.204.80:5000/gateway/api/queryTrans.do", //单笔查询请求地址
//"signCertPath" => __DIR__."/../keys/unionpay/test/sign/PM_700000000000001_acp.pfx", //签名证书路径
//"signCertPwd" => "000000", //签名证书密码
//"verifyCertPath" => __DIR__."/../keys/unionpay/test/verify/verify_sign_acp.cer", //验签证书路径
//"merId" => "xxxxxxxxx", //商户代码
],
3. 支付示例
复制代码 代码如下:$unionPay = new UnionPay();
$unionPay->config = Yii::$app->params["unionpay"];//上面的配置
$unionPay->params = [
"version" => "5.0.0", //版本号
"encoding" => "UTF-8", //编码方式
"certId" => $unionPay->getSignCertId(), //证书ID
"signature" => "", //签名
"signMethod" => "01", //签名方式
"txnType" => "01", //交易类型
"txnSubType" => "01", //交易子类
"bizType" => "000201", //产品类型
"channelType" => "08",//渠道类型
"frontUrl" => Url::toRoute(["payment/unionpayreturn"], true), //前台通知地址
"backUrl" => Url::toRoute(["payment/unionpaynotify"], true), //后台通知地址
//"frontFailUrl" => Url::toRoute(["payment/unionpayfail"], true), //失败交易前台跳转地址
"accessType" => "0", //接入类型
"merId" => Yii::$app->params["unionpay"]["merId"], //商户代码
"orderId" => $orderNo, //商户订单号
"txnTime" => date("YmdHis"), //订单发送时间
"txnAmt" => $sum * 100, //交易金额,单位分
"currencyCode" => "156", //交易币种
];
$html = $unionPay->createPostForm();
4. 异步通知示例
复制代码 代码如下:$unionPay = new UnionPay();
$unionPay->config = Yii::$app->params["unionpay"];
$unionPay->params = Yii::$app->request->post(); //银联提交的参数
if(empty($unionPay->params)) {
return "fail!";
}
if($unionPay->verifySign() && $unionPay->params["respCode"] == "00") {
//.......
}
希望本文所述对大家的php程序设计有所帮助。