Push.php 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkAdmin
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://demo.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\service\controller\api;
  15. use app\service\service\BuildService;
  16. use app\service\service\WechatService;
  17. use library\Controller;
  18. use think\Db;
  19. /**
  20. * 微信推送事件处理
  21. *
  22. * @author Anyon <zoujingli@qq.com>
  23. * @date 2016/10/18 12:38
  24. */
  25. class Push extends Controller
  26. {
  27. /**
  28. * 微信API推送事件处理
  29. * @param string $appid
  30. * @return string
  31. * @throws \think\Exception
  32. * @throws \WeChat\Exceptions\InvalidDecryptException
  33. * @throws \think\db\exception\DataNotFoundException
  34. * @throws \think\db\exception\ModelNotFoundException
  35. * @throws \think\exception\DbException
  36. * @throws \think\exception\PDOException
  37. */
  38. public function notify($appid)
  39. {
  40. if ($appid === 'wx570bc396a51b8ff8') {
  41. # 全网发布接口测试
  42. return \app\service\handler\PublishHandler::handler($appid);
  43. } else {
  44. # 接口类正常服务
  45. return \app\service\handler\ReceiveHandler::handler($appid);
  46. }
  47. }
  48. /**
  49. * 一、处理服务推送Ticket
  50. * 二、处理取消公众号授权
  51. * @return string
  52. * @throws \think\Exception
  53. * @throws \WeChat\Exceptions\InvalidResponseException
  54. * @throws \WeChat\Exceptions\LocalCacheException
  55. * @throws \think\exception\PDOException
  56. */
  57. public function ticket()
  58. {
  59. try {
  60. $server = WechatService::service();
  61. if (!($data = $server->getComonentTicket())) {
  62. return "Ticket event handling failed.";
  63. }
  64. } catch (\Exception $e) {
  65. return "Ticket event handling failed, {$e->getMessage()}";
  66. }
  67. if (!empty($data['AuthorizerAppid']) && isset($data['InfoType'])) {
  68. # 授权成功通知
  69. if ($data['InfoType'] === 'authorized') {
  70. Db::name('WechatServiceConfig')->where(['authorizer_appid' => $data['AuthorizerAppid']])->update(['is_deleted' => '0']);
  71. }
  72. # 接收取消授权服务事件
  73. if ($data['InfoType'] === 'unauthorized') {
  74. Db::name('WechatServiceConfig')->where(['authorizer_appid' => $data['AuthorizerAppid']])->update(['is_deleted' => '1']);
  75. }
  76. # 授权更新通知
  77. if ($data['InfoType'] === 'updateauthorized') {
  78. $_GET['auth_code'] = $data['PreAuthCode'];
  79. $this->applyAuth($server);
  80. }
  81. }
  82. return 'success';
  83. }
  84. /**
  85. * 网页授权
  86. * @throws \think\Exception
  87. * @throws \WeChat\Exceptions\InvalidResponseException
  88. * @throws \WeChat\Exceptions\LocalCacheException
  89. */
  90. public function oauth()
  91. {
  92. list($mode, $appid, $enurl, $sessid) = [
  93. $this->request->get('mode'), $this->request->get('state'),
  94. $this->request->get('enurl'), $this->request->get('sessid'),
  95. ];
  96. $service = WechatService::service();
  97. $result = $service->getOauthAccessToken($appid);
  98. if (empty($result['openid'])) throw new \think\Exception('网页授权失败, 无法进一步操作!');
  99. cache("{$appid}_{$sessid}_openid", $result['openid'], 3600);
  100. if (!empty($mode)) {
  101. $wechat = new \WeChat\Oauth($service->getConfig($appid));
  102. $fans = $wechat->getUserInfo($result['access_token'], $result['openid']);
  103. if (empty($fans)) throw new \think\Exception('网页授权信息获取失败, 无法进一步操作!');
  104. cache("{$appid}_{$sessid}_fans", $fans, 3600);
  105. }
  106. redirect(decode($enurl), [], 301)->send();
  107. }
  108. /**
  109. * 跳转到微信服务授权页面
  110. * @param string $redirect
  111. * @return string
  112. * @throws \think\Exception
  113. * @throws \WeChat\Exceptions\InvalidResponseException
  114. * @throws \WeChat\Exceptions\LocalCacheException
  115. * @throws \think\exception\PDOException
  116. */
  117. public function auth($redirect = '')
  118. {
  119. $fromRedirect = decode($redirect);
  120. if (empty($redirect) || empty($fromRedirect)) {
  121. return '请传入回跳Redirect参数 ( 请使用ENCODE加密 )';
  122. }
  123. # 预授权码不为空,则表示可以进行授权处理
  124. $service = WechatService::service();
  125. if (($auth_code = $this->request->get('auth_code'))) {
  126. return $this->applyAuth($service, $fromRedirect);
  127. }
  128. # 生成微信授权链接,使用刷新跳转到授权网页
  129. $url = url("@service/api.push/auth/{$redirect}", false, true, true);
  130. if (($redirect = $service->getAuthRedirect($url))) {
  131. ob_clean();
  132. header("Refresh:0;url={$redirect}");
  133. return "<script>window.location.href='{$redirect}';</script><a href='{$redirect}'>跳转中...</a>";
  134. }
  135. # 生成微信授权链接失败
  136. return "<h2>Failed to create authorization. Please return to try again.</h2>";
  137. }
  138. /**
  139. * 公众号授权绑定数据处理
  140. * @param \WeOpen\Service $service
  141. * @param string|null $redirect 授权成功回跳地址
  142. * @return string
  143. * @throws \WeChat\Exceptions\InvalidResponseException
  144. * @throws \WeChat\Exceptions\LocalCacheException
  145. * @throws \think\Exception
  146. * @throws \think\exception\PDOException
  147. */
  148. private function applyAuth($service, $redirect = null)
  149. {
  150. // 通过授权code换取公众号信息
  151. $result = $service->getQueryAuthorizerInfo();
  152. if (empty($result['authorizer_appid'])) {
  153. return "接收微信第三方平台授权失败! ";
  154. }
  155. // 重新通过接口查询公众号参数
  156. if (!($update = array_merge($result, $service->getAuthorizerInfo($result['authorizer_appid'])))) {
  157. return '获取授权数据失败, 请稍候再试!';
  158. }
  159. // 生成公众号授权参数
  160. $update = array_merge(BuildService::filter($update), [
  161. 'status' => '1', 'is_deleted' => '0', 'expires_in' => time() + 7000, 'create_at' => date('y-m-d H:i:s'),
  162. ]);
  163. // 微信接口APPKEY处理与更新
  164. $config = Db::name('WechatServiceConfig')->where(['authorizer_appid' => $result['authorizer_appid']])->find();
  165. $update['appkey'] = empty($config['appkey']) ? md5(uniqid('', true)) : $config['appkey'];
  166. data_save('WechatServiceConfig', $update, 'authorizer_appid');
  167. if (!empty($redirect)) { // 带上appid与appkey跳转到应用
  168. $split = stripos($redirect, '?') > 0 ? '&' : '?';
  169. $realurl = preg_replace(['/appid=\w+/i', '/appkey=\w+/i', '/(\?\&)$/i'], ['', '', ''], $redirect);
  170. return redirect("{$realurl}{$split}appid={$update['authorizer_appid']}&appkey={$update['appkey']}");
  171. }
  172. }
  173. }