WechatService.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkAdmin
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2014~2021 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: https://thinkadmin.top
  8. // +----------------------------------------------------------------------
  9. // | 开源协议 ( https://mit-license.org )
  10. // +----------------------------------------------------------------------
  11. // | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
  12. // | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
  13. // +----------------------------------------------------------------------
  14. namespace app\wechat\service;
  15. use think\admin\extend\JsonRpcClient;
  16. use think\admin\Service;
  17. use think\admin\storage\LocalStorage;
  18. use think\exception\HttpResponseException;
  19. /**
  20. * Class WechatService
  21. * @package app\wechat\serivce
  22. *
  23. * @method \WeChat\Card WeChatCard() static 微信卡券管理
  24. * @method \WeChat\Custom WeChatCustom() static 微信客服消息
  25. * @method \WeChat\Limit WeChatLimit() static 接口调用频次限制
  26. * @method \WeChat\Media WeChatMedia() static 微信素材管理
  27. * @method \WeChat\Menu WeChatMenu() static 微信菜单管理
  28. * @method \WeChat\Oauth WeChatOauth() static 微信网页授权
  29. * @method \WeChat\Pay WeChatPay() static 微信支付商户
  30. * @method \WeChat\Product WeChatProduct() static 微信商店管理
  31. * @method \WeChat\Qrcode WeChatQrcode() static 微信二维码管理
  32. * @method \WeChat\Receive WeChatReceive() static 微信推送管理
  33. * @method \WeChat\Scan WeChatScan() static 微信扫一扫接入管理
  34. * @method \WeChat\Script WeChatScript() static 微信前端支持
  35. * @method \WeChat\Shake WeChatShake() static 微信揺一揺周边
  36. * @method \WeChat\Tags WeChatTags() static 微信用户标签管理
  37. * @method \WeChat\Template WeChatTemplate() static 微信模板消息
  38. * @method \WeChat\User WeChatUser() static 微信粉丝管理
  39. * @method \WeChat\Wifi WeChatWifi() static 微信门店WIFI管理
  40. *
  41. * ----- WeMini -----
  42. * @method \WeMini\Account WeMiniAccount() static 小程序账号管理
  43. * @method \WeMini\Basic WeMiniBasic() static 小程序基础信息设置
  44. * @method \WeMini\Code WeMiniCode() static 小程序代码管理
  45. * @method \WeMini\Domain WeMiniDomain() static 小程序域名管理
  46. * @method \WeMini\Tester WeMinitester() static 小程序成员管理
  47. * @method \WeMini\User WeMiniUser() static 小程序帐号管理
  48. * --------------------
  49. * @method \WeMini\Crypt WeMiniCrypt() static 小程序数据加密处理
  50. * @method \WeMini\Delivery WeMiniDelivery() static 小程序即时配送
  51. * @method \WeMini\Image WeMiniImage() static 小程序图像处理
  52. * @method \WeMini\Logistics WeMiniLogistics() static 小程序物流助手
  53. * @method \WeMini\Message WeMiniMessage() static 小程序动态消息
  54. * @method \WeMini\Ocr WeMiniOcr() static 小程序ORC服务
  55. * @method \WeMini\Plugs WeMiniPlugs() static 小程序插件管理
  56. * @method \WeMini\Poi WeMiniPoi() static 小程序地址管理
  57. * @method \WeMini\Qrcode WeMiniQrcode() static 小程序二维码管理
  58. * @method \WeMini\Security WeMiniSecurity() static 小程序内容安全
  59. * @method \WeMini\Soter WeMiniSoter() static 小程序生物认证
  60. * @method \WeMini\Template WeMiniTemplate() static 小程序模板消息支持
  61. * @method \WeMini\Total WeMiniTotal() static 小程序数据接口
  62. * @method \WeMini\Newtmpl WeMiniNewtmpl() static 小程序订阅消息支持
  63. *
  64. * ----- WePay -----
  65. * @method \WePay\Bill WePayBill() static 微信商户账单及评论
  66. * @method \WePay\Order WePayOrder() static 微信商户订单
  67. * @method \WePay\Refund WePayRefund() static 微信商户退款
  68. * @method \WePay\Coupon WePayCoupon() static 微信商户代金券
  69. * @method \WePay\Redpack WePayRedpack() static 微信红包支持
  70. * @method \WePay\Transfers WePayTransfers() static 微信商户打款到零钱
  71. * @method \WePay\TransfersBank WePayTransfersBank() static 微信商户打款到银行卡
  72. *
  73. * ----- WeOpen -----
  74. * @method \WeOpen\Login WeOpenLogin() static 第三方微信登录
  75. * @method \WeOpen\Service WeOpenService() static 第三方服务
  76. *
  77. * ----- ThinkService -----
  78. * @method mixed ThinkServiceConfig() static 平台服务配置
  79. */
  80. class WechatService extends Service
  81. {
  82. /**
  83. * 静态初始化对象
  84. * @param string $name
  85. * @param array $arguments
  86. * @return mixed
  87. * @throws \think\Exception
  88. * @throws \think\admin\Exception
  89. * @throws \think\db\exception\DataNotFoundException
  90. * @throws \think\db\exception\DbException
  91. * @throws \think\db\exception\ModelNotFoundException
  92. */
  93. public static function __callStatic(string $name, array $arguments)
  94. {
  95. [$type, $class, $classname] = static::parseName($name);
  96. if ("{$type}{$class}" !== $name) {
  97. throw new \think\Exception("抱歉,实例 {$name} 不在符合规则!");
  98. }
  99. if (sysconf('wechat.type') === 'api' || $type === 'WePay') {
  100. if ($type === 'ThinkService') {
  101. throw new \think\Exception("抱歉,接口模式不能实例 {$classname} 对象!");
  102. }
  103. return new $classname(static::instance()->getConfig());
  104. } else {
  105. [$appid, $appkey] = [sysconf('wechat.thr_appid'), sysconf('wechat.thr_appkey')];
  106. $data = ['class' => $name, 'appid' => $appid, 'time' => time(), 'nostr' => uniqid()];
  107. $data['sign'] = md5("{$data['class']}#{$appid}#{$appkey}#{$data['time']}#{$data['nostr']}");
  108. $token = enbase64url(json_encode($data, JSON_UNESCAPED_UNICODE));
  109. $location = "https://open.cuci.cc/service/api.client/_TYPE_?not_init_session=1&token={$token}";
  110. if (class_exists('Yar_Client')) {
  111. $client = new \Yar_Client(str_replace('_TYPE_', 'yar', $location));
  112. } else {
  113. $client = new JsonRpcClient(str_replace('_TYPE_', 'jsonrpc', $location));
  114. }
  115. try {
  116. $exception = new \think\Exception($client->getMessage(), $client->getCode());
  117. } catch (\Exception $exception) {
  118. $exception = null;
  119. }
  120. if ($exception instanceof \Exception) {
  121. throw $exception;
  122. }
  123. return $client;
  124. }
  125. }
  126. /**
  127. * 解析调用对象名称
  128. * @param string $name
  129. * @return array
  130. */
  131. private static function parseName(string $name): array
  132. {
  133. foreach (['WeChat', 'WeMini', 'WeOpen', 'WePay', 'ThinkService'] as $type) {
  134. if (strpos($name, $type) === 0) {
  135. [, $class] = explode($type, $name);
  136. return [$type, $class, "\\{$type}\\{$class}"];
  137. }
  138. }
  139. return ['-', '-', $name];
  140. }
  141. /**
  142. * 获取当前微信APPID
  143. * @return string
  144. * @throws \think\Exception
  145. * @throws \think\db\exception\DataNotFoundException
  146. * @throws \think\db\exception\DbException
  147. * @throws \think\db\exception\ModelNotFoundException
  148. */
  149. public function getAppid(): string
  150. {
  151. if ($this->getType() === 'api') {
  152. return sysconf('wechat.appid');
  153. } else {
  154. return sysconf('wechat.thr_appid');
  155. }
  156. }
  157. /**
  158. * 获取接口授权模式
  159. * @return string
  160. * @throws \think\Exception
  161. * @throws \think\db\exception\DataNotFoundException
  162. * @throws \think\db\exception\DbException
  163. * @throws \think\db\exception\ModelNotFoundException
  164. */
  165. public function getType(): string
  166. {
  167. $type = strtolower(sysconf('wechat.type'));
  168. if (in_array($type, ['api', 'thr'])) return $type;
  169. throw new \think\Exception('请在后台配置微信对接授权模式');
  170. }
  171. /**
  172. * 获取公众号配置参数
  173. * @return array
  174. * @throws \think\Exception
  175. * @throws \think\admin\Exception
  176. * @throws \think\db\exception\DataNotFoundException
  177. * @throws \think\db\exception\DbException
  178. * @throws \think\db\exception\ModelNotFoundException
  179. */
  180. public function getConfig(): array
  181. {
  182. $options = [
  183. 'appid' => $this->getAppid(),
  184. 'token' => sysconf('wechat.token'),
  185. 'appsecret' => sysconf('wechat.appsecret'),
  186. 'encodingaeskey' => sysconf('wechat.encodingaeskey'),
  187. 'mch_id' => sysconf('wechat.mch_id'),
  188. 'mch_key' => sysconf('wechat.mch_key'),
  189. 'cache_path' => $this->app->getRuntimePath() . 'wechat',
  190. ];
  191. $local = LocalStorage::instance();
  192. switch (strtolower(sysconf('wechat.mch_ssl_type'))) {
  193. case 'p12':
  194. $options['ssl_p12'] = $local->path(sysconf('wechat.mch_ssl_p12'), true);
  195. break;
  196. case 'pem':
  197. $options['ssl_key'] = $local->path(sysconf('wechat.mch_ssl_key'), true);
  198. $options['ssl_cer'] = $local->path(sysconf('wechat.mch_ssl_cer'), true);
  199. break;
  200. }
  201. return $options;
  202. }
  203. /**
  204. * 通过网页授权获取粉丝信息
  205. * @param string $source 回跳URL地址
  206. * @param integer $isfull 获取资料模式
  207. * @param boolean $redirect 是否直接跳转
  208. * @return array
  209. * @throws \WeChat\Exceptions\InvalidResponseException
  210. * @throws \WeChat\Exceptions\LocalCacheException
  211. * @throws \think\Exception
  212. * @throws \think\db\exception\DataNotFoundException
  213. * @throws \think\db\exception\DbException
  214. * @throws \think\db\exception\ModelNotFoundException
  215. */
  216. public function getWebOauthInfo(string $source, $isfull = 0, $redirect = true): array
  217. {
  218. $appid = $this->getAppid();
  219. $openid = $this->app->session->get("{$appid}_openid");
  220. $userinfo = $this->app->session->get("{$appid}_fansinfo");
  221. if ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($userinfo))) {
  222. empty($userinfo) || FansService::instance()->set($userinfo, $appid);
  223. return ['openid' => $openid, 'fansinfo' => $userinfo];
  224. }
  225. if ($this->getType() === 'api') {
  226. // 解析 GET 参数
  227. parse_str(parse_url($source, PHP_URL_QUERY), $params);
  228. $getVars = [
  229. 'code' => $params['code'] ?? input('code', ''),
  230. 'rcode' => $params['rcode'] ?? input('rcode', ''),
  231. 'state' => $params['state'] ?? input('state', ''),
  232. ];
  233. $wechat = static::WeChatOauth();
  234. if ($getVars['state'] !== $appid || empty($getVars['code'])) {
  235. $params['rcode'] = enbase64url($source);
  236. $location = strstr("{$source}?", '?', true) . '?' . http_build_query($params);
  237. $oauthurl = $wechat->getOauthRedirect($location, $appid, $isfull ? 'snsapi_userinfo' : 'snsapi_base');
  238. throw new HttpResponseException($redirect ? redirect($oauthurl, 301) : response("location.href='{$oauthurl}'"));
  239. } elseif (($token = $wechat->getOauthAccessToken($getVars['code'])) && isset($token['openid'])) {
  240. $this->app->session->set("{$appid}_openid", $openid = $token['openid']);
  241. if ($isfull && isset($token['access_token'])) {
  242. $userinfo = $wechat->getUserInfo($token['access_token'], $openid);
  243. $this->app->session->set("{$appid}_fansinfo", $userinfo);
  244. empty($userinfo) || FansService::instance()->set($userinfo, $appid);
  245. }
  246. }
  247. if ($getVars['rcode']) {
  248. $location = debase64url($getVars['rcode']);
  249. throw new HttpResponseException($redirect ? redirect($location, 301) : response("location.href='{$location}'"));
  250. } elseif ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($userinfo))) {
  251. return ['openid' => $openid, 'fansinfo' => $userinfo];
  252. } else {
  253. throw new \think\Exception('Query params [rcode] not find.');
  254. }
  255. } else {
  256. $result = static::ThinkServiceConfig()->oauth($this->app->session->getId(), $source, $isfull);
  257. $this->app->session->set("{$appid}_openid", $openid = $result['openid']);
  258. $this->app->session->set("{$appid}_fansinfo", $userinfo = $result['fans']);
  259. if ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($userinfo))) {
  260. empty($userinfo) || FansService::instance()->set($userinfo, $appid);
  261. return ['openid' => $openid, 'fansinfo' => $userinfo];
  262. }
  263. if ($redirect) {
  264. throw new HttpResponseException(redirect($result['url'], 301));
  265. } else {
  266. throw new HttpResponseException(response("location.href='{$result['url']}'"));
  267. }
  268. }
  269. }
  270. /**
  271. * 获取微信网页JSSDK签名参数
  272. * @param null|string $location 签名地址
  273. * @return array
  274. * @throws \WeChat\Exceptions\InvalidResponseException
  275. * @throws \WeChat\Exceptions\LocalCacheException
  276. * @throws \think\Exception
  277. * @throws \think\db\exception\DataNotFoundException
  278. * @throws \think\db\exception\DbException
  279. * @throws \think\db\exception\ModelNotFoundException
  280. */
  281. public function getWebJssdkSign(?string $location = null): array
  282. {
  283. $location = $location ?: $this->app->request->url(true);
  284. if ($this->getType() === 'api') {
  285. return static::WeChatScript()->getJsSign($location);
  286. } else {
  287. return static::ThinkServiceConfig()->jsSign($location);
  288. }
  289. }
  290. }