iconv_app($name), 'path' => iconv_app(get_path_father($path)), 'ext' => get_path_ext($path), 'type' => 'file', 'mode' => get_mode($path), 'atime' => date($date_formate, fileatime($path)), //访问时间 'ctime' => date($date_formate, filectime($path)), //创建时间 'mtime' => date($date_formate, filemtime($path)), //最后修改时间 'is_readable' => intval(is_readable($path)), 'is_writeable' => intval(is_writeable($path)), 'size' => $size, 'size_friendly' => size_format($size, 2) ); return $info; } /** * 获取文件夹细信息 */ function folder_info($path,$date_formate=false){ if (!$date_formate) $date_formate = $GLOBALS['L']['time_type']; $info = array( 'name' => iconv_app(get_path_this($path)), 'path' => iconv_app(get_path_father($path)), 'type' => 'folder', 'mode' => get_mode($path), 'atime' => date($date_formate, fileatime($path)), //访问时间 'ctime' => date($date_formate, filectime($path)), //创建时间 'mtime' => date($date_formate, filemtime($path)), //最后修改时间 'is_readable' => intval(is_readable($path)), 'is_writeable' => intval(is_writeable($path)) ); return $info; } /** * 获取一个路径(文件夹&文件) 当前文件[夹]名 * test/11/ ==>11 test/1.c ==>1.c */ function get_path_this($path){ $path = str_replace('\\','/', rtrim(trim($path),'/')); return substr($path,strrpos($path,'/')+1); } /** * 获取一个路径(文件夹&文件) 父目录 * /test/11/==>/test/ /test/1.c ==>/www/test/ */ function get_path_father($path){ $path = str_replace('\\','/', rtrim(trim($path),'/')); return substr($path, 0, strrpos($path,'/')+1); } //自动获取不重复文件(夹)名 function get_filename_auto($path){ $i=1;$dot_index = strrpos($path,'.'); $file_pre = substr($path,0,$dot_index); $file_ext = substr($path,$dot_index); while(file_exists($path)){ $path = $file_pre.'('.$i.')'.$file_ext; $i++; } return $path; } /** * 获取扩展名 */ function get_path_ext($path){ return strtolower(end(explode('.',$path))); } /** * 获取文件夹详细信息,文件夹属性时调用,包含子文件夹数量,文件数量,总大小 */ function path_info($path){ //if (!is_dir($path)) return false; $pathinfo = _path_info_more($path);//子目录文件大小统计信息 $folderinfo = folder_info($path); return array_merge($pathinfo,$folderinfo); } /** * 检查名称是否合法 */ function path_check($path){ $check = array('/','\\',':','*','?','"','<','>','|'); $path = rtrim($path,'/'); $path = get_path_this($path); foreach ($check as $v) { if (strstr($path,$v)) { return false; } } return true; } /** * 递归获取文件夹信息: 子文件夹数量,文件数量,总大小 */ function _path_info_more($dir, &$file_num = 0, &$path_num = 0, &$size = 0){ if (!$dh = opendir($dir)) return false; while (false !== ($file = readdir($dh))) { if ($file != "." && $file != "..") { $fullpath = $dir . "/" . $file; if (!is_dir($fullpath)) { $file_num ++; $size += filesize($fullpath); } else { _path_info_more($fullpath, $file_num, $path_num, $size); $path_num ++; } } } closedir($dh); $pathinfo['file_num'] = $file_num; $pathinfo['folder_num'] = $path_num; $pathinfo['size'] = $size; $pathinfo['size_friendly'] = size_format($size); return $pathinfo; } /** * 获取多选文件信息,包含子文件夹数量,文件数量,总大小,父目录权限 */ function path_info_muti($list){ $pathinfo = array( 'file_num' => 0, 'folder_num' => 0, 'size' => 0, 'size_friendly' => '', 'father_name' => '', 'mod' => '' ); foreach ($list as $val) { if ($val['type'] == 'folder') { $pathinfo['folder_num'] ++; $temp = path_info($val['path']); $pathinfo['folder_num'] += $temp['folder_num']; $pathinfo['file_num'] += $temp['file_num']; $pathinfo['size'] += $temp['size']; }else{ $pathinfo['file_num']++; $pathinfo['size'] += filesize($val['path']); } } $pathinfo['size_friendly'] = size_format($pathinfo['size']); $pathinfo['father_name'] = get_path_father($list[0]['path']); $pathinfo['mode'] = get_mode($pathinfo['father_name']); return $pathinfo; } /** * 获取文件夹下列表信息 * dir 包含结尾/ d:/wwwroot/test/ * 传入需要读取的文件夹路径,为程序编码 */ function path_list($dir,$list_file=true,$check_children=false){ $dir = rtrim($dir,'/').'/'; if (!is_dir($dir)) return array('folderlist'=>array(),'filelist'=>array()); $dh = opendir($dir); $folderlist = array();$filelist = array();//文件夹与文件 while (false !== ($file = readdir($dh))) { if ($file != "." && $file != ".." && $file != ".svn" ) { $fullpath = $dir . $file; if (is_dir($fullpath)) { $info = folder_info($fullpath); if($check_children){ $info['isParent'] = path_haschildren($fullpath,$list_file); } $folderlist[] = $info; } else if($list_file) {//是否列出文件 $info = file_info($fullpath); if($check_children) $info['isParent'] = false; $filelist[] = $info; } } } closedir($dh); return array('folderlist' => $folderlist,'filelist' => $filelist); } // 判断文件夹是否含有子内容【区分为文件或者只筛选文件夹才算】 function path_haschildren($dir,$check_file=false){ $dir = rtrim($dir,'/').'/'; if (!$dh = @opendir($dir)) return false; while (false !== ($file = readdir($dh))){ if ($file != "." && $file != "..") { $fullpath = $dir.$file; if ($check_file) {//有子目录或者文件都说明有子内容 if(is_dir($fullpath.'/') || is_file($fullpath)) return true; }else{//只检查有没有文件 if(is_dir($fullpath.'/')) return true; } } } closedir($dh); return false; } /** * 删除文件 传入参数编码为操作系统编码. win--gbk */ function del_file($fullpath){ if (!@unlink($fullpath)) { // 删除不了,尝试修改文件权限 @chmod($fullpath, 0777); if (!@unlink($fullpath)) { return false; } } else { return true; } } /** * 删除文件夹 传入参数编码为操作系统编码. win--gbk */ function del_dir($dir){ if (!$dh = opendir($dir)) return false; while (false !== ($file = readdir($dh))) { if ($file != "." && $file != "..") { $fullpath = $dir . '/' . $file; if (!is_dir($fullpath)) { if (!unlink($fullpath)) { // 删除不了,尝试修改文件权限 chmod($fullpath, 0777); if (!unlink($fullpath)) { return false; } } } else { if (!del_dir($fullpath)) { chmod($fullpath, 0777); if (!del_dir($fullpath)) return false; } } } } closedir($dh); if (rmdir($dir)) { return true; } else { return false; } } /** * 复制文件夹 * eg:将D:/wwwroot/下面wordpress复制到 * D:/wwwroot/www/explorer/0000/del/1/ * 末尾都不需要加斜杠,复制到地址如果不加源文件夹名, * 就会将wordpress下面文件复制到D:/wwwroot/www/explorer/0000/del/1/下面 * $from = 'D:/wwwroot/wordpress'; * $to = 'D:/wwwroot/www/explorer/0000/del/1/wordpress'; */ function copy_dir($source, $dest){ $result = false; if (is_file($source)) { if ($dest[strlen($dest)-1] == '/') { $__dest = $dest . "/" . basename($source); } else { $__dest = $dest; } $result = copy($source, $__dest); chmod($__dest, 0755); } elseif (is_dir($source)) { if ($dest[strlen($dest)-1] == '/') { $dest = $dest . basename($source); mkdir($dest, 0755); } else { mkdir($dest, 0755); } $dirHandle = opendir($source); while (false !== ($file = readdir($dirHandle))) { if ($file != "." && $file != "..") { if (!is_dir($source . "/" . $file)) { $__dest = $dest . "/" . $file; } else { $__dest = $dest . "/" . $file; } $result = copy_dir($source . "/" . $file, $__dest); } } closedir($dirHandle); } else { $result = false; } return $result; } /** * 创建目录 * * @param string $dir * @param int $mode * @return bool */ function mk_dir($dir, $mode = 0777){ if (is_dir($dir) || mkdir($dir, $mode)) return true; if (!mk_dir(dirname($dir), $mode)) return false; return mkdir($dir, $mode); } /* * 获取文件&文件夹列表(支持文件夹层级) * path : 文件夹 $dir ——返回的文件夹array files ——返回的文件array * $deepest 是否完整递归;$deep 递归层级 */ function recursion_dir($path,&$dir,&$file,$deepest=-1,$deep=0){ $path = rtrim($path,'/').'/'; if (!is_array($file)) $file=array(); if (!is_array($dir)) $dir=array(); $handle=opendir($path); while(($val=readdir($handle)) !== false){ if ($val=='.' || $val=='..') continue; $value = strval($path.$val); if (is_file($value)){ $file[] = $value; }else if(is_dir($value)){ $dir[]=$value; if ($deepest==-1 || $deep<$deepest){ recursion_dir($value."/",$dir,$file,$deepest,$deep+1); } } } closedir($handle); return true; } /* * $search 为包含的字符串 * is_content 表示是否搜索文件内容;默认不搜索 * is_case 表示区分大小写,默认不区分 */ function path_search($path,$search,$is_content=false,$file_ext='',$is_case=false){ $ext_arr=explode("|",$file_ext); recursion_dir($path,$dirs,$files,-1,0); $strpos = 'stripos';//是否区分大小写 if ($is_case) $strpos = 'strpos'; $filelist = array(); $folderlist = array(); foreach($files as $f){ $ext = get_path_ext($f); $path_this = get_path_this($f); if ($file_ext !='' && !in_array($ext,$ext_arr)) continue;//文件类型不在用户限定内 if ($strpos($path_this,$search) !== false){//搜索文件名;搜到就返回;搜不到继续 $filelist[] = file_info($f); continue; } if ($is_content && is_file($f)){ $fp = fopen($f, "r"); $content = @fread($fp,filesize($f)); fclose($fp); if ($strpos($content,$search) !== false){ $filelist[] = file_info($f); } } } if ($file_ext == '') {//没限定扩展名则才搜索文件夹 foreach($dirs as $f){ $path_this = get_path_this($f); if ($strpos($path_this,$search)){ $folderlist[]= array( 'name' => iconv_app(get_path_this($f)), 'path' => iconv_app(get_path_father($f)) ); } } } return array('folderlist' => $folderlist,'filelist' => $filelist); } /** * 文件大小格式化 * * @param $ :$bytes, int 文件大小 * @param $ :$precision int 保留小数点 * @return :string */ function size_format($bytes, $precision = 2){ if ($bytes == 0) return "0 B"; $unit = array('TB' => 1099511627776, // pow( 1024, 4) 'GB' => 1073741824, // pow( 1024, 3) 'MB' => 1048576, // pow( 1024, 2) 'kB' => 1024, // pow( 1024, 1) 'B ' => 1, // pow( 1024, 0) ); foreach ($unit as $un => $mag) { if (doubleval($bytes) >= $mag) return round($bytes / $mag, $precision).' '.$un; } } /** * 判断路径是不是绝对路径 * 返回true('/foo/bar','c:\windows'). * * @return 返回true则为绝对路径,否则为相对路径 */ function path_is_absolute($path){ if (realpath($path) == $path)// *nux 的绝对路径 /home/my return true; if (strlen($path) == 0 || $path[0] == '.') return false; if (preg_match('#^[a-zA-Z]:\\\\#', $path))// windows 的绝对路径 c:\aaa\ return true; return (bool)preg_match('#^[/\\\\]#', $path); //绝对路径 运行 / 和 \绝对路径,其他的则为相对路径 } /** * 获取扩展名的文件类型 * * @param $ :$ext string 扩展名 * @return :string; */ function ext_type($ext){ $ext2type = array( 'text' => array('txt','ini','log','asc','csv','tsv','vbs','bat','cmd','inc','conf','inf'), 'code' => array('css','htm','html','php','js','c','cpp','h','java','cs','sql','xml'), 'picture' => array('jpg','jpeg','png','gif','ico','bmp','tif','tiff','dib','rle'), 'audio' => array('mp3','ogg','oga','mid','midi','ram','wav','wma','aac','ac3','aif','aiff','m3a','m4a','m4b','mka','mp1','mx3','mp2'), 'flash' => array('swf'), 'video' => array('rm','rmvb','flv','mkv','wmv','asf','avi','aiff','mp4','divx','dv','m4v','mov','mpeg','vob','mpg','mpv','ogm','ogv','qt'), 'document' => array('doc','docx','docm','dotm','odt','pages','pdf','rtf','xls','xlsx','xlsb','xlsm','ppt','pptx','pptm','odp'), 'rar_achieve' => array('rar','arj','tar','ace','gz','lzh','uue','bz2'), 'zip_achieve' => array('zip','gzip','cab','tbz','tbz2'), 'other_achieve' => array('dmg','sea','sit','sqx') ); foreach ($ext2type as $type => $exts) { if (in_array($ext, $exts)) { return $type; } } } /** * 文件下载 */ function file_download($file){ if (file_exists($file)) { header("Cache-Control: public"); header("Content-Type: application/octet-stream"); header("Content-Disposition: attachment;filename=".get_path_this($file)); header("Accept-Ranges: bytes"); $size = filesize($file); //如果有$_SERVER['HTTP_RANGE']参数 断点续传 if (isset ($_SERVER['HTTP_RANGE'])) { list ($a, $range) = explode("=", $_SERVER['HTTP_RANGE']); str_replace($range, "-", $range); $size2 = $size -1; //文件总字节数 $new_length = $size2 - $range; //获取下次下载的长度 header("HTTP/1.1 206 Partial Content"); header("Content-Length: $new_length"); //输入总长 header("Content-Range: bytes $range$size2/$size"); }else{//第一次连接 $size2 = $size -1; header("Content-Range: bytes 0-$size2/$size"); //Content-Range: bytes 0-4988927/4988928 header("Content-Length: " . $size); //输出总长 } $fp = fopen($file, "rb"); fseek($fp, $range); while (!feof($fp)) { set_time_limit(0); print (fread($fp, 1024 * 8)); //输出文件 flush(); ob_flush(); } fclose($fp); exit (); } } /** * 文件代理输出 */ function file_proxy_out($file){ if (file_exists($file)) { $mime = get_file_mime(get_path_ext($file)); header("Cache-Control:public"); header("Content-Type:".$mime); $size = filesize($file); header("Content-Length: " . $size); //输出总长 $fp = fopen($file, "rb"); fseek($fp, $range); while (!feof($fp)) { set_time_limit(0); print (fread($fp, 1024 * 8)); //输出文件 flush(); ob_flush(); } fclose($fp); exit; }else{ show_json('file not exists',false); } } /** * 文件下载到服务器 */ function file_download_this($from, $file_name){ $fp = @fopen ($from, "rb"); if ($fp){ $new_fp = @fopen ($file_name, "wb"); if ($new_fp){ while(!feof($fp)){ fwrite($new_fp, fread($fp, 1024 * 8 ), 1024 * 8 ); } }else{ return -1; } }else{ return -2; } return 1; } /** * 获取文件(夹)权限 rwx_rwx_rwx */ function get_mode($file){ $Mode = fileperms($file); $Owner = array();$Group=array();$World=array(); if ($Mode &0x1000) $Type = 'p'; // FIFO pipe elseif ($Mode &0x2000) $Type = 'c'; // Character special elseif ($Mode &0x4000) $Type = 'd'; // Directory elseif ($Mode &0x6000) $Type = 'b'; // Block special elseif ($Mode &0x8000) $Type = '-'; // Regular elseif ($Mode &0xA000) $Type = 'l'; // Symbolic Link elseif ($Mode &0xC000) $Type = 's'; // Socket else $Type = 'u'; // UNKNOWN // Determine les permissions par Groupe $Owner['r'] = ($Mode &00400) ? 'r' : '-'; $Owner['w'] = ($Mode &00200) ? 'w' : '-'; $Owner['x'] = ($Mode &00100) ? 'x' : '-'; $Group['r'] = ($Mode &00040) ? 'r' : '-'; $Group['w'] = ($Mode &00020) ? 'w' : '-'; $Group['e'] = ($Mode &00010) ? 'x' : '-'; $World['r'] = ($Mode &00004) ? 'r' : '-'; $World['w'] = ($Mode &00002) ? 'w' : '-'; $World['e'] = ($Mode &00001) ? 'x' : '-'; // Adjuste pour SUID, SGID et sticky bit if ($Mode &0x800) $Owner['e'] = ($Owner['e'] == 'x') ? 's' : 'S'; if ($Mode &0x400) $Group['e'] = ($Group['e'] == 'x') ? 's' : 'S'; if ($Mode &0x200) $World['e'] = ($World['e'] == 'x') ? 't' : 'T'; $Mode = $Type.$Owner['r'].$Owner['w'].$Owner['x'].' '. $Group['r'].$Group['w'].$Group['e'].' '. $World['r'].$World['w'].$World['e']; return $Mode; } /** * 获取可以上传的最大值 * return * byte */ function get_post_max(){ $upload = ini_get('upload_max_filesize'); $upload = $upload==''?ini_get('upload_max_size'):$upload; $post = ini_get('post_max_size'); $upload = intval($upload)*1024*1024; $post = intval($post)*1024*1024; return $upload<$post?$upload:$post; } /** * 文件上传处理。单个文件上传,多个分多次请求 * 调用demo * upload('file','D:/www/'); */ function upload($fileInput, $path = './'){ global $config,$L; if (!isset($_FILES[$fileInput])) show_json($L['upload_error_null'],false); $file = $_FILES[$fileInput]; $file_name = iconv_system($file['name']); $info = _upload($file['tmp_name'],$file['size'],$path.$file_name); show_json($info['data'],$info['code']); } function _upload($tmp_name,$size,$save_path){ global $L; if(file_exists($save_path)){ return array('code'=>false,'data'=>$L['upload_error_exists']); } if(move_uploaded_file($tmp_name,$save_path)){ return array('code'=>true,'data'=>$L['upload_success'],'path'=>$save_path); }else { return array('code'=>fasle,'data'=>$L['move_error']); } } /** * 写日志 * @param string $log 日志信息 * @param string $type 日志类型 [system|app|...] * @param string $level 日志级别 * @return boolean */ function write_log($log, $type = 'default', $level = 'log'){ $now_time = date('[y-m-d H:i:s]'); $now_day = date('Y_m_d'); // 根据类型设置日志目标位置 $target = LOG_PATH . strtolower($type) . '/'; mk_dir($target, 0755); if (! is_writable($target)) exit('path can not write!'); switch($level){// 分级写日志 case 'error': $target .= 'Error_' . $now_day . '.log';break; case 'warning': $target .= 'Warning_' . $now_day . '.log';break; case 'debug': $target .= 'Debug_' . $now_day . '.log';break; case 'info': $target .= 'Info_' . $now_day . '.log';break; case 'db': $target .= 'Db_' . $now_day . '.log';break; default: $target .= 'Log_' . $now_day . '.log';break; } //检测日志文件大小, 超过配置大小则重命名 if (file_exists($target) && filesize($target) <= 100000) { $file_name = substr(basename($target),0,strrpos(basename($target),'.log')).'.log'; rename($target, dirname($target) .'/'. $file_name); } clearstatcache(); return error_log("$now_time $log\n", 3, $target); }