common.php 27 KB

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