关于微店整合微信支付时出现的的各种问题的记录 php版

由于公司项目的需要,需要在项目里整合微信支付,公司原有的项目中已有老版本的支付功能,在此不作讨论。我需要作的就是加入新版本的支付功能

让项目里可以支持新老两种版本的功能。在微信官方下载到了新版支付的php版sdk,看到 WxPay.ub.config.php,SDKRuntimeException.php, WxPayPubHelper.php

三个文件。我需要改造的就是WxPayPubHelper.php这个文件,因为这个文件现在是单用户配置的,所以我要让它变成多用户的,在改造的过程中遇到了几点问题,也是大家

可能会经常遇到的,在这分享出来,希望能给他人一些帮助。


q1:redirect_uri 参数错误的问题 !

这个问题出现的比例比较大,在sdk包里有一个demo的文件夹,里面有一个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);
这一句就是设置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行的部分,有一个方法叫createOauthUrlForCode($redirectUrl),这个里面的$url这个变量没有进行url编码,请自己加一下,否则你的url里有参数将不能正常被识别,代码很简单:

$url = urlencode($redirectUrl); 这样就行。

进行了以上的设置后如果还出现redirect_uri的错误的话就要考虑你的支付路径权限的问题了,进入微信公众号的后台,请确保你的是服务号,并且已认证,查看方法是页面右上角。在确认了之后  点左侧菜单的 “开发者中心”,然后如图:

点击修改,填入“授权回调页面域名”,请确保这个域名和你的应用域名一致,这样应该就解决了这个问题。


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的形式一定要类似这样的,全是数字或全是字母的一定是错误

然后postXmlCurl才能正确提交,提交后就能得到正确的prepay_id了


q3:在取到了prepay_id后,在进行付款的时候提示:公众号支付使用了无效的商户号

这个问题大部分是出现在二次签名的时候,由于第二次签名和第一次签名不一致导致的

这时我们需要看WxPayPubHelper.php这个文件里的JsApi_pub这个类,这个类中的getParameters这个方法是进行第二次签名的关键,代码如下:

	/**
	 * 	作用:设置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;
		}
		//签名步骤一:按字典序排序参数
		ksort($Parameters);
		$String = $this->formatBizQueryParaMap($Parameters, false);
		//echo '【string1】'.$String.'</br>';
		//签名步骤二:在string后加入KEY
		if ($key!='') {
			$String = $String."&key=".$key;
		}else{
			$String = $String."&key=".WxPayConf_pub::KEY;
		}
		//echo "【string2】".$String."</br>";
		//签名步骤三:MD5加密
		$String = md5($String);
		//echo "【string3】 ".$String."</br>";
		//签名步骤四:所有字符转为大写
		$result_ = strtoupper($String);
		//echo "【result】 ".$result_."</br>";
		return $result_;
	}

可以看到,我只是多加了一个参数,加了一个判断,因为我不想过多的破坏原来的代码,所以这样加了。

在这样的改造完成后,我遇到的问题都解决了。希望对正在看的你能有所帮助,如果还有问题可以给我留言!


  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
/** * 微信支付帮助库 * ==================================================== * 接口分三种类型: * 【请求型接口】--Wxpay_client_ * 统一支付接口类--UnifiedOrder * 订单查询接口--OrderQuery * 退款申请接口--Refund * 退款查询接口--RefundQuery * 对账单接口--DownloadBill * 短链接转换接口--ShortUrl * 【响应型接口】--Wxpay_server_ * 通用通知接口--Notify * Native支付——请求商家获取商品信息接口--NativeCall * 【其他】 * 静态链接二维码--NativeLink * JSAPI支付--JsApi * ===================================================== * 【CommonUtil】常用工具: * trimString(),设置参数需要用到的字符处理函数 * createNoncestr(),产生随机字符串,不长于32位 * formatBizQueryParaMap(),格式化参数,签名过程需要用到 * getSign(),生成签名 * arrayToXml(),array转xml * xmlToArray(),xml转 array * postXmlCurl(),以post方式提交xml到对应的接口url * postXmlSSLCurl(),使用证书,以post方式提交xml到对应的接口url */ include_once("SDKRuntimeException.php"); include_once("WxPay.pub.config.php"); /** * 所有接口的基类 */ class Common_util_pub { function __construct() { } function trimString($value) { $ret = null; if (null != $value) { $ret = $value; if (strlen($ret) == 0) { $ret = null; } } return $ret; } /** * 作用:产生随机字符串,不长于32位 */ public function createNoncestr( $length = 32 ) { $chars = "abcdefghijklmnopqrstuvwxyz0123456789"; $str =""; for ( $i = 0; $i $v) { if($urlencode) { $v = urlencode($v); } //$buff .= strtolower($k) . "=" . $v . "&"; $buff .= $k . "=" . $v . "&"; } $reqPar; if (strlen($buff) > 0) { $reqPar = substr($buff, 0, strlen($buff)-1); } return $reqPar; } /** * 作用:生成签名 */ public function getSign($Obj) { foreach ($Obj as $k => $v) { $Parameters[$k] = $v; } //签名步骤一:按字典序排序参数 ksort($Parameters); $String = $this->formatBizQueryParaMap($Parameters, false)

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值