No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

FiletoolLogic.php 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海南赞赞网络科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 小虎哥 <1105415366@qq.com>
  11. * Date: 2018-4-3
  12. */
  13. namespace weapp\Systemdoctor\logic;
  14. use think\Model;
  15. use think\db;
  16. /**
  17. * 文件管理逻辑定义
  18. * Class CatsLogic
  19. * @package admin\Logic
  20. */
  21. class FiletoolLogic extends Model
  22. {
  23. public $globalTpCache = array();
  24. public $baseDir = ''; // 服务器站点根目录绝对路径
  25. public $maxDir = '';
  26. public $replaceImgOpArr = array(); // 替换权限
  27. public $editOpArr = array(); // 编辑权限
  28. public $renameOpArr = array(); // 改名权限
  29. public $delOpArr = array(); // 删除权限
  30. public $moveOpArr = array(); // 移动权限
  31. public $editExt = array(); // 允许新增/编辑扩展名文件
  32. public $disableFuns = array(); // 允许新增/编辑扩展名文件
  33. /**
  34. * 析构函数
  35. */
  36. function __construct() {
  37. $this->globalTpCache = tpCache('global');
  38. $this->baseDir = rtrim(ROOT_PATH, DS); // 服务器站点根目录绝对路径
  39. $this->maxDir = ''; // 默认文件管理的最大级别目录
  40. // 替换权限
  41. $this->replaceImgOpArr = array('gif','jpg','svg','jpeg','bmp','webp','png');
  42. // 编辑权限
  43. $this->editOpArr = array('txt','inc','pl','cgi','asp','xml','xsl','aspx','cfm','htm','html','php','js','css');
  44. // 改名权限
  45. // $this->renameOpArr = array('dir','gif','jpg','svg','flash','zip','exe','mp3','wmv','rm','txt','htm','js','css','other');
  46. // 删除权限
  47. // $this->delOpArr = array('dir','gif','jpg','svg','flash','zip','exe','mp3','wmv','rm','txt','htm','php','js','css','other');
  48. // 移动权限
  49. // $this->moveOpArr = array('gif','jpg','svg','flash','zip','exe','mp3','wmv','rm','txt','htm','js','css','other');
  50. // 允许新增/编辑扩展名文件
  51. $this->editExt = array_merge($this->editOpArr,['htaccess']);
  52. // 过滤php危险函数
  53. $this->disableFuns = ['phpinfo'];
  54. }
  55. /**
  56. * 编辑文件
  57. *
  58. * @access public
  59. * @param string $filename 文件名
  60. * @param string $activepath 当前路径
  61. * @param string $content 文件内容
  62. * @return string
  63. */
  64. public function editFile($filename, $activepath = '', $content = '')
  65. {
  66. $fileinfo = pathinfo($filename);
  67. $ext = strtolower($fileinfo['extension']);
  68. $filename = trim($fileinfo['filename'], '.').'.'.$fileinfo['extension'];
  69. /*不允许越过指定最大级目录的文件编辑*/
  70. $tmp_max_dir = preg_replace("#\/#i", "\/", $this->maxDir);
  71. if (!preg_match("#^".$tmp_max_dir."#i", $activepath)) {
  72. return '没有操作权限!';
  73. }
  74. /*--end*/
  75. /*允许编辑的文件类型*/
  76. if (!in_array($ext, $this->editExt)) {
  77. return '只允许操作文件类型如下:'.implode('|', $this->editExt);
  78. }
  79. /*--end*/
  80. $file = $this->baseDir."$activepath/$filename";
  81. if (!is_writable(dirname($file))) {
  82. return "请把模板文件目录设置为可写入权限!";
  83. }
  84. if ('htm' == $ext) {
  85. $content = htmlspecialchars_decode($content, ENT_QUOTES);
  86. if (preg_match('#<([^?]*)\?php#i', $content) || preg_match('#<\?(\s*)=#i', $content) || (preg_match('#<\?#i', $content) && preg_match('#\?>#i', $content)) || preg_match('#\{eyou\:php([^\}]*)\}#i', $content) || preg_match('#\{php([^\}]*)\}#i', $content)) {
  87. return "模板里不允许有php语法,为了安全考虑,请通过FTP工具进行编辑上传。";
  88. }
  89. foreach ($this->disableFuns as $key => $val) {
  90. $val_new = msubstr($val, 0, 1).'-'.msubstr($val, 1);
  91. $content = preg_replace("/(@)?".$val."(\s*)\(/i", "{$val_new}(", $content);
  92. }
  93. }
  94. $fp = fopen($file, "w");
  95. fputs($fp, $content);
  96. fclose($fp);
  97. return true;
  98. }
  99. /**
  100. * 上传文件
  101. *
  102. * @param string $dirname 新目录
  103. * @param string $activepath 当前路径
  104. * @param boolean $replace 是否替换
  105. * @param string $type 文件类型:图片image , 附件file , 视频media
  106. */
  107. public function upload($fileElementId, $activepath = '', $replace = false, $type = 'image')
  108. {
  109. $retData = [];
  110. $file = request()->file($fileElementId);
  111. if (is_object($file) && !is_array($file)) {
  112. $retData = $this->uploadfile($file, $activepath, $replace, $type);
  113. }
  114. else if (!is_object($file) && is_array($file)) {
  115. $fileArr = $file;
  116. $i = 0;
  117. $j = 0;
  118. foreach ($fileArr as $key => $fileObj) {
  119. if (empty($fileObj)) {
  120. continue;
  121. }
  122. $res = $this->uploadfile($fileObj, $activepath, $replace, $type);
  123. if(!empty($res['code']) && $res['code'] == 1) {
  124. $i++;
  125. } else {
  126. $j++;
  127. }
  128. }
  129. if ($j == 0) {
  130. $retData['code'] = 0;
  131. $retData['msg'] = "上传失败 $i 个文件到: $activepath";
  132. } else {
  133. $retData['code'] = 1;
  134. $retData['msg'] = "上传成功!";
  135. }
  136. }
  137. return $retData;
  138. }
  139. /**
  140. * 自定义上传
  141. *
  142. * @param object $file 文件对象
  143. * @param string $activepath 当前路径
  144. * @param boolean $replace 是否替换
  145. * @param string $type 文件类型:图片image , 附件file , 视频media
  146. */
  147. public function uploadfile($file, $activepath = '', $replace = false, $type = 'image')
  148. {
  149. $validate = array();
  150. /*文件类型限制*/
  151. switch ($type) {
  152. case 'image':
  153. $validate_ext = tpCache('basic.image_type');
  154. break;
  155. case 'file':
  156. $validate_ext = tpCache('basic.file_type');
  157. break;
  158. case 'media':
  159. $validate_ext = tpCache('basic.media_type');
  160. break;
  161. default:
  162. $validate_ext = tpCache('basic.image_type');
  163. break;
  164. }
  165. $validate['ext'] = explode('|', $validate_ext);
  166. /*--end*/
  167. /*文件大小限制*/
  168. $validate_size = tpCache('basic.file_size');
  169. if (!empty($validate_size)) {
  170. $validate['size'] = $validate_size * 1024 * 1024; // 单位为b
  171. }
  172. /*--end*/
  173. /*上传文件验证*/
  174. if (!empty($validate)) {
  175. $is_validate = $file->check($validate);
  176. if ($is_validate === false) {
  177. return ['code'=>0, 'msg'=>$file->getError()];
  178. }
  179. }
  180. /*--end*/
  181. $savePath = !empty($activepath) ? trim($activepath, '/') : UPLOAD_PATH.'temp';
  182. if (!file_exists($savePath)) {
  183. tp_mkdir($savePath);
  184. }
  185. if (false == $replace) {
  186. $fileinfo = $file->getInfo();
  187. $filename = pathinfo($fileinfo['name'], PATHINFO_BASENAME); //获取上传文件名
  188. } else {
  189. $filename = $replace;
  190. }
  191. $fileExt = pathinfo($filename, PATHINFO_EXTENSION); //获取上传文件扩展名
  192. if (!in_array($fileExt, $validate['ext'])) {
  193. return ['code'=>0, 'msg'=>'上传文件后缀不允许'];
  194. }
  195. // 使用自定义的文件保存规则
  196. $info = $file->move($savePath, $filename, true);
  197. if($info){
  198. return ['code'=>1, 'msg'=>'上传成功'];
  199. }else{
  200. return ['code'=>0, 'msg'=>$file->getError()];
  201. }
  202. }
  203. /**
  204. * 当前目录下的文件列表
  205. */
  206. public function getDirFile($directory, $activepath = '', &$arr_file = array()) {
  207. if (!file_exists($directory)) {
  208. return false;
  209. }
  210. $fileArr = $dirArr = $parentArr = array();
  211. $mydir = dir($directory);
  212. while(false !== $file = $mydir->read())
  213. {
  214. $filesize = $filetime = $intro = '';
  215. $filemine = 'file';
  216. if($file != "." && $file != ".." && !is_dir("$directory/$file"))
  217. {
  218. @$filesize = filesize("$directory/$file");
  219. @$filesize = format_bytes($filesize);
  220. @$filetime = filemtime("$directory/$file");
  221. }
  222. if ($file == '.')
  223. {
  224. continue;
  225. }
  226. else if($file == "..")
  227. {
  228. if($activepath == "" || $activepath == $this->maxDir) {
  229. continue;
  230. }
  231. $parentArr = array(
  232. array(
  233. 'filepath' => preg_replace("#[\/][^\/]*$#i", "", $activepath),
  234. 'filename' => '上级目录',
  235. 'filesize' => '',
  236. 'filetime' => '',
  237. 'filemine' => 'dir',
  238. 'filetype' => 'dir2',
  239. 'icon' => 'file_topdir.gif',
  240. 'intro' => '(当前目录:'.$activepath.')',
  241. 'act' => [],
  242. ),
  243. );
  244. continue;
  245. }
  246. else if(is_dir("$directory/$file"))
  247. {
  248. if(preg_match("#^_(.*)$#i", $file)) continue; #屏蔽FrontPage扩展目录和linux隐蔽目录
  249. if(preg_match("#^\.(.*)$#i", $file)) continue;
  250. $encoding = mb_detect_encoding($file,['UTF-8','GBK','BIG5','CP936']);
  251. $file = iconv($encoding,'UTF-8',$file); //转码
  252. $file_info = array(
  253. 'filepath' => $activepath.'/'.$file,
  254. 'filename' => $file,
  255. 'filesize' => '',
  256. 'filetime' => '',
  257. 'filemine' => 'dir',
  258. 'filetype' => 'dir',
  259. 'icon' => 'dir.gif',
  260. 'intro' => '',
  261. 'act' => ['rename','del'],
  262. );
  263. array_push($dirArr, $file_info);
  264. continue;
  265. }
  266. else if(preg_match("#\.(gif|png)#i",$file))
  267. {
  268. $filemine = 'image';
  269. $filetype = 'gif';
  270. $icon = 'gif.gif';
  271. $act = ['rename','del','move'];
  272. }
  273. else if(preg_match("#\.(jpg|jpeg|bmp|webp)#i",$file))
  274. {
  275. $filemine = 'image';
  276. $filetype = 'jpg';
  277. $icon = 'jpg.gif';
  278. $act = ['rename','del','move'];
  279. }
  280. else if(preg_match("#\.(svg)#i",$file))
  281. {
  282. $filemine = 'image';
  283. $filetype = 'svg';
  284. $icon = 'jpg.gif';
  285. $act = ['rename','del','move'];
  286. }
  287. else if(preg_match("#\.(swf|fla|fly)#i",$file))
  288. {
  289. $filetype = 'flash';
  290. $icon = 'flash.gif';
  291. $act = ['rename','del','move'];
  292. }
  293. else if(preg_match("#\.(zip|rar|tar.gz)#i",$file))
  294. {
  295. $filetype = 'zip';
  296. $icon = 'zip.gif';
  297. $act = ['rename','del','move'];
  298. }
  299. else if(preg_match("#\.(exe)#i",$file))
  300. {
  301. $filetype = 'exe';
  302. $icon = 'exe.gif';
  303. $act = ['rename','del','move'];
  304. }
  305. else if(preg_match("#\.(mp3|wma)#i",$file))
  306. {
  307. $filetype = 'mp3';
  308. $icon = 'mp3.gif';
  309. $act = ['rename','del','move'];
  310. }
  311. else if(preg_match("#\.(wmv|api)#i",$file))
  312. {
  313. $filetype = 'wmv';
  314. $icon = 'wmv.gif';
  315. $act = ['rename','del','move'];
  316. }
  317. else if(preg_match("#\.(rm|rmvb)#i",$file))
  318. {
  319. $filetype = 'rm';
  320. $icon = 'rm.gif';
  321. $act = ['rename','del','move'];
  322. }
  323. else if(preg_match("#\.(txt|inc|pl|cgi|asp|xml|xsl|aspx|cfm)#",$file))
  324. {
  325. $filetype = 'txt';
  326. $icon = 'txt.gif';
  327. $act = ['edit','rename','del','move'];
  328. }
  329. else if(preg_match("#\.(htm|html)#i",$file))
  330. {
  331. $filetype = 'htm';
  332. $icon = 'htm.gif';
  333. $act = ['edit','rename','del','move'];
  334. }
  335. else if(preg_match("#\.(php)#i",$file))
  336. {
  337. $filetype = 'php';
  338. $icon = 'php.gif';
  339. $act = ['edit','rename','del','move'];
  340. }
  341. else if(preg_match("#\.(js)#i",$file))
  342. {
  343. $filetype = 'js';
  344. $icon = 'js.gif';
  345. $act = ['edit','rename','del','move'];
  346. }
  347. else if(preg_match("#\.(css)#i",$file))
  348. {
  349. $filetype = 'css';
  350. $icon = 'css.gif';
  351. $act = ['edit','rename','del','move'];
  352. }
  353. else
  354. {
  355. $filetype = 'other';
  356. $icon = 'other.gif';
  357. $act = ['rename','del','move'];
  358. }
  359. $encoding = mb_detect_encoding($file,['UTF-8','GBK','BIG5','CP936']);
  360. $file = iconv($encoding,'UTF-8',$file); //转码
  361. $file_info = array(
  362. 'filepath' => $activepath.'/'.$file,
  363. 'filename' => $file,
  364. 'filesize' => $filesize,
  365. 'filetime' => $filetime,
  366. 'filemine' => $filemine,
  367. 'filetype' => $filetype,
  368. 'icon' => $icon,
  369. 'intro' => $intro,
  370. );
  371. array_push($fileArr, $file_info);
  372. }
  373. $mydir->close();
  374. $arr_file = array_merge($parentArr, $dirArr, $fileArr);
  375. return $arr_file;
  376. }
  377. /**
  378. * 将冒号符反替换为反斜杠,适用于IIS服务器在URL上的双重转义限制
  379. * @param string $filepath 相对路径
  380. * @param string $replacement 目标字符
  381. * @param boolean $is_back false为替换,true为还原
  382. */
  383. public function replace_path($activepath, $replacement = ':', $is_back = false)
  384. {
  385. return replace_path($activepath, $replacement, $is_back);
  386. }
  387. /**
  388. * 删除目录
  389. *
  390. * @param unknown_type $indir
  391. */
  392. public function RmDirFiles($indir)
  393. {
  394. if(!is_dir($indir))
  395. {
  396. return ;
  397. }
  398. $dh = dir($indir);
  399. while($filename = $dh->read())
  400. {
  401. if($filename == "." || $filename == "..")
  402. {
  403. continue;
  404. }
  405. else if(is_file("$indir/$filename"))
  406. {
  407. @unlink("$indir/$filename");
  408. }
  409. else
  410. {
  411. $this->RmDirFiles("$indir/$filename");
  412. }
  413. }
  414. $dh->close();
  415. @rmdir($indir);
  416. }
  417. }