需求: 安卓,ios端集成微信支付,PHPer(我)要提供一个接口给微信调用,(支付完成后调用),若是成功,就返回success,若是失败就返回fail,php
环境说明:thinkphp 框架开发,ios
微信的接口说明文档:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_7&index=3ajax
支付结果通用通知
应用场景
支付完成后,微信会把相关支付结果和用户信息发送给商户,商户须要接收处理,并返回应答。算法
对后台通知交互时,若是微信收到商户的应答不是成功或超时,微信认为通知失败,微信会经过必定的策略按期从新发起通知,尽量提升通知的成功率,但微信不保证通知最终能成功。 (通知频率为15/15/30/180/1800/1800/1800/1800/3600,单位:秒)thinkphp
注意:一样的通知可能会屡次发送给商户系统。商户系统必须可以正确处理重复的通知。api
推荐的作法是,当收到通知进行处理时,首先检查对应业务数据的状态,判断该通知是否已经处理过,若是没有处理过再进行处理,若是处理过直接返回结果成功。在对业务数据进行状态检查和处理以前,要采用数据锁进行并发控制,以免函数重入形成的数据混乱。数组
特别提醒:商户系统对于支付结果通知的内容必定要作签名验证,并校验返回的订单金额是否与商户侧的订单金额一致,防止数据泄漏致使出现“假通知”,形成资金损失。微信
接口连接
该连接是经过【统一下单API】中提交的参数notify_url设置,若是连接没法访问,商户将没法接收到微信通知。并发
通知url必须为直接可访问的url,不能携带参数。示例:notify_url:“https://pay.weixin.qq.com/wxpay/pay.action”app
是否须要证书
不须要。
通知参数
字段名
变量名
必填
类型
示例值
描述
返回状态码
return_code
是
String(16)
SUCCESS
SUCCESS/FAIL
此字段是通讯标识,非交易标识,交易是否成功须要查看result_code来判断
返回信息
return_msg
否
String(128)
签名失败
返回信息,如非空,为错误缘由
签名失败
参数格式校验错误
如下字段在return_code为SUCCESS的时候有返回
字段名
变量名
必填
类型
示例值
描述
应用ID
appid
是
String(32)
wx8888888888888888
微信开放平台审核经过的应用APPID
商户号
mch_id
是
String(32)
1900000109
微信支付分配的商户号
设备号
device_info
否
String(32)
013467007045764
微信支付分配的终端设备号,
随机字符串
nonce_str
是
String(32)
5K8264ILTKCH16CQ2502SI8ZNMTM67VS
随机字符串,不长于32位
签名
sign
是
String(32)
C380BEC2BFD727A4B6845133519F3AD6
签名,详见签名算法
业务结果
result_code
是
String(16)
SUCCESS
SUCCESS/FAIL
错误代码
err_code
否
String(32)
SYSTEMERROR
错误返回的信息描述
错误代码描述
err_code_des
否
String(128)
系统错误
错误返回的信息描述
用户标识
openid
是
String(128)
wxd930ea5d5a258f4f
用户在商户appid下的惟一标识
是否关注公众帐号
is_subscribe
否
String(1)
Y
用户是否关注公众帐号,Y-关注,N-未关注,仅在公众帐号类型支付有效
交易类型
trade_type
是
String(16)
APP
APP
付款银行
bank_type
是
String(16)
CMC
银行类型,采用字符串类型的银行标识,银行类型见银行列表
总金额
total_fee
是
Int
100
订单总金额,单位为分
货币种类
fee_type
否
String(8)
CNY
货币类型,符合ISO4217标准的三位字母代码,默认人民币:CNY,其余值列表详见货币类型
现金支付金额
cash_fee
是
Int
100
现金支付金额订单现金支付金额,详见支付金额
现金支付货币类型
cash_fee_type
否
String(16)
CNY
货币类型,符合ISO4217标准的三位字母代码,默认人民币:CNY,其余值列表详见货币类型
代金券或立减优惠金额
coupon_fee
否
Int
10
代金券或立减优惠金额<=订单总金额,订单总金额-代金券或立减优惠金额=现金支付金额,详见支付金额
代金券或立减优惠使用数量
coupon_count
否
Int
1
代金券或立减优惠使用数量
代金券或立减优惠ID
coupon_id_$n
否
String(20)
10000
代金券或立减优惠ID,$n为下标,从0开始编号
单个代金券或立减优惠支付金额
coupon_fee_$n
否
Int
100
单个代金券或立减优惠支付金额,$n为下标,从0开始编号
微信支付订单号
transaction_id
是
String(32)
1217752501201407033233368018
微信支付订单号
商户订单号
out_trade_no
是
String(32)
1212321211201407033568112322
商户系统的订单号,与请求一致。
商家数据包
attach
否
String(128)
123456
商家数据包,原样返回
支付完成时间
time_end
是
String(14)
20141030133525
支付完成时间,格式为yyyyMMddHHmmss,如2009年12月25日9点10分10秒表示为20091225091010。其余详见时间规则
举例以下:
(这个就是微信回调的内容。要把他转成本身要的数组,下面有代码)
1
返回参数
商户处理后同步返回给微信参数:
字段名
变量名
必填
类型
示例值
描述
返回状态码
return_code
是
String(16)
SUCCESS
SUCCESS/FAIL
SUCCESS表示商户接收通知成功并校验成功
返回信息
return_msg
否
String(128)
OK
返回信息,如非空,为错误缘由:
签名失败
参数格式校验错误
举例以下:
相信不少第一次作这个开发的人都很烦。并不知道怎么处理。
下面贴出个人代码:
//微信支付验证
//微信异步通知接口,若是我返回success,微信再也不通知,
public function weipayverify(){
//写支付记录,WEB_PATH是我网站的根目录
create_pay_log(WEB_PATH.'/Public/apilog/weipay_ajax/',date('Y-m-d').'.log');
libxml_disable_entity_loader(true);
$postStr = postdata();//接收post数据
$postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA);
$arr = object2array($postObj);//对象转成数组
ksort($arr);// 对数据进行排序
$str = ToUrlParams($arr);//对数据拼接成字符串
$user_sign = strtoupper(md5($str));
if($user_sign == $arr['sign']){//验证成功
//写支付记录
// to do...
//处理购买后的业务逻辑
//to do
}
}
// 接收post数据
/*
* 微信是用$GLOBALS['HTTP_RAW_POST_DATA'];这个函数接收post数据的
*/
function postdata(){
$receipt = $_REQUEST;
if($receipt==null){
$receipt = file_get_contents("php://input");
if($receipt == null){
$receipt = $GLOBALS['HTTP_RAW_POST_DATA'];
}
}
return $receipt;
}
//把对象转成数组
function object2array($array) {
if(is_object($array)) {
$array = (array)$array;
} if(is_array($array)) {
foreach($array as $key=>$value) {
$array[$key] = object2array($value);
}
}
return $array;
}
/**
* 格式化参数格式化成url参数
*/
private function ToUrlParams($arr)
{
$weipay_key = 'sdfasdfasdfasd';//微信的key,这个是微信支付给你的key,不要瞎填。
$buff = "";
foreach ($arr as $k => $v)
{
if($k != "sign" && $v != "" && !is_array($v)){
$buff .= $k . "=" . $v . "&";
}
}
$buff = trim($buff, "&");
return $buff.'&key='.$weipay_key;
}
代码虽然很简单。贴在这里给用的到的朋友。
再次提醒一下。这个接口是给安卓ios端用的,他们把这个接口放在支付完成回调的接口。微信会回调的。