wp5x
10年前发布

php文件和图片上传类

<?php  //文件和图片上传类  /*   使用方法:   function _upload($upload_dir){   $upload = new UploadFile();   //设置上传文件大小   $upload->maxSize=1024*1024*2;//最大2M   //设置上传文件类型   $upload->allowExts  = explode(',','jpg,gif,png,bmp');     //设置附件上传目录   $upload->savePath ='../images/'.$upload_dir."/";   $upload->saveRule = cp_uniqid;     if(!$upload->upload())   {   //捕获上传异常   $this->error($upload->getErrorMsg());   }   else   {   //取得成功上传的文件信息   return $upload->getUploadFileInfo();   }   }   */  class UploadFile{     // 上传文件的最大值   public $maxSize = -1;     // 是否支持多文件上传   public $supportMulti = true;     // 允许上传的文件后缀   //  留空不作后缀检查   public $allowExts = array();     // 允许上传的文件类型   // 留空不做检查   public $allowTypes = array();     // 使用对上传图片进行缩略图处理   public $thumb   =  false;   // 缩略图最大宽度   public $thumbMaxWidth;   // 缩略图最大高度   public $thumbMaxHeight;   // 缩略图前缀   public $thumbPrefix   =  'thumb_';   public $thumbSuffix  =  '';   // 缩略图保存路径   public $thumbPath = '';   // 缩略图文件名   public $thumbFile  = '';   // 是否移除原图   public $thumbRemoveOrigin = false;   // 压缩图片文件上传   public $zipImages = false;   // 启用子目录保存文件   public $autoSub   =  false;   // 子目录创建方式 可以使用hash date   public $subType   = 'hash';   public $dateFormat = 'Ymd';   public $hashLevel =  1; // hash的目录层次   // 上传文件保存路径   public $savePath = '';   public $autoCheck = true; // 是否自动检查附件   // 存在同名是否覆盖   public $uploadReplace = false;     // 上传文件命名规则   // 例如可以是 time uniqid com_create_guid 等   // 必须是一个无需任何参数的函数名 可以使用自定义函数   public $saveRule = '';     // 上传文件Hash规则函数名   // 例如可以是 md5_file sha1_file 等   public $hashType = 'md5_file';     // 错误信息   private $error = '';     // 上传成功的文件信息   private $uploadFileInfo ;     /**    +----------------------------------------------------------    * 架构函数    +----------------------------------------------------------    * @access public    +----------------------------------------------------------    */   public function __construct($maxSize='',$allowExts='',$allowTypes='',$savePath='',$saveRule='')   {    if(!empty($maxSize) && is_numeric($maxSize)) {     $this->maxSize = $maxSize;    }    if(!empty($allowExts)) {     if(is_array($allowExts)) {      $this->allowExts = array_map('strtolower',$allowExts);     }else {      $this->allowExts = explode(',',strtolower($allowExts));     }    }    if(!empty($allowTypes)) {     if(is_array($allowTypes)) {      $this->allowTypes = array_map('strtolower',$allowTypes);     }else {      $this->allowTypes = explode(',',strtolower($allowTypes));     }    }    if(!empty($savePath)) {     $this->savePath = $savePath;    }    if(!empty($saveRule)) {     $this->saveRule = $saveRule;    }       }     private function save($file)   {    $filename = $file['savepath'].$file['savename'];    if(!$this->uploadReplace && is_file($filename)) {     // 不覆盖同名文件     $this->error = '文件已经存在!'.$filename;     return false;    }    // 如果是图像文件 检测文件格式    if( in_array(strtolower($file['extension']),array('gif','jpg','jpeg','bmp','png','swf')) && false === getimagesize($file['tmp_name'])) {     $this->error = '非法图像文件';     return false;    }    if(!move_uploaded_file($file['tmp_name'], iconv('utf-8','gbk',$filename))) {     $this->error = '文件上传保存错误!';     return false;    }    if($this->thumb && in_array(strtolower($file['extension']),array('gif','jpg','jpeg','bmp','png'))) {     $image =  getimagesize($filename);     if(false !== $image) {      //是图像文件生成缩略图      $thumbWidth  = explode(',',$this->thumbMaxWidth);      $thumbHeight  = explode(',',$this->thumbMaxHeight);      $thumbPrefix  = explode(',',$this->thumbPrefix);      $thumbSuffix = explode(',',$this->thumbSuffix);      $thumbFile   = explode(',',$this->thumbFile);      $thumbPath    =  $this->thumbPath?$this->thumbPath:$file['savepath'];      // 生成图像缩略图      if(file_exists(dirname(__FILE__).'/Image.class.php'))      {       require_once(dirname(__FILE__).'/Image.class.php');       $realFilename  =  $this->autoSub?basename($file['savename']):$file['savename'];       for($i=0,$len=count($thumbWidth); $i<$len; $i++) {        $thumbname = $thumbPath.$thumbPrefix[$i].substr($realFilename,0,strrpos($realFilename, '.')).$thumbSuffix[$i].'.'.$file['extension'];        Image::thumb($filename,$thumbname,'',$thumbWidth[$i],$thumbHeight[$i],true);       }       if($this->thumbRemoveOrigin) {        // 生成缩略图之后删除原图        unlink($filename);       }      }     }    }    if($this->zipImags) {     // TODO 对图片压缩包在线解压      }    return true;   }     /**    +----------------------------------------------------------    * 上传文件    +----------------------------------------------------------    * @access public    +----------------------------------------------------------    * @param string $savePath  上传文件保存路径    +----------------------------------------------------------    * @return string    +----------------------------------------------------------    * @throws ThinkExecption    +----------------------------------------------------------    */   public function upload($savePath ='')   {    //如果不指定保存文件名,则由系统默认    if(empty($savePath))    $savePath = $this->savePath;    // 检查上传目录    if(!is_dir($savePath)) {     // 检查目录是否编码后的     if(is_dir(base64_decode($savePath))) {      $savePath = base64_decode($savePath);     }else{      // 尝试创建目录      if(!mkdir($savePath)){       $this->error  =  '上传目录'.$savePath.'不存在';       return false;      }     }    }else {     if(!is_writeable($savePath)) {      $this->error  =  '上传目录'.$savePath.'不可写';      return false;     }    }    $fileInfo = array();    $isUpload   = false;      // 获取上传的文件信息    // 对$_FILES数组信息处理    $files  =  $this->dealFiles($_FILES);    foreach($files as $key => $file) {     //过滤无效的上传     if(!empty($file['name'])) {      //登记上传文件的扩展信息      $file['key']          =  $key;      $file['extension']  = $this->getExt($file['name']);      $file['savepath']   = $savePath;      $file['savename']   = $this->getSaveName($file);        // 自动检查附件      if($this->autoCheck) {       if(!$this->check($file))       return false;      }        //保存上传文件      if(!$this->save($file)) return false;      /*       if(function_exists($this->hashType)) {       $fun =  $this->hashType;       $file['hash']   =  $fun(auto_charset($file['savepath'].$file['savename'],'utf-8','gbk'));       }       */      //上传成功后保存文件信息,供其他地方调用      unset($file['tmp_name'],$file['error']);      $fileInfo[] = $file;      $isUpload   = true;     }    }    if($isUpload) {     $this->uploadFileInfo = $fileInfo;     return true;    }else {     $this->error  =  '没有选择上传文件';     return false;    }   }     /**    +----------------------------------------------------------    * 转换上传文件数组变量为正确的方式    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param array $files  上传的文件变量    +----------------------------------------------------------    * @return array    +----------------------------------------------------------    */   private function dealFiles($files) {    $fileArray = array();    foreach ($files as $file){     if(is_array($file['name'])) {      $keys = array_keys($file);      $count  =  count($file['name']);      for ($i=0; $i<$count; $i++) {       foreach ($keys as $key)       $fileArray[$i][$key] = $file[$key][$i];      }     }else{      $fileArray = $files;     }     break;    }    return $fileArray;   }     /**    +----------------------------------------------------------    * 获取错误代码信息    +----------------------------------------------------------    * @access public    +----------------------------------------------------------    * @param string $errorNo  错误号码    +----------------------------------------------------------    * @return void    +----------------------------------------------------------    * @throws ThinkExecption    +----------------------------------------------------------    */   protected function error($errorNo)   {    switch($errorNo) {     case 1:      $this->error = '上传的文件超过了 php.ini 中 upload_max_filesize 选项限制的值';      break;     case 2:      $this->error = '上传文件的大小超过了 HTML 表单中 MAX_FILE_SIZE 选项指定的值';      break;     case 3:      $this->error = '文件只有部分被上传';      break;     case 4:      $this->error = '没有文件被上传';      break;     case 6:      $this->error = '找不到临时文件夹';      break;     case 7:      $this->error = '文件写入失败';      break;     default:      $this->error = '未知上传错误!';    }    return ;   }     /**    +----------------------------------------------------------    * 根据上传文件命名规则取得保存文件名    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param string $filename 数据    +----------------------------------------------------------    * @return string    +----------------------------------------------------------    */   private function getSaveName($filename)   {    $rule = $this->saveRule;    if(empty($rule)) {//没有定义命名规则,则保持文件名不变     $saveName = $filename['name'];    }else {     if(function_exists($rule)) {      //使用函数生成一个唯一文件标识号      $saveName = $rule().".".$filename['extension'];     }else {      //使用给定的文件名作为标识号      $saveName = $rule.".".$filename['extension'];     }    }    if($this->autoSub) {     // 使用子目录保存文件     $saveName   =  $this->getSubName($filename).'/'.$saveName;    }    return $saveName;   }     /**    +----------------------------------------------------------    * 获取子目录的名称    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param array $file  上传的文件信息    +----------------------------------------------------------    * @return string    +----------------------------------------------------------    */   private function getSubName($file)   {    switch($this->subType) {     case 'date':      $dir   =  date($this->dateFormat,time());      break;     case 'hash':     default:      $name = md5($file['savename']);      $dir   =  '';      for($i=0;$i<$this->hashLevel;$i++) {       $dir   .=  $name{0}.'/';      }      break;    }    if(!is_dir($file['savepath'].$dir)) {     mkdir($file['savepath'].$dir);    }    return $dir;   }     /**    +----------------------------------------------------------    * 检查上传的文件    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param array $file 文件信息    +----------------------------------------------------------    * @return boolean    +----------------------------------------------------------    */   private function check($file) {    if($file['error']!== 0) {     //文件上传失败     //捕获错误代码     $this->error($file['error']);     return false;    }      //检查文件Mime类型    if(!$this->checkType($file['type'])) {     $this->error = '上传文件MIME类型不允许!';     return false;    }    //检查文件类型    if(!$this->checkExt($file['extension'])) {     $this->error ='上传文件类型不允许';     return false;    }    //文件上传成功,进行自定义规则检查    //检查文件大小    if(!$this->checkSize($file['size'])) {     $this->error = '上传文件大小超出限制!';     return false;    }      //检查是否合法上传    if(!$this->checkUpload($file['tmp_name'])) {     $this->error = '非法上传文件!';     return false;    }    return true;   }     /**    +----------------------------------------------------------    * 检查上传的文件类型是否合法    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param string $type 数据    +----------------------------------------------------------    * @return boolean    +----------------------------------------------------------    */   private function checkType($type)   {    if(!empty($this->allowTypes))    return in_array(strtolower($type),$this->allowTypes);    return true;   }       /**    +----------------------------------------------------------    * 检查上传的文件后缀是否合法    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param string $ext 后缀名    +----------------------------------------------------------    * @return boolean    +----------------------------------------------------------    */   private function checkExt($ext)   {    if(!empty($this->allowExts))    return in_array(strtolower($ext),$this->allowExts,true);    return true;   }     /**    +----------------------------------------------------------    * 检查文件大小是否合法    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param integer $size 数据    +----------------------------------------------------------    * @return boolean    +----------------------------------------------------------    */   private function checkSize($size)   {    return !($size > $this->maxSize) || (-1 == $this->maxSize);   }     /**    +----------------------------------------------------------    * 检查文件是否非法提交    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param string $filename 文件名    +----------------------------------------------------------    * @return boolean    +----------------------------------------------------------    */   private function checkUpload($filename)   {    return is_uploaded_file($filename);   }     /**    +----------------------------------------------------------    * 取得上传文件的后缀    +----------------------------------------------------------    * @access private    +----------------------------------------------------------    * @param string $filename 文件名    +----------------------------------------------------------    * @return boolean    +----------------------------------------------------------    */   private function getExt($filename)   {    $pathinfo = pathinfo($filename);    return $pathinfo['extension'];   }     /**    +----------------------------------------------------------    * 取得上传文件的信息    +----------------------------------------------------------    * @access public    +----------------------------------------------------------    * @return array    +----------------------------------------------------------    */   public function getUploadFileInfo()   {    return $this->uploadFileInfo;   }     /**    +----------------------------------------------------------    * 取得最后一次错误信息    +----------------------------------------------------------    * @access public    +----------------------------------------------------------    * @return string    +----------------------------------------------------------    */   public function getErrorMsg()   {    return $this->error;   }    }//类定义结束  ?>