開發一個 composer 通用文件上傳包,發佈到 Packagist,並在 Laravel 中測試。php
$ git clone git@github.com:guanguans/uploadfile.git $ cd uploadfile
composer.json
文件yzm@Alert MINGW64 /i/phpstudy/WWW/uploadfile $ composer init Welcome to the Composer config generator This command will guide you through creating your composer.json config. Package name (<vendor>/<name>) [yzm/try-make-package]: guanguans/uploadfile Description []: 一個通用文件上傳包 Author [guanguans <53222411@qq.com>, n to skip]: guanguans <yzmguanguan@gmail.com> Minimum Stability []: dev Package Type (e.g. library, project, metapackage, composer-plugin) []: l ibrary License []: MIT Define your dependencies. Would you like to define your dependencies (require) interactively [yes] ? yes Search for a package: php Enter the version constraint to require (or leave blank to use the lates t version): >=5.4.0 Search for a package: Would you like to define your dev dependencies (require-dev) interactive ly [yes]? yes Search for a package: php Enter the version constraint to require (or leave blank to use the lates t version): >=5.4.0 Search for a package: { "name": "guanguans/uploadfile", "description": "一個通用文件上傳包", "type": "library", "require": { "php": ">=5.4" }, "require-dev": { "php": ">=5.4" }, "license": "MIT", "authors": [ { "name": "guanguans", "email": "yzmguanguan@gmail.com" } ], "minimum-stability": "dev" } Do you confirm generation [yes]? yes
yzm@Alert MINGW64 /i/phpstudy/WWW/uploadfile $ composer init Welcome to the Composer config generator This command will guide you through creating your composer.json config. // 1. 輸入項目命名空間 // 注意<vendor>/<name> 必需要符合 [a-z0-9_.-]+/[a-z0-9_.-]+ Package name (<vendor>/<name>) [dell/htdocs]: yourname/projectname // 2. 項目描述 Description []: 這是一個測試 // 3. 輸入做者信息,能夠直接回車 Author [guanguans <53222411@qq.com>, n to skip]: // 4. 輸入最低穩定版本,stable, RC, beta, alpha, dev Minimum Stability []: dev // 5. 輸入項目類型, Package Type (e.g. library, project, metapackage, composer-plugin) []: library // 6. 輸入受權類型 License []: > Define your dependencies. // 7. 輸入依賴信息 Would you like to define your dependencies (require) interactively [yes]? // 若是須要依賴,則輸入要安裝的依賴 Search for a package: php // 輸入版本號 Enter the version constraint to require (or leave blank to use the latest version): >=5.4.0 // 如需多個,則重複以上兩個步驟 // 8. 是否須要require-dev, Would you like to define your dev dependencies (require-dev) interactively [yes]? // 操做同上 { "name": "guanguans/uploadfile", "description": "一個通用文件上傳包", "type": "library", "require": { "php": ">=5.4" }, "require-dev": { "php": ">=5.4" }, "license": "MIT", "authors": [ { "name": "guanguans", "email": "yzmguanguan@gmail.com" } ], "minimum-stability": "dev" } // 9. 是否生成composer.json Do you confirm generation [yes]? yes
在上一步生成的composer.json
中追加css
"autoload": { "psr-4": { "Guanguans\\": "src/" } }
uploadfile/src/UploadFile.php
├─uploadfile │ ├─src │ │ ├─UploadFile.php │ └─composer.json
<?php /** * 通用文件上傳類 * @author guanguans <yzmguanguan@gmail.com> */ namespace Guanguans; // 注意命名空間與 composer.json 中的一致 class UploadFile { private $config = [ 'maxSize' => -1, // 上傳文件的最大值 'supportMulti' => true, // 是否支持多文件上傳 'allowExts' => [], // 容許上傳的文件後綴 留空不做後綴檢查 'allowTypes' => [], // 容許上傳的文件類型 留空不作檢查 'thumb' => false, // 使用對上傳圖片進行縮略圖處理 'imageClassPath' => 'ORG.Util.Image', // 圖庫類包路徑 'thumbMaxWidth' => '',// 縮略圖最大寬度 'thumbMaxHeight' => '',// 縮略圖最大高度 'thumbPrefix' => 'thumb_',// 縮略圖前綴 'thumbSuffix' => '', 'thumbPath' => '',// 縮略圖保存路徑 'thumbFile' => '',// 縮略圖文件名 'thumbExt' => '',// 縮略圖擴展名 'thumbRemoveOrigin' => false,// 是否移除原圖 'thumbType' => 1, // 縮略圖生成方式 1 按設置大小截取 0 按原圖等比例縮略 '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=[]) { 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']), ['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']), ['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; } if(1 == $this->thumbType){ Image::thumb2($filename,$thumbPath.$thumbname.'.'.$thumbExt,'',$thumbWidth[$i],$thumbHeight[$i],true); }else{ 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 = []; $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 = []; 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 = []; 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 = []; $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; } }
composer install
,這時會生成vendor
目錄,及其餘文件yzm@Alert MINGW64 /i/phpstudy/WWW/uploadfile $ composer install
uploadfile/test/UpploadFileTest.php
、uploadfile/test/UpploadFile.html
<?php require_once '../vendor/autoload.php'; use Guanguans\UploadFile; $upload = new UploadFile(); $upload->maxSize = 1*1024*1024; // 默認爲-1,不限制上傳大小 $upload->savePath = './upload/'; // 上傳根目錄 $upload->saveRule = 'uniqid'; // 上傳文件的文件名保存規則 $upload->uploadReplace = true; // 若是存在同名文件是否進行覆蓋 $upload->autoSub = true; // 上傳子目錄開啓 $upload->subType = 'date'; // 上傳子目錄命名規則 $upload->allowExts = ['jpg', 'png']; // 容許類型 if ($upload->upload()) { var_dump($upload->getUploadFileInfo()); } else { var_dump($upload->getErrorMsg()); }
<!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8"> <title>uploadfile test</title> <link href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet"> </head> <body> <form action="UpploadfileTest.php" method="post" enctype="multipart/form-data"> <div class="form-group"> <label>單文件上傳</label> <input type="file" name="uploadfile"> </div> <button type="submit" class="btn btn-primary">Submit</button> </form> <hr> <form action="UpploadfileTest.php" method="post" enctype="multipart/form-data"> <div class="form-group"> <label>多文件上傳</label> <input type="file" name="uploadfile[]"> <input type="file" name="uploadfile[]"> </div> <button type="submit" class="btn btn-primary">Submit</button> </form> </body> </html>
4.2.3 本地瀏覽器訪問uploadfile/test/UpploadFile.html
進行測試html
README.md
、LICENSE
、.gitignore
等文件,項目最終結構以下:個人包GitHub地址 ├─uploadfile 擴展包根目錄 │ ├─src 擴展包代碼目錄 │ │ ├─UploadFile.php │ ├─test 測試目錄 │ │ ├─uploadfile.html │ │ ├─UpploadfileTest.php │ ├─.gitignore │ ├─composer.json │ ├─LICENSE │ └─README.md
git add . git commit -m 'init' git tag v1.0.0 // 記住打一個版本號 git push origin master git push v1.0.0
https://github.com/guanguans/uploadfile
檢測正常的話,會出現 Submit 按鈕,再點擊一下 Submit 按鈕,咱們的包就提交到 Packagist 上了laravel
上面提交上的包提交的包,當咱們更新 GitHub 倉庫時,Packagist 上面的的包並不會自動更新,如今咱們來設置一下自動更新git
setting
,選擇 Integrations & services
,添加 packagist service
,點擊 Test service
移步 Packagist 包主頁,發現已經沒有了紅色的圈住的提示,說明設置自動更新成功。github
我以 Laravel 中使用舉例thinkphp
composer create-project laravel/laravel cd laravel composer require guanguans/uploadfile
本文爲琯琯原創文章,轉載無需和我聯繫,但請註明來自琯琯博客 https://guanguans.cnjson