Sin descripción
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.

FilemanagerLogic.php 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  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 FilemanagerLogic 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 = $this->globalTpCache['web_templets_dir']; // 默认文件管理的最大级别目录
  40. // 替换权限
  41. $this->replaceImgOpArr = array('gif','jpg','svg');
  42. // 编辑权限
  43. $this->editOpArr = array('txt','htm','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('htm','js','css','txt');
  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. ),
  242. );
  243. continue;
  244. }
  245. else if(is_dir("$directory/$file"))
  246. {
  247. if(preg_match("#^_(.*)$#i", $file)) continue; #屏蔽FrontPage扩展目录和linux隐蔽目录
  248. if(preg_match("#^\.(.*)$#i", $file)) continue;
  249. $file_info = array(
  250. 'filepath' => $activepath.'/'.$file,
  251. 'filename' => $file,
  252. 'filesize' => '',
  253. 'filetime' => '',
  254. 'filemine' => 'dir',
  255. 'filetype' => 'dir',
  256. 'icon' => 'dir.gif',
  257. 'intro' => '',
  258. );
  259. array_push($dirArr, $file_info);
  260. continue;
  261. }
  262. else if(preg_match("#\.(gif|png)#i",$file))
  263. {
  264. $filemine = 'image';
  265. $filetype = 'gif';
  266. $icon = 'gif.gif';
  267. }
  268. else if(preg_match("#\.(jpg|jpeg|bmp|webp)#i",$file))
  269. {
  270. $filemine = 'image';
  271. $filetype = 'jpg';
  272. $icon = 'jpg.gif';
  273. }
  274. else if(preg_match("#\.(svg)#i",$file))
  275. {
  276. $filemine = 'image';
  277. $filetype = 'svg';
  278. $icon = 'jpg.gif';
  279. }
  280. else if(preg_match("#\.(swf|fla|fly)#i",$file))
  281. {
  282. $filetype = 'flash';
  283. $icon = 'flash.gif';
  284. }
  285. else if(preg_match("#\.(zip|rar|tar.gz)#i",$file))
  286. {
  287. $filetype = 'zip';
  288. $icon = 'zip.gif';
  289. }
  290. else if(preg_match("#\.(exe)#i",$file))
  291. {
  292. $filetype = 'exe';
  293. $icon = 'exe.gif';
  294. }
  295. else if(preg_match("#\.(mp3|wma)#i",$file))
  296. {
  297. $filetype = 'mp3';
  298. $icon = 'mp3.gif';
  299. }
  300. else if(preg_match("#\.(wmv|api)#i",$file))
  301. {
  302. $filetype = 'wmv';
  303. $icon = 'wmv.gif';
  304. }
  305. else if(preg_match("#\.(rm|rmvb)#i",$file))
  306. {
  307. $filetype = 'rm';
  308. $icon = 'rm.gif';
  309. }
  310. else if(preg_match("#\.(txt|inc|pl|cgi|asp|xml|xsl|aspx|cfm)#",$file))
  311. {
  312. $filetype = 'txt';
  313. $icon = 'txt.gif';
  314. }
  315. else if(preg_match("#\.(htm|html)#i",$file))
  316. {
  317. $filetype = 'htm';
  318. $icon = 'htm.gif';
  319. }
  320. else if(preg_match("#\.(php)#i",$file))
  321. {
  322. $filetype = 'php';
  323. $icon = 'php.gif';
  324. }
  325. else if(preg_match("#\.(js)#i",$file))
  326. {
  327. $filetype = 'js';
  328. $icon = 'js.gif';
  329. }
  330. else if(preg_match("#\.(css)#i",$file))
  331. {
  332. $filetype = 'css';
  333. $icon = 'css.gif';
  334. }
  335. else
  336. {
  337. $filetype = 'other';
  338. $icon = 'other.gif';
  339. }
  340. $file_info = array(
  341. 'filepath' => $activepath.'/'.$file,
  342. 'filename' => $file,
  343. 'filesize' => $filesize,
  344. 'filetime' => $filetime,
  345. 'filemine' => $filemine,
  346. 'filetype' => $filetype,
  347. 'icon' => $icon,
  348. 'intro' => $intro,
  349. );
  350. array_push($fileArr, $file_info);
  351. }
  352. $mydir->close();
  353. $arr_file = array_merge($parentArr, $dirArr, $fileArr);
  354. return $arr_file;
  355. }
  356. /**
  357. * 将冒号符反替换为反斜杠,适用于IIS服务器在URL上的双重转义限制
  358. * @param string $filepath 相对路径
  359. * @param string $replacement 目标字符
  360. * @param boolean $is_back false为替换,true为还原
  361. */
  362. public function replace_path($activepath, $replacement = ':', $is_back = false)
  363. {
  364. return replace_path($activepath, $replacement, $is_back);
  365. }
  366. }