V3微信这文档咋就没写app微信支付 服务器端的处理呢?我找半天没找到,幸亏有先辈已经搞过,网上留下了blog,才解决了问题
注意的是: 签名必须二次签名,、
以下getPrePayOrder()是生成 预生成订单 然后将二次签名后的json串 返回给Android客户端即可 echo $sign2 = $this->getOrder($res['prepay_id']); ;
然后问题就接踵而至了,那么 用户下单微信支付时 用户取消了,订单成为了 待付款的订单 这个时候 用户在待付款的订单中再次调起支付咋整?
然后就在微信开放平台找, 我去没找到待付款订单的处理相关资料,后来看到有个 orderquery的,然后是查询 预生成订单啥的一个接口,但是这就是一个大深坑
我去审核了n便的 参数,没问题呀,为啥就是拿不到 第一次调起的json串 呢?prepay_id为啥就是null呢? 依然未果;第二天继续找了一上午,未果,在此时中午时分,
我加了微信支付群,然后我问的是app支付调起,他们回答的是微信公众号的支付,我也是醉了,但是有个人说了一句: 再次调起一下不可以吗? 我一下有种试试的心态,
然后去看调起预生成订单的接口文档:https://api.mch.weixin.qq.com/pay/unifiedorder 呵呵,跟们就没说,相同订单号、参数再次调起的字眼,然后我和Android的客户端开始试试,发现Android返回-1,然后就感觉不行应该,因为文档根本就没说 之前生成的订单号,同样的订单号再去生成 一次预付款的订单,我个人想想就不靠谱吧,因为服务器端已经生成过一次了,相同订单号再去生成不会冲突?但是抱着试试的心态,审核了一遍参数,返现金额好像和之前订单有出入,然后换成了 数据库的代付款的金额,参数完全和 第一次调起 预生成订单一模一样了,然后Android去调起,我来个去,Android成功调起了,fuck 不可思议,我只能说我醉了,可能是我能力不足吧
header("Content-type:text/html;charset=utf-8");
// 本类由系统自动生成,仅供测试用途
class WxpayAction extends Action {
protected function _initialize(){ }
/*
配置参数
*/
var $config = array(
'appid' => "***************", /*微信开放平台上的应用id*/
'mch_id' => "************", /*微信申请成功之后邮件中的商户id*/
'api_key' => "*******************", /*在微信商户平台上自己设定的api密钥 32位*/
'notify_url' => 'http://www.wechat.cn/app/commerce_wechat/notify.json' /*自定义的回调程序地址id*/
);
//获取预支付订单
public function getPrePayOrder(){
$url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
$notify_url = $this->config["notify_url"];
$body ="APP支付测试";
$out_trade_no = mt_rand();
$total_fee ='1';
$onoce_str = $this->getRandChar(32);
$data["appid"] = $this->config["appid"];
$data["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转成数组
$res = $this->xmlstr_to_array($response);
$sign2 = $this->getOrder($res['prepay_id']);
if(!empty($sign2)) echo json_encode(array('status'=>1,'data'=>$sign2));
else echo json_encode(array('status'=>0,'data'=>"请确保参数合法性!"));
}
// 待付款订单再次生成预生成订单 调起,覆盖之前的调起
public function orderquery($osn,$feedeal,$type=""){ //$osn是第一次生成的也是数据库待付款的订单号 $feedeal也是之前的金额
$url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
$notify_url = $this->config["notify_url"];
$body ="快逸生活钱包充值";
$attach = "快逸生活钱包充值";
$out_trade_no = $osn;
$total_fee =$feedeal;
$onoce_str = $this->getRandChar(32);
//判断是微信钱包充值调起,还是购物调起
if($type!='' && $type==1){
$notify_url = "http://www.24789.com/Pay/wx_pay_order";
$body = "快逸生活订单支付";
$attach = "快逸生活订单支付";
}
$data["appid"] = $this->config["appid"];
$data["attach"]=$attach;
$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*100;
$data["trade_type"] = "APP";
$s = $this->getSign($data, false);
$data["sign"] = $s;
$xml = $this->arrayToXml($data); //echo json_encode(array('status'=>0,'data'=>$xml,'one'=>$data));exit;
$response = $this->postXmlCurl($xml, $url);
//将微信返回的结果xml转成数组
$res = $this->xmlstr_to_array($response);
$sign2 = $this->getOrder($res['prepay_id']);
if(!empty($sign2)) return json_encode(array('status'=>1,'data'=>$sign2));
else return json_encode(array('status'=>0,'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_;
}
//执行第二次签名,才能返回给客户端使用
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 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;
}
/*
获取当前服务器的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;
}
/**
xml转成数组
*/
function xmlstr_to_array($xmlstr) {
$doc = new DOMDocument();
$doc->loadXML($xmlstr);
return $this->domnode_to_array($doc->documentElement);
}
//数组转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;
}
//将数组转成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;
}
//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;
}
}
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;
}
}