最近因为项目要接入微信App支付(没做过,不了解),然后就开网上一番狂搜+看官方文档,那是一个乱七八糟。
微信在2014年9月10号更新出了v3的版本,结果我竟然拿着v2版本在那里调试=》被坑。
回归到正题:希望接下来的朋友少走一些弯路。
微信总共有三个平台:
公众号平台:https://mp.weixin.qq.com/
商户平台:https://pay.weixin.qq.com
开放平台:https://open.weixin.qq.com/
这里所提到的微信App支付则是用到开放平台。
支付帐号申请下来后,收到财付通的一封邮件
效果如下:
https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=3_1
基本上,app支付的流程就是
1、统一下单(由自己的服务器处理) =》得到预支付订单
2、发起支付(客户端)
3、支付成功回调(服务器端)
https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=9_1
这里就说一下统一下单和支付回调两个后台需要做的操作:(这里只说如何接入,中间碰到的问题解决,会在此文章一并截图或文字提到)
调试了好几天,总算调通让前端能调起支付界面了,整理了一下代码如下:
WechatAppPay.php:
<?php
/**
* convert xml string to php array - useful to get a serializable value
*
* @param string $xmlstr
* @return array
* @author Adrien aka Gaarf
*/
class WxPayHelper{
/*
配置参数
*/
var $config = array(
'appid' => "wxe7380efe00000000", /*微信开放平台上的应用id*/
'mch_id' => "1200000000", /*微信申请成功之后邮件中的商户id*/
'api_key' => "3a9ce8d8a17cc85bcbaf32544e9cc781", /*在微信商户平台上自己设定的api密钥 32位*/
'notify_url' => 'http://www.wechat.cn/app/commerce_wechat/notify.json' /*自定义的回调程序地址id*/
);
public function __construct() {
}
//获取预支付订单
public function getPrePayOrder($body, $out_trade_no, $total_fee){
$url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
$notify_url = $this->config["notify_url"];
$onoce_str = $this->getRandChar(32);
$data["appid"] = $this->config["appid"];
$data["body"] = $body;
$data["mch_id"] = $this->config['mch_id'];
$data["nonce_str"] = $onoce_str;
$data["notify_url"] = $notify_url;
$data["out_trade_no"] = $out_trade_no;
$data["spbill_create_ip"] = $this->get_client_ip();
$data["total_fee"] = $total_fee;
$data["trade_type"] = "APP";
$s = $this->getSign($data, false);
$data["sign"] = $s;
$xml = $this->arrayToXml($data);
$response = $this->postXmlCurl($xml, $url);
//将微信返回的结果xml转成数组
return $this->xmlstr_to_array($response);
}
//执行第二次签名,才能返回给客户端使用
public function getOrder($prepayId){
$data["appid"] = $this->config["appid"];
$data["noncestr"] = $this->getRandChar(32);;
$data["package"] = "Sign=WXPay";
$data["partnerid"] = $this->config['mch_id'];
$data["prepayid"] = $prepayId;
$data["timestamp"] = time();
$s = $this->getSign($data, false);
$data["sign"] = $s;
return $data;
}
/*
生成签名
*/
function getSign($Obj)
{
foreach ($Obj as $k => $v)
{
$Parameters[strtolower($k)] = $v;
}
//签名步骤一:按字典序排序参数
ksort($Parameters);
$String = $this->formatBizQueryParaMap($Parameters, false);
//echo "【string】 =".$String."</br>";
//签名步骤二:在string后加入KEY
$String = $String."&key=".$this->config['api_key'];
//echo "<textarea style='width: 50%; height: 150px;'>$String</textarea> <br />";
//签名步骤三:MD5加密
$result_ = strtoupper(md5($String));
return $result_;
}
//获取指定长度的随机字符串
function getRandChar($length){
$str = null;
$strPol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";
$max = strlen($strPol)-1;
for($i=0;$i<$length;$i++){
$str.=$strPol[rand(0,$max)];//rand($min,$max)生成介于min和max两个数之间的一个随机整数
}
return $str;
}
//数组转xml
function arrayToXml($arr)
{
$xml = "<xml>";
foreach ($arr as $key=>$val)
{
if (is_numeric($val))
{
$xml.="<".$key.">".$val."</".$key.">";
}
else
$xml.="<".$key."><![CDATA[".$val."]]></".$key.">";
}
$xml.="</xml>";
return $xml;
}
//post https请求,CURLOPT_POSTFIELDS xml格式
function postXmlCurl($xml,$url,$second=30)
{
//初始化curl
$ch = curl_init();
//超时时间
curl_setopt($ch,CURLOPT_TIMEOUT,$second);
//这里设置代理,如果有的话
//curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
//curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
curl_setopt($ch,CURLOPT_URL, $url);
curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);
//设置header
curl_setopt($ch, CURLOPT_HEADER, FALSE);
//要求结果为字符串且输出到屏幕上
curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
//post提交方式
curl_setopt($ch, CURLOPT_POST, TRUE);
curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
//运行curl
$data = curl_exec($ch);
//返回结果
if($data)
{
curl_close($ch);
return $data;
}
else
{
$error = curl_errno($ch);
echo "curl出错,错误码:$error"."<br>";
echo "<a href='http://curl.haxx.se/libcurl/c/libcurl-errors.html'>错误原因查询</a></br>";
curl_close($ch);
return false;
}
}
/*
获取当前服务器的IP
*/
function get_client_ip()
{
if ($_SERVER['REMOTE_ADDR']) {
$cip = $_SERVER['REMOTE_ADDR'];
} elseif (getenv("REMOTE_ADDR")) {
$cip = getenv("REMOTE_ADDR");
} elseif (getenv("HTTP_CLIENT_IP")) {
$cip = getenv("HTTP_CLIENT_IP");
} else {
$cip = "unknown";
}
return $cip;
}
//将数组转成uri字符串
function formatBizQueryParaMap($paraMap, $urlencode)
{
$buff = "";
ksort($paraMap);
foreach ($paraMap as $k => $v)
{
if($urlencode)
{
$v = urlencode($v);
}
$buff .= strtolower($k) . "=" . $v . "&";
}
$reqPar;
if (strlen($buff) > 0)
{
$reqPar = substr($buff, 0, strlen($buff)-1);
}
return $reqPar;
}
/**
xml转成数组
*/
function xmlstr_to_array($xmlstr) {
$doc = new DOMDocument();
$doc->loadXML($xmlstr);
return $this->domnode_to_array($doc->documentElement);
}
function domnode_to_array($node) {
$output = array();
switch ($node->nodeType) {
case XML_CDATA_SECTION_NODE:
case XML_TEXT_NODE:
$output = trim($node->textContent);
break;
case XML_ELEMENT_NODE:
for ($i=0, $m=$node->childNodes->length; $i<$m; $i++) {
$child = $node->childNodes->item($i);
$v = $this->domnode_to_array($child);
if(isset($child->tagName)) {
$t = $child->tagName;
if(!isset($output[$t])) {
$output[$t] = array();
}
$output[$t][] = $v;
}
elseif($v) {
$output = (string) $v;
}
}
if(is_array($output)) {
if($node->attributes->length) {
$a = array();
foreach($node->attributes as $attrName => $attrNode) {
$a[$attrName] = (string) $attrNode->value;
}
$output['@attributes'] = $a;
}
foreach ($output as $t => $v) {
if(is_array($v) && count($v)==1 && $t!='@attributes') {
$output[$t] = $v[0];
}
}
}
break;
}
return $output;
}
}
?>
建议参考的朋友直接把WechatAppPay.php拷贝放到你的项目中做调用即可。
注意的地方:
post必须支持https,且参数格式必须是xml
sign签名的参数包括所有$data,除了自己
$data[“spbill_create_ip”]不能随便设定一个ip地址,不要以为调试方便随便设定,结果返回签名错误坑你没商量。一定要是程序执行时所在的服务器ip地址,所以使用get_client_ip()获取就好。
api_key是需要自己进入商户平台设定的,邮件不会发给你哦
使用在线随机程序产生32个字符就好了
相当重要的是:返回给各户端发起支付时,还要进行二次签名
$WxPayHelper->getOrder
第二步:App端支付完成后该调用第一步请求预支付订单时填写的notify_url了。
因为不知道以哪种方式验证回调结果是否为微信的,所以我就没做签名验证的处理
直接拿到结果去做订单的处理了《这里对回调不做过多的解释只说下大体,有疑问的朋友可以参考下http://wyong.blog.51cto.com/1115465/1692322这位朋友的博客,分析的很透彻》
分享一下简单的处理代码:
function services_commerce_wechat_wx_notify(){
require_once 'WechatAppPay.php';
$postStr = $GLOBALS['HTTP_RAW_POST_DATA'];//这里拿到微信返回的数据结果
$WechatAppPay = new WxPayHelper();
$getData = $WechatAppPay->xmlstr_to_array($postStr);//为了方便我就直接把结果转成数组,看个人爱好了
//回调通知服务器数据正常$getData['result_code'] == 'SUCCESS'
if(!empty($getData['return_code']) && $getData['return_code'] == 'SUCCESS'){
$order_no = $getData['out_trade_no'];
$arr_str=explode("Z",$order_no);
if(!empty($arr_str)){
$order_no1 = $arr_str[1];
$order = commerce_order_load($order_no1);
global $user;
if ($user->uid == 0) {
$user = user_load($order->uid);
}
commerce_quick_wechat_notify_submit1($order, $getData);
}
else {
//error, no order id process
}
}
}
function commerce_quick_wechat_notify_submit1($order, $notify) {
$flag='';
if($order->status=='checkout_complete'||$order->status=='completed'){
echo 'The bill is finished.';
}
// Handle trade types of cases.
switch ($notify['return_code']) {
// Transaction successful.
case 'SUCCESS':
$flag=commerce_order_status_update($order, 'checkout_complete');
break;
}
if($flag){
commerce_checkout_complete($order);
//header("Content-type: text/xml; charset=utf-8");
require_once 'WechatAppPay.php';
$WechatAppPay = new WxPayHelper();
$return = array(
'return_code' => 'SUCCESS',
'return_ok' => 'OK',
);
$xml = $WechatAppPay->arrayToXml($return);
return $xml;
} else{
echo 'fail';
die();
}
}
因为我们用的框架是drupal,请根据你自己的框架来处理这些代码了。
碰到一个巨坑的问题(至今不知什么原因)网上也几乎没搜到结果。
就是回调结果转换成数组之后是这样的:
但我单独打印print_r($getData['result_code'])这两个字段的时候总是给我返回1,请教朋友也说微信官方也没这个结果的,无法解释。折腾了两天找原因。最后手贱似的清理了一下缓存。就莫名其妙的得到了success了。
文章会有不足的地方,请多多见谅指教,只为分享出来,让后续做的朋友少绕一些圈子。
请在评论中指出()or发我邮箱754634469@qq.com。
转载于:https://blog.51cto.com/laok8/1716997