MpService.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  1. <?php
  2. namespace App\Services;
  3. /**
  4. * 微信服务管理-服务类
  5. * @author laravel开发员
  6. * @since 2020/11/11
  7. * Class WechatService
  8. * @package App\Services
  9. */
  10. class MpService extends BaseService
  11. {
  12. // 静态对象
  13. protected static $instance = null;
  14. protected $debug = true;
  15. protected $expireTime = 7200; // 缓存日志时长
  16. protected $mpAppid = '';
  17. protected $mpAppSecret = '';
  18. // 接口地址
  19. protected $apiUrls = [
  20. // 授权登录
  21. 'auth'=> 'https://api.weixin.qq.com/sns/oauth2/access_token?appid=%s&secret=%s&code=%s&grant_type=authorization_code',
  22. // 授权跳转地址
  23. 'authorize'=>'https://open.weixin.qq.com/connect/oauth2/authorize?appid=%s&redirect_uri=%s&response_type=code&scope=snsapi_userinfo&state=%s#wechat_redirect',
  24. // 获取token
  25. 'getToken'=>'https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s',
  26. // 获取二维码
  27. 'getQrcode'=>'https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token=%s&scene=%s&page=%s&env_version=%s',
  28. // 生成小程序分享链接
  29. 'getSchemeLink'=>'https://api.weixin.qq.com/wxa/generatescheme?access_token=%s',
  30. // 短链接
  31. 'getShortLink'=>'https://api.weixin.qq.com/wxa/genwxashortlink?access_token=%s',
  32. // 获取用户信息
  33. 'getUserInfo'=>'https://api.weixin.qq.com/sns/jscode2session',
  34. // 获取手机号
  35. 'getPhoneNumber'=>'https://api.weixin.qq.com/wxa/business/getuserphonenumber?access_token=%s',
  36. // 获取运力公司列表
  37. 'getDeliveryCompanyList'=>'https://api.weixin.qq.com/product/delivery/get_company_list?access_token=%s',
  38. // 获取支持的快递公司列表
  39. 'getDelivery'=>'https://api.weixin.qq.com/cgi-bin/express/local/business/delivery/getall?access_token=%s',
  40. // 上传发货信息,同步发货状态
  41. 'deliverySend'=>'https://api.weixin.qq.com/wxa/sec/order/upload_shipping_info?access_token=%s',
  42. ];
  43. public function __construct()
  44. {
  45. $this->mpAppid = ConfigService::make()->getConfigByCode('wechat_mp_appid');
  46. $this->mpAppSecret = ConfigService::make()->getConfigByCode('wechat_mp_appsecret');
  47. }
  48. /**
  49. * 静态入口
  50. * @return static|null
  51. */
  52. public static function make()
  53. {
  54. if (!self::$instance) {
  55. self::$instance = new static();
  56. }
  57. return self::$instance;
  58. }
  59. /**
  60. * web 授权
  61. * @param $code
  62. * @return array|false|mixed|string|string[]
  63. */
  64. public function auth($code)
  65. {
  66. try {
  67. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  68. $this->error = '小程序参数未配置';
  69. return false;
  70. }
  71. // 没有code参数
  72. if(empty($code)){
  73. $uri=urlencode(env("WEB_URL")."/v1/mpAuth");
  74. $state=get_random_code(16);
  75. $url= sprintf($this->apiUrls['authorize'], $this->mpAppid, $this->mpAppSecret, $uri, $state);
  76. return redirect($url);
  77. }
  78. $cacheKey = "caches:mpApp:mp_{$this->mpAppid}:";
  79. $url = sprintf($this->apiUrls['auth'],$this->mpAppid, $this->mpAppSecret, $code);
  80. $result = httpRequest($url, '', 'get','',5);
  81. $this->saveLog($cacheKey.'auth:request', ['url'=>$url,'code'=>$code,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  82. if(empty($result)){
  83. $this->error = '授权登录失败';
  84. return "<script>alert('微信授权失败');window.close()</script>";
  85. }
  86. return $result;
  87. }catch (\Exception $e){
  88. $this->error = $e->getMessage();
  89. $this->saveLog($cacheKey.'auth:error', ['code'=>$code,'error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  90. return false;
  91. }
  92. }
  93. /**
  94. * 获取 access_token
  95. * @param false $refresh
  96. * @return false|mixed|string
  97. */
  98. public function getAccessToken($refresh = false)
  99. {
  100. try {
  101. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  102. $this->error = '小程序参数未配置';
  103. return false;
  104. }
  105. $cacheKey = "caches:mpApp:mp_{$this->mpAppid}:";
  106. $tokenData = RedisService::get($cacheKey.'access_token');
  107. $token = isset($tokenData['access_token'])? $tokenData['access_token'] : '';
  108. if($token && !$refresh){
  109. return $token;
  110. }
  111. $url = sprintf($this->apiUrls['getToken'], $this->mpAppid, $this->mpAppSecret);
  112. $result = httpRequest($url,'', 'get','',5);
  113. $this->saveLog($cacheKey.'tokens:request', ['url'=>$url,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  114. $token = isset($result['access_token'])? $result['access_token'] : '';
  115. if(empty($result) || empty($token)){
  116. $this->error = '获取小程序TOKEN失败';
  117. return false;
  118. }
  119. $result['date'] = date('Y-m-d H:i:s');
  120. RedisService::set($cacheKey.'access_token', $result, 7000);
  121. return $token;
  122. }catch (\Exception $e){
  123. $this->error = $e->getMessage();
  124. $this->saveLog($cacheKey.'tokens:error', ['error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  125. return false;
  126. }
  127. }
  128. /**
  129. * 小程序二维码
  130. * @param $page 页面
  131. * @param $scene 场景参数
  132. * @param string $version 类型:release-永久
  133. * @param false $refresh
  134. * @return false|string
  135. */
  136. public function getMiniQrcode($page, $scene, $version='release', $refresh=false)
  137. {
  138. if (!in_array($version,['release','trial','develop'])) {
  139. $version='release';
  140. }
  141. try {
  142. if(empty($page) || empty($scene)){
  143. $this->error = '缺少二维码参数';
  144. return false;
  145. }
  146. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  147. $this->error = '小程序参数未配置';
  148. return false;
  149. }
  150. if(!$token = $this->getAccessToken())
  151. {
  152. $this->error = '获取token失败';
  153. return false;
  154. }
  155. $cacheKey = "caches:members:mp_{$this->mpAppid}:";
  156. $filePath = base_path('public/uploads');
  157. $qrFile = '/qrcodes/mp_'.date("YmdHis")."_".md5($page.$scene).".png";
  158. $qrKey = md5(date("Ym").$page.$scene);
  159. if(RedisService::get($cacheKey.$qrKey) && file_exists($filePath.'/'.$qrFile) && !$refresh){
  160. return $qrFile;
  161. }
  162. if(!is_dir($filePath.'/qrcodes/')){
  163. @mkdirs($filePath.'/qrcodes/');
  164. }
  165. $data=['page' => $page,'scene'=>$scene,'check_path'=>false,'env_version'=>$version];
  166. $url = sprintf($this->apiUrls['getQrcode'], $token, $scene, $page, $version);
  167. $result = curl_post($url, json_encode($data));
  168. $datas = $result? json_decode($result, true) : [];
  169. $this->saveLog($cacheKey.'qrcode:request', ['page'=>$page,'scene'=>$scene,'url'=>$url,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  170. $errcode = isset($datas['errcode'])? $datas['errcode'] : '';
  171. $errmsg = isset($datas['errmsg'])? $datas['errmsg'] : '';
  172. if($errcode){
  173. $this->error = $errmsg? $errmsg : '获取二维码失败';
  174. return false;
  175. }
  176. file_put_contents($filePath.'/'.$qrFile, $result);
  177. if(!file_exists($filePath.'/'.$qrFile)){
  178. $this->error = '生成二维码失败';
  179. return false;
  180. }
  181. RedisService::set($cacheKey.$qrKey, ['page'=>$page,'scene'=>$scene,'qrcode'=>$qrFile,'date'=>date('Y-m-d H:i:s')], 30 * 86400);
  182. return $qrFile;
  183. }catch (\Exception $e){
  184. $this->error = $e->getMessage();
  185. $this->saveLog($cacheKey.'qrcode:error', ['page'=>$page,'scene'=>$scene,'error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  186. return false;
  187. }
  188. }
  189. /**
  190. * 小程序分享链接
  191. * @param $path 页面路径
  192. * @param $query 地址参数
  193. * @param false $refresh
  194. * @return false|string
  195. */
  196. public function getMiniShareLink($path, $query,$isExpire=false,$linkType=2, $refresh=false)
  197. {
  198. try {
  199. if(empty($path) || empty($query)){
  200. $this->error = '缺少链接参数';
  201. return false;
  202. }
  203. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  204. $this->error = '小程序参数未配置';
  205. return false;
  206. }
  207. if(!$token = $this->getAccessToken())
  208. {
  209. $this->error = '获取token失败';
  210. return false;
  211. }
  212. $cacheKey = "caches:members:mpShare_{$this->mpAppid}:{$linkType}_".md5($path.$query);
  213. $link = RedisService::get($cacheKey);
  214. if($link && !$refresh){
  215. return $link;
  216. }
  217. if($linkType == 1){
  218. $data = [
  219. 'jump_wxa'=>[
  220. 'path'=>$path,
  221. 'query'=>$query,
  222. ],
  223. 'is_expire'=> $isExpire,
  224. ];
  225. }else if($linkType ==2){
  226. $data=[
  227. 'page_url'=> $path,
  228. 'page_title'=> $query,
  229. 'is_permanent'=>$isExpire
  230. ];
  231. }
  232. $linkTypes = [1=>'getSchemeLink',2=>'getShortLink'];
  233. $name = isset($linkTypes[$linkType])?$linkTypes[$linkType]: 'getShortLink';
  234. $url = sprintf($this->apiUrls[$name], $token);
  235. $result = curl_post($url, json_encode($data));
  236. $datas = $result? json_decode($result, true) : [];
  237. $this->saveLog($cacheKey.'_result', ['path'=>$path,'query'=>$query,'linkType'=>$linkType,'url'=>$url,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  238. $errcode = isset($datas['errcode'])? $datas['errcode'] : '';
  239. $errmsg = isset($datas['errmsg'])? $datas['errmsg'] : '';
  240. $link =isset($datas['link'])? $datas['link'] : '';
  241. if($errcode || empty($link)){
  242. $this->error = $errmsg? '获取失败:'.$errmsg : '获取链接失败';
  243. return false;
  244. }
  245. RedisService::set($cacheKey,$link, rand(3600,7200));
  246. return $link;
  247. }catch (\Exception $e){
  248. $this->error = $e->getMessage();
  249. $this->saveLog($cacheKey.'_error', ['path'=>$path,'query'=>$query,'linkType'=>$linkType,'error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  250. return false;
  251. }
  252. }
  253. /**
  254. * 获取用户信息
  255. * @param $code
  256. * @return array|false|mixed|string[]
  257. */
  258. public function getUserinfo($code)
  259. {
  260. try {
  261. if(empty($code)){
  262. $this->error = '缺少授权参数';
  263. return false;
  264. }
  265. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  266. $this->error = '小程序参数未配置';
  267. return false;
  268. }
  269. $data=[
  270. 'appid' => $this->mpAppid,
  271. 'secret'=> $this->mpAppSecret,
  272. 'js_code'=>$code,
  273. 'grant_type'=>'authorization_code'
  274. ];
  275. $cacheKey = "caches:mpApp:mp_{$this->mpAppid}:";
  276. $url = $this->apiUrls['getUserInfo'];
  277. $result = httpRequest($url, $data,'get','',5);
  278. $this->saveLog($cacheKey.'userInfo:request', ['code'=>$code,'url'=>$url,'query'=>$data,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  279. if(empty($result)){
  280. $this->error = '获取用户信息失败';
  281. return false;
  282. }
  283. return $result;
  284. }catch (\Exception $e){
  285. $this->error = $e->getMessage();
  286. $this->saveLog($cacheKey.'userInfo:error', ['code'=>$code,'error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  287. return false;
  288. }
  289. }
  290. /**
  291. * 获取用户手机号码
  292. * @param $code
  293. * @return array|false|mixed|string[]
  294. */
  295. public function getPhoneNumber($code)
  296. {
  297. try {
  298. if(empty($code)){
  299. $this->error = '缺少授权参数';
  300. return false;
  301. }
  302. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  303. $this->error = '小程序参数未配置';
  304. return false;
  305. }
  306. if(!$token = $this->getAccessToken())
  307. {
  308. $this->error = '获取token失败';
  309. return false;
  310. }
  311. $cacheKey = "caches:mpApp:mp_{$this->mpAppid}:";
  312. $url = sprintf($this->apiUrls['getPhoneNumber'], $token);
  313. $result = httpRequest($url, json_encode(['code'=>$code],256),'post','',5);
  314. $this->saveLog($cacheKey.'phone:request', ['code'=>$code,'url'=>$url,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  315. if(empty($result)){
  316. $this->error = '获取用户手机号失败';
  317. return false;
  318. }
  319. return $result;
  320. }catch (\Exception $e){
  321. $this->error = $e->getMessage();
  322. $this->saveLog($cacheKey.'phone:error', ['code'=>$code,'error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  323. return false;
  324. }
  325. }
  326. /**
  327. * 只带 access_token API 请求
  328. * @param string $apiName 接口名称
  329. * @param $data 接口数据
  330. * @param $requestType 请求方式:post或get
  331. * @return false|mixed|string
  332. */
  333. public function requestApi(string $apiName, $data=[], $requestType = 'post')
  334. {
  335. try {
  336. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  337. $this->error = '小程序参数未配置';
  338. return false;
  339. }
  340. if(!$token = $this->getAccessToken())
  341. {
  342. $this->error = '获取token失败';
  343. return false;
  344. }
  345. $cacheKey = "caches:mpApp:mp_{$this->mpAppid}:";
  346. $url = sprintf($this->apiUrls[$apiName], $token);
  347. $result = httpRequest($url,$data?json_encode($data,256):'', $requestType,'',5);
  348. $this->saveLog($cacheKey."{$apiName}:request", ['url'=>$url,'data'=>$data,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  349. return $result;
  350. }catch (\Exception $e){
  351. $this->error = $e->getMessage();
  352. $this->saveLog($cacheKey."{$apiName}:error", ['data'=>$data,'error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  353. return false;
  354. }
  355. }
  356. /**
  357. * 获取 客服接口 access_token
  358. * @param false $refresh
  359. * @return false|mixed|string
  360. */
  361. public function getServiceToken($refresh = false)
  362. {
  363. try {
  364. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  365. $this->error = '小程序参数未配置';
  366. return false;
  367. }
  368. $cacheKey = "caches:mpApp:mp_{$this->mpAppid}:";
  369. $tokenData = RedisService::get($cacheKey.'kf_access_token');
  370. $token = isset($tokenData['access_token'])? $tokenData['access_token'] : '';
  371. if($token && !$refresh){
  372. return $token;
  373. }
  374. $url = sprintf($this->apiUrls['getServiceToken'], $this->mpAppid, $this->mpAppSecret);
  375. $result = httpRequest($url,'', 'get','',5);
  376. $this->saveLog($cacheKey.'kfTokens:request', ['url'=>$url,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  377. $token = isset($result['access_token'])? $result['access_token'] : '';
  378. if(empty($result) || empty($token)){
  379. $this->error = '获取小程序客服TOKEN失败';
  380. return false;
  381. }
  382. $result['date'] = date('Y-m-d H:i:s');
  383. RedisService::set($cacheKey.'kf_access_token', $result, 7000);
  384. return $token;
  385. }catch (\Exception $e){
  386. $this->error = $e->getMessage();
  387. $this->saveLog($cacheKey.'kfTokens:error', ['error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  388. return false;
  389. }
  390. }
  391. /**
  392. * 获取客服地址
  393. * @return array|false|mixed|string[]
  394. */
  395. public function getServiceUrl()
  396. {
  397. try {
  398. if(empty($this->mpAppid) || empty($this->mpAppSecret)){
  399. $this->error = '小程序参数未配置';
  400. return false;
  401. }
  402. if(!$token = $this->getServiceToken())
  403. {
  404. $this->error = '获取token失败';
  405. return false;
  406. }
  407. $cacheKey = "caches:mpApp:mp_{$this->mpAppid}:";
  408. $url = sprintf($this->apiUrls['getServiceUrl'], $token);
  409. $result = httpRequest($url, '','post','',5);
  410. $this->saveLog($cacheKey.'service:request', ['url'=>$url,'result'=>$result,'date'=>date('Y-m-d H:i:s')]);
  411. if(empty($result)){
  412. $this->error = '获取小程序客服地址失败';
  413. return false;
  414. }
  415. return $result;
  416. }catch (\Exception $e){
  417. $this->error = $e->getMessage();
  418. $this->saveLog($cacheKey.'service:error', ['error'=>$this->error,'trace'=>$e->getTrace(),'date'=>date('Y-m-d H:i:s')]);
  419. return false;
  420. }
  421. }
  422. /**
  423. * 检验数据的真实性,并且获取解密后的明文.
  424. * @param $encryptedData string 加密的用户数据
  425. * @param $iv string 与用户数据一同返回的初始向量
  426. * @param $sessionKey string 解密会话KEY
  427. *
  428. * @return int 成功0,失败返回对应的错误码
  429. */
  430. public function decryptData($encryptedData, $iv, $sessionKey)
  431. {
  432. if (strlen($sessionKey) != 24) {
  433. $this->error = -41001;
  434. return false;
  435. }
  436. $aesKey=base64_decode($sessionKey);
  437. if (strlen($iv) != 24) {
  438. $this->error = -41002;
  439. return false;
  440. }
  441. $aesIV=base64_decode($iv);
  442. $aesCipher=base64_decode($encryptedData);
  443. $result=openssl_decrypt( $aesCipher, "AES-128-CBC", $aesKey, 1, $aesIV);
  444. $dataObj=json_decode( $result);
  445. if( $dataObj == NULL )
  446. {
  447. $this->error = -41003;
  448. return false;
  449. }
  450. if( $dataObj->watermark->appid != $this->mpAppid)
  451. {
  452. $this->error = -41003;
  453. return false;
  454. }
  455. return $dataObj;
  456. }
  457. /**
  458. * 保存日志
  459. * @param $cackekey
  460. * @param $data
  461. * @param $time
  462. */
  463. public function saveLog($cackekey, $data, $time=0)
  464. {
  465. if($this->debug){
  466. RedisService::set($cackekey, $data, $time?$time : $this->expireTime);
  467. }
  468. }
  469. }