Nessuna descrizione
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 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  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 app\admin\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. $security = tpSetting('security');
  67. if (empty($security['security_ask_open']) || empty($security['security_answer'])) {
  68. return '需要开启密保问题设置';
  69. } else {
  70. $admin_id = session('?admin_id') ? (int)session('admin_id') : 0;
  71. $admin_info = Db::name('admin')->field('admin_id,last_ip')->where(['admin_id'=>$admin_id])->find();
  72. // 当前管理员二次安全验证过的IP地址
  73. $security_answerverify_ip = !empty($security['security_answerverify_ip']) ? $security['security_answerverify_ip'] : '-1';
  74. // 同IP不验证
  75. if (empty($admin_info) || $admin_info['last_ip'] != $security_answerverify_ip) {
  76. return '出于安全考虑,请勿非法越过密保答案验证';
  77. }
  78. }
  79. if (!filename_preg_match()) {
  80. return '文件名称含有非法入侵字符!';
  81. }
  82. $fileinfo = pathinfo($filename);
  83. $ext = strtolower($fileinfo['extension']);
  84. $filename = trim($fileinfo['filename'], '.').'.'.$fileinfo['extension'];
  85. /*不允许越过指定最大级目录的文件编辑*/
  86. $tmp_max_dir = preg_replace("#\/#i", "\/", $this->maxDir);
  87. if (!preg_match("#^".$tmp_max_dir."#i", $activepath)) {
  88. return '没有操作权限!';
  89. }
  90. /*--end*/
  91. /*允许编辑的文件类型*/
  92. if (!in_array($ext, $this->editExt)) {
  93. return '只允许操作文件类型如下:'.implode('|', $this->editExt);
  94. }
  95. /*--end*/
  96. $file = $this->baseDir."$activepath/$filename";
  97. if (!is_writable(dirname($file))) {
  98. return "请把模板文件目录设置为可写入权限!";
  99. }
  100. if ('htm' == $ext) {
  101. $content = htmlspecialchars_decode($content, ENT_QUOTES);
  102. 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) || preg_match('#(\s+)language(\s*)=(\s*)("|\')?php("|\')?#i', $content)) {
  103. return "模板里不允许有php语法,为了安全考虑,请通过FTP工具进行编辑上传。";
  104. }
  105. foreach ($this->disableFuns as $key => $val) {
  106. $val_new = msubstr($val, 0, 1).'-'.msubstr($val, 1);
  107. $content = preg_replace("/(@)?".$val."(\s*)\(/i", "{$val_new}(", $content);
  108. }
  109. }
  110. $fp = fopen($file, "w");
  111. fputs($fp, $content);
  112. fclose($fp);
  113. return true;
  114. }
  115. /**
  116. * 上传文件
  117. *
  118. * @param string $dirname 新目录
  119. * @param string $activepath 当前路径
  120. * @param boolean $replace 是否替换
  121. * @param string $type 文件类型:图片image , 附件file , 视频media
  122. */
  123. public function upload($fileElementId, $activepath = '', $replace = false, $type = 'image')
  124. {
  125. $retData = [];
  126. $file = request()->file($fileElementId);
  127. if (is_object($file) && !is_array($file)) {
  128. $retData = $this->uploadfile($file, $activepath, $replace, $type);
  129. }
  130. else if (!is_object($file) && is_array($file)) {
  131. $fileArr = $file;
  132. $i = 0;
  133. $j = 0;
  134. foreach ($fileArr as $key => $fileObj) {
  135. if (empty($fileObj)) {
  136. continue;
  137. }
  138. $res = $this->uploadfile($fileObj, $activepath, $replace, $type);
  139. if(!empty($res['code']) && $res['code'] == 1) {
  140. $i++;
  141. } else {
  142. $j++;
  143. }
  144. }
  145. if ($j == 0) {
  146. $retData['code'] = 0;
  147. $retData['msg'] = "上传失败 $i 个文件到: $activepath";
  148. } else {
  149. $retData['code'] = 1;
  150. $retData['msg'] = "上传成功!";
  151. }
  152. }
  153. return $retData;
  154. }
  155. /**
  156. * 自定义上传
  157. *
  158. * @param object $file 文件对象
  159. * @param string $activepath 当前路径
  160. * @param boolean $replace 是否替换
  161. * @param string $type 文件类型:图片image , 附件file , 视频media
  162. */
  163. public function uploadfile($file, $activepath = '', $replace = false, $type = 'image')
  164. {
  165. $validate = array();
  166. /*文件类型限制*/
  167. switch ($type) {
  168. case 'image':
  169. $validate_ext = tpCache('basic.image_type');
  170. break;
  171. case 'file':
  172. $validate_ext = tpCache('basic.file_type');
  173. break;
  174. case 'media':
  175. $validate_ext = tpCache('basic.media_type');
  176. break;
  177. default:
  178. $validate_ext = tpCache('basic.image_type');
  179. break;
  180. }
  181. $validate['ext'] = explode('|', $validate_ext);
  182. /*--end*/
  183. /*文件大小限制*/
  184. $validate_size = tpCache('basic.file_size');
  185. if (!empty($validate_size)) {
  186. $validate['size'] = $validate_size * 1024 * 1024; // 单位为b
  187. }
  188. /*--end*/
  189. /*上传文件验证*/
  190. if (!empty($validate)) {
  191. $is_validate = $file->check($validate);
  192. if ($is_validate === false) {
  193. return ['code'=>0, 'msg'=>$file->getError()];
  194. }
  195. }
  196. /*--end*/
  197. $savePath = !empty($activepath) ? trim($activepath, '/') : UPLOAD_PATH.'temp';
  198. if (!file_exists($savePath)) {
  199. tp_mkdir($savePath);
  200. }
  201. if (false == $replace) {
  202. $fileinfo = $file->getInfo();
  203. $filename = pathinfo($fileinfo['name'], PATHINFO_BASENAME); //获取上传文件名
  204. } else {
  205. $filename = $replace;
  206. }
  207. $fileExt = pathinfo($filename, PATHINFO_EXTENSION); //获取上传文件扩展名
  208. if (!in_array($fileExt, $validate['ext'])) {
  209. return ['code'=>0, 'msg'=>'上传文件后缀不允许'];
  210. }
  211. // 使用自定义的文件保存规则
  212. $info = $file->move($savePath, $filename, true);
  213. if($info){
  214. return ['code'=>1, 'msg'=>'上传成功'];
  215. }else{
  216. return ['code'=>0, 'msg'=>$file->getError()];
  217. }
  218. }
  219. /**
  220. * 当前目录下的文件列表
  221. */
  222. public function getDirFile($directory, $activepath = '', &$arr_file = array()) {
  223. if (!file_exists($directory)) {
  224. return false;
  225. }
  226. $fileArr = $dirArr = $parentArr = array();
  227. $mydir = dir($directory);
  228. while(false !== $file = $mydir->read())
  229. {
  230. $filesize = $filetime = $intro = '';
  231. $filemine = 'file';
  232. if($file != "." && $file != ".." && !is_dir("$directory/$file"))
  233. {
  234. @$filesize = filesize("$directory/$file");
  235. @$filesize = format_bytes($filesize);
  236. @$filetime = filemtime("$directory/$file");
  237. }
  238. if ($file == '.')
  239. {
  240. continue;
  241. }
  242. else if($file == "..")
  243. {
  244. if($activepath == "" || $activepath == $this->maxDir) {
  245. continue;
  246. }
  247. $parentArr = array(
  248. array(
  249. 'filepath' => preg_replace("#[\/][^\/]*$#i", "", $activepath),
  250. 'filename' => '上级目录',
  251. 'filesize' => '',
  252. 'filetime' => '',
  253. 'filemine' => 'dir',
  254. 'filetype' => 'dir2',
  255. 'icon' => 'file_topdir.gif',
  256. 'intro' => '(当前目录:'.$activepath.')',
  257. ),
  258. );
  259. continue;
  260. }
  261. else if(is_dir("$directory/$file"))
  262. {
  263. if(preg_match("#^_(.*)$#i", $file)) continue; #屏蔽FrontPage扩展目录和linux隐蔽目录
  264. if(preg_match("#^\.(.*)$#i", $file)) continue;
  265. $file_info = array(
  266. 'filepath' => $activepath.'/'.$file,
  267. 'filename' => $file,
  268. 'filesize' => '',
  269. 'filetime' => '',
  270. 'filemine' => 'dir',
  271. 'filetype' => 'dir',
  272. 'icon' => 'dir.gif',
  273. 'intro' => '',
  274. );
  275. array_push($dirArr, $file_info);
  276. continue;
  277. }
  278. else if(preg_match("#\.(gif|png)#i",$file))
  279. {
  280. $filemine = 'image';
  281. $filetype = 'gif';
  282. $icon = 'gif.gif';
  283. }
  284. else if(preg_match("#\.(jpg|jpeg|bmp|webp)#i",$file))
  285. {
  286. $filemine = 'image';
  287. $filetype = 'jpg';
  288. $icon = 'jpg.gif';
  289. }
  290. else if(preg_match("#\.(svg)#i",$file))
  291. {
  292. $filemine = 'image';
  293. $filetype = 'svg';
  294. $icon = 'jpg.gif';
  295. }
  296. else if(preg_match("#\.(swf|fla|fly)#i",$file))
  297. {
  298. $filetype = 'flash';
  299. $icon = 'flash.gif';
  300. }
  301. else if(preg_match("#\.(zip|rar|tar.gz)#i",$file))
  302. {
  303. $filetype = 'zip';
  304. $icon = 'zip.gif';
  305. }
  306. else if(preg_match("#\.(exe)#i",$file))
  307. {
  308. $filetype = 'exe';
  309. $icon = 'exe.gif';
  310. }
  311. else if(preg_match("#\.(mp3|wma)#i",$file))
  312. {
  313. $filetype = 'mp3';
  314. $icon = 'mp3.gif';
  315. }
  316. else if(preg_match("#\.(wmv|api)#i",$file))
  317. {
  318. $filetype = 'wmv';
  319. $icon = 'wmv.gif';
  320. }
  321. else if(preg_match("#\.(rm|rmvb)#i",$file))
  322. {
  323. $filetype = 'rm';
  324. $icon = 'rm.gif';
  325. }
  326. else if(preg_match("#\.(txt|inc|pl|cgi|asp|xml|xsl|aspx|cfm)#",$file))
  327. {
  328. $filetype = 'txt';
  329. $icon = 'txt.gif';
  330. }
  331. else if(preg_match("#\.(htm|html)#i",$file))
  332. {
  333. $filetype = 'htm';
  334. $icon = 'htm.gif';
  335. }
  336. else if(preg_match("#\.(php)#i",$file))
  337. {
  338. $filetype = 'php';
  339. $icon = 'php.gif';
  340. }
  341. else if(preg_match("#\.(js)#i",$file))
  342. {
  343. $filetype = 'js';
  344. $icon = 'js.gif';
  345. }
  346. else if(preg_match("#\.(css)#i",$file))
  347. {
  348. $filetype = 'css';
  349. $icon = 'css.gif';
  350. }
  351. else
  352. {
  353. $filetype = 'other';
  354. $icon = 'other.gif';
  355. }
  356. $file_info = array(
  357. 'filepath' => $activepath.'/'.$file,
  358. 'filename' => $file,
  359. 'filesize' => $filesize,
  360. 'filetime' => $filetime,
  361. 'filemine' => $filemine,
  362. 'filetype' => $filetype,
  363. 'icon' => $icon,
  364. 'intro' => $intro,
  365. );
  366. array_push($fileArr, $file_info);
  367. }
  368. $mydir->close();
  369. $arr_file = array_merge($parentArr, $dirArr, $fileArr);
  370. return $arr_file;
  371. }
  372. /**
  373. * 将冒号符反替换为反斜杠,适用于IIS服务器在URL上的双重转义限制
  374. * @param string $filepath 相对路径
  375. * @param string $replacement 目标字符
  376. * @param boolean $is_back false为替换,true为还原
  377. */
  378. public function replace_path($activepath, $replacement = ':', $is_back = false)
  379. {
  380. $activepath = replace_path($activepath, $replacement, $is_back);
  381. $activepath = preg_replace('/<(\w+)([^\>]*)>([^<]*)<\/(\w+)>/i', '', $activepath);
  382. $activepath = preg_replace('/(<|>|;|\(|\)|\!)/i', '', $activepath);
  383. return $activepath;
  384. }
  385. }