Common.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. <?php
  2. namespace app\api\controller;
  3. use app\common\controller\Api;
  4. use app\common\exception\UploadException;
  5. use app\common\library\Upload;
  6. use app\common\model\Area;
  7. use app\common\model\Category;
  8. use app\common\model\Selection;
  9. use app\common\model\Version;
  10. use fast\Random;
  11. use think\Config;
  12. use think\Hook;
  13. /**
  14. * 公共接口
  15. */
  16. class Common extends Api
  17. {
  18. protected $noNeedLogin = '*';
  19. protected $noNeedRight = '*';
  20. /**
  21. * 加载初始化
  22. *
  23. * @param string $version 版本号
  24. * @param string $lng 经度
  25. * @param string $lat 纬度
  26. */
  27. public function init()
  28. {
  29. if ($version = $this->request->request('version')) {
  30. $lng = $this->request->request('lng');
  31. $lat = $this->request->request('lat');
  32. //配置信息
  33. $upload = Config::get('upload');
  34. //如果非服务端中转模式需要修改为中转
  35. if ($upload['storage'] != 'local' && isset($upload['uploadmode']) && $upload['uploadmode'] != 'server') {
  36. //临时修改上传模式为服务端中转
  37. set_addon_config($upload['storage'], ["uploadmode" => "server"], false);
  38. $upload = \app\common\model\Config::upload();
  39. // 上传信息配置后
  40. Hook::listen("upload_config_init", $upload);
  41. $upload = Config::set('upload', array_merge(Config::get('upload'), $upload));
  42. }
  43. $upload['cdnurl'] = $upload['cdnurl'] ? $upload['cdnurl'] : cdnurl('', true);
  44. $upload['uploadurl'] = preg_match("/^((?:[a-z]+:)?\/\/)(.*)/i", $upload['uploadurl']) ? $upload['uploadurl'] : url($upload['storage'] == 'local' ? '/api/common/upload' : $upload['uploadurl'], '', false, true);
  45. $content = [
  46. 'citydata' => Area::getCityFromLngLat($lng, $lat),
  47. 'versiondata' => Version::check($version),
  48. 'uploaddata' => $upload,
  49. 'coverdata' => Config::get("cover"),
  50. ];
  51. $this->success('', $content);
  52. } else {
  53. $this->error(__('Invalid parameters'));
  54. }
  55. }
  56. /**
  57. * 上传文件
  58. * @ApiMethod (POST)
  59. * @param File $file 文件流
  60. */
  61. public function upload()
  62. {
  63. Config::set('default_return_type', 'json');
  64. //必须设定cdnurl为空,否则cdnurl函数计算错误
  65. Config::set('upload.cdnurl', '');
  66. $chunkid = $this->request->post("chunkid");
  67. if ($chunkid) {
  68. if (!Config::get('upload.chunking')) {
  69. $this->error(__('Chunk file disabled'));
  70. }
  71. $action = $this->request->post("action");
  72. $chunkindex = $this->request->post("chunkindex/d");
  73. $chunkcount = $this->request->post("chunkcount/d");
  74. $filename = $this->request->post("filename");
  75. $method = $this->request->method(true);
  76. if ($action == 'merge') {
  77. $attachment = null;
  78. //合并分片文件
  79. try {
  80. $upload = new Upload();
  81. $attachment = $upload->merge($chunkid, $chunkcount, $filename);
  82. } catch (UploadException $e) {
  83. $this->error($e->getMessage());
  84. }
  85. $this->success(__('Uploaded successful'), ['url' => $attachment->url, 'fullurl' => cdnurl($attachment->url, true)]);
  86. } elseif ($method == 'clean') {
  87. //删除冗余的分片文件
  88. try {
  89. $upload = new Upload();
  90. $upload->clean($chunkid);
  91. } catch (UploadException $e) {
  92. $this->error($e->getMessage());
  93. }
  94. $this->success();
  95. } else {
  96. //上传分片文件
  97. //默认普通上传文件
  98. $file = $this->request->file('file');
  99. try {
  100. $upload = new Upload($file);
  101. $upload->chunk($chunkid, $chunkindex, $chunkcount);
  102. } catch (UploadException $e) {
  103. $this->error($e->getMessage());
  104. }
  105. $this->success();
  106. }
  107. } else {
  108. $attachment = null;
  109. //默认普通上传文件
  110. $file = $this->request->file('file');
  111. try {
  112. $upload = new Upload($file);
  113. $attachment = $upload->upload();
  114. } catch (UploadException $e) {
  115. $this->error($e->getMessage());
  116. }
  117. $this->success(__('Uploaded successful'), ['url' => $attachment->url, 'fullurl' => cdnurl($attachment->url, true)]);
  118. }
  119. }
  120. /**
  121. * 获取系统配置
  122. * @ApiParams (name='site',description='officialwx官方微信 money狗粮充值配置 score喜欢数量充值配置 PrivacyAgreement隐私协议 VerifyAuthorization验证授权书 用户服务协议UserServiceAgreement 关于我们AboutUs 狗粮用户协议DogFoodUserAgreement 公众号图片
  123. official_account_img 公众号文本
  124. official_account_text')
  125. */
  126. public function system_configuration(){
  127. $this->success('请求成功',config('site.'.input('site')));
  128. }
  129. /**
  130. * 获取分类
  131. * @ApiParams (name='type',description='education学历分类 tag标签 reason_cancellation注销列表')
  132. */
  133. public function category(){
  134. $this->success('请求成功',Category::where('type',input('type'))->select());
  135. }
  136. /**
  137. * 生成订单号
  138. * @return string
  139. */
  140. function getOrderId()
  141. {
  142. return date('Ymd') . str_pad(mt_rand(1, 99999), 5, '0', STR_PAD_LEFT);
  143. }
  144. /**
  145. * 获取openid(已废弃)
  146. * @ApiParams (name="code")
  147. */
  148. function getOpenid($code_ = ''){
  149. $code = $code_;
  150. $appid = 'wx5e111b790c719cae';
  151. $secret = 'a4e2a6d2a4fd06bd9e4ba279fd95e2d8';
  152. $get_token_url = 'https://api.weixin.qq.com/sns/oauth2/access_token?appid='.$appid.'&secret='.$secret.'&code='.$code.'&grant_type=authorization_code';
  153. $ch = curl_init();
  154. curl_setopt($ch,CURLOPT_URL,$get_token_url);
  155. curl_setopt($ch,CURLOPT_HEADER,0);
  156. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1 );
  157. curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
  158. $res = curl_exec($ch);
  159. curl_close($ch);
  160. $json_obj = json_decode($res,true);
  161. if($json_obj['openid']){
  162. return $json_obj['openid'];
  163. }else{
  164. $this->error('请重新登录');
  165. }
  166. $this->error('0',$json_obj);
  167. }
  168. /**
  169. * 拉取wx用户信息
  170. */
  171. function getUserinfo($code_ = ''){
  172. $code = $code_;
  173. $appid = 'wx5e111b790c719cae';
  174. $secret = 'a4e2a6d2a4fd06bd9e4ba279fd95e2d8';
  175. //获取access_token openid
  176. $get_token_url = 'https://api.weixin.qq.com/sns/oauth2/access_token?appid='.$appid.'&secret='.$secret.'&code='.$code.'&grant_type=authorization_code';
  177. $wx_info_json_obj = json_decode(self::curl($get_token_url),true);
  178. //获取用户信息
  179. $get_userinfo_url = 'https://api.weixin.qq.com/sns/userinfo?access_token='.$wx_info_json_obj['access_token'].'&openid='.$wx_info_json_obj['openid'].'&lang=zh_CN';
  180. $get_userinfo_json_obj = json_decode(self::curl($get_userinfo_url),true);
  181. if($get_userinfo_json_obj){
  182. return $get_userinfo_json_obj;
  183. }else{
  184. $this->error('请重新登录');
  185. }
  186. $this->error('0',$get_userinfo_json_obj);
  187. }
  188. /**
  189. * curl
  190. */
  191. function curl($url = ''){
  192. $ch = curl_init();
  193. curl_setopt($ch,CURLOPT_URL,$url);
  194. curl_setopt($ch,CURLOPT_HEADER,0);
  195. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1 );
  196. curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
  197. $res = curl_exec($ch);
  198. curl_close($ch);
  199. return $res;
  200. }
  201. /**
  202. * 选择项
  203. * @ApiParams (name="type",description="1年龄 2身高 3计划结婚")
  204. */
  205. public function selection(){
  206. $this->success('',Selection::all(['type'=>input('type'),'status'=>1]));
  207. }
  208. /**
  209. * 获取所有地区
  210. * @return void
  211. * @throws \think\db\exception\DataNotFoundException
  212. * @throws \think\db\exception\ModelNotFoundException
  213. * @throws \think\exception\DbException
  214. */
  215. public function getAllArea()
  216. {
  217. $field = ['id', 'pid', 'shortname', 'name', 'first'];
  218. $list = Area::with(['children' => function ($query) use ($field) {
  219. $query->field($field)->with(['children' => function ($query) use ($field) {
  220. $query->field($field);
  221. }]);
  222. }])
  223. ->where('pid', 0)->field($field)->select();
  224. $this->success('获取成功', $list);
  225. }
  226. /**
  227. * 获取所有的省市
  228. * @return void
  229. * @throws \think\db\exception\DataNotFoundException
  230. * @throws \think\db\exception\ModelNotFoundException
  231. * @throws \think\exception\DbException
  232. */
  233. public function get_area_city()
  234. {
  235. $field = ['id', 'pid', 'shortname', 'name', 'first'];
  236. $list = Area::with(['children' => function ($query) use ($field) {
  237. $query->field($field);
  238. }])
  239. ->where('pid', 0)->field($field)->select();
  240. $this->success('获取成功', $list);
  241. }
  242. /**
  243. * 根据城市id获取下一级
  244. * @ApiMethod (GET)
  245. * @ApiParams (name="pid", type="string", description="城市id")
  246. */
  247. public function get_area()
  248. {
  249. $pid = $this->request->get('pid', 0);
  250. $field = ['id', 'pid', 'shortname', 'name'];
  251. $list = Area::where('pid', $pid)->field($field)->select();
  252. $this->success('获取成功', $list);
  253. }
  254. /**
  255. * 生成32随机字符串
  256. */
  257. function getRandomString($len, $chars=null)
  258. {
  259. if (is_null($chars)) {
  260. $chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
  261. }
  262. mt_srand(10000000*(double)microtime());
  263. for ($i = 0, $str = '', $lc = strlen($chars)-1; $i < $len; $i++) {
  264. $str .= $chars[mt_rand(0, $lc)];
  265. }
  266. return $str;
  267. }
  268. /**
  269. * 微信配置分享
  270. * @ApiMethod (POST)
  271. * @ApiParams (name='url')
  272. * @return void
  273. */
  274. public function get_sign(){
  275. // 微信 JS 接口签名校验工具: https://mp.weixin.qq.com/debug/cgi-bin/sandbox?t=jsapisign
  276. $appid = 'wx5e111b790c719cae';
  277. $secret = 'a4e2a6d2a4fd06bd9e4ba279fd95e2d8';
  278. // 获取token
  279. $token_data = file_get_contents(__DIR__.'/wechat_token.txt');
  280. $time = 7200;
  281. if (!empty($token_data)) {
  282. $token_data = json_decode($token_data, true);
  283. $time = time() - $token_data['time'];
  284. }
  285. if ($time > 3600) {
  286. $token_url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid={$appid}&secret={$secret}";
  287. $token_res = $this->https_request($token_url);
  288. $token_res = json_decode($token_res, true);
  289. $token = $token_res['access_token'];
  290. $data = array(
  291. 'time' =>time(),
  292. 'token' =>$token
  293. );
  294. $res = file_put_contents(__DIR__.'wechat_token.txt', json_encode($data));
  295. // if ($res) {
  296. // echo '更新 token 成功';
  297. // }
  298. } else {
  299. $token = $token_data['token'];
  300. }
  301. // 获取ticket
  302. $ticket_data = file_get_contents(__DIR__.'/wechat_ticket.txt');
  303. $time1 = 7200;
  304. if (!empty($ticket_data)) {
  305. $ticket_data = json_decode($ticket_data, true);
  306. $time1 = time() - $ticket_data['time'];
  307. }
  308. if ($time1 > 3600) {
  309. $ticket_url = "https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token={$token}&type=jsapi";
  310. $ticket_res = $this->https_request($ticket_url);
  311. $ticket_res = json_decode($ticket_res, true);
  312. $ticket = $ticket_res['ticket'];
  313. $data = array(
  314. 'time' =>time(),
  315. 'ticket' =>$ticket
  316. );
  317. $res = file_put_contents(__DIR__.'wechat_ticket.txt', json_encode($data));
  318. // if ($res) {
  319. // echo '更新 ticket 成功';
  320. // }
  321. } else {
  322. $ticket = $ticket_data['ticket'];
  323. }
  324. // 进行sha1签名
  325. $timestamp = time();
  326. $nonceStr = $this->createNonceStr();
  327. // 注意 URL 动态获取(也可以写死).
  328. $protocol = (!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] !== 'off' || $_SERVER['SERVER_PORT'] == 443) ? "https://" : "http://";
  329. // $url = "$protocol$_SERVER[HTTP_HOST]$_SERVER[REQUEST_URI]"; // 调用JSSDK的页面地址
  330. $this->request->post('url')?$url = $this->request->post('url'):$this->error('参数缺失');
  331. //$url = $_SERVER['HTTP_REFERER']; // 前后端分离的, 获取请求地址(此值不准确时可以通过其他方式解决)
  332. $str = "jsapi_ticket={$ticket}&noncestr={$nonceStr}&timestamp={$timestamp}&url={$url}";
  333. $sha_str = sha1($str);
  334. $signPackage = array(
  335. "appId" => $appid,
  336. "nonceStr" => $nonceStr,
  337. "timestamp" => $timestamp,
  338. "signature" => $sha_str,
  339. );
  340. $this->success($signPackage);
  341. }
  342. //随机字符串
  343. public function createNonceStr($length = 16) {
  344. $chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
  345. $str = "";
  346. for ($i = 0; $i < $length; $i++) {
  347. $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
  348. }
  349. return $str;
  350. }
  351. // 模拟 http 请求
  352. public function https_request($url, $data = null){
  353. // curl 初始化
  354. $curl = curl_init();
  355. // curl 设置
  356. curl_setopt($curl, CURLOPT_URL, $url);
  357. curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, FALSE);
  358. curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, FALSE);
  359. // 判断 $data get or post
  360. if ( !empty($data) ) {
  361. curl_setopt($curl, CURLOPT_POST, 1);
  362. curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
  363. }
  364. curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
  365. // 执行
  366. $res = curl_exec($curl);
  367. curl_close($curl);
  368. return $res;
  369. }
  370. public function fasong(){
  371. echo Date('Y-m-d H:i:s',time());
  372. }
  373. /**
  374. * 发送微信模板消息
  375. * @param $order_id
  376. * @param $template
  377. * @return false|string|\think\response\Json
  378. */
  379. public function sendWechatMessage($openid,$template_id,$data,$url)
  380. {
  381. $data = array(
  382. "touser" => $openid, //openid
  383. "template_id" => $template_id, //模板id
  384. "url" => $url,
  385. "data" => $data //模板数据
  386. );
  387. //下单用户发送
  388. $access_token = $this->access_token();
  389. if ($access_token == 400){
  390. return ['code'=>0,'message'=>'获取access_token失败'];
  391. }
  392. $urls = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=".$access_token;
  393. $result = $this->http_post_json($urls,json_encode($data));//发送请求
  394. if($result[0] == 200){
  395. return ['code'=>1,'data'=>'推送成功'];
  396. }
  397. return ['code'=>0,'message'=>$result[1]['errmsg']];
  398. }
  399. public function http_post_json($url, $jsonStr)
  400. {
  401. $ch = curl_init();
  402. curl_setopt($ch, CURLOPT_POST, 1);
  403. curl_setopt($ch, CURLOPT_URL, $url);
  404. curl_setopt($ch, CURLOPT_POSTFIELDS, $jsonStr);
  405. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
  406. curl_setopt($ch, CURLOPT_HTTPHEADER, array(
  407. 'Content-Type: application/json; charset=utf-8',
  408. 'Content-Length: ' . strlen($jsonStr)
  409. )
  410. );
  411. $response = curl_exec($ch);
  412. $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
  413. curl_close($ch);
  414. return [$httpCode,$response];
  415. // return json_encode(array($httpCode, $response),true);
  416. //"[200,"{\"errcode\":0,\"errmsg\":\"ok\",\"msgid\":2696398438942572550}"]"
  417. }
  418. //获取access_token
  419. public function access_token(){
  420. $appId = 'wx5e111b790c719cae';
  421. $appSecret = 'a4e2a6d2a4fd06bd9e4ba279fd95e2d8';
  422. $url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=".$appId."&secret=".$appSecret;
  423. $ch = curl_init();//初始化curl
  424. curl_setopt($ch, CURLOPT_URL,$url); //要访问的地址
  425. curl_setopt($ch,CURLOPT_RETURNTRANSFER,1);
  426. curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);//跳过证书验证
  427. curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false); // 从证书中检查SSL加密算法是否存在
  428. $data = json_decode(curl_exec($ch),true);
  429. if(curl_errno($ch)){
  430. var_dump(curl_error($ch)); //打印错误信息
  431. }
  432. curl_close($ch);//关闭curl
  433. return $data['access_token'];
  434. }
  435. //根据身份证号码获取生日
  436. function get_birthday($idcard) {
  437. if(empty($idcard)) return null;
  438. $bir = substr($idcard, 6, 8);
  439. $year = (int) substr($bir, 0, 4);
  440. $month = (int) substr($bir, 4, 2);
  441. $day = (int) substr($bir, 6, 2);
  442. return $year . '-' . $month . '-' . $day;
  443. }
  444. }