当前位置:首页 > PHP教程 > php高级应用 > 列表

php银联网页支付实现方法

发布:smiling 来源: PHP粉丝网  添加日期:2021-05-15 15:33:27 浏览: 评论:0 

这篇文章主要介绍了php银联网页支付实现方法,实例分析了php操作银联网支付接口的技巧,具有一定参考借鉴价值,需要的朋友可以参考下

本文实例讲述了php银联网页支付实现方法。分享给大家供大家参考。具体分析如下:

这里介绍的银联WAP支付功能,仅限消费功能。

1. PHP代码如下:

  1. <?php 
  2.  
  3. namespace common\services; 
  4. class UnionPay 
  5.     /** 
  6.      * 支付配置 
  7.      * @var array 
  8.      */ 
  9.     public $config = []; 
  10.     /** 
  11.      * 支付参数,提交到银联对应接口的所有参数 
  12.      * @var array 
  13.      */ 
  14.     public $params = []; 
  15.     /** 
  16.      * 自动提交表单模板 
  17.      * @var string 
  18.      */ 
  19.     private $formTemplate = <<<'HTML' 
  20. <!DOCTYPE HTML> 
  21. <html> 
  22. <head> 
  23.     <meta charset="utf-8"
  24.     <title>支付</title> 
  25. </head> 
  26. <body> 
  27.     <div style="text-align:center">跳转中...</div> 
  28.     <form id="pay_form" name="pay_form" action="%s" method="post"
  29.         %s 
  30.     </form> 
  31.     <script type="text/javascript"
  32.         document.onreadystatechange = function(){ 
  33.             if(document.readyState == "complete") { 
  34.                 document.pay_form.submit(); 
  35.             } 
  36.         }; 
  37.     </script> 
  38. </body> 
  39. </html> 
  40. HTML; 
  41. /** 
  42. * 构建自动提交HTML表单 
  43. * @return string 
  44. */ 
  45. public function createPostForm() 
  46.         $this->params['signature'] = $this->sign(); 
  47.         $input = ''
  48.         foreach($this->params as $key => $item) { 
  49.             $input .= "\t\t<input type=\"hidden\" name=\"{$key}\" value=\"{$item}\">\n"
  50.         } 
  51.         return sprintf($this->formTemplate, $this->config['frontUrl'], $input); 
  52. /** 
  53. * 验证签名 
  54. * 验签规则: 
  55. * 除signature域之外的所有项目都必须参加验签 
  56. * 根据key值按照字典排序,然后用&拼接key=value形式待验签字符串; 
  57. * 然后对待验签字符串使用sha1算法做摘要; 
  58. * 用银联公钥对摘要和签名信息做验签操作 
  59. * 
  60. * @throws \Exception 
  61. * @return bool 
  62. */ 
  63. public function verifySign() 
  64.         $publicKey = $this->getVerifyPublicKey(); 
  65.         $verifyArr = $this->filterBeforSign(); 
  66.         ksort($verifyArr); 
  67.         $verifyStr = $this->arrayToString($verifyArr); 
  68.         $verifySha1 = sha1($verifyStr); 
  69.         $signature = base64_decode($this->params['signature']); 
  70.         $result = openssl_verify($verifySha1$signature$publicKey); 
  71.         if($result === -1) { 
  72.             throw new \Exception('Verify Error:'.openssl_error_string()); 
  73.         } 
  74.         return $result === 1 ? true : false; 
  75. /** 
  76. * 取签名证书ID(SN) 
  77. * @return string 
  78. */ 
  79. public function getSignCertId() 
  80.         return $this->getCertIdPfx($this->config['signCertPath']); 
  81. }   
  82. /** 
  83. * 签名数据 
  84. * 签名规则: 
  85. * 除signature域之外的所有项目都必须参加签名 
  86. * 根据key值按照字典排序,然后用&拼接key=value形式待签名字符串; 
  87. * 然后对待签名字符串使用sha1算法做摘要; 
  88. * 用银联颁发的私钥对摘要做RSA签名操作 
  89. * 签名结果用base64编码后放在signature域 
  90. * 
  91. * @throws \InvalidArgumentException 
  92. * @return multitype|string 
  93. */ 
  94. private function sign() { 
  95.         $signData = $this->filterBeforSign(); 
  96.         ksort($signData); 
  97.         $signQueryString = $this->arrayToString($signData); 
  98.         if($this->params['signMethod'] == 01) { 
  99.             //签名之前先用sha1处理 
  100.             //echo $signQueryString;exit; 
  101.             $datasha1 = sha1($signQueryString); 
  102.             $signed = $this->rsaSign($datasha1); 
  103.         } else { 
  104.             throw new \InvalidArgumentException('Nonsupport Sign Method'); 
  105.         } 
  106.         return $signed
  107. /** 
  108. * 数组转换成字符串 
  109. * @param array $arr 
  110. * @return string 
  111. */ 
  112. private function arrayToString($arr
  113.         $str = ''
  114.         foreach($arr as $key => $value) { 
  115.             $str .= $key.'='.$value.'&'
  116.         } 
  117.         return substr($str, 0, strlen($str) - 1); 
  118. /** 
  119. * 过滤待签名数据 
  120. * signature域不参加签名 
  121. * 
  122. * @return array 
  123. */ 
  124. private function filterBeforSign() 
  125.         $tmp = $this->params; 
  126.         unset($tmp['signature']); 
  127.         return $tmp
  128. /** 
  129. * RSA签名数据,并base64编码 
  130. * @param string $data 待签名数据 
  131. * @return mixed 
  132. */ 
  133. private function rsaSign($data
  134.         $privatekey = $this->getSignPrivateKey(); 
  135.         $result = openssl_sign($data$signature$privatekey); 
  136.         if($result) { 
  137.             return base64_encode($signature); 
  138.         } 
  139.         return false; 
  140. /** 
  141. * 取.pfx格式证书ID(SN) 
  142. * @return string 
  143. */ 
  144. private function getCertIdPfx($path
  145.         $pkcs12certdata = file_get_contents($path); 
  146.         openssl_pkcs12_read($pkcs12certdata$certs$this->config['signCertPwd']); 
  147.         $x509data = $certs['cert']; 
  148.         openssl_x509_read($x509data); 
  149.         $certdata = openssl_x509_parse($x509data); 
  150.         return $certdata['serialNumber']; 
  151. /** 
  152. * 取.cer格式证书ID(SN) 
  153. * @return string 
  154. */ 
  155. private function getCertIdCer($path
  156.         $x509data = file_get_contents($path); 
  157.         openssl_x509_read($x509data); 
  158.         $certdata = openssl_x509_parse($x509data); 
  159.         return $certdata['serialNumber']; 
  160. /** 
  161. * 取签名证书私钥 
  162. * @return resource 
  163. */ 
  164. private function getSignPrivateKey() 
  165.         $pkcs12 = file_get_contents($this->config['signCertPath']); 
  166.         openssl_pkcs12_read($pkcs12$certs$this->config['signCertPwd']); 
  167.         return $certs['pkey']; 
  168. /** 
  169. * 取验证签名证书 
  170. * @throws \InvalidArgumentException 
  171. * @return string 
  172. */ 
  173. private function getVerifyPublicKey() 
  174.         //先判断配置的验签证书是否银联返回指定的证书是否一致 
  175.         if($this->getCertIdCer($this->config['verifyCertPath']) != $this->params['certId']) { 
  176.             throw new \InvalidArgumentException('Verify sign cert is incorrect'); 
  177.         } 
  178.         return file_get_contents($this->config['verifyCertPath']);       
  179.     } 

2. 配置示例,代码如下:

  1. //银联支付设置 
  2.  'unionpay' => [ 
  3.      //测试环境参数 
  4.      'frontUrl' => 'https://101.231.204.80:5000/gateway/api/frontTransReq.do', //前台交易请求地址 
  5.      //'singleQueryUrl' => 'https://101.231.204.80:5000/gateway/api/queryTrans.do', //单笔查询请求地址 
  6.      'signCertPath' => __DIR__.'/../keys/unionpay/test/sign/700000000000001_acp.pfx'//签名证书路径 
  7.      'signCertPwd' => '000000'//签名证书密码 
  8.      'verifyCertPath' => __DIR__.'/../keys/unionpay/test/verify/verify_sign_acp.cer'//验签证书路径 
  9.      'merId' => 'xxxxxxx'
  10.      //正式环境参数 
  11.      //'frontUrl' => 'https://101.231.204.80:5000/gateway/api/frontTransReq.do', //前台交易请求地址 
  12.      //'singleQueryUrl' => 'https://101.231.204.80:5000/gateway/api/queryTrans.do', //单笔查询请求地址 
  13.      //'signCertPath' => __DIR__.'/../keys/unionpay/test/sign/PM_700000000000001_acp.pfx', //签名证书路径 
  14.      //'signCertPwd' => '000000', //签名证书密码 
  15.      //'verifyCertPath' => __DIR__.'/../keys/unionpay/test/verify/verify_sign_acp.cer', //验签证书路径 
  16.      //'merId' => 'xxxxxxxxx', //商户代码 
  17.  ], 

3. 支付示例:

  1. $unionPay = new UnionPay(); 
  2. $unionPay->config = Yii::$app->params['unionpay'];//上面的配置 
  3. $unionPay->params = [ 
  4.     'version' => '5.0.0'//版本号 
  5.     'encoding' => 'UTF-8'//编码方式 
  6.     'certId' => $unionPay->getSignCertId(), //证书ID 
  7.     'signature' => ''//签名 
  8.     'signMethod' => '01'//签名方式 
  9.     'txnType' => '01'//交易类型 
  10.     'txnSubType' => '01'//交易子类 
  11.     'bizType' => '000201'//产品类型 
  12.     'channelType' => '08',//渠道类型 
  13.     'frontUrl' => Url::toRoute(['payment/unionpayreturn'], true), //前台通知地址 
  14.     'backUrl' => Url::toRoute(['payment/unionpaynotify'], true), //后台通知地址 
  15.     //'frontFailUrl' => Url::toRoute(['payment/unionpayfail'], true), //失败交易前台跳转地址 
  16.     'accessType' => '0'//接入类型 
  17.     'merId' => Yii::$app->params['unionpay']['merId'], //商户代码 
  18.     'orderId' => $orderNo//商户订单号 
  19.     'txnTime' => date('YmdHis'), //订单发送时间 
  20.     'txnAmt' => $sum * 100, //交易金额,单位分 
  21.     'currencyCode' => '156'//交易币种 
  22. ]; 
  23. $html = $unionPay->createPostForm(); 

4. 异步通知示例:

  1. $unionPay = new UnionPay(); 
  2. $unionPay->config = Yii::$app->params['unionpay']; 
  3. $unionPay->params = Yii::$app->request->post(); //银联提交的参数 
  4. if(emptyempty($unionPay->params)) { 
  5.     return 'fail!'
  6. if($unionPay->verifySign() && $unionPay->params['respCode'] == '00') { 
  7.     //....... 
  8. }

Tags: php银联网页支付

分享到: