Push.php 7.5 KB

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