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

php实现微信支付之企业付款

发布:smiling 来源: PHP粉丝网  添加日期:2021-09-21 19:33:13 浏览: 评论:0 

网上的很多PHP微信支付接入教程都颇为复杂,且需要配置和引入较多的文件,本人通过整理后给出一个单文件版的,希望可以给各位想接入微信支付的带来些许帮助和借鉴意义。

直接运行该文件即可给指定的微信用户转账。

需要注意的事项:

1.微信企业付款到零钱要求必传证书,需要到这里账户中心->账户设置->API安全->下载证书,然后修改代码中的证书路径

2.该文件需放到支付授权目录下,可以在微信支付商户平台->产品中心->开发配置中设置。

3.如提示签名错误可以通过微信支付签名验证工具进行验证:微信公众平台支付接口调试工具

4.错误码参照:参照地址

代码如下:

  1. <?php 
  2. /** 
  3.  * 关于微信企业付款的说明 
  4.  * 1.微信企业付款要求必传证书,需要到https://pay.weixin.qq.com 账户中心->账户设置->API安全->下载证书,证书路径在第207行和210行修改 
  5.  * 2.错误码参照 :https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=14_2 
  6.  */ 
  7. header('Content-type:text/html; Charset=utf-8'); 
  8. $mchid = 'xxxxx';  //微信支付商户号 PartnerID 通过微信支付商户资料审核后邮件发送 
  9. $appid = 'xxxxx'//微信支付申请对应的公众号的APPID 
  10. $appKey = 'xxxxx'//微信支付申请对应的公众号的APP Key 
  11. $apiKey = 'xxxxx'//https://pay.weixin.qq.com 帐户设置-安全设置-API安全-API密钥-设置API密钥 
  12.  
  13. //①、获取当前访问页面的用户openid(如果给指定用户转账,则直接填写指定用户的openid) 
  14. $wxPay = new WxpayService($mchid,$appid,$appKey,$apiKey); 
  15. $openId = $wxPay->GetOpenid(); //获取openid 
  16. if(!$openIdexit('获取openid失败'); 
  17. //②、付款 
  18. $outTradeNo = uniqid(); //订单号 
  19. $payAmount = 1;  //转账金额,单位:元。转账最小金额为1元 
  20. $trueName = '张三';  //收款人真实姓名 
  21. $result = $wxPay->createJsBizPackage($openId,$payAmount,$outTradeNo,$trueName); 
  22. echo 'success'
  23. class WxpayService 
  24.  protected $mchid
  25.  protected $appid
  26.  protected $appKey
  27.  protected $apiKey
  28.  public $data = null; 
  29.  
  30.  public function __construct($mchid$appid$appKey,$key
  31.  { 
  32.  $this->mchid = $mchid
  33.  $this->appid = $appid
  34.  $this->appKey = $appKey
  35.  $this->apiKey = $key
  36.  } 
  37.  
  38.  /** 
  39.  * 通过跳转获取用户的openid,跳转流程如下: 
  40.  * 1、设置自己需要调回的url及其其他参数,跳转到微信服务器https://open.weixin.qq.com/connect/oauth2/authorize 
  41.  * 2、微信服务处理完成之后会跳转回用户redirect_uri地址,此时会带上一些参数,如:code 
  42.  * @return 用户的openid 
  43.  */ 
  44.  public function GetOpenid() 
  45.  { 
  46.  //通过code获得openid 
  47.  if (!isset($_GET['code'])){ 
  48.   //触发微信返回code码 
  49.   $scheme = $_SERVER['HTTPS']=='on' ? 'https://' : 'http://'
  50.   $baseUrl = urlencode($scheme.$_SERVER['HTTP_HOST'].$_SERVER['PHP_SELF'].$_SERVER['QUERY_STRING']); 
  51.   $url = $this->__CreateOauthUrlForCode($baseUrl); 
  52.   Header("Location: $url"); 
  53.   exit(); 
  54.  } else { 
  55.   //获取code码,以获取openid 
  56.   $code = $_GET['code']; 
  57.   $openid = $this->getOpenidFromMp($code); 
  58.   return $openid
  59.  } 
  60.  } 
  61.  
  62.  /** 
  63.  * 通过code从工作平台获取openid机器access_token 
  64.  * @param string $code 微信跳转回来带上的code 
  65.  * @return openid 
  66.  */ 
  67.  public function GetOpenidFromMp($code
  68.  { 
  69.  $url = $this->__CreateOauthUrlForOpenid($code); 
  70.  $res = self::curlGet($url); 
  71.  //取出openid 
  72.  $data = json_decode($res,true); 
  73.  $this->data = $data
  74.  $openid = $data['openid']; 
  75.  return $openid
  76.  } 
  77.  
  78.  /** 
  79.  * 构造获取open和access_toke的url地址 
  80.  * @param string $code,微信跳转带回的code 
  81.  * @return 请求的url 
  82.  */ 
  83.  private function __CreateOauthUrlForOpenid($code
  84.  { 
  85.  $urlObj["appid"] = $this->appid; 
  86.  $urlObj["secret"] = $this->appKey; 
  87.  $urlObj["code"] = $code
  88.  $urlObj["grant_type"] = "authorization_code"
  89.  $bizString = $this->ToUrlParams($urlObj); 
  90.  return "https://api.weixin.qq.com/sns/oauth2/access_token?".$bizString
  91.  } 
  92.  
  93.  /** 
  94.  * 构造获取code的url连接 
  95.  * @param string $redirectUrl 微信服务器回跳的url,需要url编码 
  96.  * @return 返回构造好的url 
  97.  */ 
  98.  private function __CreateOauthUrlForCode($redirectUrl
  99.  { 
  100.  $urlObj["appid"] = $this->appid; 
  101.  $urlObj["redirect_uri"] = "$redirectUrl"
  102.  $urlObj["response_type"] = "code"
  103.  $urlObj["scope"] = "snsapi_base"
  104.  $urlObj["state"] = "STATE"."#wechat_redirect"
  105.  $bizString = $this->ToUrlParams($urlObj); 
  106.  return "https://open.weixin.qq.com/connect/oauth2/authorize?".$bizString
  107.  } 
  108.  
  109.  /** 
  110.  * 拼接签名字符串 
  111.  * @param array $urlObj 
  112.  * @return 返回已经拼接好的字符串 
  113.  */ 
  114.  private function ToUrlParams($urlObj
  115.  { 
  116.  $buff = ""
  117.  foreach ($urlObj as $k => $v
  118.  { 
  119.   if($k != "sign"$buff .= $k . "=" . $v . "&"
  120.  } 
  121.  $buff = trim($buff"&"); 
  122.  return $buff
  123.  } 
  124.  
  125.  /** 
  126.  * 企业付款 
  127.  * @param string $openid 调用【网页授权获取用户信息】接口获取到用户在该公众号下的Openid 
  128.  * @param float $totalFee 收款总费用 单位元 
  129.  * @param string $outTradeNo 唯一的订单号 
  130.  * @param string $orderName 订单名称 
  131.  * @param string $notifyUrl 支付结果通知url 不要有问号 
  132.  * @param string $timestamp 支付时间 
  133.  * @return string 
  134.  */ 
  135.  public function createJsBizPackage($openid$totalFee$outTradeNo,$trueName
  136.  { 
  137.  $config = array
  138.   'mch_id' => $this->mchid, 
  139.   'appid' => $this->appid, 
  140.   'key' => $this->apiKey, 
  141.  ); 
  142.  $unified = array
  143.   'mch_appid' => $config['appid'], 
  144.   'mchid' => $config['mch_id'], 
  145.   'nonce_str' => self::createNonceStr(), 
  146.   'openid' => $openid
  147.   'check_name'=>'FORCE_CHECK'//校验用户姓名选项。NO_CHECK:不校验真实姓名,FORCE_CHECK:强校验真实姓名 
  148.   're_user_name'=>$trueName,   //收款用户真实姓名(不支持给非实名用户打款) 
  149.   'partner_trade_no' => $outTradeNo
  150.   'spbill_create_ip' => '127.0.0.1'
  151.   'amount' => intval($totalFee * 100), //单位 转为分 
  152.   'desc'=>'付款',  //企业付款操作说明信息 
  153.  ); 
  154.  $unified['sign'] = self::getSign($unified$config['key']); 
  155.  $responseXml = $this->curlPost('https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers', self::arrayToXml($unified)); 
  156.  $unifiedOrder = simplexml_load_string($responseXml'SimpleXMLElement', LIBXML_NOCDATA); 
  157.  if ($unifiedOrder === false) { 
  158.   die('parse xml error'); 
  159.  } 
  160.  if ($unifiedOrder->return_code != 'SUCCESS') { 
  161.   die($unifiedOrder->return_msg); 
  162.  } 
  163.  if ($unifiedOrder->result_code != 'SUCCESS') { 
  164.   die($unifiedOrder->err_code); 
  165.  } 
  166.  return true; 
  167.  } 
  168.  
  169.  public static function curlGet($url = ''$options = array()) 
  170.  { 
  171.  $ch = curl_init($url); 
  172.  curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); 
  173.  curl_setopt($ch, CURLOPT_TIMEOUT, 30); 
  174.  if (!emptyempty($options)) { 
  175.   curl_setopt_array($ch$options); 
  176.  } 
  177.  //https请求 不验证证书和host 
  178.  curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); 
  179.  curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false); 
  180.  $data = curl_exec($ch); 
  181.  curl_close($ch); 
  182.  return $data
  183.  } 
  184.  
  185.  public function curlPost($url = ''$postData = ''$options = array()) 
  186.  { 
  187.  if (is_array($postData)) { 
  188.   $postData = http_build_query($postData); 
  189.  } 
  190.  $ch = curl_init(); 
  191.  curl_setopt($ch, CURLOPT_URL, $url); 
  192.  curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); 
  193.  curl_setopt($ch, CURLOPT_POST, 1); 
  194.  curl_setopt($ch, CURLOPT_POSTFIELDS, $postData); 
  195.  curl_setopt($ch, CURLOPT_TIMEOUT, 30); //设置cURL允许执行的最长秒数 
  196.  if (!emptyempty($options)) { 
  197.   curl_setopt_array($ch$options); 
  198.  } 
  199.  //https请求 不验证证书和host 
  200.  curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); 
  201.  curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false); 
  202.  
  203.  //第一种方法,cert 与 key 分别属于两个.pem文件 
  204.  //默认格式为PEM,可以注释 
  205.  curl_setopt($ch,CURLOPT_SSLCERTTYPE,'PEM'); 
  206.  curl_setopt($ch,CURLOPT_SSLCERT,getcwd().'/cert/apiclient_cert.pem'); 
  207.  //默认格式为PEM,可以注释 
  208.  curl_setopt($ch,CURLOPT_SSLKEYTYPE,'PEM'); 
  209.  curl_setopt($ch,CURLOPT_SSLKEY,getcwd().'/cert/apiclient_key.pem'); 
  210.  //第二种方式,两个文件合成一个.pem文件 
  211. // curl_setopt($ch,CURLOPT_SSLCERT,getcwd().'/all.pem'); 
  212.  $data = curl_exec($ch); 
  213.  curl_close($ch); 
  214.  return $data
  215.  } 
  216.  
  217.  public static function createNonceStr($length = 16) 
  218.  { 
  219.  $chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'
  220.  $str = ''
  221.  for ($i = 0; $i < $length$i++) { 
  222.   $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1); 
  223.  } 
  224.  return $str
  225.  } 
  226.  public static function arrayToXml($arr
  227.  { 
  228.  $xml = "<xml>"
  229.  foreach ($arr as $key => $val) { 
  230.   if (is_numeric($val)) { 
  231.   $xml .= "<" . $key . ">" . $val . "</" . $key . ">"
  232.   } else 
  233.   $xml .= "<" . $key . "><![CDATA[" . $val . "]]></" . $key . ">"
  234.  } 
  235.  $xml .= "</xml>"
  236.  return $xml
  237.  } 
  238.  
  239.  public static function getSign($params$key
  240.  { 
  241.  ksort($params, SORT_STRING); 
  242.  $unSignParaString = self::formatQueryParaMap($params, false); 
  243.  $signStr = strtoupper(md5($unSignParaString . "&key=" . $key)); 
  244.  return $signStr
  245.  } 
  246.  protected static function formatQueryParaMap($paraMap$urlEncode = false) 
  247.  { 
  248.  $buff = ""
  249.  ksort($paraMap); 
  250.  foreach ($paraMap as $k => $v) { 
  251.   if (null != $v && "null" != $v) { 
  252.   if ($urlEncode) { 
  253.    $v = urlencode($v); 
  254.   } 
  255.   $buff .= $k . "=" . $v . "&"
  256.   } 
  257.  } 
  258.  $reqPar = ''
  259.  if (strlen($buff) > 0) { 
  260.   $reqPar = substr($buff, 0, strlen($buff) - 1); 
  261.  } 
  262.  return $reqPar
  263.  } 
  264. ?>

Tags: php微信支付 php企业付款

分享到: