common.php 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374
  1. <?php
  2. /**
  3. * Niushop商城系统 - 团队十年电商经验汇集巨献!
  4. * =========================================================
  5. * Copy right 2019-2029 山西牛酷信息科技有限公司, 保留所有权利。
  6. * ----------------------------------------------
  7. * 官方网址: https://www.niushop.com.cn
  8. * 这不是一个自由软件!您只能在不用于商业目的的前提下对程序代码进行修改和使用。
  9. * 任何企业和个人不允许对程序代码以任何形式任何目的再发布。
  10. * =========================================================
  11. */
  12. // 应用公共文件
  13. // 除了 E_NOTICE,报告其他所有错误
  14. //error_reporting(E_ALL ^ E_NOTICE);
  15. //error_reporting(E_ERROR | E_WARNING | E_PARSE);
  16. error_reporting(E_NOTICE);
  17. use extend\QRcode as QRcode;
  18. use think\facade\Session;
  19. use think\facade\Event;
  20. use app\model\system\Addon;
  21. /*****************************************************基础函数*********************************************************/
  22. /**
  23. * 把返回的数据集转换成Tree
  24. *
  25. * @param array $list
  26. * 要转换的数据集
  27. * @param string $pid
  28. * parent标记字段
  29. * @param string $level
  30. * level标记字段
  31. * @return array
  32. */
  33. function list_to_tree($list, $pk = 'id', $pid = 'pid', $child = '_child', $root = 0)
  34. {
  35. // 创建Tree
  36. $tree = [];
  37. if (!is_array($list)) :
  38. return false;
  39. endif;
  40. // 创建基于主键的数组引用
  41. $refer = [];
  42. foreach ($list as $key => $data) {
  43. $refer[ $data[ $pk ] ] = &$list[ $key ];
  44. $refer[ $data[ $pk ] ][ $child ] = [];
  45. $refer[ $data[ $pk ] ]['child_num'] = 0;
  46. }
  47. foreach ($refer as $key => $data) {
  48. // 判断是否存在parent
  49. $parentId = $data[ $pid ];
  50. if ($root == $parentId) {
  51. $tree[ $key ] = &$refer[ $key ];
  52. } else if (isset($refer[ $parentId ])) {
  53. is_object($refer[ $parentId ]) && $refer[ $parentId ] = $refer[ $parentId ]->toArray();
  54. $parent = &$refer[ $parentId ];
  55. $parent[ $child ][ $key ] = &$refer[ $key ];
  56. $parent['child_num']++;
  57. }
  58. }
  59. return $tree;
  60. }
  61. /**
  62. * 把返回的数据集转换成Tree (不要key)
  63. *
  64. * @param array $list
  65. * 要转换的数据集
  66. * @param string $pid
  67. * parent标记字段
  68. * @param string $level
  69. * level标记字段
  70. * @return array
  71. */
  72. function list_to_tree_value($list, $pk = 'id', $pid = 'pid', $child = '_child', $root = 0)
  73. {
  74. // 创建Tree
  75. $tree = [];
  76. if (!is_array($list)) :
  77. return false;
  78. endif;
  79. // 创建基于主键的数组引用
  80. $refer = [];
  81. foreach ($list as $key => $data) {
  82. $refer[ $data[ $pk ] ] = &$list[ $key ];
  83. if ($data[$pid] == $root){
  84. $refer[ $data[ $pk ] ][ $child ] = [];
  85. }
  86. //$refer[ $data[ $pk ] ]['child_num'] = 0;
  87. }
  88. //echo "<pre>";print_r($refer);exit;
  89. foreach ($refer as $key => $data) {
  90. // 判断是否存在parent
  91. $parentId = $data[ $pid ];
  92. //echo "<pre>";print_r($parentId);exit;
  93. if ($root == $parentId) {
  94. $tree[] = &$refer[ $key ];
  95. } else if (isset($refer[ $parentId ])) {
  96. is_object($refer[ $parentId ]) && $refer[ $parentId ] = $refer[ $parentId ]->toArray();
  97. $parent = &$refer[ $parentId ];
  98. $parent[ $child ][] = &$refer[ $key ];
  99. //$parent['child_num']++;
  100. }
  101. }
  102. //echo "<pre>";print_r($tree);exit;
  103. return $tree;
  104. }
  105. /**
  106. * 将list_to_tree的树还原成列表
  107. *
  108. * @param array $tree
  109. * 原来的树
  110. * @param string $child
  111. * 孩子节点的键
  112. * @param string $order
  113. * 排序显示的键,一般是主键 升序排列
  114. * @param array $list
  115. * 过渡用的中间数组,
  116. * @return array 返回排过序的列表数组
  117. */
  118. function tree_to_list($tree, $child = '_child', $order = 'id', &$list = array())
  119. {
  120. if (is_array($tree)) {
  121. foreach ($tree as $key => $value) {
  122. $reffer = $value;
  123. if (isset($reffer[ $child ])) {
  124. unset($reffer[ $child ]);
  125. tree_to_list($value[ $child ], $child, $order, $list);
  126. }
  127. $list[] = $reffer;
  128. }
  129. $list = list_sort_by($list, $order, $sortby = 'asc');
  130. }
  131. return $list;
  132. }
  133. /**
  134. * 对查询结果集进行排序
  135. *
  136. * @access public
  137. * @param array $list
  138. * 查询结果
  139. * @param string $field
  140. * 排序的字段名
  141. * @param array $sortby
  142. * 排序类型
  143. * asc正向排序 desc逆向排序 nat自然排序
  144. * @return array
  145. */
  146. function list_sort_by($list, $field, $sortby = 'asc')
  147. {
  148. if (is_array($list)) {
  149. $refer = $resultSet = array();
  150. foreach ($list as $i => $data)
  151. $refer[ $i ] = &$data[ $field ];
  152. switch ($sortby) {
  153. case 'asc': // 正向排序
  154. asort($refer);
  155. break;
  156. case 'desc': // 逆向排序
  157. arsort($refer);
  158. break;
  159. case 'nat': // 自然排序
  160. natcasesort($refer);
  161. break;
  162. }
  163. foreach ($refer as $key => $val)
  164. $resultSet[] = &$list[ $key ];
  165. return $resultSet;
  166. }
  167. return false;
  168. }
  169. /**
  170. * 对象转化为数组
  171. * @param object $obj
  172. */
  173. function object_to_array($obj)
  174. {
  175. if (is_object($obj)) {
  176. $obj = (array) $obj;
  177. }
  178. if (is_array($obj)) {
  179. foreach ($obj as $key => $value) {
  180. $obj[ $key ] = object_to_array($value);
  181. }
  182. }
  183. return $obj;
  184. }
  185. /**
  186. * 系统加密方法
  187. *
  188. * @param string $data
  189. * 要加密的字符串
  190. * @param string $key
  191. * 加密密钥
  192. * @param int $expire
  193. * 过期时间 单位 秒
  194. * @return string
  195. */
  196. function encrypt($data, $key = '', $expire = 0)
  197. {
  198. $key = md5(empty ($key) ? 'niucloud' : $key);
  199. $data = base64_encode($data);
  200. $x = 0;
  201. $len = strlen($data);
  202. $l = strlen($key);
  203. $char = '';
  204. for ($i = 0; $i < $len; $i++) {
  205. if ($x == $l)
  206. $x = 0;
  207. $char .= substr($key, $x, 1);
  208. $x++;
  209. }
  210. $str = sprintf('%010d', $expire ? $expire + time() : 0);
  211. for ($i = 0; $i < $len; $i++) {
  212. $str .= chr(ord(substr($data, $i, 1)) + (ord(substr($char, $i, 1))) % 256);
  213. }
  214. return str_replace(array(
  215. '+',
  216. '/',
  217. '='
  218. ), array(
  219. '-',
  220. '_',
  221. ''
  222. ), base64_encode($str));
  223. }
  224. /**
  225. * 系统解密方法
  226. *
  227. * @param string $data
  228. * 要解密的字符串 (必须是encrypt方法加密的字符串)
  229. * @param string $key
  230. * 加密密钥
  231. * @return string
  232. */
  233. function decrypt($data, $key = '')
  234. {
  235. $key = md5(empty ($key) ? 'niucloud' : $key);
  236. $data = str_replace(array(
  237. '-',
  238. '_'
  239. ), array(
  240. '+',
  241. '/'
  242. ), $data);
  243. $mod4 = strlen($data) % 4;
  244. if ($mod4) {
  245. $data .= substr('====', $mod4);
  246. }
  247. $data = base64_decode($data);
  248. $expire = substr($data, 0, 10);
  249. $data = substr($data, 10);
  250. if ($expire > 0 && $expire < time()) {
  251. return '';
  252. }
  253. $x = 0;
  254. $len = strlen($data);
  255. $l = strlen($key);
  256. $char = $str = '';
  257. for ($i = 0; $i < $len; $i++) {
  258. if ($x == $l)
  259. $x = 0;
  260. $char .= substr($key, $x, 1);
  261. $x++;
  262. }
  263. for ($i = 0; $i < $len; $i++) {
  264. if (ord(substr($data, $i, 1)) < ord(substr($char, $i, 1))) {
  265. $str .= chr((ord(substr($data, $i, 1)) + 256) - ord(substr($char, $i, 1)));
  266. } else {
  267. $str .= chr(ord(substr($data, $i, 1)) - ord(substr($char, $i, 1)));
  268. }
  269. }
  270. return base64_decode($str);
  271. }
  272. /**
  273. * 数据签名认证
  274. */
  275. function data_auth_sign($data)
  276. {
  277. // 数据类型检测
  278. if (!is_array($data)) {
  279. $data = (array) $data;
  280. }
  281. ksort($data); // 排序
  282. $code = http_build_query($data); // url编码并生成query字符串
  283. $sign = sha1($code); // 生成签名
  284. return $sign;
  285. }
  286. /**
  287. * 重写md5加密方式
  288. *
  289. * @param unknown $str
  290. * @return string
  291. */
  292. function data_md5($str)
  293. {
  294. return '' === $str ? '' : md5(md5($str) . 'NiuCloud');
  295. }
  296. /**
  297. * 时间戳转时间
  298. */
  299. function time_to_date($time_stamp, $format = 'Y-m-d H:i:s')
  300. {
  301. if ($time_stamp > 0) {
  302. $time = date($format, $time_stamp);
  303. } else {
  304. $time = "";
  305. }
  306. return $time;
  307. }
  308. /**
  309. * 时间转时间戳
  310. */
  311. function date_to_time($date)
  312. {
  313. $time_stamp = strtotime($date);
  314. return $time_stamp;
  315. }
  316. /**
  317. * 获取唯一随机字符串
  318. * 创建时间:2018年8月7日15:54:16
  319. */
  320. function unique_random($len = 10)
  321. {
  322. $str = 'qwertyuiopasdfghjklzxcvbnm';
  323. str_shuffle($str);
  324. $res = 'nc_' . substr(str_shuffle($str), 0, $len) . date('is');
  325. return $res;
  326. }
  327. /**
  328. * 生成随机数
  329. * @param int $length
  330. * @return string
  331. */
  332. function random_keys($length)
  333. {
  334. $pattern = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLOMNOPQRSTUVWXYZ';
  335. $key = '';
  336. for ($i = 0; $i < $length; $i++) {
  337. $key .= $pattern{mt_rand(0, 35)}; //生成php随机数
  338. }
  339. return $key;
  340. }
  341. /**
  342. * 发送HTTP请求方法,目前只支持CURL发送请求
  343. *
  344. * @param string $url
  345. * 请求URL
  346. * @param array $params
  347. * 请求参数
  348. * @param string $method
  349. * 请求方法GET/POST
  350. * @return array $data 响应数据
  351. */
  352. function http($url, $timeout = 30, $header = array())
  353. {
  354. if (!function_exists('curl_init')) {
  355. throw new Exception('server not install curl');
  356. }
  357. $ch = curl_init();
  358. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  359. curl_setopt($ch, CURLOPT_HEADER, true);
  360. curl_setopt($ch, CURLOPT_URL, $url);
  361. curl_setopt($ch, CURLOPT_TIMEOUT, $timeout);
  362. if (!empty($header)) {
  363. curl_setopt($ch, CURLOPT_HTTPHEADER, $header);
  364. }
  365. $data = curl_exec($ch);
  366. list ($header, $data) = explode("\r\n\r\n", $data);
  367. $http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
  368. if ($http_code == 301 || $http_code == 302) {
  369. $matches = array();
  370. preg_match('/Location:(.*?)\n/', $header, $matches);
  371. $url = trim(array_pop($matches));
  372. curl_setopt($ch, CURLOPT_URL, $url);
  373. curl_setopt($ch, CURLOPT_HEADER, false);
  374. $data = curl_exec($ch);
  375. }
  376. if ($data == false) {
  377. curl_close($ch);
  378. }
  379. @curl_close($ch);
  380. return $data;
  381. }
  382. /**
  383. * 替换数组元素
  384. * @param array $array 数组
  385. * @param array $replace 替换元素['key' => 'value', 'key' => 'value']
  386. */
  387. function replace_array_element($array, $replace)
  388. {
  389. foreach ($replace as $k => $v) {
  390. if ($v == "unset" || $v == "") {
  391. unset($array[ $k ]);
  392. } else {
  393. $array[ $k ] = $v;
  394. }
  395. }
  396. return $array;
  397. }
  398. /**
  399. * 过滤特殊符号
  400. * 创建时间:2018年1月30日15:39:32
  401. * @param unknown $string
  402. * @return mixed
  403. */
  404. function ihtmlspecialchars($string)
  405. {
  406. if (is_array($string)) {
  407. foreach ($string as $key => $val) {
  408. $string[ $key ] = ihtmlspecialchars($val);
  409. }
  410. } else {
  411. $string = preg_replace('/&amp;((#(d{3,5}|x[a-fa-f0-9]{4})|[a-za-z][a-z0-9]{2,5});)/', '&\1',
  412. str_replace(array( '&', '"', '<', '>' ), array( '&amp;', '&quot;', '&lt;', '&gt;' ), $string));
  413. }
  414. return $string;
  415. }
  416. /********************************************* 插件,站点相关函数 ************************************************************************************
  417. *
  418. * /**
  419. * 插件显示内容里生成访问插件的url
  420. *
  421. * @param string $url
  422. * url
  423. * @param array $param
  424. * 参数
  425. * 格式:addon_url('HelloWorld://sitehome/Game/index', [])
  426. */
  427. function addon_url($url, $param = array())
  428. {
  429. if (strpos($url, 'http://') === 0 || strpos($url, 'https://') === 0) {
  430. return $url;
  431. }
  432. $parse_url = parse_url($url);
  433. $addon = isset($parse_url['scheme']) ? $parse_url['scheme'] : '';
  434. $controller = isset($parse_url['host']) ? $parse_url['host'] : '';
  435. $action = trim($parse_url['path'], '/');
  436. /* 解析URL带的参数 */
  437. if (isset($parse_url['query'])) {
  438. parse_str($parse_url['query'], $query);
  439. $param = array_merge($query, $param);
  440. }
  441. $url = $addon . '/' . $controller . '/' . $action;
  442. if (empty($addon)) {
  443. $url = $controller . '/' . $action;
  444. if (empty($controller)) {
  445. $url = $action;
  446. }
  447. }
  448. return url($url, $param);
  449. }
  450. /**
  451. * Url生成(重写url函数)
  452. * @param string $url 路由地址
  453. */
  454. function url(string $url = '', $vars = [])
  455. {
  456. if (!empty($vars)) {
  457. if (is_array($vars)) {
  458. $vars = http_build_query($vars);
  459. }
  460. $tag = REWRITE_MODULE ? '?' : '&';
  461. $var_url = $tag . $vars;
  462. } else {
  463. $var_url = '';
  464. }
  465. $url = $url . '.html';
  466. return ROOT_URL . '/' . $url . $var_url;
  467. }
  468. /**
  469. * 解析url的插件,模块,控制器,方法
  470. * @param unknown $url
  471. */
  472. function url_action($url)
  473. {
  474. if (empty($url)) {
  475. return [
  476. 'addon' => '',
  477. 'model' => 'index',
  478. 'controller' => 'index',
  479. 'action' => 'index'
  480. ];
  481. }
  482. if (!strstr($url, '://')) {
  483. $url_array = explode('/', $url);
  484. return [
  485. 'addon' => '',
  486. 'model' => $url_array[0],
  487. 'controller' => $url_array[1],
  488. 'action' => $url_array[2]
  489. ];
  490. } else {
  491. $url_addon_array = explode('://', $url);
  492. $addon = $url_addon_array[0];
  493. $url_array = explode('/', $url_addon_array[1]);
  494. return [
  495. 'addon' => $addon,
  496. 'model' => $url_array[0],
  497. 'controller' => $url_array[1],
  498. 'action' => $url_array[2]
  499. ];
  500. }
  501. }
  502. /**
  503. * 检测插件是否存在
  504. * @param unknown $name
  505. * @return number
  506. */
  507. function addon_is_exit($name)
  508. {
  509. $addon_model = new Addon();
  510. $addon_data = $addon_model->getAddonList([], 'name');
  511. $addons = array_column($addon_data['data'], 'name');
  512. if(in_array($name, $addons))
  513. {
  514. return 1;
  515. }else{
  516. return 0;
  517. }
  518. }
  519. /***************************************************niucloud系统函数***************************************************/
  520. /**
  521. * 处理事件
  522. *
  523. * @param string $event
  524. * 钩子名称
  525. * @param mixed $args
  526. * 传入参数
  527. * @param bool $once
  528. * 只获取一个有效返回值
  529. * @return void
  530. */
  531. function event($event, $args = [], $once = false)
  532. {
  533. $res = Event::trigger($event, $args);
  534. if (is_array($res)) {
  535. $res = array_filter($res);
  536. sort($res);
  537. }
  538. //只返回一个结果集
  539. if ($once) {
  540. return isset($res[0]) ? $res[0] : '';
  541. }
  542. return $res;
  543. }
  544. /**
  545. * 错误返回值函数
  546. * @param int $code
  547. * @param string $message
  548. * @param string $data
  549. * @return array
  550. */
  551. function error($code = -1, $message = '', $data = '')
  552. {
  553. return [
  554. 'code' => $code,
  555. 'message' => $message,
  556. 'data' => $data
  557. ];
  558. }
  559. /**
  560. * 返回值函数
  561. * @param int $code
  562. * @param string $message
  563. * @param string $data
  564. * @return array
  565. */
  566. function success($code = 0, $message = '', $data = '')
  567. {
  568. return [
  569. 'code' => $code,
  570. 'message' => $message,
  571. 'data' => $data
  572. ];
  573. }
  574. /**
  575. * 实例化Model
  576. *
  577. * @param string $name
  578. * Model名称
  579. */
  580. function model($table = '')
  581. {
  582. return new \app\model\Model($table);
  583. }
  584. /**
  585. * 获取带有表前缀的表名
  586. * @param string $table
  587. */
  588. function table($table = '')
  589. {
  590. return config('database.connections.prefix') . $table;
  591. }
  592. /**
  593. * 获取图片的真实路径
  594. *
  595. * @param string $path 图片初始路径
  596. * @param string $type 类型 big、mid、small
  597. * @return string 图片的真实路径
  598. */
  599. function img($path, $type = '')
  600. {
  601. $start = strripos($path, '.');
  602. $type = $type ? '_' . $type : '';
  603. $first = explode("/", $path);
  604. $path = substr_replace($path, $type, $start, 0);
  605. if (stristr($path, "http://") === false && stristr($path, "https://") === false) {
  606. if (is_numeric($first[0])) {
  607. $true_path = __ROOT__ . '/upload/' . $path;
  608. } else {
  609. $true_path = __ROOT__ . '/' . $path;
  610. }
  611. } else {
  612. $true_path = $path;
  613. }
  614. return $true_path;
  615. }
  616. /**
  617. * 获取标准二维码格式
  618. *
  619. * @param string $url
  620. * @param string $path
  621. * @param string $ext
  622. */
  623. function qrcode($url, $path, $qrcode_name)
  624. {
  625. if (!is_dir($path)) {
  626. $mode = intval('0777', 8);
  627. mkdir($path, $mode, true);
  628. chmod($path, $mode);
  629. }
  630. $path = $path . '/' . $qrcode_name . '.png';
  631. if (file_exists($path)) {
  632. unlink($path);
  633. }
  634. QRcode::png($url, $path, '', 4, 1);
  635. return $path;
  636. }
  637. /**
  638. * 前端页面api请求(通过api接口实现)
  639. * @param string $method
  640. * @param array $params
  641. * @return mixed
  642. */
  643. function api($method, $params = [])
  644. {
  645. //本地访问
  646. $data = get_api_data($method, $params);
  647. return $data;
  648. }
  649. /**
  650. * 获取Api类
  651. *
  652. * @param string $method
  653. */
  654. function get_api_data($method, $params)
  655. {
  656. $method_array = explode('.', $method);
  657. if ($method_array[0] == 'System') {
  658. $class_name = 'app\\api\\controller\\' . $method_array[1];
  659. if (!class_exists($class_name)) {
  660. return error();
  661. }
  662. $api_model = new $class_name($params);
  663. } else {
  664. $class_name = "addon\\{$method_array[0]}\\api\\controller\\" . $method_array[1];
  665. if (!class_exists($class_name)) {
  666. return error();
  667. }
  668. $api_model = new $class_name($params);
  669. }
  670. $function = $method_array[2];
  671. $data = $api_model->$function($params);
  672. return $data;
  673. }
  674. /**
  675. * 根据年份计算生肖
  676. * @param unknown $year
  677. */
  678. function get_zodiac($year)
  679. {
  680. $animals = array(
  681. '鼠', '牛', '虎', '兔', '龙', '蛇', '马', '羊', '猴', '鸡', '狗', '猪'
  682. );
  683. $key = ($year - 1900) % 12;
  684. return $animals[ $key ];
  685. }
  686. /**
  687. * 计算.星座
  688. * @param int $month 月份
  689. * @param int $day 日期
  690. * @return str
  691. */
  692. function get_constellation($month, $day)
  693. {
  694. $constellations = array(
  695. '水瓶座', '双鱼座', '白羊座', '金牛座', '双子座', '巨蟹座',
  696. '狮子座', '处女座', '天秤座', '天蝎座', '射手座', '摩羯座'
  697. );
  698. if ($day <= 22) {
  699. if (1 != $month) {
  700. $constellation = $constellations[ $month - 2 ];
  701. } else {
  702. $constellation = $constellations[11];
  703. }
  704. } else {
  705. $constellation = $constellations[ $month - 1 ];
  706. }
  707. return $constellation;
  708. }
  709. /**
  710. * 数组键名转化为数字
  711. * @param $data
  712. */
  713. function arr_key_to_int($data, $clild_name)
  714. {
  715. $temp_data = array_values($data);
  716. foreach ($temp_data as $k => $v) {
  717. if (!empty($v[ $clild_name ])) {
  718. $temp_data[ $k ][ $clild_name ] = arr_key_to_int($v[ $clild_name ], $clild_name);
  719. }
  720. }
  721. return $temp_data;
  722. }
  723. /**
  724. * 以天为单位 计算间隔内的日期数组
  725. * @param $start_time
  726. * @param $end_time
  727. * @return array
  728. */
  729. function period_group($start_time, $end_time, $format = 'Ymd')
  730. {
  731. $type_time = 3600 * 24;
  732. $data = [];
  733. for ($i = $start_time; $i <= $end_time; $i += $type_time) {
  734. $data[] = date($format, $i);
  735. }
  736. return $data;
  737. }
  738. /**
  739. * 数组删除另一个数组
  740. * @param $arr
  741. * @param $del_arr
  742. * @return mixed
  743. */
  744. function arr_del_arr($arr, $del_arr)
  745. {
  746. foreach ($arr as $k => $v) {
  747. if (in_array($v, $del_arr)) {
  748. unset($arr[ $k ]);
  749. }
  750. }
  751. sort($arr);
  752. return $arr;
  753. }
  754. /**
  755. * 检测登录(应用于h5网页检测登录)
  756. * @param unknown $url
  757. */
  758. function check_auth($url = '')
  759. {
  760. $access_token = Session::get("access_token_" . request()->siteid());
  761. if (empty($access_token)) {
  762. if (!empty($url)) {
  763. Session::set("redirect_login_url", $url);
  764. }
  765. //尚未登录(直接跳转)
  766. return error(url('wap/login/login'));
  767. }
  768. $member_info = cache("member_info_" . request()->siteid() . $access_token);
  769. if (empty($member_info)) {
  770. $member_info = api("System.Member.memberInfo", [ 'access_token' => $access_token ]);
  771. if ($member_info['code'] == 0) {
  772. $member_info = $member_info['data'];
  773. cache("member_info_" . request()->siteid() . $access_token, $member_info);
  774. }
  775. }
  776. $member_info['access_token'] = $access_token;
  777. return success($member_info);
  778. }
  779. /**
  780. * 分割sql语句
  781. * @param string $content sql内容
  782. * @param bool $string 如果为真,则只返回一条sql语句,默认以数组形式返回
  783. * @param array $replace 替换前缀,如:['my_' => 'me_'],表示将表前缀my_替换成me_
  784. * @return array|string 除去注释之后的sql语句数组或一条语句
  785. */
  786. function parse_sql($content = '', $string = false, $replace = [])
  787. {
  788. // 纯sql内容
  789. $pure_sql = [];
  790. // 被替换的前缀
  791. $from = '';
  792. // 要替换的前缀
  793. $to = '';
  794. // 替换表前缀
  795. if (!empty($replace)) {
  796. $to = current($replace);
  797. $from = current(array_flip($replace));
  798. }
  799. if ($content != '') {
  800. // 多行注释标记
  801. $comment = false;
  802. // 按行分割,兼容多个平台
  803. $content = str_replace([ "\r\n", "\r" ], "\n", $content);
  804. $content = explode("\n", trim($content));
  805. // 循环处理每一行
  806. foreach ($content as $key => $line) {
  807. // 跳过空行
  808. if ($line == '') {
  809. continue;
  810. }
  811. // 跳过以#或者--开头的单行注释
  812. if (preg_match("/^(#|--)/", $line)) {
  813. continue;
  814. }
  815. // 跳过以/**/包裹起来的单行注释
  816. if (preg_match("/^\/\*(.*?)\*\//", $line)) {
  817. continue;
  818. }
  819. // 多行注释开始
  820. if (substr($line, 0, 2) == '/*') {
  821. $comment = true;
  822. continue;
  823. }
  824. // 多行注释结束
  825. if (substr($line, -2) == '*/') {
  826. $comment = false;
  827. continue;
  828. }
  829. // 多行注释没有结束,继续跳过
  830. if ($comment) {
  831. continue;
  832. }
  833. // 替换表前缀
  834. if ($from != '') {
  835. $line = str_replace('`' . $from, '`' . $to, $line);
  836. }
  837. // sql语句
  838. $pure_sql[] = $line;
  839. }
  840. // 只返回一条语句
  841. if ($string) {
  842. return implode($pure_sql, "");
  843. }
  844. // 以数组形式返回sql语句
  845. $pure_sql = implode($pure_sql, "\n");
  846. $pure_sql = explode(";\n", $pure_sql);
  847. }
  848. return $pure_sql;
  849. }
  850. /**
  851. * 执行sql
  852. * @param unknown $sql_name
  853. */
  854. function execute_sql($sql_name)
  855. {
  856. $sql_string = file_get_contents($sql_name);
  857. $sql_string = str_replace("{{prefix}}", config("database.connections.mysql.prefix"), $sql_string);
  858. if ($sql_string) {
  859. $sql = explode(";\n", str_replace("\r", "\n", $sql_string));
  860. foreach ($sql as $value) {
  861. $value = trim($value);
  862. if (!empty($value)) {
  863. \think\facade\Db::execute($value);
  864. }
  865. }
  866. }
  867. }
  868. /**
  869. * 检测目录读写权限
  870. */
  871. function check_dir_iswritable($dir)
  872. {
  873. $testDir = $dir;
  874. sp_dir_create($testDir);
  875. if (sp_testwrite($testDir)) {
  876. return true;
  877. } else {
  878. return false;
  879. }
  880. }
  881. /**
  882. * 检查测试文件是否可写入
  883. */
  884. function sp_testwrite($d)
  885. {
  886. $tfile = "_test.txt";
  887. $fp = @fopen($d . "/" . $tfile, "w");
  888. if (!$fp) {
  889. return false;
  890. }
  891. fclose($fp);
  892. $rs = @unlink($d . "/" . $tfile);
  893. if ($rs) {
  894. return true;
  895. }
  896. return false;
  897. }
  898. /**
  899. * 检查文件是否创建
  900. */
  901. function sp_dir_create($path, $mode = 0777)
  902. {
  903. if (is_dir($path))
  904. return true;
  905. $ftp_enable = 0;
  906. $path = sp_dir_path($path);
  907. $temp = explode('/', $path);
  908. $cur_dir = '';
  909. $max = count($temp) - 1;
  910. for ($i = 0; $i < $max; $i++) {
  911. $cur_dir .= $temp[ $i ] . '/';
  912. if (@is_dir($cur_dir))
  913. continue;
  914. @mkdir($cur_dir, 0777, true);
  915. @chmod($cur_dir, 0777);
  916. }
  917. return is_dir($path);
  918. }
  919. /**
  920. * 判断目录是否为空
  921. * @param $dir
  922. * @return bool
  923. */
  924. function dir_is_empty($dir)
  925. {
  926. $handle = opendir($dir);
  927. while (false !== ($entry = readdir($handle))) {
  928. if ($entry != "." && $entry != "..") {
  929. return FALSE;
  930. }
  931. }
  932. return TRUE;
  933. }
  934. /**
  935. * 创建文件夹
  936. *
  937. * @param string $path 文件夹路径
  938. * @param int $mode 访问权限
  939. * @param bool $recursive 是否递归创建
  940. * @return bool
  941. */
  942. function dir_mkdir($path = '', $mode = 0777, $recursive = true)
  943. {
  944. clearstatcache();
  945. if (!is_dir($path)) {
  946. mkdir($path, $mode, $recursive);
  947. return chmod($path, $mode);
  948. }
  949. return true;
  950. }
  951. /**
  952. * 文件夹文件拷贝
  953. *
  954. * @param string $src 来源文件夹
  955. * @param string $dst 目的地文件夹
  956. * @return bool
  957. */
  958. function dir_copy($src = '', $dst = '')
  959. {
  960. if (empty($src) || empty($dst)) {
  961. return false;
  962. }
  963. $dir = opendir($src);
  964. dir_mkdir($dst);
  965. while (false !== ($file = readdir($dir))) {
  966. if (($file != '.') && ($file != '..')) {
  967. if (is_dir($src . '/' . $file)) {
  968. dir_copy($src . '/' . $file, $dst . '/' . $file);
  969. } else {
  970. copy($src . '/' . $file, $dst . '/' . $file);
  971. }
  972. }
  973. }
  974. closedir($dir);
  975. return true;
  976. }
  977. /**查询存在目录
  978. * @param $dir
  979. */
  980. function sp_exist_dir($dir)
  981. {
  982. $is_exist = false;
  983. $is_write = false;
  984. while (!$is_exist) {
  985. $dir = dirname($dir);
  986. if (is_dir($dir) || $dir == ".") {
  987. $is_exist = true;
  988. if (is_writeable($dir)) {
  989. $is_write = true;
  990. }
  991. }
  992. }
  993. return $is_write;
  994. }
  995. /**
  996. * 拼接字符串
  997. * @param $string
  998. * @param $delimiter 分割字符
  999. * @param $value
  1000. */
  1001. function string_split($string, $delimiter, $value)
  1002. {
  1003. return empty($string) ? $value : $string . $delimiter . $value;
  1004. }
  1005. /**
  1006. * $str为要进行截取的字符串,$length为截取长度(汉字算一个字,字母算半个字
  1007. * @param $str
  1008. * @param $length
  1009. * @return string
  1010. */
  1011. function str_sub($str, $length = 10)
  1012. {
  1013. return mb_substr($str, 0, $length, 'UTF-8') . "...";
  1014. }
  1015. /**
  1016. * 删除缓存文件使用
  1017. * @param $dir
  1018. */
  1019. function rmdirs($dir)
  1020. {
  1021. $dir = 'runtime/' . $dir;
  1022. $dh = opendir($dir);
  1023. while ($file = readdir($dh)) {
  1024. if ($file != "." && $file != "..") {
  1025. $fullpath = $dir . "/" . $file;
  1026. if (is_dir($fullpath)) {
  1027. rmdirs($fullpath);
  1028. } else {
  1029. unlink($fullpath);
  1030. }
  1031. }
  1032. }
  1033. closedir($dh);
  1034. }
  1035. /**
  1036. * 以天为单位 计算间隔内的日期数组
  1037. * @param $srart_time
  1038. * @param $end_time
  1039. * @return array
  1040. */
  1041. function periodGroup($srart_time, $end_time, $format = 'Ymd')
  1042. {
  1043. $type_time = 3600 * 24;
  1044. $data = [];
  1045. for ($i = $srart_time; $i <= $end_time; $i += $type_time) {
  1046. $data[] = date($format, $i);
  1047. }
  1048. return $data;
  1049. }
  1050. //解决个别中文乱码
  1051. function mbStrreplace($content, $to_encoding = "UTF-8", $from_encoding = "GBK")
  1052. {
  1053. $content = mb_convert_encoding($content, $to_encoding, $from_encoding);
  1054. $str = mb_convert_encoding(" ", $to_encoding, $from_encoding);
  1055. $content = mb_eregi_replace($str, " ", $content);
  1056. $content = mb_convert_encoding($content, $from_encoding, $to_encoding);
  1057. $content = trim($content);
  1058. return $content;
  1059. }
  1060. /**
  1061. * 将非UTF-8字符集的编码转为UTF-8
  1062. *
  1063. * @param mixed $mixed 源数据
  1064. *
  1065. * @return mixed utf-8格式数据
  1066. */
  1067. function charset2utf8($mixed)
  1068. {
  1069. if (is_array($mixed)) {
  1070. foreach ($mixed as $k => $v) {
  1071. if (is_array($v)) {
  1072. $mixed[ $k ] = charsetToUTF8($v);
  1073. } else {
  1074. $encode = mb_detect_encoding($v, array( 'ASCII', 'UTF-8', 'GB2312', 'GBK', 'BIG5' ));
  1075. if ($encode == 'EUC-CN') {
  1076. $mixed[ $k ] = iconv('GBK', 'UTF-8', $v);
  1077. }
  1078. }
  1079. }
  1080. } else {
  1081. $encode = mb_detect_encoding($mixed, array( 'ASCII', 'UTF-8', 'GB2312', 'GBK', 'BIG5' ));
  1082. if ($encode == 'EUC-CN') {
  1083. $mixed = iconv('GBK', 'UTF-8', $mixed);
  1084. }
  1085. }
  1086. return $mixed;
  1087. }
  1088. /**
  1089. * 过滤bom
  1090. * @param $filename
  1091. */
  1092. function check_bom ($filename) {
  1093. $contents = file_get_contents($filename);
  1094. $charset[1] = substr($contents, 0, 1);
  1095. $charset[2] = substr($contents, 1, 1);
  1096. $charset[3] = substr($contents, 2, 1);
  1097. if (ord($charset[1]) == 239 && ord($charset[2]) == 187 && ord($charset[3]) == 191) {
  1098. $rest = substr($contents, 3);
  1099. return $rest;
  1100. }else{
  1101. return $contents;
  1102. }
  1103. }
  1104. /**
  1105. * 判断 文件/目录 是否可写(取代系统自带的 is_writeable 函数)
  1106. *
  1107. * @param string $file 文件/目录
  1108. * @return boolean
  1109. */
  1110. function is_write($file)
  1111. {
  1112. if (is_dir($file)) {
  1113. $dir = $file;
  1114. if ($fp = @fopen("$dir/test.txt", 'w')) {
  1115. @fclose($fp);
  1116. @unlink("$dir/test.txt");
  1117. $writeable = true;
  1118. } else {
  1119. $writeable = false;
  1120. }
  1121. } else {
  1122. if ($fp = @fopen($file, 'a+')) {
  1123. @fclose($fp);
  1124. $writeable = true;
  1125. } else {
  1126. $writeable = false;
  1127. }
  1128. }
  1129. return $writeable;
  1130. }
  1131. /**
  1132. * 是否是url链接
  1133. * @param unknown $string
  1134. * @return boolean
  1135. */
  1136. function is_url($string)
  1137. {
  1138. if (strstr($string, 'http://') === false && strstr($string, 'https://') === false) {
  1139. return false;
  1140. } else {
  1141. return true;
  1142. }
  1143. }
  1144. /**
  1145. * 根据两点间的经纬度计算距离
  1146. * @param float $lng1
  1147. * @param float $lat1
  1148. * @param float $lng2
  1149. * @param float $lat2
  1150. * @return number
  1151. */
  1152. function getDistance(float $lng1, float $lat1, float $lng2, float $lat2)
  1153. {
  1154. if (($lng1 == $lng2) && ($lat1 == $lat2)) {
  1155. return 0;
  1156. }
  1157. //将角度转为狐度
  1158. $radLat1 = deg2rad($lat1);//deg2rad()函数将角度转换为弧度
  1159. $radLat2 = deg2rad($lat2);
  1160. $radLng1 = deg2rad($lng1);
  1161. $radLng2 = deg2rad($lng2);
  1162. $a = $radLat1 - $radLat2;
  1163. $b = $radLng1 - $radLng2;
  1164. $s = 2 * asin(sqrt(pow(sin($a / 2), 2) + cos($radLat1) * cos($radLat2) * pow(sin($b / 2), 2))) * 6378.137 * 1000;
  1165. return $s;
  1166. }
  1167. /**
  1168. * 获取token
  1169. */
  1170. function getToken(){
  1171. // $token = cache('hx_token');
  1172. // if (!$token){
  1173. $data = array(
  1174. "grant_type" => "client_credentials",
  1175. "client_id" => 'YXA6pmsN9WTbQQuf2no5YmnNhA',
  1176. "client_secret" => 'YXA6Gp3RiQOG_AW4stlt9gMqTAxyVKw'
  1177. );
  1178. $url = "https://a1.easemob.com/1118220221135762/mendian/token";
  1179. $rs = json_decode(curl($url, $data), true);
  1180. cache('hx_token',$rs['access_token'],$rs['expires_in']-1500);
  1181. $token = $rs['access_token'];
  1182. // }
  1183. return $token;
  1184. }
  1185. /**
  1186. * 环信注册
  1187. */
  1188. function huanxin_zhuce($username,$nickname){
  1189. $param = array (
  1190. "username" => $username,
  1191. "password" => '12345678',
  1192. "nickname" => $nickname
  1193. );
  1194. $url = "https://a1.easemob.com/1118220221135762/mendian/users";
  1195. $res = huanxin_curl_request($url, json_encode($param));
  1196. $tokenResult = json_decode($res, true);
  1197. //dump($tokenResult);
  1198. }
  1199. /**
  1200. * 添加好友
  1201. */
  1202. function add_friend($owner_username,$friend_username){
  1203. $param = array ();
  1204. $header = array(
  1205. 'Authorization: Bearer '.getToken()
  1206. );
  1207. $url = "https://a1.easemob.com/1118220221135762/mendian/users/".$owner_username."/contacts/users/".$friend_username;
  1208. $res = huanxin_curl_request($url,$param,$header,'POST');
  1209. $Result = json_decode($res, true);
  1210. //print_r($Result);
  1211. }
  1212. /**
  1213. * 好友列表
  1214. */
  1215. function friend_list($owner_username){
  1216. $param = array ();
  1217. $header = array(
  1218. 'Authorization: Bearer '.getToken()
  1219. );
  1220. $url = "https://a1.easemob.com/1118220221135762/mendian/users/".$owner_username."/contacts/users";
  1221. $res = huanxin_curl_request($url,$param,$header,'GET');
  1222. $Result = json_decode($res, true);
  1223. return $Result['data'];
  1224. }
  1225. function huanxin_curl_request($url, $body, $header = array(), $method = "POST") {
  1226. array_push ( $header, 'Accept:application/json' );
  1227. array_push ( $header, 'Content-Type:application/json' );
  1228. $ch = curl_init ();
  1229. curl_setopt ( $ch, CURLOPT_CONNECTTIMEOUT, 60 );
  1230. curl_setopt ( $ch, CURLOPT_URL, $url );
  1231. curl_setopt ( $ch, CURLOPT_RETURNTRANSFER, 1 );
  1232. // curl_setopt($ch, $method, 1);
  1233. switch (strtoupper($method)) {
  1234. case "GET" :
  1235. curl_setopt ( $ch, CURLOPT_HTTPGET, true );
  1236. break;
  1237. case "POST" :
  1238. curl_setopt ( $ch, CURLOPT_POST, true );
  1239. break;
  1240. case "PUT" :
  1241. curl_setopt ( $ch, CURLOPT_CUSTOMREQUEST, "PUT" );
  1242. break;
  1243. case "DELETE" :
  1244. curl_setopt ( $ch, CURLOPT_CUSTOMREQUEST, "DELETE" );
  1245. break;
  1246. }
  1247. curl_setopt ( $ch, CURLOPT_USERAGENT, 'SSTS Browser/1.0' );
  1248. curl_setopt ( $ch, CURLOPT_ENCODING, 'gzip' );
  1249. curl_setopt ( $ch, CURLOPT_SSL_VERIFYPEER, FALSE );
  1250. curl_setopt ( $ch, CURLOPT_SSL_VERIFYHOST, 2 );
  1251. if (isset ( $body {3} ) > 0) {
  1252. curl_setopt ( $ch, CURLOPT_POSTFIELDS, $body );
  1253. }
  1254. if (count ( $header ) > 0) {
  1255. curl_setopt ( $ch, CURLOPT_HTTPHEADER, $header );
  1256. }
  1257. $ret = curl_exec ( $ch );
  1258. $err = curl_error ( $ch );
  1259. curl_close ( $ch );
  1260. if ($err) {
  1261. return $err;
  1262. }
  1263. return $ret;
  1264. }
  1265. function curl($url, $data, $header = false, $method = "POST")
  1266. {
  1267. $ch = curl_init($url);
  1268. curl_setopt($ch, CURLOPT_URL, $url);
  1269. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
  1270. if ($header) {
  1271. curl_setopt($ch, CURLOPT_HTTPHEADER, $header);
  1272. }
  1273. curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method);
  1274. if ($data) {
  1275. curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($data));
  1276. }
  1277. curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
  1278. curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
  1279. $ret = curl_exec($ch);
  1280. return $ret;
  1281. }