WechatAuthenticatedService.php 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. <?php
  2. /**
  3. * @Name
  4. * @Description
  5. * @Author 刘学玺
  6. * @Date 2024/3/25 17:00
  7. */
  8. namespace App\Http\Services\Frontend\Client\Auth;
  9. use App\Exceptions\ApiException;
  10. use App\Http\Services\Frontend\Client\Common\AuthService;
  11. use App\Http\Services\Frontend\Client\Common\SmsService;
  12. use App\Http\Services\Service;
  13. use App\Models\Member\User;
  14. use App\Models\WechatUser;
  15. use EasyWeChat\Kernel\Exceptions\InvalidArgumentException;
  16. use Illuminate\Http\JsonResponse;
  17. use Illuminate\Http\Response;
  18. use Illuminate\Support\Facades\Cache;
  19. use Illuminate\Support\Facades\DB;
  20. use Illuminate\Support\Str;
  21. use Overtrue\LaravelWeChat\EasyWeChat;
  22. class WechatAuthenticatedService extends Service
  23. {
  24. /**
  25. * @throws InvalidArgumentException
  26. */
  27. public function oauth($data): array
  28. {
  29. $wechat = EasyWeChat::officialAccount();
  30. $oauth = $wechat->getOAuth();
  31. $scopes = $data['scopes'] ?: 'snsapi_base';
  32. //callback_url 是授权回调的URL
  33. //生成完整的授权URL
  34. $redirectUrl = $oauth->scopes([$scopes])->redirect($data['redirect_url']);
  35. return ['redirectUrl' => $redirectUrl];
  36. }
  37. /**
  38. * @throws InvalidArgumentException
  39. */
  40. public function callback(array $params): array
  41. {
  42. $wechat = EasyWeChat::officialAccount();
  43. $oauth = $wechat->getOAuth();
  44. // 获取 OAuth 授权用户信息
  45. $wechat_user = $oauth->userFromCode($params['code']);
  46. //$wechat_user = $user->toArray();
  47. $data = [
  48. 'openid' => $wechat_user->getId() ?? $wechat_user->getTokenResponse()['openid'],
  49. 'bindUser' => true
  50. ];
  51. $avatar = $wechat_user->getAvatar();
  52. $avatar && ($data['avatar'] = $avatar);
  53. $nickname = $wechat_user->getNickname();
  54. $nickname && ($data['nickname'] = $nickname);
  55. $userIsExists = User::query()->where('openid', $data['openid'])->exists();
  56. if (!$userIsExists) {
  57. // $user_id = User::query()->create($data)->id;
  58. // $user = User::query()->find($user_id);
  59. $data['bindUser'] = false;
  60. }
  61. // $isExits = $this->hasOpenID($data['openid']);
  62. // 是否已存在
  63. // !$isExits && $this->create($data);
  64. // $wechatUser = WechatUser::query()->with('user.info')->where('openid', $data['openid'])->first();
  65. // return $this->success('', ['wechatUser' => $wechatUser->toArray()]);
  66. return $data;
  67. }
  68. private function hasOpenID($openId): bool
  69. {
  70. return false;
  71. // return WechatUser::query()->where('openid', $openId)->exists();
  72. }
  73. /**
  74. * @throws ApiException
  75. */
  76. public function bind(array $params): void
  77. {
  78. $mobile = $params['mobile'];
  79. $code = $params['code'];
  80. $openid = $params['open_id'];
  81. // 验证验证码
  82. $verifyCodeResult = (new SmsService())->verifyCode($mobile, $code);
  83. !$verifyCodeResult && self::error('验证码错误!', 400);
  84. // 获取微信信息
  85. // $data = [
  86. // 'openid' => $params['openid'],
  87. // 'avatar' => $params['avatar'],
  88. // 'nickname' => $params['nickname'],
  89. // ];
  90. $register_ip = request()->getClientIp();
  91. // 获取绑定用户
  92. User::query()->updateOrCreate(['openid' => $openid], ['openid' => $openid, 'mobile' => $mobile, 'register_ip' => $register_ip]);
  93. }
  94. /**
  95. * @throws ApiException
  96. */
  97. public function store($data): array
  98. {
  99. // 公众号OpenID
  100. $openID = $data['open_id'];
  101. $autoRegister = $data['auto_register'] ?? 0;
  102. $user = User::query()->where('openid', $openID)->first();
  103. if (!$user && $autoRegister) {
  104. // 获取微信信息
  105. // 创建用户信息
  106. $user_id = User::query()->create(['openid' => $openID])->id;
  107. $user = User::query()->find($user_id);
  108. }
  109. $token = (new AuthService)->store($user);
  110. return ['token' => $token];
  111. }
  112. public function create1($data): Response
  113. {
  114. // 判断验证码
  115. (new SmsService())->verify($data['mobile'], $data['code']);
  116. $where = ['mobile' => $data['mobile'], 'type' => 0];
  117. $exists = User::query()->where($where)->exists();
  118. if (!$exists) User::query()->create($where);
  119. // 绑定公众号
  120. $user = User::query()->where($where)->first();
  121. $user->open_id = $data['openid'];
  122. $user->save();
  123. return (new AuthService)->store($user);
  124. }
  125. // 发送短信
  126. public function send($data): Response
  127. {
  128. $mobile = $data['mobile'];
  129. $sms = new SmsService();
  130. return $sms->send($mobile, 0, 2);
  131. }
  132. /**
  133. * 公众号签名
  134. * @throws ApiException
  135. */
  136. public function getSignature(array $data): array
  137. {
  138. $wechat = EasyWeChat::officialAccount();
  139. $ticket = $wechat->getTicket();
  140. $jsapi_ticket = $ticket->getTicket();
  141. // 当前URL
  142. $url = request('url',request()->header('referer'));
  143. // 生成签名
  144. $nonceStr = 'xiaoding'.Str::random(20); // 请用随机算法生成
  145. $timestamp = time(); // 当前时间戳
  146. $signatureStr = "jsapi_ticket={$jsapi_ticket}&noncestr={$nonceStr}&timestamp={$timestamp}&url={$url}";
  147. $signature = sha1($signatureStr);
  148. $wechatConfig = $wechat->getConfig();
  149. // 数据用于前端显示
  150. $signPackage = [
  151. "appId" => $wechatConfig['app_id'],
  152. "nonceStr" => $nonceStr,
  153. "timestamp" => $timestamp,
  154. "url" => $url,
  155. "signature" => $signature,
  156. "rawString" => $signatureStr
  157. ];
  158. // $sign = $this->makeSha1Sign($signData); // 生成微信签名
  159. //
  160. // $signData['signature'] = $sign;
  161. return $signPackage;
  162. //
  163. //
  164. // $access_token = $this->getAccessToken();
  165. //
  166. // // 这里的$token就是你在微信公众平台设置的Token
  167. // // 这里的$timestamp就是时间戳
  168. // // 这里的$nonce就是随机数
  169. // // 这里的$msg就是发送给微信服务器的数据
  170. //
  171. // // 将token、timestamp、nonce三个参数按字典序排序
  172. // $array = array($token, $timestamp, $nonce, $msg);
  173. // sort($array, SORT_STRING);
  174. //
  175. // // 将排序后的三个参数拼接成一个字符串
  176. // $str = implode($array);
  177. //
  178. // // 使用sha1算法计算出摘要
  179. // $signature = sha1($str);
  180. //
  181. // // 转换成小写
  182. // $signature = strtolower($signature);
  183. //
  184. // return $signature;
  185. }
  186. //生成 sha1 签名
  187. private function makeSha1Sign($array): string
  188. {
  189. //升序数组的键
  190. ksort($array);
  191. $params = urldecode(http_build_query($array));
  192. //字符串SHA1
  193. return sha1($params);
  194. }
  195. //获取微信access_token
  196. /**
  197. * @throws ApiException
  198. */
  199. private function getAccessToken()
  200. {
  201. $tokeUrl = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=" . config('easywechat.official_account.default.app_id') . "&secret=" . config('easywechat.official_account.default.secret');
  202. $tokenResp = file_get_contents($tokeUrl);
  203. !$tokenResp && self::error('token 服务器返回失败');
  204. $tokenData = json_decode($tokenResp, true);
  205. if (!isset($tokenData['access_token'])) self::error($tokenData['errmsg']);
  206. return $tokenData['access_token'];
  207. }
  208. //获取微信ticket
  209. /**
  210. * @throws ApiException
  211. */
  212. private function getTicket($access_token)
  213. {
  214. $ticketUrl = 'https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token=' . $access_token . '&type=jsapi';
  215. $ticketResp = file_get_contents($ticketUrl);
  216. if (!$ticketResp) return self::error('ticket 获取失败');
  217. $ticketData = json_decode($ticketResp, true);
  218. if (!isset($ticketData['ticket'])) return self::error('ticket 解析错误');
  219. $ticket = $ticketData['ticket'];
  220. cache::set('wechart_share_ticket', $ticket, 7200);
  221. return $ticket;
  222. }
  223. }