MpService.php 18 KB

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