微信开发之企业付款到银行卡接口开发的示例代码


Posted in Javascript onSeptember 18, 2018

微信支付已上线企业付款至银行卡功能。商户可以将商户号余额付款至指定的收款银行账户。通过指定收款银行账户户名、卡号,以及收款银行信息即可实现付款。功能目前为灰度开放,已灰度新资金流直连普通商户及普通受理模式子商户,其他商户类型将在后续开放。

一、微信商户号须开通此功能;

微信开发之企业付款到银行卡接口开发的示例代码

二、应用环境TP5+Mysql+Centos

三、编写代码

3.1、Model层,Wechat.php

<?php
namespace app\home\model;
use think\Cache;

class Wechat extends Model
{
  private $appid;      //公众号APPID
  private $appsecret;    //公众号appsecret
  private $mchid;      //商户号
  private $key;       //支付密钥
  private $sslcert;     //证书保存的绝对路径
  private $sslkey;     //证书保存的绝对路径
  
  public function __construct($appid,$appsecret,$mchid,$key,$sslcert,$sslkey)
  {
    parent::__construct();
    $this->appid = $appid;
    $this->appsecret = $appsecret;
    $this->mchid = $mchid;
    $this->key = $key;
    $this->sslcert = $sslcert;
    $this->sslkey = $sslkey;
  }
  /*
  * 企业付款到银行卡接口
   * @params string $out_trade_no : 商户订单号
   * @params int $amount : 付款金额,单位分
   * @params string $enc_bank_no : 收款方银行卡号
   * @params string $enc_true_name : 收款方用户名
   * @params string $bank_name : 收款方开户行,根据银行名称获取银行编号bank_code
   * @params string $desc : 付款备注
   * return string $payment_no :支付成功的订单号
  */
  public function payForBank($out_trade_no,$amount,$enc_bank_no,$enc_true_name,$bank_name,$desc='企业付款到银行卡')
  {
    $data['amount'] = $amount;
    $data['bank_code'] = $this->getBankCode($bank_code);
    $data['desc'] = $desc;
    $data['enc_bank_no'] = $this->publicEncrypt($enc_bank_no);
    $data['enc_true_name'] = $this->publicEncrypt($enc_true_name);
    $data['mch_id'] = $this->mchid;
    $data['nonce_str'] = $this->random(12);
    $data['partner_trade_no'] = $out_trade_no;
    $sign = $this->getParam($data);

    $dataXML="<xml>
    <amount>".$data['amount']."</amount>
    <bank_code>".$data['bank_code']."</bank_code>
    <desc>".$data['desc']."</desc>
    <enc_bank_no>".$data['enc_bank_no']."</enc_bank_no>
    <enc_true_name>".$data['enc_true_name']."</enc_true_name>
    <mch_id>".$data['mch_id']."</mch_id>
    <nonce_str>".$data['nonce_str']."</nonce_str>
    <partner_trade_no>".$data['partner_trade_no']."</partner_trade_no>
    <sign>".$sign."</sign>
    </xml>";

    $url = 'https://api.mch.weixin.qq.com/mmpaysptrans/pay_bank';
    $ret = $this->httpsPost($url,$dataXML,true);
    if($ret['return_code'] == 'SUCCESS' && $ret['result_code'] == 'SUCCESS' && $ret['err_code'] == 'SUCCESS'){
      return $ret['payment_no'];
    }else{
      $this->errorLog('微信付款到银行卡失败,appid:'.$this->appid,$ret);
      return false;
    }
  }
  /*
   * 查询付款到银行卡状态
   * @params string $out_trade_no : 商户订单号
   * return array $ret:查询状态
   * */
  public function queryBank($out_trade_no)
  {
    $data['mch_id'] = $this->mchid;
    $data['nonce_str'] = $this->random(12);
    $data['partner_trade_no'] = $out_trade_no;
    $sign = $this->getParam($data);
    $dataXML="<xml>
    <mch_id>".$data['mch_id']."</mch_id>
    <nonce_str>".$data['nonce_str']."</nonce_str>
    <partner_trade_no>".$data['partner_trade_no']."</partner_trade_no>
    <sign>".$sign."</sign>
    </xml>";
    $url = 'https://api.mch.weixin.qq.com/mmpaysptrans/query_bank';
    $ret = $this->httpsPost($url,$dataXML,true);
    if($ret['return_code'] == 'SUCCESS' && $ret['result_code'] == 'SUCCESS' && $ret['err_code'] == 'SUCCESS'){
      return $ret;
    }else{
      $this->errorLog('查询微信付款到银行卡失败,appid:'.$this->appid.',订单号:'.$out_trade_no,$ret);
      return false;
    }
  }
  /*
   * 银行编号列表,详情参考:https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=24_4
   * @params string $bank_name : 银行名称,4个汉字
   * return int $bank_code : 银行编码
   * */
  private function getBankCode($bank_name)
  {
    $bank_code = 0;
    switch ($bank_name){
      case '工商银行':  $bank_code = 1002; break;
      case '农业银行':  $bank_code = 1005; break;
      case '中国银行':  $bank_code = 1026; break;
      case '建设银行':  $bank_code = 1003; break;
      case '招商银行':  $bank_code = 1001; break;
      case '邮储银行':  $bank_code = 1066; break;
      case '交通银行':  $bank_code = 1020; break;
      case '浦发银行':  $bank_code = 1004; break;
      case '民生银行':  $bank_code = 1006; break;
      case '兴业银行':  $bank_code = 1009; break;
      case '平安银行':  $bank_code = 1010; break;
      case '中信银行':  $bank_code = 1021; break;
      case '华夏银行':  $bank_code = 1025; break;
      case '广发银行':  $bank_code = 1027; break;
      case '光大银行':  $bank_code = 1022; break;
      case '北京银行':  $bank_code = 1032; break;
      case '宁波银行':  $bank_code = 1056; break;
    }
    return $bank_code;
  }
  /**
   * 公钥加密,银行卡号和姓名需要RSA算法加密
   * @param string $data  需要加密的字符串,银行卡/姓名
   * @return null|string  加密后的字符串
   */
  private function publicEncrypt($data)
  {
    // 进行加密
    $pubkey = openssl_pkey_get_public(file_get_contents(ROOT_PATH.'data/cert/public_pkcs8.pem'));
    $encrypt_data = '';
    $encrypted = '';
    $r = openssl_public_encrypt($data,$encrypt_data,$pubkey,OPENSSL_PKCS1_OAEP_PADDING);
    if($r){//加密成功,返回base64编码的字符串
      return base64_encode($encrypted.$encrypt_data);
    }else{
      return false;
    }
  }
  /*
   * 获取公钥,格式为PKCS#1 转PKCS#8
   * openssl rsa -RSAPublicKey_in -in  <filename> -out <out_put_filename>
   * */
  private function get_pub_key()
  {
    $rsafile = ROOT_PATH.'data/cert/'.$this->appid.'_publicrsa.pem';
    if(!is_file($rsafile)){
      $data['mch_id'] = $this->mchid;
      $data['nonce_str'] = $this->random(12);
      $sign = $this->getParam($data);
      $dataXML="<xml>
      <mch_id>".$data['mch_id']."</mch_id>
      <nonce_str>".$data['nonce_str']."</nonce_str>
      <sign>".$sign."</sign>
      </xml>";
      $url = 'https://fraud.mch.weixin.qq.com/risk/getpublickey';
      $ret = $this->httpsPost($url,$dataXML,true);
      if($ret['return_code'] == 'SUCCESS' && isset($ret['pub_key'])){
        file_put_contents($rsafile,$ret['pub_key']);
        return $ret['pub_key'];
      }else{
        return null;
      }
    }else{
      return file_get_contents($rsafile);
    }
  }
  /*
  * 发起POST网络请求
  * @params string $url : 请求的url链接地址
  * @params string $data : 数据包
  * @params bool $ssl : 是否加载证书
  * return array $result : 返回的数据结果 
  */
  private function httpsPost($url,$data,$ssl = false)
  {
    $ch = curl_init ();
    curl_setopt ( $ch, CURLOPT_URL, $url );
    curl_setopt ( $ch, CURLOPT_CUSTOMREQUEST, "POST" );
    curl_setopt ( $ch, CURLOPT_SSL_VERIFYPEER, FALSE );
    curl_setopt ( $ch, CURLOPT_SSL_VERIFYHOST, FALSE );
    if($ssl) {
      curl_setopt ( $ch,CURLOPT_SSLCERT,$this->sslcert);
      curl_setopt ( $ch,CURLOPT_SSLKEY,$this->sslkey);
    }
    curl_setopt ( $ch, CURLOPT_FOLLOWLOCATION, 1 );
    curl_setopt ( $ch, CURLOPT_AUTOREFERER, 1 );
    curl_setopt ( $ch, CURLOPT_POSTFIELDS, $data );
    curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, true );
    $result = curl_exec($ch);
    if (curl_errno($ch)) {
      return 'Errno: '.curl_error($ch);
    }
    curl_close($ch);
    return $this->xmlToArray($result);
  }
  //对参数排序,生成MD5加密签名
  private function getParam($paramArray, $isencode=false)
  {
    $paramStr = '';
    ksort($paramArray);
    $i = 0;
    foreach ($paramArray as $key => $value)
    {
      if ($key == 'Signature'){
        continue;
      }
      if ($i == 0){
        $paramStr .= '';
      }else{
        $paramStr .= '&';
      }
      $paramStr .= $key . '=' . ($isencode?urlencode($value):$value);
      ++$i;
    }
    $stringSignTemp=$paramStr."&key=".$this->key;
    $sign=strtoupper(md5($stringSignTemp));
    return $sign;
  }
  /*
  * 将xml转换成数组
  * @params xml $xml : xml数据
  * return array $data : 返回数组
  */
  private function xmlToArray($xml)
  {
    //禁止引用外部xml实体
    libxml_disable_entity_loader(true);
    $xmlstring = simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA);
    $val = json_decode(json_encode($xmlstring),true);
    return $val;
  }
  /*
  *  随机字符串
   * @param int $length 长度
   * @param string $type 类型
   * @param int $convert 转换大小写 1大写 0小写
   * @return string
  */
  private function random($length=10,$type='letter',$convert=false)
  {
    $config = array(
      'number'=>'1234567890',
      'letter'=>'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ',
      'string'=>'abcdefghjkmnpqrstuvwxyzABCDEFGHJKMNPQRSTUVWXYZ23456789',
      'all'=>'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890'
    );
  
    if(!isset($config[$type])) $type = 'letter';
    $string = $config[$type];
  
    $code = '';
    $strlen = strlen($string) -1;
    for($i = 0; $i < $length; $i++){
      $code .= $string{mt_rand(0, $strlen)};
    }
    if(!empty($convert)){
      $code = ($convert > 0)? strtoupper($code) : strtolower($code);
    }
    return $code;
  }
  /*
  * 日志记录
  * @params string $msg : 文字描述
  * @params array $ret : 调用接口返回的数组
  */
  private function errorLog($msg,$ret)
  {
    $path = ROOT_PATH.'runtime/error/';
    if(!is_dir($path)) mkdir($path,0777);
    file_put_contents(ROOT_PATH . 'runtime/error/wxpay.log', "[" . date('Y-m-d H:i:s') . "] ".$msg."," .json_encode($ret).PHP_EOL, FILE_APPEND);
  }
}

3.2、Controller控制层调用,Wxpay.php

<?php
namespace app\home\controller;
use app\home\model\Wechat;
class Wxpay extends Controller
{
  //企业付款到银行卡
  public function payBank()
  {
    $appid = 'wx****d4';
    $appsecret = '37***f0';
    $mchid = '13***2';
    $key = '53***e8';
    $sslcert = ROOT_PATH.'data/cert/apiclient_cert.pem';
    $sslkey = ROOT_PATH.'data/cert/apiclient_key.pem';
    $out_trade_no = date('Ymdhis', time()).substr(floor(microtime()*1000),0,1).rand(0,9);
    $money = 100;
    $enc_bank_no = '62***44';
    $enc_true_name = '张**';
    $bank_name = '中国银行';
    $desc = '企业付款到银行卡测试';
    $wxapi = new Wechat($appid,$appsecret,$mchid,$key,$sslcert,$sslkey);
    $payment_no = $wxapi->payForBank($out_trade_no,$money,$enc_bank_no,$enc_true_name,$bank_name,$desc);
    if($payment_no){
       echo '微信付款到银行卡操作成功,微信订单号:'.$payment_no;
    } else {
      echo '微信付款到银行卡操作失败,请检查日记';
    }
  }
}

四、付款截图

微信开发之企业付款到银行卡接口开发的示例代码

五、注意事项

1、获取的rsa为PKCS1格式需要转成PKCS8,需要执行以下命令

$filename : 从微信服务器上获取的公钥数组

$out_put_filename :转换成PKCS8后的文件保存路径

openssl rsa -RSAPublicKey_in -in <filename> -out <out_put_filename>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScipt基本教程之前言
Jan 16 Javascript
使用jQuery操作Cookies的实现代码
Oct 09 Javascript
jquery.artwl.thickbox.js  一个非常简单好用的jQuery弹出层插件
Mar 01 Javascript
JavaScript判断textarea值是否为空并给出相应提示
Sep 04 Javascript
jQuery实现伸展与合拢panel的方法
Apr 30 Javascript
javascript实现瀑布流动态加载图片原理
Aug 12 Javascript
基于JavaScript实现窗口拖动效果
Jan 18 Javascript
jQuery中ajax获取数据赋值给页面的实例
Dec 31 jQuery
vue-image-crop基于Vue的移动端图片裁剪组件示例
Aug 28 Javascript
详解Vue-axios 设置请求头问题
Dec 06 Javascript
Vue的生命周期操作示例
Sep 17 Javascript
javascript实现贪吃蛇小练习
Jul 05 Javascript
vue自定义tap指令及tap事件的实现
Sep 18 #Javascript
Vue条件循环判断+计算属性+绑定样式v-bind的实例
Sep 18 #Javascript
浅谈vue同一页面中拥有两个表单时,的验证问题
Sep 18 #Javascript
记React connect的几种写法(小结)
Sep 18 #Javascript
Vue循环组件加validate多表单验证的实例
Sep 18 #Javascript
vue2.0 + ele的循环表单及验证字段方法
Sep 18 #Javascript
JavaScript指定断点操作实例教程
Sep 18 #Javascript
You might like
使用PHP获取汉字的拼音(全部与首字母)
2013/06/27 PHP
PHP使用PHPMailer发送邮件的简单使用方法
2013/11/12 PHP
ThinkPHP框架任意代码执行漏洞的利用及其修复方法
2014/07/04 PHP
ThinkPHP和UCenter接口冲突的解决方法
2016/07/25 PHP
关于Yii2框架跑脚本时内存泄漏问题的分析与解决
2019/12/01 PHP
jQuery 数据缓存data(name, value)详解及实现
2010/01/04 Javascript
前端开发部分总结[兼容性、DOM操作、跨域等](持续更新)
2010/03/04 Javascript
jquery 学习之二 属性 文本与值(text,val)
2010/11/25 Javascript
jquery 实现input输入什么div图层显示什么
2014/06/15 Javascript
JS+CSS实现仿触屏手机拨号盘界面及功能模拟完整实例
2015/05/16 Javascript
Javascript中神奇的this
2016/01/20 Javascript
基于JavaScript实现带数据验证和复选框的表单提交
2017/08/23 Javascript
nodejs连接mysql数据库及基本知识点详解
2018/03/20 NodeJs
vue移动端弹框组件的实例
2018/09/25 Javascript
JavaScript适配器模式原理与用法实例详解
2020/03/09 Javascript
Nuxt配置Element-UI按需引入的操作方法
2020/07/06 Javascript
vue组件讲解(is属性的用法)模板标签替换操作
2020/09/04 Javascript
[01:36]DOTA2完美大师赛趣味视频之与队友相处的十万个技巧
2017/11/19 DOTA
python数据分析数据标准化及离散化详解
2018/02/26 Python
python2与python3的print及字符串格式化小结
2018/11/30 Python
python根据url地址下载小文件的实例
2018/12/18 Python
python创建子类的方法分析
2019/11/28 Python
python logging 日志的级别调整方式
2020/02/21 Python
python中的时区问题
2021/01/14 Python
python tkinter实现下载进度条及抖音视频去水印原理
2021/02/07 Python
教你使用Canvas处理图片的方法
2017/11/28 HTML / CSS
PAUL HEWITT手表美国站:德国北部时尚生活配饰品牌,船锚元素
2017/11/18 全球购物
日本最大的药妆连锁店:Matsukiyo松本清药妆店
2017/11/23 全球购物
为您搜罗全球潮流時尚品牌:HBX
2019/12/04 全球购物
JYSK加拿大:购买家具、床垫、家居装饰等
2020/02/14 全球购物
好的自荐信的要求
2013/10/30 职场文书
淘宝客服自我总结鉴定
2014/01/25 职场文书
关爱留守儿童倡议书
2014/04/15 职场文书
流动人口婚育证明
2014/10/19 职场文书
学生病假条怎么写
2015/08/17 职场文书
《乌鸦喝水》教学反思
2016/02/19 职场文书