微信授权获取用户信息时code回调了两
通过调用微信授权,首先要获取 code
,微信将携带 code
回调至开发者指定的地址。但有部分安卓机型(截止目前发现机型有:小米)会提示 errcode:40163, errmsg: code been used
错误。
Nginx 请求日志
1 2 3 4 5 6 7 8
| 10.10.10.10 - - [23/Jul/2021:17:32:37 +0800] - - - - - "GET /wap/index HTTP/1.1" 10.10.10.10 - - [23/Jul/2021:17:32:37 +0800] - - - - - "GET /wap/index?code=021Svm1w3V3BNW22W51w39nFIa1Svm2F&state=1 HTTP/1.1"
10.10.10.10 - - [23/Jul/2021:17:34:24 +0800] - - - 117.136.78.68 - "GET /wap/index HTTP/1.1" 10.10.10.10 - - [23/Jul/2021:17:34:25 +0800] - - - 117.136.78.68 - "GET /wap/index?code=0512dXkl2E9Or74M1yml2DwBNK42dXkV&state=1 HTTP/1.1" 10.10.10.10 - - [23/Jul/2021:17:34:25 +0800] - - - - - "GET /wap/index?code=0512dXkl2E9Or74M1yml2DwBNK42dXkV&state=1 HTTP/1.1"
|
代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33
| $sign = $this->getg('s', 0); $code = $this->getg('code', ''); $state = $this->getg('state', ''); file_put_contents('/tmp/sign-' . $sign . '.txt', date('Y-m-d H:i:s', time()) . '-' . $state . "-" . $code . PHP_EOL, FILE_APPEND); if(empty($code)) { $state = md5(uniqid()); $redirect_uri = urlencode('http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI']); if ($sign == 2 || $sign == 3) { $redirect_uri = urlencode('https://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI']); } $url ="https://open.weixin.qq.com/connect/oauth2/authorize?appid={$this->AppId}&redirect_uri={$redirect_uri}&response_type=code&scope=snsapi_userinfo&state={$state}#wechat_redirect"; if ($sign == 1 || $sign == 2) { Header("Location:" . $url); exit; } echo "<div style='width:100%;height:500px;line-height:500px;margin:0 auto;'><a href='${url}'>点击跳转</a></div>"; exit; }
$oauth2Url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid={$this->AppId}&secret={$this->AppSecret}&code={$code}&grant_type=authorization_code"; $oauth2 = $this->curlGet($oauth2Url); $oauth2 = json_decode($oauth2, true); echo '<pre>'; var_dump($oauth2); echo '</pre>';
$access_token = $oauth2["access_token"]; $openid = $oauth2['openid']; $get_user_info_url = "https://api.weixin.qq.com/sns/userinfo?access_token={$access_token}&openid={$openid}&lang=zh_CN"; $userInfo = $this->curlGet($get_user_info_url); $userInfo = json_decode($userInfo, true); echo '<pre>'; var_dump($userInfo); echo '</pre>'; exit;
|
测试
安卓手机挂代理测试可正常获取用户微信信息。
无代理,使用 https 回调 code
也能正常获取到用户微信信息。
无代理,使用 <a href="跳转微信获取code链接地址">跳转</a>
方式,点击跳转还是会出现两次回调 code
。
此方式可通过数据线将手机与电脑链接,使用 chrome://inspect
测试发现,只请求了一次 code
回调,证明不是微信浏览器客户端做的两次回调 code
,很有可能是微信客户端的代理IP做的了。
解决方案
- 继续使用 http 请求,不去理会
code
请求两次的问题,使用 cookie
、session
会话保持,保存第一次 code
获取的用户微信信息,第二次请求时先判断 cookie
、session
中是否存在用户微信信息,若存在则直接进行业务处理,不再走微信授权逻辑。
- 使用
https
回调 code
,可保证只回调一次 code
。