使用php实现快钱支付功能(涉及到接口)


Posted in PHP onJuly 01, 2013

本项目用zend framework框架实现的
modules/default/controllers/IndexController.php
IndexController.php

<?php
class IndexController extends Zend_Controller_Action
{
    public function init()
    {
        /* Initialize action controller here */
    }
    public function indexAction()
    {      /*模拟订单 
      *$MockOrder是从数据库取出来的信息,它包含一些块钱Request的信息。这里我写死了。
      *orderId订单号,数据库表的主键(唯一)。    //必要字段
      *usr_idtype证件类型,根据自己需要。
      *usr_idcode证件号,根据自己需要。
      *etx_status是否优惠,根据自己需要。
      *time_create验证是否符合优惠的时间,根据自己需要。
      *ets_license套餐代码如同商品类别,根据自己需要。
      *contact_type联系方式类型,固定选择值1,2。1电子邮件,2手机号,根据自己需要,块钱那边可以为空。
      *contact_text联系方式,根据contact_type来填写,根据自己需要,块钱那边可以为空。
      *etsPrice套餐价格及商品价格,根据自己需要。
      *orderPrice实际价格,根据自己需要。
      *orderAmount订单实际支付金额,这个要加手续费的。  //必要字段
      *orderTime订单时间。        //必要字段
      *paySuccess订单是否支付成功。      //必要字段
      *buySuccess账号是否生成功,根据自己需要
      *payTime订单支付成功时间。       //必要字段
      *总之凡是跟订单有关的都是必要字段
      *orderId、orderAmount、orderTime为 Request所需字段
      *paySuccess、payTime为Response所需字段
      */
  $MockOrder = array();
  $MockOrder['orderId'] = '100000125';//订单号。--必要
  $MockOrder['usr_idtype'] = '1';//证件类型,身份证
  $MockOrder['usr_idcode'] = '371111199011111111';//身份证号
  $MockOrder['etx_status'] = '0';//是否优惠,否
  $MockOrder['time_create'] = '1352338189';//验证是否优惠时间
  $MockOrder['ets_license'] = '1';//套餐代码及商品类别
  $MockOrder['contact_type'] = '1';//联系方式类型 1,邮箱
  $MockOrder['contact_text'] = 'x@163.com';//联系方式,邮箱
  $MockOrder['etsPrice'] = '30800';//套餐价格及商品价格
  $MockOrder['orderPrice'] = '30800';//实际价格
  $MockOrder['orderAmount'] = '31100';//订单实际支付价格,加手续费的。--必要
  $MockOrder['orderTime'] = '1352338199';//订单生成时间。--必要
  $MockOrder['paySuccess'] = '0';//订单是否支付成功。--必要
  $MockOrder['buySuccess'] = '0';//账号是否生成成功
  $MockOrder['payTime'] = '0';//订单支付时间。--必要
  //BillRequest就是快钱那边需要的的一些参数
  $this->view->BillRequest = new Application_Model_BillRequest($MockOrder);
  Zend_Debug::dump($this->view->BillRequest);exit;  
    }
    //bgUrl地址指向这里
    public function receiveAction()
    {
     //receive数据库设计
     /*用$MockReceive数组模拟
      * $MockReceive = array();
      * $MockReceive['id']主键;
      * $MockReceive['orderId']商户订单号;
      * $MockReceive['receiveTime']接受时间;
      * $MockReceive['queryString']http_build_encode($_REQUEST);
      * $MockReceive['dealId']快钱交易号;
      * $MockReceive['bankDealId']银行交易号;
      * $MockReceive['payResult']处理结果10:支付成功;11:支付失败;
      * $MockReceive['dealTime']快钱交易时间;
      * $MockReceive['payAmount']订单实际支付金额;
      * $MockReceive['fee']费用;
      * $MockReceive['errCode']错误代码;
      */
     
     /*$_REQUEST是快钱那边返回来的数据
      * merchantAcctId人民币账号,与提交订单时的块钱账号保持一致。
      * version网关版本,固定值:v2.0,与提交订单时的网关版本号保持一致。
      * language网页显示语言种类,1中文显示,与提交订单时的网页显示语言种类保持一致
      * signType签名类型,4PKI签名,与提交订单时的签名类型保持一致
      * payType支付方式,00全部,与提交订单时的支付方式保持一致
      * bankId银行代码
      * orderId商户订单号,与提交订单时的商户订单号保持一致
      * orderTime商户订单提交时间,与提交订单时的商户订单提交时间保持一致
      * orderAmount商户订单金额,与提交订单时的商户订单金额保持一致。
      * dealId快钱交易号
      * bankDealId银行交易号
      * dealTime快钱交易时间
      * payAmount订单实际支付金额
      * fee费用
      * ext1扩展字段1,与提交订单时的扩展字段1保持一致
      * ext2扩展字段2,与提交订单时的扩展字段2保持一致
      * payResult处理结果 10:支付成功;11:支付失败
      * errCode错误代码,可为空
      * signMsg签名字符串
      */
  $BillResponse = new Application_Model_BillResponse($_REQUEST);
  //$BillResponse->checkSignMsg验证签名字符串是否正确,防止bug漏洞等
  if($BillResponse->checkSignMsg){
   //判断订单支付是否成功
   if($BillResponse->isSuccess){
    //返回给快钱,快钱会按照redirecturl地址跳到新页面,这里是成功页面
    return "<result>1</result><redirecturl>http://99bill/default/index/sucess</redirecturl>";exit;
   }else{
    //返回给快钱,快钱会按照redirecturl地址跳到新页面,这个是失败页面
    return "<result>1</result><redirecturl>http://99bill/default/index/fail</redirecturl>";exit;
   }
  }
  //返回给快钱,快钱会按照redirecturl地址跳到新页面,这个是失败页面
  return "<result>1</result><redirecturl>http://99bill/default/index/fail</redirecturl>";exit;
    }
    //redirecturl地址
    //成功
    public function success()
    {
    }
    //失败
    public function fail()
    {
    }
}

modules/default/views/scripts/index/index.phtml
https://www.99bill.com/gateway/recvMerchantInfoAction.htm
<?php $BillRequest = (array)$this->BillRequest;?>
<div style="display:none;">
<form name="kqPay" action="https://www.99bill.com/gateway/recvMerchantInfoAction.htm" method="post">
<?php foreach($BillRequest as $key => $val):?>
 <input type="hidden" name="<?php echo $key;?>" value="<?php echo $val;?>"/>
<?php endforeach;?>
 <input type="submit" name="submit" value="提交到快钱" id="kqPay">
</form>
</div>
<script>
document.getElementById('kqPay').click();
</script>

models/BillRequest.php
BillRequest.php
<?php
class Application_Model_BillRequest
{
 public function __construct($MockOrder){
  /*
   * 人民币网关账号。
   *第一种方式:该账号为11位人民币网关商户编号+01,该参数必填。01对应工商银行。
   *第二种方式:该账号为16位人民币网关商户
   */
  $this->merchantAcctId = "1001011111101"; 
  //服务器接收支付结果的后台地址,该参数务必填写,绝对路径//不能为空。
  $this->bgUrl = "http://99bill/default/index/receive";
  //商户订单号,以下采用时间来定义订单号,商户可以根据自己订单号的定义规则来定义该值//不能为空。
  $this->orderId = 'TOLPC'.sprintf("%09d", $MockOrder['orderId']);
  //订单金额,金额以“分”为单位,商户测试以1分测试即可,切勿以大金额测试,该参数必填//不能为空
  $this->orderAmount =$MockOrder['orderAmount'];
  //订单提交时间,格式:yyyyMMddHHmmss,如:20071117020101//不能为空。
  $this->orderTime = date("YmdHis", $MockOrder['orderTime']);
  //支付人姓名,可以为空。
  $this->payerName= ""; 
  //支付人联系类型,1 代表电子邮件方式;2 代表手机联系方式。可以为空。
  $this->payerContactType =  "";
  //支付人联系方式,与payerContactType设置对应,payerContactType为1,则填写邮箱地址;payerContactType为2,则填写手机号码。可以为空。
  $this->payerContact =  "";
  //商品名称,可以为空。
  $this->productName= "TOLPC";
  //商品数量,可以为空。
  $this->productNum = "1";
  //商品代码,可以为空。
  $this->productId = $MockOrder['ets_license'];
  //商品描述,可以为空。
  $this->productDesc = "";
  //支付方式,一般为00,代表所有的支付方式。如果是银行直连商户,该值为10,必填//不能为空
  $this->payType = "00";
  //编码方式,1代表 UTF-8; 2 代表 GBK; 3代表 GB2312 默认为1,该参数必填//不能为空
  $this->inputCharset = "1";
  //网关版本,固定值:v2.0,该参数必填//不能为空
  $this->version =  "v2.0";
  //语言种类,1代表中文显示,2代表英文显示。默认为1,该参数必填//不能为空
  $this->language =  "1";
  //签名类型,该值为4,代表PKI加密方式,该参数必填//不能为空
  $this->signType =  "4";
  //接收支付结果的页面地址,该参数一般置为空即可。
  $this->pageUrl = "";
  //扩展字段1,商户可以传递自己需要的参数,支付完快钱会原值返回,可以为空。
  $this->ext1 = $MockOrder['orderId'];
  //扩展自段2,商户可以传递自己需要的参数,支付完快钱会原值返回,可以为空。
  $this->ext2 = $MockOrder['orderTime'];
  //银行代码,如果payType为00,该值可以为空;如果payType为10,该值必须填写,具体请参考银行列表。
  $this->bankId = "";
  //同一订单禁止重复提交标志,实物购物车填1,虚拟产品用0。1代表只能提交一次,0代表在支付不成功情况下可以再提交。可为空。
  $this->redoFlag = "";
  //快钱合作伙伴的帐户号,即商户编号,可为空。
  $this->pid = "";  //快钱提供的request参数。
  $KeyOrders = array('inputCharset','pageUrl','bgUrl','version','language','signType','merchantAcctId','payerName','payerContactType','payerContact',
   'orderId','orderAmount','orderTime','productName','productNum','productId','productDesc','ext1','ext2','payType','bankId','redoFlag','pid',);
  //判断快钱提供的request参数的值是否为空,把非空的参数及值重新组建数组
  foreach($KeyOrders as $key){
   if(''==$this->{$key}){continue;}
   $params[$key] = $this->{$key};
  }
  //http_build_query()生成URL-encode之后的请求字符串
  //urldecode()还原未编码的字符串
  //getSignMsg() PKI加密,也可使用MD5加密
  //MD5加密方式  strtoupper(md5(urldecode(http_build_query($params))));这种不常用了。
  //常用PKI加密
  $this->signMsg = $this->getSignMsg(urldecode(http_build_query($params)));
 }
 //PKI加密技术
 public function getSignMsg($param){
  //99bill-rsa.pem是快钱的一个CA证书
  //本地随机生成一个KEY,用此KEY加密数据 KEY为$priv_key_id
  $priv_key_id = openssl_get_privatekey(file_get_contents("99bill-rsa.pem", "r"));
  //用$priv_key_id给$param数据加密。
  //计算一个签名字符串$param通过使用SHA1哈希加密,随后$priv_key_id私钥加密。数据本身是不加密的。
  openssl_sign($param, $signMsg, $priv_key_id, OPENSSL_ALGO_SHA1);
  //从存储器上释放$priv_key_id
  openssl_free_key($priv_key_id);
  //使用base64对数据进行编码
  return base64_encode($signMsg);
 }
}

models/BillResponse.php
BillResponse.php
<?php
class Application_Model_BillResponse
{
 /*
  * __construct()构造函数
  * 生成19个参数及值,可能有一个参数的值为空,$this->errCode的值可能为空
  */
 public function __construct($response){
  $KeyOrders = array('merchantAcctId','version','language','signType','payType','bankId','orderId','orderTime','orderAmount',
   'dealId','bankDealId','dealTime','payAmount','fee','ext1','ext2','payResult','errCode', 'signMsg');
  foreach($KeyOrders as $key){
   $this->{$key} = $response[$key];
  }
 }
 /*
  * 检查签名字符串
  * 快钱返回的签名字符串是$this->signMsg
  * 使用base64对前面字符串进行解码
  * 验证使用快钱给的公钥验证
  * 快钱那边他们把返回来的参数值不为空的使用私钥加密生成了$this->signMsg
  * 快钱给了我们私钥对应的公钥,我们使用这个公钥来验证。1成功,0失败,-1错误。
  */
 public function checkSignMsg(){
  $KeyOrders = array('merchantAcctId','version','language','signType','payType','bankId','orderId','orderTime','orderAmount',
   'dealId','bankDealId','dealTime','payAmount','fee','ext1','ext2','payResult','errCode',);
  foreach($KeyOrders as $key){
   if(''==$this->{$key}){continue;}
   $params[$key] = $this->{$key};
  }
  //$pub_key_id 公钥
  $pub_key_id = openssl_get_publickey(file_get_contents("99bill-rsa.cer", "r"));
  return openssl_verify(urldecode(http_build_query($params)), base64_decode($this->signMsg), $pub_key_id); 
 }
 public function isSuccess(){
  //$this->payResult成功时10,失败时11
  return '10'==$this->payResult;
 }
 public function getOrderId(){
  return str_replace('XXX', '', $this->orderId);
 }
}

需要一个公钥和一个私钥,这个不是一对的
都是一半
99bill-rsa.cer
99bill-rsa.pem
PHP 相关文章推荐
自动生成文章摘要的代码[PHP 版本]
Mar 20 PHP
PHP学习笔记 (1) 环境配置与代码调试
Jun 19 PHP
PHP文章按日期(月日)SQL归档语句
Nov 29 PHP
解析CodeIgniter自定义配置文件
Jun 18 PHP
慎用preg_replace危险的/e修饰符(一句话后门常用)
Jun 19 PHP
PHP中fwrite与file_put_contents性能测试代码
Aug 02 PHP
php 发送带附件邮件示例
Jan 23 PHP
ThinkPHP基于PHPExcel导入Excel文件的方法
Oct 15 PHP
PHP文件上传判断file是否己选择上传文件的方法
Nov 10 PHP
php之header的不同用法总结(实例讲解)
Nov 28 PHP
PHP实现的简单组词算法示例
Apr 10 PHP
详解PHP 7.4 中数组延展操作符语法知识点
Jul 19 PHP
在wamp集成环境下升级php版本(实现方法)
Jul 01 #PHP
mongo Table类文件 获取MongoCursor(游标)的实现方法分析
Jul 01 #PHP
php5.3 注意事项说明
Jul 01 #PHP
file_get_contents(&quot;php://input&quot;, &quot;r&quot;)实例介绍
Jul 01 #PHP
如何给phpcms v9增加类似于phpcms 2008中的关键词表
Jul 01 #PHP
解析php做推送服务端实现ios消息推送
Jul 01 #PHP
php ios推送(代码)
Jul 01 #PHP
You might like
php创建多级目录的方法
2015/03/24 PHP
javascript 日期时间 转换的方法
2013/02/21 Javascript
javascript解决innerText浏览器兼容问题思路代码
2013/05/17 Javascript
JavaScript中instanceof运算符的用法总结
2013/11/19 Javascript
javascript history对象(历史记录)使用方法(实现浏览器前进后退)
2014/01/07 Javascript
JavaScript使用HTML5的window.postMessage实现跨域通信例子
2014/04/11 Javascript
详谈JS中实现种子随机数及作用
2016/07/19 Javascript
javascript的几种写法总结
2016/09/30 Javascript
浅谈js数组和splice的用法
2016/12/04 Javascript
vue 根据数组中某一项的值进行排序的方法
2018/08/30 Javascript
详解@angular/cli 改变默认启动端口两种方式
2018/11/29 Javascript
JS实现的图片选择顺序切换和循环切换功能示例【测试可用】
2018/12/28 Javascript
微信小程序云开发之使用云存储
2019/05/17 Javascript
[51:11]2014 DOTA2国际邀请赛中国区预选赛5.21 LGD-CDEC VS DT
2014/05/22 DOTA
[08:47]2018国际邀请赛 OG战队举杯时刻
2018/08/29 DOTA
[01:57]2018年度DOTA2最具潜力解说-完美盛典
2018/12/16 DOTA
[54:10]完美世界DOTA2联赛PWL S2 Magma vs FTD 第二场 11.29
2020/12/03 DOTA
Python最长公共子串算法实例
2015/03/07 Python
Python的多态性实例分析
2015/07/07 Python
Python优先队列实现方法示例
2017/09/21 Python
python中map的基本用法示例
2018/09/10 Python
Python中的取模运算方法
2018/11/10 Python
Python字典的基本用法实例分析【创建、增加、获取、修改、删除】
2019/03/05 Python
利用pyinstaller打包exe文件的基本教程
2019/05/02 Python
解决Django 在ForeignKey中出现 non-nullable field错误的问题
2019/08/06 Python
Python+Opencv实现把图片、视频互转的示例
2020/12/17 Python
一款纯css3实现的非常实用的鼠标悬停特效演示
2014/11/05 HTML / CSS
复古风格的女装和装饰品:ModCloth
2017/12/29 全球购物
印尼第一大家居、生活和家具电子商务:Ruparupa
2019/11/25 全球购物
专科毕业生学习生活的自我评价
2013/10/26 职场文书
应届生如何写自荐信
2014/01/05 职场文书
敬老模范事迹
2014/05/21 职场文书
2014年审计工作总结
2014/11/17 职场文书
工作简报格式范文
2015/07/21 职场文书
js基础语法与maven项目配置教程案例
2021/07/15 Javascript
详细介绍python操作RabbitMq
2022/04/12 Python