UploadFile.class.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631
  1. <?php
  2. /**
  3. +------------------------------------------------------------------------------
  4. * 文件上传类
  5. +------------------------------------------------------------------------------
  6. * @category ORG
  7. * @package ORG
  8. * @subpackage Net
  9. * @author liu21st <liu21st@gmail.com>
  10. * @version $Id$
  11. +------------------------------------------------------------------------------
  12. */
  13. class UploadFile extends Think
  14. {//类定义开始
  15. // 上传文件的最大值
  16. public $maxSize = -1;
  17. // 是否支持多文件上传
  18. public $supportMulti = true;
  19. // 允许上传的文件后缀
  20. // 留空不作后缀检查
  21. public $allowExts = array();
  22. // 允许上传的文件类型
  23. // 留空不做检查
  24. public $allowTypes = array();
  25. // 使用对上传图片进行缩略图处理
  26. public $thumb = false;
  27. // 图库类包路径
  28. public $imageClassPath = 'ORG.Util.Image';
  29. // 缩略图最大宽度
  30. public $thumbMaxWidth;
  31. // 缩略图最大高度
  32. public $thumbMaxHeight;
  33. // 缩略图前缀
  34. public $thumbPrefix = 'thumb_';
  35. public $thumbSuffix = '';
  36. // 缩略图保存路径
  37. public $thumbPath = '';
  38. // 缩略图文件名
  39. public $thumbFile = '';
  40. // 是否移除原图
  41. public $thumbRemoveOrigin = false;
  42. // 压缩图片文件上传
  43. public $zipImages = false;
  44. // 启用子目录保存文件
  45. public $autoSub = false;
  46. // 子目录创建方式 可以使用hash date
  47. public $subType = 'hash';
  48. public $dateFormat = 'Ymd';
  49. public $hashLevel = 1; // hash的目录层次
  50. // 上传文件保存路径
  51. public $savePath = '';
  52. public $autoCheck = true; // 是否自动检查附件
  53. // 存在同名是否覆盖
  54. public $uploadReplace = false;
  55. // 上传文件命名规则
  56. // 例如可以是 time uniqid com_create_guid 等
  57. // 必须是一个无需任何参数的函数名 可以使用自定义函数
  58. public $saveRule = '';
  59. // 上传文件Hash规则函数名
  60. // 例如可以是 md5_file sha1_file 等
  61. public $hashType = 'md5_file';
  62. // 错误信息
  63. private $error = '';
  64. // 上传成功的文件信息
  65. private $uploadFileInfo ;
  66. /**
  67. +----------------------------------------------------------
  68. * 架构函数
  69. +----------------------------------------------------------
  70. * @access public
  71. +----------------------------------------------------------
  72. */
  73. public function __construct($maxSize='',$allowExts='',$allowTypes='',$savePath='',$saveRule='')
  74. {
  75. if(!empty($maxSize) && is_numeric($maxSize)) {
  76. $this->maxSize = $maxSize;
  77. }
  78. if(!empty($allowExts)) {
  79. if(is_array($allowExts)) {
  80. $this->allowExts = array_map('strtolower',$allowExts);
  81. }else {
  82. $this->allowExts = explode(',',strtolower($allowExts));
  83. }
  84. }
  85. if(!empty($allowTypes)) {
  86. if(is_array($allowTypes)) {
  87. $this->allowTypes = array_map('strtolower',$allowTypes);
  88. }else {
  89. $this->allowTypes = explode(',',strtolower($allowTypes));
  90. }
  91. }
  92. if(!empty($saveRule)) {
  93. $this->saveRule = $saveRule;
  94. }else{
  95. $this->saveRule = C('UPLOAD_FILE_RULE');
  96. }
  97. $this->savePath = $savePath;
  98. }
  99. /**
  100. +----------------------------------------------------------
  101. * 上传一个文件
  102. +----------------------------------------------------------
  103. * @access public
  104. +----------------------------------------------------------
  105. * @param mixed $name 数据
  106. * @param string $value 数据表名
  107. +----------------------------------------------------------
  108. * @return string
  109. +----------------------------------------------------------
  110. * @throws ThinkExecption
  111. +----------------------------------------------------------
  112. */
  113. private function save($file)
  114. {
  115. $filename = $file['savepath'].$file['savename'];
  116. if(!$this->uploadReplace && is_file($filename)) {
  117. // 不覆盖同名文件
  118. $this->error = '文件已经存在!'.$filename;
  119. return false;
  120. }
  121. // 如果是图像文件 检测文件格式
  122. if( in_array(strtolower($file['extension']),array('gif','jpg','jpeg','bmp','png','swf')) && false === getimagesize($file['tmp_name'])) {
  123. $this->error = '非法图像文件';
  124. return false;
  125. }
  126. if(in_array(strtolower($file['extension']),array('gif','jpg','jpeg','bmp','png')) && getimagesize($file['tmp_name'])) {
  127. $tmp_imagesize = @getimagesize($file['tmp_name']);
  128. list($tmp_width, $tmp_height, $tmp_type) = (array)$tmp_imagesize;
  129. $tmp_size = $tmp_width * $tmp_height;
  130. if($tmp_size > 16777216 || $tmp_size < 4 || empty($tmp_type) || strpos($tmp_imagesize['mime'], 'flash') > 0) {
  131. @unlink($file['tmp_name']);
  132. $this->error = '非法图像文件';
  133. return false;
  134. }
  135. }
  136. if(!move_uploaded_file($file['tmp_name'], auto_charset($filename,'utf-8','gbk'))) {
  137. $this->error = '文件上传保存错误!';
  138. return false;
  139. }
  140. if($this->thumb && in_array(strtolower($file['extension']),array('gif','jpg','jpeg','bmp','png'))) {
  141. $image = getimagesize($filename);
  142. if(false !== $image) {
  143. //是图像文件生成缩略图
  144. $thumbWidth = explode(',',$this->thumbMaxWidth);
  145. $thumbHeight = explode(',',$this->thumbMaxHeight);
  146. $thumbPrefix = explode(',',$this->thumbPrefix);
  147. $thumbSuffix = explode(',',$this->thumbSuffix);
  148. $thumbFile = explode(',',$this->thumbFile);
  149. $thumbPath = $this->thumbPath?$this->thumbPath:$file['savepath'];
  150. // 生成图像缩略图
  151. import($this->imageClassPath);
  152. $realFilename = $this->autoSub?basename($file['savename']):$file['savename'];
  153. for($i=0,$len=count($thumbWidth); $i<$len; $i++) {
  154. $thumbname = $thumbPath.$thumbPrefix[$i].substr($realFilename,0,strrpos($realFilename, '.')).$thumbSuffix[$i].'.'.$file['extension'];
  155. Image::thumb($filename,$thumbname,'',$thumbWidth[$i],$thumbHeight[$i],true);
  156. }
  157. if($this->thumbRemoveOrigin) {
  158. // 生成缩略图之后删除原图
  159. unlink($filename);
  160. }
  161. }
  162. }
  163. if($this->zipImags) {
  164. // TODO 对图片压缩包在线解压
  165. }
  166. return true;
  167. }
  168. /**
  169. +----------------------------------------------------------
  170. * 上传所有文件
  171. +----------------------------------------------------------
  172. * @access public
  173. +----------------------------------------------------------
  174. * @param string $savePath 上传文件保存路径
  175. +----------------------------------------------------------
  176. * @return string
  177. +----------------------------------------------------------
  178. * @throws ThinkExecption
  179. +----------------------------------------------------------
  180. */
  181. public function upload($savePath ='')
  182. {
  183. //如果不指定保存文件名,则由系统默认
  184. if(empty($savePath))
  185. $savePath = $this->savePath;
  186. // 检查上传目录
  187. if(!is_dir($savePath)) {
  188. // 检查目录是否编码后的
  189. if(is_dir(base64_decode($savePath))) {
  190. $savePath = base64_decode($savePath);
  191. }else{
  192. // 尝试创建目录
  193. if(!mkdir($savePath)){
  194. $this->error = '上传目录'.$savePath.'不存在';
  195. return false;
  196. }
  197. }
  198. }else {
  199. if(!is_writeable($savePath)) {
  200. $this->error = '上传目录'.$savePath.'不可写';
  201. return false;
  202. }
  203. }
  204. $fileInfo = array();
  205. $isUpload = false;
  206. // 获取上传的文件信息
  207. // 对$_FILES数组信息处理
  208. $files = $this->dealFiles($_FILES);
  209. foreach($files as $key => $file) {
  210. //过滤无效的上传
  211. if(!empty($file['name'])) {
  212. //登记上传文件的扩展信息
  213. $file['key'] = $key;
  214. $file['extension'] = strtolower($this->getExt($file['name']));
  215. $file['savepath'] = $savePath;
  216. $file['savename'] = strtolower($this->getSaveName($file));
  217. // 自动检查附件
  218. if($this->autoCheck) {
  219. if(!$this->check($file))
  220. return false;
  221. }
  222. //保存上传文件
  223. if(!$this->save($file)) return false;
  224. if(function_exists($this->hashType)) {
  225. $fun = $this->hashType;
  226. $file['hash'] = $fun(auto_charset($file['savepath'].$file['savename'],'utf-8','gbk'));
  227. }
  228. //上传成功后保存文件信息,供其他地方调用
  229. unset($file['tmp_name'],$file['error']);
  230. $fileInfo[] = $file;
  231. $isUpload = true;
  232. }
  233. }
  234. if($isUpload) {
  235. $this->uploadFileInfo = $fileInfo;
  236. return true;
  237. }else {
  238. $this->error = '没有选择上传文件';
  239. return false;
  240. }
  241. }
  242. /**
  243. +----------------------------------------------------------
  244. * 上传单个上传字段中的文件 支持多附件
  245. +----------------------------------------------------------
  246. * @access public
  247. +----------------------------------------------------------
  248. * @param array $file 上传文件信息
  249. * @param string $savePath 上传文件保存路径
  250. +----------------------------------------------------------
  251. * @return string
  252. +----------------------------------------------------------
  253. * @throws ThinkExecption
  254. +----------------------------------------------------------
  255. */
  256. public function uploadOne($file,$savePath=''){
  257. //如果不指定保存文件名,则由系统默认
  258. if(empty($savePath))
  259. $savePath = $this->savePath;
  260. // 检查上传目录
  261. if(!is_dir($savePath)) {
  262. // 尝试创建目录
  263. if(!mk_dir($savePath)){
  264. $this->error = '上传目录'.$savePath.'不存在';
  265. return false;
  266. }
  267. }else {
  268. if(!is_writeable($savePath)) {
  269. $this->error = '上传目录'.$savePath.'不可写';
  270. return false;
  271. }
  272. }
  273. //过滤无效的上传
  274. if(!empty($file['name'])) {
  275. $fileArray = array();
  276. if(is_array($file['name'])) {
  277. $keys = array_keys($file);
  278. $count = count($file['name']);
  279. for ($i=0; $i<$count; $i++) {
  280. foreach ($keys as $key)
  281. $fileArray[$i][$key] = $file[$key][$i];
  282. }
  283. }else{
  284. $fileArray[] = $file;
  285. }
  286. $info = array();
  287. foreach ($fileArray as $key=>$file){
  288. //登记上传文件的扩展信息
  289. $file['extension'] = $this->getExt($file['name']);
  290. $file['savepath'] = $savePath;
  291. $file['savename'] = $this->getSaveName($file);
  292. // 自动检查附件
  293. if($this->autoCheck) {
  294. if(!$this->check($file))
  295. return false;
  296. }
  297. //保存上传文件
  298. if(!$this->save($file)) return false;
  299. if(function_exists($this->hashType)) {
  300. $fun = $this->hashType;
  301. $file['hash'] = $fun(auto_charset($file['savepath'].$file['savename'],'utf-8','gbk'));
  302. }
  303. unset($file['tmp_name'],$file['error']);
  304. $info[] = $file;
  305. }
  306. // 返回上传的文件信息
  307. return $info;
  308. }else {
  309. $this->error = '没有选择上传文件';
  310. return false;
  311. }
  312. }
  313. /**
  314. +----------------------------------------------------------
  315. * 转换上传文件数组变量为正确的方式
  316. +----------------------------------------------------------
  317. * @access private
  318. +----------------------------------------------------------
  319. * @param array $files 上传的文件变量
  320. +----------------------------------------------------------
  321. * @return array
  322. +----------------------------------------------------------
  323. */
  324. private function dealFiles($files) {
  325. $fileArray = array();
  326. $n = 0;
  327. foreach ($files as $file){
  328. if(is_array($file['name'])) {
  329. $keys = array_keys($file);
  330. $count = count($file['name']);
  331. for ($i=0; $i<$count; $i++) {
  332. foreach ($keys as $key)
  333. $fileArray[$n][$key] = $file[$key][$i];
  334. $n++;
  335. }
  336. }else{
  337. $fileArray[$n] = $file;
  338. $n++;
  339. }
  340. }
  341. return $fileArray;
  342. }
  343. /**
  344. +----------------------------------------------------------
  345. * 获取错误代码信息
  346. +----------------------------------------------------------
  347. * @access public
  348. +----------------------------------------------------------
  349. * @param string $errorNo 错误号码
  350. +----------------------------------------------------------
  351. * @return void
  352. +----------------------------------------------------------
  353. * @throws ThinkExecption
  354. +----------------------------------------------------------
  355. */
  356. protected function error($errorNo)
  357. {
  358. switch($errorNo) {
  359. case 1:
  360. $this->error = '上传的文件超过了 php.ini 中 upload_max_filesize 选项限制的值';
  361. break;
  362. case 2:
  363. $this->error = '上传文件的大小超过了 HTML 表单中 MAX_FILE_SIZE 选项指定的值';
  364. break;
  365. case 3:
  366. $this->error = '文件只有部分被上传';
  367. break;
  368. case 4:
  369. $this->error = '没有文件被上传';
  370. break;
  371. case 6:
  372. $this->error = '找不到临时文件夹';
  373. break;
  374. case 7:
  375. $this->error = '文件写入失败';
  376. break;
  377. default:
  378. $this->error = '未知上传错误!';
  379. }
  380. return ;
  381. }
  382. /**
  383. +----------------------------------------------------------
  384. * 根据上传文件命名规则取得保存文件名
  385. +----------------------------------------------------------
  386. * @access private
  387. +----------------------------------------------------------
  388. * @param string $filename 数据
  389. +----------------------------------------------------------
  390. * @return string
  391. +----------------------------------------------------------
  392. */
  393. private function getSaveName($filename)
  394. {
  395. $rule = $this->saveRule;
  396. if(empty($rule)) {//没有定义命名规则,则保持文件名不变
  397. $saveName = $filename['name'];
  398. }else {
  399. if(function_exists($rule)) {
  400. //使用函数生成一个唯一文件标识号
  401. $saveName = $rule().".".$filename['extension'];
  402. }else {
  403. //使用给定的文件名作为标识号
  404. $saveName = $rule.".".$filename['extension'];
  405. }
  406. }
  407. if($this->autoSub) {
  408. // 使用子目录保存文件
  409. $filename['savename'] = $saveName;
  410. $saveName = $this->getSubName($filename).'/'.$saveName;
  411. }
  412. return $saveName;
  413. }
  414. /**
  415. +----------------------------------------------------------
  416. * 获取子目录的名称
  417. +----------------------------------------------------------
  418. * @access private
  419. +----------------------------------------------------------
  420. * @param array $file 上传的文件信息
  421. +----------------------------------------------------------
  422. * @return string
  423. +----------------------------------------------------------
  424. */
  425. private function getSubName($file)
  426. {
  427. switch($this->subType) {
  428. case 'date':
  429. $dir = date($this->dateFormat,time());
  430. break;
  431. case 'hash':
  432. default:
  433. $name = md5($file['savename']);
  434. $dir = '';
  435. for($i=0;$i<$this->hashLevel;$i++) {
  436. $dir .= $name{$i}.'/';
  437. }
  438. break;
  439. }
  440. if(!is_dir($file['savepath'].$dir)) {
  441. mk_dir($file['savepath'].$dir);
  442. }
  443. return $dir;
  444. }
  445. /**
  446. +----------------------------------------------------------
  447. * 检查上传的文件
  448. +----------------------------------------------------------
  449. * @access private
  450. +----------------------------------------------------------
  451. * @param array $file 文件信息
  452. +----------------------------------------------------------
  453. * @return boolean
  454. +----------------------------------------------------------
  455. */
  456. private function check($file) {
  457. if($file['error']!== 0) {
  458. //文件上传失败
  459. //捕获错误代码
  460. $this->error($file['error']);
  461. return false;
  462. }
  463. //文件上传成功,进行自定义规则检查
  464. //检查文件大小
  465. if(!$this->checkSize($file['size'])) {
  466. $this->error = '上传文件大小不符!';
  467. return false;
  468. }
  469. //检查文件Mime类型
  470. if(!$this->checkType($file['type'])) {
  471. $this->error = '上传文件MIME类型不允许!';
  472. return false;
  473. }
  474. //检查文件类型
  475. if(!$this->checkExt($file['extension'])) {
  476. $this->error ='上传文件类型不允许';
  477. return false;
  478. }
  479. //检查是否合法上传
  480. if(!$this->checkUpload($file['tmp_name'])) {
  481. $this->error = '非法上传文件!';
  482. return false;
  483. }
  484. return true;
  485. }
  486. /**
  487. +----------------------------------------------------------
  488. * 检查上传的文件类型是否合法
  489. +----------------------------------------------------------
  490. * @access private
  491. +----------------------------------------------------------
  492. * @param string $type 数据
  493. +----------------------------------------------------------
  494. * @return boolean
  495. +----------------------------------------------------------
  496. */
  497. private function checkType($type)
  498. {
  499. if(!empty($this->allowTypes))
  500. return in_array(strtolower($type),$this->allowTypes);
  501. return true;
  502. }
  503. /**
  504. +----------------------------------------------------------
  505. * 检查上传的文件后缀是否合法
  506. +----------------------------------------------------------
  507. * @access private
  508. +----------------------------------------------------------
  509. * @param string $ext 后缀名
  510. +----------------------------------------------------------
  511. * @return boolean
  512. +----------------------------------------------------------
  513. */
  514. private function checkExt($ext)
  515. {
  516. if(!empty($this->allowExts))
  517. return in_array(strtolower($ext),$this->allowExts,true);
  518. return true;
  519. }
  520. /**
  521. +----------------------------------------------------------
  522. * 检查文件大小是否合法
  523. +----------------------------------------------------------
  524. * @access private
  525. +----------------------------------------------------------
  526. * @param integer $size 数据
  527. +----------------------------------------------------------
  528. * @return boolean
  529. +----------------------------------------------------------
  530. */
  531. private function checkSize($size)
  532. {
  533. return !($size > $this->maxSize) || (-1 == $this->maxSize);
  534. }
  535. /**
  536. +----------------------------------------------------------
  537. * 检查文件是否非法提交
  538. +----------------------------------------------------------
  539. * @access private
  540. +----------------------------------------------------------
  541. * @param string $filename 文件名
  542. +----------------------------------------------------------
  543. * @return boolean
  544. +----------------------------------------------------------
  545. */
  546. private function checkUpload($filename)
  547. {
  548. return is_uploaded_file($filename);
  549. }
  550. /**
  551. +----------------------------------------------------------
  552. * 取得上传文件的后缀
  553. +----------------------------------------------------------
  554. * @access private
  555. +----------------------------------------------------------
  556. * @param string $filename 文件名
  557. +----------------------------------------------------------
  558. * @return boolean
  559. +----------------------------------------------------------
  560. */
  561. private function getExt($filename)
  562. {
  563. $pathinfo = pathinfo($filename);
  564. return $pathinfo['extension'];
  565. }
  566. /**
  567. +----------------------------------------------------------
  568. * 取得上传文件的信息
  569. +----------------------------------------------------------
  570. * @access public
  571. +----------------------------------------------------------
  572. * @return array
  573. +----------------------------------------------------------
  574. */
  575. public function getUploadFileInfo()
  576. {
  577. return $this->uploadFileInfo;
  578. }
  579. /**
  580. +----------------------------------------------------------
  581. * 取得最后一次错误信息
  582. +----------------------------------------------------------
  583. * @access public
  584. +----------------------------------------------------------
  585. * @return string
  586. +----------------------------------------------------------
  587. */
  588. public function getErrorMsg()
  589. {
  590. return $this->error;
  591. }
  592. }//类定义结束
  593. ?>