UtilService.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
  8. // +----------------------------------------------------------------------
  9. // | Author: CRMEB Team <admin@crmeb.com>
  10. // +----------------------------------------------------------------------
  11. namespace service;
  12. use think\Request;
  13. use service\JsonService;
  14. class UtilService
  15. {
  16. /**
  17. * 获取POST请求的数据
  18. * @param $params
  19. * @param null $request
  20. * @param bool $suffix
  21. * @return array
  22. */
  23. public static function postMore($params, $request = null, $suffix = false)
  24. {
  25. if ($request === null) $request = Request::instance();
  26. $p = [];
  27. $i = 0;
  28. foreach ($params as $param) {
  29. if (!is_array($param)) {
  30. $p[$suffix == true ? $i++ : $param] = $request->param($param);
  31. } else {
  32. if (!isset($param[1])) $param[1] = null;
  33. if (!isset($param[2])) $param[2] = '';
  34. if (is_array($param[0])) {
  35. $name = is_array($param[1]) ? $param[0][0] . '/a' : $param[0][0] . '/' . $param[0][1];
  36. $keyName = $param[0][0];
  37. } else {
  38. $name = is_array($param[1]) ? $param[0] . '/a' : $param[0];
  39. $keyName = $param[0];
  40. }
  41. $p[$suffix == true ? $i++ : (isset($param[3]) ? $param[3] : $keyName)] = $request->param($name, $param[1], $param[2]);
  42. }
  43. }
  44. return $p;
  45. }
  46. /**
  47. * 获取请求的数据
  48. * @param $params
  49. * @param null $request
  50. * @param bool $suffix
  51. * @return array
  52. */
  53. public static function getMore($params, $request = null, $suffix = false)
  54. {
  55. if ($request === null) $request = Request::instance();
  56. $p = [];
  57. $i = 0;
  58. foreach ($params as $param) {
  59. if (!is_array($param)) {
  60. $p[$suffix == true ? $i++ : $param] = $request->param($param);
  61. } else {
  62. if (!isset($param[1])) $param[1] = null;
  63. if (!isset($param[2])) $param[2] = '';
  64. if (is_array($param[0])) {
  65. $name = is_array($param[1]) ? $param[0][0] . '/a' : $param[0][0] . '/' . $param[0][1];
  66. $keyName = $param[0][0];
  67. } else {
  68. $name = is_array($param[1]) ? $param[0] . '/a' : $param[0];
  69. $keyName = $param[0];
  70. }
  71. $p[$suffix == true ? $i++ : (isset($param[3]) ? $param[3] : $keyName)] = $request->param($name, $param[1], $param[2]);
  72. }
  73. }
  74. return $p;
  75. }
  76. public static function encrypt($string, $operation = false, $key = '', $expiry = 0)
  77. {
  78. // 动态密匙长度,相同的明文会生成不同密文就是依靠动态密匙
  79. $ckey_length = 6;
  80. // 密匙
  81. $key = md5($key);
  82. // 密匙a会参与加解密
  83. $keya = md5(substr($key, 0, 16));
  84. // 密匙b会用来做数据完整性验证
  85. $keyb = md5(substr($key, 16, 16));
  86. // 密匙c用于变化生成的密文
  87. $keyc = $ckey_length ? ($operation == false ? substr($string, 0, $ckey_length) :
  88. substr(md5(microtime()), -$ckey_length)) : '';
  89. // 参与运算的密匙
  90. $cryptkey = $keya . md5($keya . $keyc);
  91. $key_length = strlen($cryptkey);
  92. // 明文,前10位用来保存时间戳,解密时验证数据有效性,10到26位用来保存$keyb(密匙b),
  93. //解密时会通过这个密匙验证数据完整性
  94. // 如果是解码的话,会从第$ckey_length位开始,因为密文前$ckey_length位保存 动态密匙,以保证解密正确
  95. $string = $operation == false ? base64_decode(substr($string, $ckey_length)) :
  96. sprintf('%010d', $expiry ? $expiry + time() : 0) . substr(md5($string . $keyb), 0, 16) . $string;
  97. $string_length = strlen($string);
  98. $result = '';
  99. $box = range(0, 255);
  100. $rndkey = array();
  101. // 产生密匙簿
  102. for ($i = 0; $i <= 255; $i++) {
  103. $rndkey[$i] = ord($cryptkey[$i % $key_length]);
  104. }
  105. // 用固定的算法,打乱密匙簿,增加随机性,好像很复杂,实际上对并不会增加密文的强度
  106. for ($j = $i = 0; $i < 256; $i++) {
  107. $j = ($j + $box[$i] + $rndkey[$i]) % 256;
  108. $tmp = $box[$i];
  109. $box[$i] = $box[$j];
  110. $box[$j] = $tmp;
  111. }
  112. // 核心加解密部分
  113. for ($a = $j = $i = 0; $i < $string_length; $i++) {
  114. $a = ($a + 1) % 256;
  115. $j = ($j + $box[$a]) % 256;
  116. $tmp = $box[$a];
  117. $box[$a] = $box[$j];
  118. $box[$j] = $tmp;
  119. // 从密匙簿得出密匙进行异或,再转成字符
  120. $result .= chr(ord($string[$i]) ^ ($box[($box[$a] + $box[$j]) % 256]));
  121. }
  122. if ($operation == false) {
  123. // 验证数据有效性,请看未加密明文的格式
  124. if ((substr($result, 0, 10) == 0 || substr($result, 0, 10) - time() > 0) &&
  125. substr($result, 10, 16) == substr(md5(substr($result, 26) . $keyb), 0, 16)) {
  126. return substr($result, 26);
  127. } else {
  128. return '';
  129. }
  130. } else {
  131. // 把动态密匙保存在密文里,这也是为什么同样的明文,生产不同密文后能解密的原因
  132. // 因为加密后的密文可能是一些特殊字符,复制过程可能会丢失,所以用base64编码
  133. return $keyc . str_replace('=', '', base64_encode($result));
  134. }
  135. }
  136. /**
  137. * 路径转url路径
  138. * @param $path
  139. * @return string
  140. */
  141. public static function pathToUrl($path)
  142. {
  143. return trim(str_replace(DS, '/', $path), '.');
  144. }
  145. /**
  146. * url转换路径
  147. * @param $url
  148. * @return string
  149. */
  150. public static function urlToPath($url)
  151. {
  152. return ROOT_PATH . 'public/' . trim(str_replace('/', DS, $url), DS);
  153. }
  154. public static function timeTran($time)
  155. {
  156. $t = time() - $time;
  157. $f = array(
  158. '31536000' => '年',
  159. '2592000' => '个月',
  160. '604800' => '星期',
  161. '86400' => '天',
  162. '3600' => '小时',
  163. '60' => '分钟',
  164. '1' => '秒'
  165. );
  166. foreach ($f as $k => $v) {
  167. if (0 != $c = floor($t / (int)$k)) {
  168. return $c . $v . '前';
  169. }
  170. }
  171. }
  172. /**
  173. * 分级排序
  174. * @param $data
  175. * @param int $pid
  176. * @param string $field
  177. * @param string $pk
  178. * @param string $html
  179. * @param int $level
  180. * @param bool $clear
  181. * @return array
  182. */
  183. public static function sortListTier($data, $pid = 0, $field = 'pid', $pk = 'id', $html = '|-----', $level = 1, $clear = true)
  184. {
  185. static $list = [];
  186. if ($clear) $list = [];
  187. foreach ($data as $k => $res) {
  188. if ($res[$field] == $pid) {
  189. $res['html'] = str_repeat($html, $level);
  190. $list[] = $res;
  191. unset($data[$k]);
  192. self::sortListTier($data, $res[$pk], $field, $pk, $html, $level + 1, false);
  193. }
  194. }
  195. return $list;
  196. }
  197. /**
  198. * 删除公告资源
  199. * @param $url
  200. * @return \StdClass
  201. */
  202. public static function rmPublicResource($url, $isPath = false)
  203. {
  204. $path = $isPath ? $url : realpath(self::urlToPath($url));
  205. if (!$path) return JsonService::fail('删除文件不存在!');
  206. if (!file_exists($path)) return JsonService::fail('删除路径不合法!');
  207. // if(0 !== strpos($path,ROOT_PATH.'public/uploads/')) return JsonService::fail('删除路径不合法!');
  208. if (!unlink($path)) return JsonService::fail('删除文件失败!');
  209. return JsonService::successful();
  210. }
  211. /**
  212. * 是否为微信内部浏览器
  213. * @return bool
  214. */
  215. public static function isWechatBrowser()
  216. {
  217. return (strpos($_SERVER['HTTP_USER_AGENT'], 'MicroMessenger') !== false);
  218. }
  219. /**
  220. * 匿名处理
  221. * @param $name
  222. * @return string
  223. */
  224. public static function anonymity($name)
  225. {
  226. $strLen = mb_strlen($name, 'UTF-8');
  227. $min = 3;
  228. if ($strLen <= 1)
  229. return '*';
  230. if ($strLen <= $min)
  231. return mb_substr($name, 0, 1, 'UTF-8') . str_repeat('*', $min - 1);
  232. else
  233. return mb_substr($name, 0, 1, 'UTF-8') . str_repeat('*', $strLen - 1) . mb_substr($name, -1, 1, 'UTF-8');
  234. }
  235. /**
  236. * 身份证验证
  237. * @param $card
  238. * @return bool
  239. */
  240. public static function setCard($card)
  241. {
  242. $city = [11 => "北京", 12 => "天津", 13 => "河北", 14 => "山西", 15 => "内蒙古", 21 => "辽宁", 22 => "吉林", 23 => "黑龙江 ", 31 => "上海", 32 => "江苏", 33 => "浙江", 34 => "安徽", 35 => "福建", 36 => "江西", 37 => "山东", 41 => "河南", 42 => "湖北 ", 43 => "湖南", 44 => "广东", 45 => "广西", 46 => "海南", 50 => "重庆", 51 => "四川", 52 => "贵州", 53 => "云南", 54 => "西藏 ", 61 => "陕西", 62 => "甘肃", 63 => "青海", 64 => "宁夏", 65 => "新疆", 71 => "台湾", 81 => "香港", 82 => "澳门", 91 => "国外 "];
  243. $tip = "";
  244. $match = "/^\d{6}(18|19|20)?\d{2}(0[1-9]|1[012])(0[1-9]|[12]\d|3[01])\d{3}(\d|X)$/";
  245. $pass = true;
  246. if (!$card || !preg_match($match, $card)) {
  247. //身份证格式错误
  248. $pass = false;
  249. } else if (!$city[substr($card, 0, 2)]) {
  250. //地址错误
  251. $pass = false;
  252. } else {
  253. //18位身份证需要验证最后一位校验位
  254. if (strlen($card) == 18) {
  255. $card = str_split($card);
  256. //∑(ai×Wi)(mod 11)
  257. //加权因子
  258. $factor = [7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2];
  259. //校验位
  260. $parity = [1, 0, 'X', 9, 8, 7, 6, 5, 4, 3, 2];
  261. $sum = 0;
  262. $ai = 0;
  263. $wi = 0;
  264. for ($i = 0; $i < 17; $i++) {
  265. $ai = $card[$i];
  266. $wi = $factor[$i];
  267. $sum += $ai * $wi;
  268. }
  269. $last = $parity[$sum % 11];
  270. if ($parity[$sum % 11] != $card[17]) {
  271. // $tip = "校验位错误";
  272. $pass = false;
  273. }
  274. } else {
  275. $pass = false;
  276. }
  277. }
  278. if (!$pass) return false;/* 身份证格式错误*/
  279. return true;/* 身份证格式正确*/
  280. }
  281. /*
  282. * 判断手机机型
  283. * @return int 1=ios,2=android,3=其他
  284. * */
  285. public static function getDeviceType()
  286. {
  287. if (strpos($_SERVER['HTTP_USER_AGENT'], 'iPhone') || strpos($_SERVER['HTTP_USER_AGENT'], 'iPad')) {
  288. return 1;
  289. } else if (strpos($_SERVER['HTTP_USER_AGENT'], 'Android')) {
  290. return 2;
  291. } else {
  292. return 3;
  293. }
  294. }
  295. /**随机数
  296. * @param bool $prefix 前缀
  297. * @param int $len 随机长度
  298. * @param bool $random
  299. * @return string
  300. */
  301. public static function makeRandomNumber($prefix = false, $len = 6, $random = false)
  302. {
  303. if (!$prefix) {
  304. $prefix = "";
  305. }
  306. switch ($len) {
  307. case 5:
  308. $one_random = mt_rand(11111, 99999);
  309. $max = 99999;
  310. break;
  311. case 6:
  312. $one_random = mt_rand(111111, 999999);
  313. $max = 999999;
  314. break;
  315. }
  316. $date_random = date('ymd', time());
  317. $random_tmp = strlen($one_random);
  318. $two_randow = str_pad(mt_rand(1, $max), $random_tmp, '0', STR_PAD_LEFT);
  319. if (!$random) {
  320. return $two_randow;
  321. } else {
  322. return $prefix . $one_random . $date_random . $two_randow;
  323. }
  324. }
  325. }