使用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中使用Oracle数据库(1)
Oct 09 PHP
Adodb的十个实例(清晰版)
Dec 31 PHP
PHP 字符串 小常识
Jun 05 PHP
php基础学习之变量的使用
Jun 09 PHP
php 抽象类的简单应用
Sep 06 PHP
php使用date和strtotime函数输出指定日期的方法
Nov 14 PHP
php+xml实现在线英文词典查询的方法
Jan 23 PHP
PHP通过串口实现发送短信
Jul 08 PHP
详解php框架Yaf路由重写
Jun 20 PHP
PHP 实现文件压缩解压操作的方法
Jun 14 PHP
php array_chunk()函数用法与注意事项
Jul 12 PHP
关于laravel5.5的定时任务详解(demo)
Oct 23 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
简单的页面缓冲技术
2006/10/09 PHP
php天翼开放平台短信发送接口实现方法
2014/12/22 PHP
php使用Jpgraph绘制饼状图的方法
2015/06/10 PHP
php简单构造json多维数组的方法示例
2017/06/08 PHP
浅析PHP开发规范
2018/02/05 PHP
浅析PHP类的反射来实现依赖注入过程
2018/02/06 PHP
php中访问修饰符的知识点总结
2019/01/27 PHP
用javascript实现给出的盒子的序列是否可连为一矩型
2007/08/30 Javascript
在网页中使用document.write时遭遇的奇怪问题
2010/08/24 Javascript
js 操作select与option(示例讲解)
2013/12/20 Javascript
javascript替换已有元素replaceChild()使用介绍
2014/04/03 Javascript
将字符串中由空格隔开的每个单词首字母大写
2014/04/06 Javascript
jQuery取消ajax请求的方法
2015/06/09 Javascript
JSON 对象未定义错误的解决方法
2016/09/29 Javascript
Js apply方法详解
2017/02/16 Javascript
详解Vue方法与事件
2017/03/09 Javascript
[02:11]2014DOTA2 TI专访VG战队Fenrir:队伍气氛良好
2014/07/11 DOTA
[36:17]DOTA2上海特级锦标赛 - VGL音乐会全集
2016/03/06 DOTA
Python中使用hashlib模块处理算法的教程
2015/04/28 Python
Python实现绘制双柱状图并显示数值功能示例
2018/06/23 Python
python http基本验证方法
2018/12/26 Python
python批量从es取数据的方法(文档数超过10000)
2018/12/27 Python
在python tkinter中Canvas实现进度条显示的方法
2019/06/14 Python
Python reshape的用法及多个二维数组合并为三维数组的实例
2020/02/07 Python
python如何实现读取并显示图片(不需要图形界面)
2020/07/08 Python
用python实现前向分词最大匹配算法的示例代码
2020/08/06 Python
selenium与xpath之获取指定位置的元素的实现
2021/01/26 Python
全球最大的在线旅游公司:Expedia
2017/11/16 全球购物
家庭睡衣和家庭用品:Little Blue House
2018/03/18 全球购物
社会实践感言
2014/01/25 职场文书
我的中国梦演讲稿小学篇
2014/08/19 职场文书
幼儿园教师自我评价
2015/03/04 职场文书
水电施工员岗位职责
2015/04/11 职场文书
小学教师教育随笔
2015/08/14 职场文书
苹果发布了MagSafe固件更新,可以不外接电源实现最高7.5W充电
2022/04/21 数码科技
Spring Security动态权限的实现方法详解
2022/06/16 Java/Android