523 lines
18 KiB
PHP
523 lines
18 KiB
PHP
|
<?php
|
|||
|
// +----------------------------------------------------------------------
|
|||
|
// | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
|
|||
|
// +----------------------------------------------------------------------
|
|||
|
// | Copyright (c) 2009 http://thinkphp.cn All rights reserved.
|
|||
|
// +----------------------------------------------------------------------
|
|||
|
// | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
|
|||
|
// +----------------------------------------------------------------------
|
|||
|
// | Author: liu21st <liu21st@gmail.com>
|
|||
|
// +----------------------------------------------------------------------
|
|||
|
namespace Org\Net;
|
|||
|
/**
|
|||
|
* 文件上传类
|
|||
|
* @category ORG
|
|||
|
* @package ORG
|
|||
|
* @subpackage Net
|
|||
|
* @author liu21st <liu21st@gmail.com>
|
|||
|
*/
|
|||
|
class UploadFile {//类定义开始
|
|||
|
|
|||
|
private $config = array(
|
|||
|
'maxSize' => -1, // 上传文件的最大值
|
|||
|
'supportMulti' => true, // 是否支持多文件上传
|
|||
|
'allowExts' => array(), // 允许上传的文件后缀 留空不作后缀检查
|
|||
|
'allowTypes' => array(), // 允许上传的文件类型 留空不做检查
|
|||
|
'thumb' => false, // 使用对上传图片进行缩略图处理
|
|||
|
'imageClassPath' => 'ORG.Util.Image', // 图库类包路径
|
|||
|
'thumbMaxWidth' => '',// 缩略图最大宽度
|
|||
|
'thumbMaxHeight' => '',// 缩略图最大高度
|
|||
|
'thumbPrefix' => 'thumb_',// 缩略图前缀
|
|||
|
'thumbSuffix' => '',
|
|||
|
'thumbPath' => '',// 缩略图保存路径
|
|||
|
'thumbFile' => '',// 缩略图文件名
|
|||
|
'thumbExt' => '',// 缩略图扩展名
|
|||
|
'thumbRemoveOrigin' => false,// 是否移除原图
|
|||
|
'zipImages' => false,// 压缩图片文件上传
|
|||
|
'autoSub' => false,// 启用子目录保存文件
|
|||
|
'subType' => 'hash',// 子目录创建方式 可以使用hash date custom
|
|||
|
'subDir' => '', // 子目录名称 subType为custom方式后有效
|
|||
|
'dateFormat' => 'Ymd',
|
|||
|
'hashLevel' => 1, // hash的目录层次
|
|||
|
'savePath' => '',// 上传文件保存路径
|
|||
|
'autoCheck' => true, // 是否自动检查附件
|
|||
|
'uploadReplace' => false,// 存在同名是否覆盖
|
|||
|
'saveRule' => 'uniqid',// 上传文件命名规则
|
|||
|
'hashType' => 'md5_file',// 上传文件Hash规则函数名
|
|||
|
);
|
|||
|
|
|||
|
// 错误信息
|
|||
|
private $error = '';
|
|||
|
// 上传成功的文件信息
|
|||
|
private $uploadFileInfo ;
|
|||
|
|
|||
|
public function __get($name){
|
|||
|
if(isset($this->config[$name])) {
|
|||
|
return $this->config[$name];
|
|||
|
}
|
|||
|
return null;
|
|||
|
}
|
|||
|
|
|||
|
public function __set($name,$value){
|
|||
|
if(isset($this->config[$name])) {
|
|||
|
$this->config[$name] = $value;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public function __isset($name){
|
|||
|
return isset($this->config[$name]);
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 架构函数
|
|||
|
* @access public
|
|||
|
* @param array $config 上传参数
|
|||
|
*/
|
|||
|
public function __construct($config=array()) {
|
|||
|
if(is_array($config)) {
|
|||
|
$this->config = array_merge($this->config,$config);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 上传一个文件
|
|||
|
* @access public
|
|||
|
* @param mixed $name 数据
|
|||
|
* @param string $value 数据表名
|
|||
|
* @return string
|
|||
|
*/
|
|||
|
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'))) {
|
|||
|
$info = getimagesize($file['tmp_name']);
|
|||
|
if(false === $info || ('gif' == strtolower($file['extension']) && empty($info['bits']))){
|
|||
|
$this->error = '非法图像文件';
|
|||
|
return false;
|
|||
|
}
|
|||
|
}
|
|||
|
if(!move_uploaded_file($file['tmp_name'], $this->autoCharset($filename,'utf-8','gbk'))) {
|
|||
|
$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:dirname($filename).'/';
|
|||
|
$thumbExt = $this->thumbExt ? $this->thumbExt : $file['extension']; //自定义缩略图扩展名
|
|||
|
// 生成图像缩略图
|
|||
|
import($this->imageClassPath);
|
|||
|
for($i=0,$len=count($thumbWidth); $i<$len; $i++) {
|
|||
|
if(!empty($thumbFile[$i])) {
|
|||
|
$thumbname = $thumbFile[$i];
|
|||
|
}else{
|
|||
|
$prefix = isset($thumbPrefix[$i])?$thumbPrefix[$i]:$thumbPrefix[0];
|
|||
|
$suffix = isset($thumbSuffix[$i])?$thumbSuffix[$i]:$thumbSuffix[0];
|
|||
|
$thumbname = $prefix.basename($filename,'.'.$file['extension']).$suffix;
|
|||
|
}
|
|||
|
Image::thumb($filename,$thumbPath.$thumbname.'.'.$thumbExt,'',$thumbWidth[$i],$thumbHeight[$i],true);
|
|||
|
}
|
|||
|
if($this->thumbRemoveOrigin) {
|
|||
|
// 生成缩略图之后删除原图
|
|||
|
unlink($filename);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
if($this->zipImags) {
|
|||
|
// TODO 对图片压缩包在线解压
|
|||
|
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 上传所有文件
|
|||
|
* @access public
|
|||
|
* @param string $savePath 上传文件保存路径
|
|||
|
* @return string
|
|||
|
*/
|
|||
|
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'])) {
|
|||
|
//登记上传文件的扩展信息
|
|||
|
if(!isset($file['key'])) $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($this->autoCharset($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 public
|
|||
|
* @param array $file 上传文件信息
|
|||
|
* @param string $savePath 上传文件保存路径
|
|||
|
* @return string
|
|||
|
*/
|
|||
|
public function uploadOne($file,$savePath=''){
|
|||
|
//如果不指定保存文件名,则由系统默认
|
|||
|
if(empty($savePath))
|
|||
|
$savePath = $this->savePath;
|
|||
|
// 检查上传目录
|
|||
|
if(!is_dir($savePath)) {
|
|||
|
// 尝试创建目录
|
|||
|
if(!mkdir($savePath,0777,true)){
|
|||
|
$this->error = '上传目录'.$savePath.'不存在';
|
|||
|
return false;
|
|||
|
}
|
|||
|
}else {
|
|||
|
if(!is_writeable($savePath)) {
|
|||
|
$this->error = '上传目录'.$savePath.'不可写';
|
|||
|
return false;
|
|||
|
}
|
|||
|
}
|
|||
|
//过滤无效的上传
|
|||
|
if(!empty($file['name'])) {
|
|||
|
$fileArray = array();
|
|||
|
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[] = $file;
|
|||
|
}
|
|||
|
$info = array();
|
|||
|
foreach ($fileArray as $key=>$file){
|
|||
|
//登记上传文件的扩展信息
|
|||
|
$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($this->autoCharset($file['savepath'].$file['savename'],'utf-8','gbk'));
|
|||
|
}
|
|||
|
unset($file['tmp_name'],$file['error']);
|
|||
|
$info[] = $file;
|
|||
|
}
|
|||
|
// 返回上传的文件信息
|
|||
|
return $info;
|
|||
|
}else {
|
|||
|
$this->error = '没有选择上传文件';
|
|||
|
return false;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 转换上传文件数组变量为正确的方式
|
|||
|
* @access private
|
|||
|
* @param array $files 上传的文件变量
|
|||
|
* @return array
|
|||
|
*/
|
|||
|
private function dealFiles($files) {
|
|||
|
$fileArray = array();
|
|||
|
$n = 0;
|
|||
|
foreach ($files as $key=>$file){
|
|||
|
if(is_array($file['name'])) {
|
|||
|
$keys = array_keys($file);
|
|||
|
$count = count($file['name']);
|
|||
|
for ($i=0; $i<$count; $i++) {
|
|||
|
$fileArray[$n]['key'] = $key;
|
|||
|
foreach ($keys as $_key){
|
|||
|
$fileArray[$n][$_key] = $file[$_key][$i];
|
|||
|
}
|
|||
|
$n++;
|
|||
|
}
|
|||
|
}else{
|
|||
|
$fileArray[$key] = $file;
|
|||
|
}
|
|||
|
}
|
|||
|
return $fileArray;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 获取错误代码信息
|
|||
|
* @access public
|
|||
|
* @param string $errorNo 错误号码
|
|||
|
* @return void
|
|||
|
*/
|
|||
|
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) {
|
|||
|
// 使用子目录保存文件
|
|||
|
$filename['savename'] = $saveName;
|
|||
|
$saveName = $this->getSubName($filename).$saveName;
|
|||
|
}
|
|||
|
return $saveName;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 获取子目录的名称
|
|||
|
* @access private
|
|||
|
* @param array $file 上传的文件信息
|
|||
|
* @return string
|
|||
|
*/
|
|||
|
private function getSubName($file) {
|
|||
|
switch($this->subType) {
|
|||
|
case 'custom':
|
|||
|
$dir = $this->subDir;
|
|||
|
break;
|
|||
|
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{$i}.'/';
|
|||
|
}
|
|||
|
break;
|
|||
|
}
|
|||
|
if(!is_dir($file['savepath'].$dir)) {
|
|||
|
mkdir($file['savepath'].$dir,0777,true);
|
|||
|
}
|
|||
|
return $dir;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 检查上传的文件
|
|||
|
* @access private
|
|||
|
* @param array $file 文件信息
|
|||
|
* @return boolean
|
|||
|
*/
|
|||
|
private function check($file) {
|
|||
|
if($file['error']!== 0) {
|
|||
|
//文件上传失败
|
|||
|
//捕获错误代码
|
|||
|
$this->error($file['error']);
|
|||
|
return false;
|
|||
|
}
|
|||
|
//文件上传成功,进行自定义规则检查
|
|||
|
//检查文件大小
|
|||
|
if(!$this->checkSize($file['size'])) {
|
|||
|
$this->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->checkUpload($file['tmp_name'])) {
|
|||
|
$this->error = '非法上传文件!';
|
|||
|
return false;
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
// 自动转换字符集 支持数组转换
|
|||
|
private function autoCharset($fContents, $from='gbk', $to='utf-8') {
|
|||
|
$from = strtoupper($from) == 'UTF8' ? 'utf-8' : $from;
|
|||
|
$to = strtoupper($to) == 'UTF8' ? 'utf-8' : $to;
|
|||
|
if (strtoupper($from) === strtoupper($to) || empty($fContents) || (is_scalar($fContents) && !is_string($fContents))) {
|
|||
|
//如果编码相同或者非字符串标量则不转换
|
|||
|
return $fContents;
|
|||
|
}
|
|||
|
if (function_exists('mb_convert_encoding')) {
|
|||
|
return mb_convert_encoding($fContents, $to, $from);
|
|||
|
} elseif (function_exists('iconv')) {
|
|||
|
return iconv($from, $to, $fContents);
|
|||
|
} else {
|
|||
|
return $fContents;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* 检查上传的文件类型是否合法
|
|||
|
* @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;
|
|||
|
}
|
|||
|
}
|