自从学习微信开发就一直遇到大大小小的bug,每次的问题都是绞尽脑汁啊。
最近测试刚完成的网页授权获取微信用户信息功能突然出现了很多人都关注的错误,而且网上都没有得到解决的错误?
现在终于得到完全解决,给走在路上或正在路上出现问题的伙伴一个走捷径的方法。
问题描述:网页授权获取微信用户信息 错误40029:不合法的oauth_code?
在第一步:用户同意授权获取code,通过code获取access_token时,有时会出现40029错误。
经过调试会发现问题出现在redirect_uri=REDIRECT_URI当跳转到授权链接后,微信会发出两次转向至redirect_uri的相同请求(两次带进来的code是相同的)。
第一次的code后已经成功换取得openid以及access_token;
第二次转向到redirect_uri时,该code已经失效(code只能使用一次),从而导致了40029:不合法的oauth_code的错误,不能再获取到access_token。
可这种情况只是偶尔发生,过一会儿再进入又正常了 !
======================================================================
解决方案:
$code=$this->input->get("code");//获取code
$dlzt=$this->input->cookie("dlzt"); //重点 ->此cookie用于记录code是否重复提交
if (empty($code)) //不用说 通过empty函数判断code是否设置值,没有从小跳转微信code获取页面或逻辑页面
{
redirect("/wap/hy/index_hy"); //我这里跳转的是逻辑页面,视为code不存在视为不合法访问,因为正常访问不会出现没有code。
exit();
}
if(!empty($dlzt)) //上面说了 记录code是否重复提交,此处就是判断这个$dlzt存在了不执行获取用户信息,因为用户信息已经获取过了。
{
//重要说明:
这里放如果存在二次转向到redirect_uri时,在这里进行第二次跳转的逻辑判断,我此处是进行登录获取用户信息,因为第一次已经登录完成了,登录状态和相关逻辑都已经成功执行一次,第二次我是直接跳入会员中心,就避免了第二次重复登录出现code无限的错误了。
delete_cookie("dlzt");//删除我们设置的$dlzt 这个cookie值,删除好处是,用户重新打开页面或下次访问页面还是第一次访问状态,不会出现因为有值而导致不执行获取微信用户信息出错。
}else
{
//此处是完整的通过code来获取accesstoken 及获取用户资料。
// 通过code 获取 accessToken 和 openid
$res = $this->Weixin_model->getAccessToken($code);//这是我封装的方法你们调用自己的方法。
if(!empty($res['errcode']))//判断是否获取到accesstoken 防止偶尔的错误处理加个判断有好处。
{
redirect('https://open.weixin.qq.com/connect/oauth2/authorize?appid=********&redirect_uri=http://www.baidu.com&response_type=code&scope=snsapi_userinfo&state=STATE#wechat_redirect');
exit();
}
$token=$res['access_token'];
$openid=$res['openid'];
//------ 检测accesstoken是否有效
if (!$this->Weixin_model->checkAccessToken($token, $openid))
{
// (假设token过期需要从小获取自己去判断)
$restk = $this->Weixin_model->refreshToken($res['refresh_token']);
$token=$restk['access_token'];
}
//获取用户信息 (scope为 snsapi_userinfo)
$userInfo = $this->Weixin_model->getUserInfo($token, $openid);
$nickname=$userInfo['nickname'];//昵称
$headimgurl=$userInfo['headimgurl'];//用户头像
$sex=$userInfo['sex'];//性别
$dlzt=$this->Login_model->Openid_login_hy($res['openid'],$nickname,$headimgurl,$sex);
$this->input->set_cookie("dlzt",$dlzt,120);
}