BasicWeChat.php 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | WeChatDeveloper
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2014~2022 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://think.ctolog.com
  8. // +----------------------------------------------------------------------
  9. // | 开源协议 ( https://mit-license.org )
  10. // +----------------------------------------------------------------------
  11. // | github开源项目:https://github.com/zoujingli/WeChatDeveloper
  12. // +----------------------------------------------------------------------
  13. namespace WeChat\Contracts;
  14. use WeChat\Exceptions\InvalidArgumentException;
  15. use WeChat\Exceptions\InvalidResponseException;
  16. /**
  17. * Class BasicWeChat
  18. * @package WeChat\Contracts
  19. */
  20. class BasicWeChat
  21. {
  22. /**
  23. * 当前微信配置
  24. * @var DataArray
  25. */
  26. public $config;
  27. /**
  28. * 访问AccessToken
  29. * @var string
  30. */
  31. public $access_token = '';
  32. /**
  33. * 当前请求方法参数
  34. * @var array
  35. */
  36. protected $currentMethod = [];
  37. /**
  38. * 当前模式
  39. * @var bool
  40. */
  41. protected $isTry = false;
  42. /**
  43. * 静态缓存
  44. * @var static
  45. */
  46. protected static $cache;
  47. /**
  48. * 注册代替函数
  49. * @var string
  50. */
  51. protected $GetAccessTokenCallback;
  52. /**
  53. * BasicWeChat constructor.
  54. * @param array $options
  55. */
  56. public function __construct(array $options)
  57. {
  58. if (empty($options['appid'])) {
  59. throw new InvalidArgumentException("Missing Config -- [appid]");
  60. }
  61. if (empty($options['appsecret'])) {
  62. throw new InvalidArgumentException("Missing Config -- [appsecret]");
  63. }
  64. if (isset($options['GetAccessTokenCallback']) && is_callable($options['GetAccessTokenCallback'])) {
  65. $this->GetAccessTokenCallback = $options['GetAccessTokenCallback'];
  66. }
  67. if (!empty($options['cache_path'])) {
  68. Tools::$cache_path = $options['cache_path'];
  69. }
  70. $this->config = new DataArray($options);
  71. }
  72. /**
  73. * 静态创建对象
  74. * @param array $config
  75. * @return static
  76. */
  77. public static function instance(array $config)
  78. {
  79. $key = md5(get_called_class() . serialize($config));
  80. if (isset(self::$cache[$key])) return self::$cache[$key];
  81. return self::$cache[$key] = new static($config);
  82. }
  83. /**
  84. * 获取访问 AccessToken
  85. * @return string
  86. * @throws \WeChat\Exceptions\InvalidResponseException
  87. * @throws \WeChat\Exceptions\LocalCacheException
  88. */
  89. public function getAccessToken()
  90. {
  91. if (!empty($this->access_token)) {
  92. return $this->access_token;
  93. }
  94. $cache = $this->config->get('appid') . '_access_token';
  95. $this->access_token = Tools::getCache($cache);
  96. if (!empty($this->access_token)) {
  97. return $this->access_token;
  98. }
  99. // 处理开放平台授权公众号获取AccessToken
  100. if (!empty($this->GetAccessTokenCallback) && is_callable($this->GetAccessTokenCallback)) {
  101. $this->access_token = call_user_func_array($this->GetAccessTokenCallback, [$this->config->get('appid'), $this]);
  102. if (!empty($this->access_token)) {
  103. Tools::setCache($cache, $this->access_token, 7000);
  104. }
  105. return $this->access_token;
  106. }
  107. list($appid, $secret) = [$this->config->get('appid'), $this->config->get('appsecret')];
  108. $url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid={$appid}&secret={$secret}";
  109. $result = Tools::json2arr(Tools::get($url));
  110. if (!empty($result['access_token'])) {
  111. Tools::setCache($cache, $result['access_token'], 7000);
  112. }
  113. return $this->access_token = $result['access_token'];
  114. }
  115. /**
  116. * 设置外部接口 AccessToken
  117. * @param string $accessToken
  118. * @throws \WeChat\Exceptions\LocalCacheException
  119. * @author 高一平 <iam@gaoyiping.com>
  120. *
  121. * 当用户使用自己的缓存驱动时,直接实例化对象后可直接设置 AccessToken
  122. * - 多用于分布式项目时保持 AccessToken 统一
  123. * - 使用此方法后就由用户来保证传入的 AccessToken 为有效 AccessToken
  124. */
  125. public function setAccessToken($accessToken)
  126. {
  127. if (!is_string($accessToken)) {
  128. throw new InvalidArgumentException("Invalid AccessToken type, need string.");
  129. }
  130. $cache = $this->config->get('appid') . '_access_token';
  131. Tools::setCache($cache, $this->access_token = $accessToken);
  132. }
  133. /**
  134. * 清理删除 AccessToken
  135. * @return bool
  136. */
  137. public function delAccessToken()
  138. {
  139. $this->access_token = '';
  140. return Tools::delCache($this->config->get('appid') . '_access_token');
  141. }
  142. /**
  143. * 以GET获取接口数据并转为数组
  144. * @param string $url 接口地址
  145. * @return array
  146. * @throws InvalidResponseException
  147. * @throws \WeChat\Exceptions\LocalCacheException
  148. */
  149. protected function httpGetForJson($url)
  150. {
  151. try {
  152. return Tools::json2arr(Tools::get($url));
  153. } catch (InvalidResponseException $exception) {
  154. if (isset($this->currentMethod['method']) && empty($this->isTry)) {
  155. if (in_array($exception->getCode(), ['40014', '40001', '41001', '42001'])) {
  156. [$this->delAccessToken(), $this->isTry = true];
  157. return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
  158. }
  159. }
  160. throw new InvalidResponseException($exception->getMessage(), $exception->getCode());
  161. }
  162. }
  163. /**
  164. * 以POST获取接口数据并转为数组
  165. * @param string $url 接口地址
  166. * @param array $data 请求数据
  167. * @param bool $buildToJson
  168. * @return array
  169. * @throws InvalidResponseException
  170. * @throws \WeChat\Exceptions\LocalCacheException
  171. */
  172. protected function httpPostForJson($url, array $data, $buildToJson = true)
  173. {
  174. try {
  175. $options = [];
  176. if ($buildToJson) $options['headers'] = ['Content-Type: application/json'];
  177. return Tools::json2arr(Tools::post($url, $buildToJson ? Tools::arr2json($data) : $data, $options));
  178. } catch (InvalidResponseException $exception) {
  179. if (!$this->isTry && in_array($exception->getCode(), ['40014', '40001', '41001', '42001'])) {
  180. [$this->delAccessToken(), $this->isTry = true];
  181. return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
  182. }
  183. throw new InvalidResponseException($exception->getMessage(), $exception->getCode());
  184. }
  185. }
  186. /**
  187. * 注册当前请求接口
  188. * @param string $url 接口地址
  189. * @param string $method 当前接口方法
  190. * @param array $arguments 请求参数
  191. * @return string
  192. * @throws \WeChat\Exceptions\InvalidResponseException
  193. * @throws \WeChat\Exceptions\LocalCacheException
  194. */
  195. protected function registerApi(&$url, $method, $arguments = [])
  196. {
  197. $this->currentMethod = ['method' => $method, 'arguments' => $arguments];
  198. if (empty($this->access_token)) $this->access_token = $this->getAccessToken();
  199. return $url = str_replace('ACCESS_TOKEN', urlencode($this->access_token), $url);
  200. }
  201. /**
  202. * 接口通用POST请求方法
  203. * @param string $url 接口URL
  204. * @param array $data POST提交接口参数
  205. * @param bool $isBuildJson
  206. * @return array
  207. * @throws InvalidResponseException
  208. * @throws \WeChat\Exceptions\LocalCacheException
  209. */
  210. public function callPostApi($url, array $data, $isBuildJson = true)
  211. {
  212. $this->registerApi($url, __FUNCTION__, func_get_args());
  213. return $this->httpPostForJson($url, $data, $isBuildJson);
  214. }
  215. /**
  216. * 接口通用GET请求方法
  217. * @param string $url 接口URL
  218. * @return array
  219. * @throws InvalidResponseException
  220. * @throws \WeChat\Exceptions\LocalCacheException
  221. */
  222. public function callGetApi($url)
  223. {
  224. $this->registerApi($url, __FUNCTION__, func_get_args());
  225. return $this->httpGetForJson($url);
  226. }
  227. }