UtilService.php 11 KB

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