企业🤖AI智能体构建引擎,智能编排和调试,一键部署,支持私有化部署方案 广告
# [开发 Composer 包详细步骤](https://segmentfault.com/a/1190000013947602) * https://segmentfault.com/a/1190000013947602 开发一个 composer 通用文件上传包,发布到 Packagist,并在 Laravel 中测试。 ![](https://segmentfault.com/img/remote/1460000013948273?w=600&h=356) ## 一、GitHub 创建一个名[uploadfile](https://github.com/guanguans/uploadfile)新仓库,并克隆至本地。 ~~~ $ git clone git@github.com:guanguans/uploadfile.git $ cd uploadfile ~~~ ## 二、初始化项目,生成`composer.json`文件 ### 2.1 步骤 ~~~ 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 ~~~ ### 2.2 步骤解释 ~~~ 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`中追加 ~~~ "autoload": { "psr-4": { "Guanguans\\": "src/" } } ~~~ ## 四、构建项目 ### 4.1 新建`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; } } ~~~ ### 4.2 测试 #### 4.2.1 终端下执行`composer install`,这时会生成`vendor`目录,及其他文件 ~~~ yzm@Alert MINGW64 /i/phpstudy/WWW/uploadfile $ composer install ~~~ #### 4.2.2 新建`uploadfile/test/UpploadFileTest.php`、`uploadfile/test/UpploadFile.html` * UpploadFileTest.php ~~~ <?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()); } ~~~ * UpploadFile.html ~~~ <!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`进行测试 ![](https://segmentfault.com/img/remote/1460000013947613?w=918&h=742) ## 五、添加`README.md`、`LICENSE`、`.gitignore`等文件,项目最终结构如下:[我的包GitHub地址](https://github.com/guanguans/uploadfile) ~~~ ├─uploadfile 扩展包根目录 │ ├─src 扩展包代码目录 │ │ ├─UploadFile.php │ ├─test 测试目录 │ │ ├─uploadfile.html │ │ ├─UpploadfileTest.php │ ├─.gitignore │ ├─composer.json │ ├─LICENSE │ └─README.md ~~~ ## 六、推送到 GitHub ~~~ git add . git commit -m 'init' git tag v1.0.0 // 记住打一个版本号 git push origin master git push v1.0.0 ~~~ ## 七、将 GitHub 上的包提交到[Packagist](https://packagist.org/) 1. 首先要在[Packagist](https://packagist.org/)上注册账号并登录(可以用 GitHub 直接登录) 2. 点击顶部导航条中的 Summit 按钮 3. 在输入框中输入 GitHub 上的刚才包地址,如:`https://github.com/guanguans/uploadfile` 4. 然后点击 Check 按钮 Packagist 会去检测此仓库地址的代码是否符合 Composer 的 Package 包的要求 检测正常的话,会出现 Submit 按钮,再点击一下 Submit 按钮,我们的包就提交到 Packagist 上了 ![](https://segmentfault.com/img/remote/1460000013947614) ![](https://segmentfault.com/img/remote/1460000013947615) ## 八、设置 composer 包自动更新 上面提交上的包提交的包,当我们更新 GitHub 仓库时,Packagist 上面的的包并不会自动更新,现在我们来设置一下自动更新 ### 8.1 复制[Profile](https://packagist.org/profile/)API Token ![](https://segmentfault.com/img/remote/1460000013947616) ### 8.2 打开 GitHub 项目`setting`,选择`Integrations & services`,添加`packagist service`,点击`Test service` ![](https://segmentfault.com/img/remote/1460000013947617) ![](https://segmentfault.com/img/remote/1460000013947618) ![](https://segmentfault.com/img/remote/1460000013947619) ![](https://segmentfault.com/img/remote/1460000013947620) ![](https://segmentfault.com/img/remote/1460000013947621) ### 8.3 验证是否已经自动更新 移步 Packagist 包主页,发现已经没有了红色的圈住的提示,说明设置自动更新成功。 ![](https://segmentfault.com/img/remote/1460000013947622) ## 九、项目中使用 我以 Laravel 中使用举例 ~~~ composer create-project laravel/laravel cd laravel composer require guanguans/uploadfile ~~~ ![](https://segmentfault.com/img/remote/1460000013947623) ## 其他 * 本文通用上传类由 ThinkPHP 中[UploadFile.class.php](https://github.com/top-think/thinkphp/blob/3.0/ThinkPHP/Extend/Library/ORG/Net/UploadFile.class.php)修改 * 本文首发**[琯琯博客](https://guanguans.cn/)**,可前往浏览更多文章。