Wechat.php 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. <?php
  2. namespace app\index\controller;
  3. use app\common\controller\Frontend;
  4. use app\common\library\Sms;
  5. use fast\Random;
  6. use think\exception\HttpResponseException;
  7. use think\Request;
  8. use think\Response;
  9. class Wechat extends Frontend
  10. {
  11. protected $noNeedLogin = '*';
  12. private $wechat = null;
  13. private $_error = '';
  14. private $has_mobile = false;
  15. private $is_notice = false;
  16. public function __construct(Request $request = null)
  17. {
  18. parent::__construct($request);
  19. $this->wechat = new \app\common\controller\Wechat();
  20. }
  21. /**
  22. * 在公网接口处重写此方法:验证服务器有效性
  23. * @return string
  24. */
  25. public function serverValidation()
  26. {
  27. $TOKEN = 'I8cezsHeF1buiCBPwD';
  28. $signature = $_GET["signature"] ?? "";
  29. $timestamp = $_GET["timestamp"] ?? "";
  30. $nonce = $_GET["nonce"] ?? "";
  31. $tmpArr = array($TOKEN, $timestamp, $nonce);
  32. sort($tmpArr, SORT_STRING);
  33. $tmpStr = implode($tmpArr);
  34. $tmpStr = sha1($tmpStr);
  35. if ($tmpStr == $signature) {
  36. return $_GET["echostr"] ?? '';
  37. }
  38. return 'error';
  39. }
  40. /**
  41. * 在公网接口处重写此方法:引导用户进入授权页,并跳转到目标页
  42. * @param string $url 授权后需要跳转的页面地址,必须进行base64编码
  43. * @author fuyelk <fuyelk@fuyelk.com>
  44. */
  45. public function bootToUrl($url = '')
  46. {
  47. $redirect_uri = request()->domain() . '/index/wechat/wechatRedirect?redirect=' . $url;
  48. $appid = $this->wechat->appid;
  49. $redirect_uri = urlencode($redirect_uri);
  50. // $scope = 'snsapi_base'; // 静默授权,只能获取openid
  51. $scope = 'snsapi_userinfo'; // 授权弹窗,可获取用户昵称头像
  52. $wechat_authorize = "http://2.ipfsfil168.com/get-weixin-code.html?appid={$appid}&redirect_uri={$redirect_uri}&response_type=code&scope={$scope}&state=123#wechat_redirect";
  53. $this->redirect($wechat_authorize);
  54. }
  55. /**
  56. * 在公网接口处重写此方法:用户授权后微信将用户重定向至此接口并携带code,
  57. * 使用code可获取access_token及用户openid,完成业务逻辑处理后拼接上openid将用户重定向指指定页面
  58. */
  59. public function wechatRedirect()
  60. {
  61. $code = input('code');
  62. $redirect = input('redirect');
  63. // 获取用户信息
  64. if ($code) {
  65. $wechat_url = 'https://api.weixin.qq.com/sns/oauth2/access_token' .
  66. '?appid=' . $this->wechat->appid .
  67. '&secret=' . $this->wechat->appsecret .
  68. '&code=' . input('code') .
  69. '&grant_type=authorization_code';
  70. try {
  71. $res = $this->wechat->http_request($wechat_url);
  72. } catch (\Exception $e) {
  73. $redirect = $redirect . (strpos($redirect, '?') ? '&' : '?') . 'status=0&token=&errmsg=' . $e->getMessage();
  74. $this->redirect($redirect ? urldecode($redirect) : '/h5/#/');
  75. exit();
  76. }
  77. $res = json_decode($res, true);
  78. if (empty($res['openid'])) {
  79. $redirect = $redirect . (strpos($redirect, '?') ? '&' : '?') . 'status=0&token=&errmsg=登录失败';
  80. $this->redirect($redirect ? urldecode($redirect) : '/h5/#/');
  81. exit();
  82. }
  83. $redirect = base64_decode($redirect);
  84. $redirect = htmlspecialchars_decode($redirect);
  85. $query = parse_url(strstr($redirect, '?'), PHP_URL_QUERY);
  86. parse_str($query, $params);
  87. // 获取用户信息
  88. if ('snsapi_userinfo' == $res['scope']) {
  89. $userinfo = file_get_contents("https://api.weixin.qq.com/sns/userinfo?access_token={$res['access_token']}&openid={$res['openid']}&lang=zh_CN");
  90. $userinfo = json_decode($userinfo, true);
  91. if (!empty($userinfo['errcode']) && !empty($userinfo['errmsg'])) {
  92. $redirect = $redirect . (strpos($redirect, '?') ? '&' : '?') . 'status=0&token=&errmsg=' . $userinfo['errmsg'];
  93. $this->redirect($redirect ? urldecode($redirect) : '/h5/#/');
  94. exit();
  95. }
  96. // [openid] => ou6NC6kBSloLS94gUhCyKeguIaYI
  97. // [nickname] => fuyelk
  98. // [sex] => 1 // 1=男,2=女,0=未知
  99. // [language] => zh_CN
  100. // [city] => 郑州
  101. // [province] => 河南
  102. // [country] => 中国
  103. // [headimgurl] => https://thirdwx.qlogo.cn/mmopen/vi_32/Q0j4TwGTfTKY3E1A2MFK46y1fNv7OElhtQFefO4mrBy8kcQoFCHP9diaULlpeINYB1FJ7KNh4fx5v3rzAJ2LzFg/132
  104. $invite_code = $params['sharecode'] ?? '';
  105. $is_channel = $params['channel'] ?? '';
  106. $token = $this->login($userinfo['openid'], $userinfo['nickname'], $userinfo['headimgurl'], $invite_code, $is_channel);
  107. if (false === $token) {
  108. $errmsg = $this->_error ?: '登录失败';
  109. $redirect = $redirect . (strpos($redirect, '?') ? '&' : '?') . 'status=0&token=&errmsg=' . $errmsg;
  110. $this->redirect($redirect ? urldecode($redirect) : '/h5/#/');
  111. exit();
  112. }
  113. } else {
  114. $errmsg = $this->_error ?: '仅支持snsapi_userinfo';
  115. $redirect = $redirect . (strpos($redirect, '?') ? '&' : '?') . 'status=0&token=&errmsg=' . $errmsg;
  116. $this->redirect($redirect ? urldecode($redirect) : '/h5/#/');
  117. exit();
  118. }
  119. // 绑定过手机,直接去首页
  120. if ($this->has_mobile) {
  121. $this->redirect('/h5/#/?status=1&errmsg=&token=' . $token . "&is_notice=". $this->is_notice);
  122. }
  123. $redirect = $redirect . (strpos($redirect, '?') ? '&' : '?') . 'status=1&errmsg=&token=' . $token . "&is_notice=". $this->is_notice;
  124. $this->redirect($redirect ? urldecode($redirect) : '/h5/#/');
  125. }
  126. $errmsg = $this->_error ?: '授权失败';
  127. $redirect = $redirect . (strpos($redirect, '?') ? '&' : '?') . 'status=0&token=&errmsg=' . $errmsg;
  128. $this->redirect($redirect ? urldecode($redirect) : '/h5/#/');
  129. }
  130. /**
  131. * 登录
  132. * @param string $openid 登录
  133. * @param string $invite_code
  134. * @return bool|mixed
  135. * @author fuyelk <fuyelk@fuyelk.com>
  136. */
  137. private function login($openid, $nickname = '', $avatar = '', $invite_code = '', $is_channel ='')
  138. {
  139. $user = \app\common\model\User::getByWechatOpenid($openid);
  140. if ($user) {
  141. if ($user->status != 'normal') {
  142. $this->_error = '账号被锁定';
  143. return false;
  144. }
  145. //如果已经有账号则直接登录
  146. $ret = $this->auth->direct($user->id);
  147. } else {
  148. $extend = [
  149. 'invite_code' => $invite_code,
  150. 'wechat_openid' => $openid,
  151. 'nickname' => $nickname,
  152. 'avatar' => $avatar,
  153. ];
  154. $ret = $this->auth->register('', Random::alnum(), '', '', $extend);
  155. if($ret && $is_channel == '49ba59abbe56e057'){ //赠送10金币
  156. \app\common\model\User::where(['id'=>$this->auth->id])->setInc("coin", 10);
  157. $this->is_notice = 1;
  158. }
  159. }
  160. if ($ret) {
  161. $data = $this->auth->getUserinfo();
  162. $this->has_mobile = !empty($data['mobile']) ? true : false;
  163. return $data['token'];
  164. }
  165. $this->_error = $this->auth->getError();
  166. return false;
  167. }
  168. /**
  169. * 微信登录并重定向
  170. * @author fuyelk <fuyelk@fuyelk.com>
  171. */
  172. public function wechatLoginRedirect()
  173. {
  174. $redirect = input('redirect');
  175. if (empty('')) {
  176. throw new HttpResponseException(Response::create(['code' => 403, 'msg' => '重定向地址为空', 'data' => null], 'json', 200));
  177. }
  178. $this->redirect('/index/wechat/bootToUrl?url=' . base64_encode($redirect));
  179. }
  180. }