关于微店整合微信支付时出现的的各种问题的记录 php版
来源:程序员人生 发布时间:2015-03-07 13:27:10 阅读次数:5029次
由于公司项目的需要,需要在项目里整合微信支付,公司原本的项目中已有老版本的支付功能,在此不作讨论。我需要作的就是加入新版本的支付功能
让项目里可以支持新老两种版本的功能。在微信官方下载到了新版支付的php版sdk,看到 WxPay.ub.config.php,SDKRuntimeException.php, WxPayPubHelper.php
3个文件。我需要改造的就是WxPayPubHelper.php这个文件,由于这个文件现在是单用户配置的,所以我要让它变成多用户的,在改造的进程中遇到了几点问题,也是大家
可能会常常遇到的,在这分享出来,希望能给他人1些帮助。
q1:redirect_uri 参数毛病的问题 !
这个问题出现的比例比较大,在sdk包里有1个demo的文件夹,里面有1个js_api_call.php文件,在他的13行里有以下代码:
//使用jsapi接口
$jsApi = new JsApi_pub();
//=========步骤1:网页授权获得用户openid============
//通过code取得openid
if (!isset($_GET['code']))
{
//触发微信返回code码
$url = $jsApi->createOauthUrlForCode(WxPayConf_pub::JS_API_CALL_URL);
Header("Location: $url");
}else
{
//获得code码,以获得openid
$code = $_GET['code'];
$jsApi->setCode($code);
$openid = $jsApi->getOpenId();
}
$url = $jsApi->createOauthUrlForCode(WxPayConf_pub::JS_API_CALL_URL);
这1句就是设置redirect_uri参数的地方,我把它改造成了自己的地址,代码以下:
$url = $jsApi->createOauthUrlForCode( $site_domain.base64_decode($_GET['uri']));
为何要用base64编码呢,由于我的url是像这样的 :
index.php?g=Wap&m=Wxpay&a=newpay&token=$1&wecha_id=$2&orderid=$3
这个url又臭又长,而且很多&符,没法辨认出哪部份是自己的url了,还有最重要的部份 WxPayHubHelper.php在大概822行的部份,有1个方法叫createOauthUrlForCode($redirectUrl),这个里面的$url这个变量没有进行url编码,请自己加1下,否则你的url里有参数将不能正常被辨认,代码很简单:
$url = urlencode($redirectUrl); 这样就行。
进行了以上的设置后如果还出现redirect_uri的毛病的话就要斟酌你的支付路径权限的问题了,进入微信公众号的后台,请确保你的是服务号,并且已认证,查看方法是页面右上角。在确认了以后 点左边菜单的 “开发者中心”,然后如图:
点击修改,填入“授权回调页面域名”,请确保这个域名和你的利用域名1致,这样应当就解决了这个问题。
q2:没法获得prepay_id的问题
首选请确保你 WxPayPubHelper.php这个文件里 UnifiedOrder_pub 这个类的getPrepayId这个方法能正常使用。
此方法代码以下:
/**
* 获得prepay_id
*/
function getPrepayId()
{
$this->postXml();
$this->result = $this->xmlToArray($this->response);
$prepay_id = $this->result["prepay_id"];
return $prepay_id;
}
查看posXml这个方法后发现调用了自己的createXml这个方法
postXml方法以下:
/**
* 作用:post要求xml
*/
function postXml()
{
$xml = $this->createXml();
$this->response = $this->postXmlCurl($xml,$this->url,$this->curl_timeout);
return $this->response;
}
这时候需要确认你的$xml这个变量情势以下:
<xml><openid><![CDATA[o-aubjtEkotIj_GNXXXXqV8_XXXXX]]></openid>
<body>201501201705551389</body>
<out_trade_no><![CDATA[wx7143c4a87a86c8ac142174XXX]]></out_trade_no>
<total_fee>1</total_fee>
<notify_url><![CDATA[http://demo.com/paydemo/demo/notify_url.php]]></notify_url>
<trade_type><![CDATA[JSAPI]]></trade_type>
<appid><![CDATA[wx7143c4axxxxxxxx]]></appid>
<mch_id>10020230</mch_id>
<spbill_create_ip><![CDATA[192.168.1.110]]></spbill_create_ip>
<nonce_str><![CDATA[imfgrg1g2odbl6mbiacmlkxxxxxxx]]></nonce_str>
<sign><![CDATA[CBE62F36806A3E1D98CB2311XXXXXX]]></sign></xml>
特别注意 openid的情势1定要类似这样的,全是数字或全是字母的1定是毛病
然后postXmlCurl才能正确提交,提交后就可以得到正确的prepay_id了
q3:在取到了prepay_id后,在进行付款的时候提示:公众号支付使用了无效的商户号
这个问题大部份是出现在2次签名的时候,由于第2次签名和第1次签名不1致致使的
这时候我们需要看WxPayPubHelper.php这个文件里的JsApi_pub这个类,这个类中的getParameters这个方法是进行第2次签名的关键,代码以下:
/**
* 作用:设置jsapi的参数
*/
public function getParameters()
{
$jsApiObj["appId"] = WxPayConf_pub::APPID;
$timeStamp = time();
$jsApiObj["timeStamp"] = "$timeStamp";
$jsApiObj["nonceStr"] = $this->createNoncestr();
$jsApiObj["package"] = "prepay_id=$this->prepay_id";
$jsApiObj["signType"] = "MD5";
$jsApiObj["paySign"] = $this->getSign($jsApiObj);
$this->parameters = json_encode($jsApiObj);
return $this->parameters;
}
由于我需要根据数据表里的设置来动态的调用key,所以我的这个签名要符合多个商家的,代码改动以下:
/**
* 作用:设置jsapi的参数
*/
public function getParameters()
{
$jsApiObj["appId"] = $this->config['appid'];
$timeStamp = time();
$jsApiObj["timeStamp"] = "$timeStamp";
$jsApiObj["nonceStr"] = $this->createNoncestr();
$jsApiObj["package"] = "prepay_id=$this->prepay_id";
$jsApiObj["signType"] = "MD5";
$jsApiObj["paySign"] = $this->getSign($jsApiObj, $this->config['key']);
$this->parameters = json_encode($jsApiObj);
return $this->parameters;
}
getSign这个方法原型不帖了,改动后以下:
/**
* 作用:生成签名
*/
public function getSign($Obj, $key ='')
{
foreach ($Obj as $k => $v)
{
$Parameters[$k] = $v;
}
//签名步骤1:按字典序排序参数
ksort($Parameters);
$String = $this->formatBizQueryParaMap($Parameters, false);
//echo '【string1】'.$String.'</br>';
//签名步骤2:在string后加入KEY
if ($key!='') {
$String = $String."&key=".$key;
}else{
$String = $String."&key=".WxPayConf_pub::KEY;
}
//echo "【string2】".$String."</br>";
//签名步骤3:MD5加密
$String = md5($String);
//echo "【string3】 ".$String."</br>";
//签名步骤4:所有字符转为大写
$result_ = strtoupper($String);
//echo "【result】 ".$result_."</br>";
return $result_;
}
可以看到,我只是多加了1个参数,加了1个判断,由于我不想过量的破坏原来的代码,所以这样加了。
在这样的改造完成后,我遇到的问题都解决了。希望对正在看的你能有所帮助,如果还有问题可以给我留言!
生活不易,码农辛苦
如果您觉得本网站对您的学习有所帮助,可以手机扫描二维码进行捐赠