阿里云OSS图片上传类

本文涉及的产品
对象存储 OSS,20GB 3个月
对象存储 OSS,恶意文件检测 1000次 1年
对象存储 OSS,内容安全 1000次 1年
简介:

1.阿里云基本函数

/**
 * 把本地变量的内容到文件
 * 简单上传,上传指定变量的内存值作为object的内容
 */
public function putObject($imgPath,$object)
{   
    $content = file_get_contents($imgPath); // 把当前文件的内容获取到传入文件中
    $options = array();
    try {
        $this->ossClient->putObject($this->bucket, $object, $content, $options);
    } catch (OssException $e) {
        return $e->getMessage();
    }
    return TRUE;
}

/**
 * 上传指定的本地文件内容
 */
public function uploadFile($imgPath,$object) //$_FILES['img']['tmp_name']
{
    $filePath  = $imgPath;
    $options = array();
    try {
        $this->ossClient->uploadFile($this->bucket, $object, $filePath, $options);
    } catch (OssException $e) {
        return $e->getMessage();
    }
    return TRUE;
}

// 删除对象
public function deleteObject($object) {
    try {
        $this->ossClient->deleteObject($this->bucket, $object);
    } catch (OssException $e) {
        return $e->getMessage();
    }
    return TRUE;
}

// 判断对象是否存在
public function doesObjectExist($object) {
    try {
        $result = $this->ossClient->doesObjectExist($this->bucket, $object);
    } catch (OssException $e) {
        return $e->getMessage();
    }
    return $result;
}

// 批量删除对象
public function deleteObjects($objects) {
    try {
        $this->ossClient->deleteObjects($this->bucket, $objects);
    } catch (OssException $e) {
        return $e->getMessage();
    }
    return TRUE;
}

/**
 * 获取object的内容
 *
 * @param OssClient $ossClient OssClient实例
 * @param string $bucket 存储空间名称
 * @return null
 */
public function getObject($object)
{
    $options = array();
    try {
        $content = $this->ossClient->getObject($this->bucket, $object, $options);
    } catch (OssException $e) {
        return $e->getMessage();
    }
    // file_get_contents
    return $content;
}

2.基本配置与辅助函数

public $ossClient,$bucket;
private $configinfo =   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->configinfo[$name])) {
        return $this->configinfo[$name];
    }
    return null;
}

public function __set($name,$value){
    if(isset($this->configinfo[$name])) {
        $this->configinfo[$name]    =   $value;
    }
}

public function __isset($name){
    return isset($this->configinfo[$name]);
}

/**
 * 架构函数
 * @access public
 * @param array $config  上传参数
 */
public function __construct($config=array()) {
    if(is_array($config)) {
        $this->config   =   array_merge($this->config,$config);
    }

    $this->bucket = C('OSS_TEST_BUCKET');
    $this->ossClient = new OssClient(C('OSS_ACCESS_ID'), C('OSS_ACCESS_KEY'), C('OSS_ENDPOINT'), false);
}

3.主函数

/**
 * 上传所有文件
 * @access public
 * @param string $savePath  上传文件保存路径
 * @return string
 */
public function upload($savePath ='') {
    //如果不指定保存文件名,则由系统默认
    if(empty($savePath)) {
        $savePath = $this->savePath;
    }

    $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;
    }
}

4.核心处理函数

/**
 * 上传一个文件
 * @access public
 * @param mixed $name 数据
 * @param string $value  数据表名
 * @return string
 */
private function save($file) {
    $filename = $file['savepath'].$file['savename'];
    if(!$this->uploadReplace && $this->doesObjectExist($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(!$this->putObject($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(C('OSS_IMG_URL').'/'.$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;
                }
                $this->thumb(C('OSS_IMG_URL').'/'.$filename,$thumbPath.$thumbname.'.'.$thumbExt,'',$thumbWidth[$i],$thumbHeight[$i],true);                    
            }
            if($this->thumbRemoveOrigin) {
                // 生成缩略图之后删除原图
                $this->deleteObject($filename);
            }
        }
    }
    if($this->zipImags) {
        // TODO 对图片压缩包在线解压

    }
    return true;
}


/**
 * 生成缩略图
 * @static
 * @access public
 * @param string $image  原图
 * @param string $type 图像格式
 * @param string $thumbname 缩略图文件名
 * @param string $maxWidth  宽度
 * @param string $maxHeight  高度
 * @param string $position 缩略图保存目录
 * @param boolean $interlace 启用隔行扫描
 * @return void
 */
public function thumb($image, $thumbname, $type='', $maxWidth=200, $maxHeight=50, $interlace=true) {
    // 获取原图信息
    $info = Image::getImageInfo($image);
    if ($info !== false) {
        $srcWidth = $info['width'];
        $srcHeight = $info['height'];
        $type = empty($type) ? $info['type'] : $type;
        $type = strtolower($type);
        $interlace = $interlace ? 1 : 0;
        unset($info);
        $scale = min($maxWidth / $srcWidth, $maxHeight / $srcHeight); // 计算缩放比例
        if ($scale >= 1) {
            // 超过原图大小不再缩略
            $width = $srcWidth;
            $height = $srcHeight;
        } else {
            // 缩略图尺寸
            $width = (int) ($srcWidth * $scale);
            $height = (int) ($srcHeight * $scale);
        }

        // 载入原图
        $createFun = 'ImageCreateFrom' . ($type == 'jpg' ? 'jpeg' : $type);
        if(!function_exists($createFun)) {
            return false;
        }
        $srcImg = $createFun($image);


        //创建缩略图
        if ($type != 'gif' && function_exists('imagecreatetruecolor'))
            $thumbImg = imagecreatetruecolor($width, $height);
        else
            $thumbImg = imagecreate($width, $height);
          //png和gif的透明处理 by luofei614
        if('png'==$type){
            imagealphablending($thumbImg, false);//取消默认的混色模式(为解决阴影为绿色的问题)
            imagesavealpha($thumbImg,true);//设定保存完整的 alpha 通道信息(为解决阴影为绿色的问题)    
        }elseif('gif'==$type){
            $trnprt_indx = imagecolortransparent($srcImg);
             if ($trnprt_indx >= 0) {
                    //its transparent
                   $trnprt_color = imagecolorsforindex($srcImg , $trnprt_indx);
                   $trnprt_indx = imagecolorallocate($thumbImg, $trnprt_color['red'], $trnprt_color['green'], $trnprt_color['blue']);
                   imagefill($thumbImg, 0, 0, $trnprt_indx);
                   imagecolortransparent($thumbImg, $trnprt_indx);
          }
        }
        // 复制图片
        if (function_exists("ImageCopyResampled"))
            imagecopyresampled($thumbImg, $srcImg, 0, 0, 0, 0, $width, $height, $srcWidth, $srcHeight);
        else
            imagecopyresized($thumbImg, $srcImg, 0, 0, 0, 0, $width, $height, $srcWidth, $srcHeight);

        // 对jpeg图形设置隔行扫描
        if ('jpg' == $type || 'jpeg' == $type)
            imageinterlace($thumbImg, $interlace);

        imagePNG($thumbImg,'Uploads/file.png'); // 中转站
        // 生成图片
        $this->putObject('Uploads/file.png',$thumbname);
        imagedestroy($thumbImg);
        imagedestroy($srcImg);
        return $thumbname;
    }
    return false;
}

5.辅助函数

    /**
     * 转换上传文件数组变量为正确的方式
     * @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 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;
    }

总结:与普通上传的区别在于,它是全部通过阿里云的oss接口来处理文件保存的。普通上传是把本地文件移动到服务器上,而它则是把文件移动到阿里云服务器上。
缩略图思路,
a.上传图片至服务器
b.获取图片进行处理
c.上传处理好的图片至服务器
d.根据配置,删除或者不删除服务器的原图(OSS)

'Uploads/file.png'// 中转站// 生成图片'Uploads/file.png''Uploads/file.png'
相关实践学习
借助OSS搭建在线教育视频课程分享网站
本教程介绍如何基于云服务器ECS和对象存储OSS,搭建一个在线教育视频课程分享网站。
相关文章
|
8天前
|
分布式计算 大数据 MaxCompute
MaxCompute产品使用合集之使用pyodps读取OSS(阿里云对象存储)中的文件的步骤是什么
MaxCompute作为一款全面的大数据处理平台,广泛应用于各类大数据分析、数据挖掘、BI及机器学习场景。掌握其核心功能、熟练操作流程、遵循最佳实践,可以帮助用户高效、安全地管理和利用海量数据。以下是一个关于MaxCompute产品使用的合集,涵盖了其核心功能、应用场景、操作流程以及最佳实践等内容。
|
8天前
|
Java Serverless 应用服务中间件
Serverless 应用引擎操作报错合集之阿里云函数计算中出现“找不到类”的错误如何解决
Serverless 应用引擎(SAE)是阿里云提供的Serverless PaaS平台,支持Spring Cloud、Dubbo、HSF等主流微服务框架,简化应用的部署、运维和弹性伸缩。在使用SAE过程中,可能会遇到各种操作报错。以下是一些常见的报错情况及其可能的原因和解决方法。
37 1
Serverless 应用引擎操作报错合集之阿里云函数计算中出现“找不到类”的错误如何解决
|
3天前
|
存储 弹性计算 数据库
阿里云oss备份网站数据的详细步骤
该教程指导如何使用阿里云OSS备份网站数据。首先,注册阿里云账号并购买40GB的OSS存储空间。创建Bucket,选择与服务器相同的区域和私有权限。安装阿里云OSS插件,获取AccessKey信息。在宝塔面板中设置计划任务进行网站或数据库备份,选择内网域名以节省流量。备份完成后,通过文件管理器检查OSS中是否有备份文件。下载备份文件需点击文件名,然后打开文件URL。
|
8天前
|
安全 Linux 对象存储
数据备份到阿里云oss上,以防勒索病毒的破坏
数据备份到阿里云oss上,以防勒索病毒的破坏
24 0
|
8天前
|
运维 Serverless 开发工具
Serverless 应用引擎产品使用之阿里云函数计算中在哪里可以找到函数入口设置的类如何解决
阿里云Serverless 应用引擎(SAE)提供了完整的微服务应用生命周期管理能力,包括应用部署、服务治理、开发运维、资源管理等功能,并通过扩展功能支持多环境管理、API Gateway、事件驱动等高级应用场景,帮助企业快速构建、部署、运维和扩展微服务架构,实现Serverless化的应用部署与运维模式。以下是对SAE产品使用合集的概述,包括应用管理、服务治理、开发运维、资源管理等方面。
24 1
|
8天前
|
网络协议 对象存储
阿里云oss配置自有域名
阿里云oss配置自有域名
33 1
|
8天前
|
存储 JSON 前端开发
Javaweb之SpringBootWeb案例之阿里云OSS服务集成的详细解析
Javaweb之SpringBootWeb案例之阿里云OSS服务集成的详细解析
31 0
|
8天前
|
存储 开发工具 对象存储
Javaweb之SpringBootWeb案例之阿里云OSS服务入门的详细解析
Javaweb之SpringBootWeb案例之阿里云OSS服务入门的详细解析
27 0
|
8天前
|
存储 文字识别 安全
Javaweb之SpringBootWeb案例之阿里云OSS服务的详细解析
Javaweb之SpringBootWeb案例之阿里云OSS服务的详细解析
29 0
|
8天前
|
存储 安全 Java
全面了解阿里云OSS使用方法
本文介绍了阿里云对象存储(OSS)的准备工作、服务开通、优势、使用步骤以及代码实现。首先,用户需要注册阿里云账号并进行实名认证,然后开通OSS服务,创建存储空间。阿里云OSS提供高可靠、安全、低成本的云存储,适合各种数据存储和分发场景。接着,通过官方SDK学习如何上传文件,并提供了Java代码示例。最后,展示了如何在Spring Boot项目中集成阿里云OSS,实现文件上传功能。

热门文章

最新文章