分布式文件存储库MinIO可还行?

在传统的单体应用架构中,一个应用程序对应一台服务器,提供单进程服务。

但是随着业务的升级,技术的更新迭代,分布式、集群架构、微服务等现已俨然成为主流。

几乎所有的项目都会与文件挂钩,例如OA系统的报表文件,电商系统的商品图片等等...

我们来看一下传统的(单体应用)文件存储与读取方式

分布式文件存储库MinIO可还行?_第1张图片

 一台主机对于N个客户端,如果是小项目还好,稍微大点的项目,服务器分分钟崩溃

进而演进为分布式架构

分布式文件存储库MinIO可还行?_第2张图片

 将每一个大的模块进行拆分,以前单进程支撑的系统现在多进程协同。将文件操作也进行剥离,部署到不同的服务器进行维护,各司其职,减轻不同模块服务器的压力.

话又说回来了,怎样实现呢?

这里的话我主要说一下分布式文件的存储服务MinIO。当然还有其他的一些中间件及工具,大家随意即可。

首先介绍一下MinIO,MinIO是一款高性能的对象资源存储库,而且自身很轻量。

GitHub上start数已经有20多K了

官网:https://min.io
GitHub:https://github.com/minio

分布式文件存储库MinIO可还行?_第3张图片

 而且MinIO支持多种语言,也提供了丰富的API.

OK!现在要动手了。

用它肯定要先安装它,我本次操作在Linux下。

首先在Docker中拉取一个镜像并运行

docker pull minio/minio
docker run -p 9000:9000 minio/minio server /data

它会分配给你密钥,用作登录。这个密钥在后续项目中也会用到。

分布式文件存储库MinIO可还行?_第4张图片

 在浏览器中输入ip:port如果正常显示,就证明你安装成功了。如下:

分布式文件存储库MinIO可还行?_第5张图片

 我们可以点击加号,创建一个桶(文件夹)

分布式文件存储库MinIO可还行?_第6张图片

 注意文件夹的名称不能大写

 我们可以上传一张图片

分布式文件存储库MinIO可还行?_第7张图片

 找到它的链接就可以在浏览器中访问了

分布式文件存储库MinIO可还行?_第8张图片

 接下来在代码中进行CRUD了

 先在项目中引入Minio包

PM> Install-Package Minio

我的是webapi的项目,你可随意

上传文件:

        /// 
        /// 上传文件
        /// 
        /// 
        /// 
        [HttpPost]
        public async Task Upload(IFormFile file)
        {
            try
            {
                var minio = new MinioClient(_configuration["Minio:endpoint"], _configuration["Minio:accessKey"], _configuration["Minio:secretKey"]);
                var data = await FileUpload.Run(minio, file);
                return new JsonResult(new
                {
                    success = true,
                    fileUrl = data.fileUrl,
                    message = data.message
                });
            }
            catch (Exception ex)
            {
                return new JsonResult(new
                {
                    success = false,
                    fileUrl = default(string),
                    message = ex.Message
                });
            }
        }
        public async static Task Run(MinioClient minio, IFormFile formFile)
        {
            var bucketName = "picfile";
            var location = "us-east-1";
            var objectName = $"pic/{DateTime.Now.ToString("yyyy-MM-dd")}/" + Guid.NewGuid().ToString().Replace("-", string.Empty);
            var contentType = "image/jpeg";
            try
            {
                //判断桶(文件夹)是否存在
                bool found = await minio.BucketExistsAsync(bucketName);
                if (!found)
                {
                    //新增桶(文件夹)
                    await minio.MakeBucketAsync(bucketName, location);
                }
                //文件大小
                var len = formFile.Length;
                //打开请求流以读取上传的文件
                var stream = formFile.OpenReadStream();
                //上传文件到桶(文件夹).
                await minio.PutObjectAsync(bucketName, objectName, stream, len, contentType, null, null);
                //返回url
                var url = await minio.PresignedGetObjectAsync(bucketName, objectName, 3600 * 24 * 7);
                //var data = await minio.StatObjectAsync("picfile", objectName);
                //... 对数据库进行操作,例如存入文件名与桶(文件夹)的名称
                return new FileData { success = true, fileUrl = url, message = "上传成功" };
            }
            catch (MinioException ex)
            {
                return new FileData { success = false, fileUrl = default(string), message = ex.Message };
            }
        }

密钥和ip这些配置项我们可以写在appsettings.json中

分布式文件存储库MinIO可还行?_第9张图片

 在控制器中已注入的形式使用

分布式文件存储库MinIO可还行?_第10张图片 下载文件:

        /// 
        /// 下载文件
        /// 
        /// 桶(文件夹)名称
        /// 文件名称
        /// 
        [HttpGet]
        public async Task DownLoadFile(string bucketName, string objextName)
        {
            try
            {
                if (string.IsNullOrEmpty(bucketName) && string.IsNullOrEmpty(objextName)) return new JsonResult(new { message = "桶(文件夹)名称和文件名称不能为空!" });
                //实例化minio客户端
                var minio = new MinioClient(_configuration["Minio:endpoint"], _configuration["Minio:accessKey"], _configuration["Minio:secretKey"]);
                //获得图片的链接
                var url = await minio.PresignedGetObjectAsync("picfile", objextName, 3600 * 24 * 7);
                //实例化WebClient 对象
                var webClient = new WebClient();
                //根据图片链接转成byte字节数组
                var dataByte = webClient.DownloadData(url);
Stream stream = new MemoryStream(dataByte); return File(stream, "application/vnd.android.package-archive", $"图片下载{DateTime.Now.ToString("yyyyMMddHHmmss")}.jpg");
} catch (Exception ex) { return new JsonResult(new { message = ex.Message, success = false }); } }

移除文件:

        /// 
        /// 移除文件
        /// 
        /// 
        /// 
        /// 
        [HttpDelete]
        public async Task DeleteFile(string bucketName, string objextName)
        {
            try
            {
                if (string.IsNullOrEmpty(bucketName) && string.IsNullOrEmpty(objextName)) return new JsonResult(new { message = "桶(文件夹)名称和文件名称不能为空!" });
                //实例化minio客户端
                var minio = new MinioClient(_configuration["Minio:endpoint"], _configuration["Minio:accessKey"], _configuration["Minio:secretKey"]);
                //移除文件
                await minio.RemoveObjectAsync(bucketName, objextName);
                return new JsonResult(new
                {
                    message = "移除资源文件成功",
                    success = true
                });
            }
            catch (Exception ex)
            {
                return new JsonResult(new
                {
                    message = ex.Message,
                    success = false
                });
            }
        }

这里的话只列举以上几个较常用的api,MinIO其实还提供了很多api,大家可以慢慢研究。

分布式文件存储库MinIO可还行?_第11张图片到此MinIO就介绍完了,大家可以根据自己需要再进行扩展。

中文官网:http://docs.minio.org.cn

分布式文件存储库MinIO可还行?_第12张图片

你可能感兴趣的:(分布式文件存储库MinIO可还行?)