(目录)
对象存储服务MinIO
1 MinIO简介
MinIO
基于Apache License v2.0开源协议的对象存储服务
,可以做为云存储的解决方案用来保存海量的图片,视频,文档。由于采用Golang实现,服务端可以工作在Windows,Linux, OS X和FreeBSD上。配置简单,基本是复制可执行程序,单行命令可以运行起来。
MinIO兼容亚马逊S3云存储服务接口,非常适合于存储大容量非结构化的数据,例如图片、视频、日志文件、备份数据和容器/虚拟机镜像等
,而一个对象文件可以是任意大小,从几kb到最大5T不等。
S3 ( Simple Storage Service简单存储服务)
基本概念:
bucket
– 类比于文件系统的目录
Object
– 类比文件系统的文件
Keys
– 类比文件名
官网文档:http://docs.minio.org.cn/docs/
2 MinIO特点
数据保护
Minio使用Minio Erasure Code(纠删码)来防止硬件故障。即便损坏一半以上的driver,但是仍然可以从中恢复。
高性能
作为高性能对象存储,在标准硬件条件下它能达到55GB/s的读、35GB/s的写速率
可扩容
不同MinIO集群可以组成联邦,并形成一个全局的命名空间,并跨越多个数据中心
SDK支持
基于Minio轻量的特点,它得到类似Java、Python或Go等语言的sdk支持
有操作页面
面向用户友好的简单操作界面,非常方便的管理Bucket及里面的文件资源
功能简单
这一设计原则让MinIO不容易出错、更快启动
丰富的API
支持文件资源的分享连接及分享链接的过期策略、存储桶操作、文件列表访问及文件上传下载的基本功能等。
文件变化主动通知
存储桶(Bucket)如果发生改变,比如上传对象和删除对象,可以使用存储桶事件通知机制进行监控,并通过以下方式发布出去:AMQP、MQTT、Elasticsearch、Redis、NATS、MySQL、Kafka、Webhooks等。
3 开箱使用
3.1 安装启动
拉取minio镜像
docker pull minio/minio:RELEASE.2021-06-14T01-29-23Z
使用docker进行环境部署和启动
docker run -p 9090:9000 --name minio -d --restart=always -e "MINIO_ACCESS_KEY=minio" -e "MINIO_SECRET_KEY=minio123" -v /home/data:/data -v /home/config:/root/.minio minio/minio:RELEASE.2021-06-14T01-29-23Z server /data
3.2 管理控制台
假设我们的服务器地址为http://192.168.200.130:9090
我们在地址栏输入:http://http://192.168.200.130:9090/ 即可进入登录界面。
Access
Key为minio
Secret_key
为minio123
进入系统后可以看到主界面
点击右下角的“+”号 ,点击下面的图标,创建一个桶
开启 读写权限
4 封装MinIO为starter
4.1 编写file-spring-boot-starter模块
新增依赖
<dependency>
<groupId>io.minio</groupId>
<artifactId>minio</artifactId>
<version>7.1.0</version>
</dependency>
4.2 编写配置类
MinIOConfig
import io.minio.MinioClient;
import lombok.Data;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Data
@Configuration
@EnableConfigurationProperties({MinIOConfigProperties.class})
//当引入FileStorageService接口时
@ConditionalOnClass(FileStorageService.class)
public class MinIOConfig {
@Autowired
private MinIOConfigProperties minIOConfigProperties;
@Bean
public MinioClient buildMinioClient() {
return MinioClient
.builder()
.credentials(minIOConfigProperties.getAccessKey(),
minIOConfigProperties.getSecretKey())
.endpoint(minIOConfigProperties.getEndpoint())
.build();
}
}
MinIOConfigProperties
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import java.io.Serializable;
@Data
@ConfigurationProperties(prefix = "file.minio") // 文件上传 配置前缀file.minio
public class MinIOConfigProperties implements Serializable {
private String accessKey;
private String secretKey;
private String bucket;
private String endpoint;
private String readPath;
}
4.3 编写 Service
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
/**
*
* @Description 文件上传接口
*/
public interface FileStorageService {
/**
* @Description 文件上传
* @param prefix 文件上传前缀
* @param filename 文件名称
* @param inputStream 文件流
* @return pathUrl 全路径
*/
String store(String prefix, String filename, InputStream inputStream);
/**
* @Description 文件上传
* @param prefix 文件上传前缀
* @param filename 文件名称
* @param contentType 文件类型 "image/jpg" 或"text/html"
* @param inputStream 文件流
* @return pathUrl 全路径
*/
String store(String prefix, String filename,String contentType, InputStream inputStream);
/**
* @Description 文件删除
* @param pathUrl 全路径
* @throws Exception
*/
void delete(String pathUrl);
/**
* @Description 批量文件删除
* @param pathUrls 全路径集合
* @throws Exception
*/
void deleteBatch(List<String> pathUrls);
/**
* @Description 下载文件
* @param pathUrl 全路径
* @return
*/
InputStream downloadFile(String pathUrl);
/**
* @Description 获取文件文本内容
* @param pathUrl 全路径
* @return
* @throws IOException
*/
String getFileContent(String pathUrl) throws IOException;
}
4.4 编写实现类
import com.heima.file.config.MinIOConfig;
import com.heima.file.config.MinIOConfigProperties;
import com.heima.file.service.FileStorageService;
import io.minio.GetObjectArgs;
import io.minio.MinioClient;
import io.minio.PutObjectArgs;
import io.minio.RemoveObjectArgs;
import io.minio.RemoveObjectsArgs;
import io.minio.messages.DeleteObject;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Import;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
import java.io.IOException;
import java.io.InputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.LinkedList;
import java.util.List;
@Slf4j
@EnableConfigurationProperties(MinIOConfigProperties.class)
@Import(MinIOConfig.class)
@Component("minIOFileStorageService")
// 创建bean对象,bean name=minIOFileStorageService
public class MinIOFileStorageService implements FileStorageService {
@Autowired
private MinioClient minioClient;
@Autowired
private MinIOConfigProperties minIOConfigProperties;
private final static String separator = "/";
/**
* @param dirPath
* @param filename yyyy/mm/dd/file.jpg
* @return
*/
public String builderFilePath(String dirPath,String filename) {
StringBuilder stringBuilder = new StringBuilder(50);
if(!StringUtils.isEmpty(dirPath)){
stringBuilder.append(dirPath).append(separator);
}
SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd");
String todayStr = sdf.format(new Date());
stringBuilder.append(todayStr).append(separator);
stringBuilder.append(filename);
return stringBuilder.toString();
}
/**
* 上传图片文件
* @param prefix 文件前缀
* @param filename 文件名
* @param contentType 文件类型 图片:"image/jpg" html:"text/html"
* @param inputStream 文件流
* @return 文件全路径
*/
@Override
public String store(String prefix, String filename, String contentType, InputStream inputStream) {
String filePath = builderFilePath(prefix, filename);
try {
PutObjectArgs putObjectArgs = PutObjectArgs.builder()
.object(filePath)
.contentType(contentType)
.bucket(minIOConfigProperties.getBucket()).stream(inputStream,inputStream.available(),-1)
.build();
minioClient.putObject(putObjectArgs);
StringBuilder urlPath = new StringBuilder();
urlPath.append(minIOConfigProperties.getBucket());
urlPath.append(separator);
urlPath.append(filePath);
return urlPath.toString();
}catch (Exception ex){
log.error("minio put file error.",ex);
throw new RuntimeException("上传文件失败");
}
}
/**
* @param prefix 文件上传前缀
* @param filename 文件名称
* @param inputStream 文件流
* @return pathUrl 全路径
* @Description 文件上传
*/
@Override
public String store(String prefix, String filename, InputStream inputStream) {
return this.store(prefix,filename,"image/jpg",inputStream);
}
/**
* 上传html文件
* @param prefix 文件前缀
* @param filename 文件名
* @param inputStream 文件流
* @return 文件全路径
*/
public String uploadHtmlFile(String prefix, String filename,InputStream inputStream) {
String filePath = builderFilePath(prefix, filename);
try {
PutObjectArgs putObjectArgs = PutObjectArgs.builder()
.object(filePath)
.contentType("text/html")
.bucket(minIOConfigProperties.getBucket()).stream(inputStream,inputStream.available(),-1)
.build();
minioClient.putObject(putObjectArgs);
StringBuilder urlPath = new StringBuilder(minIOConfigProperties.getReadPath());
urlPath.append(minIOConfigProperties.getBucket());
urlPath.append(separator);
urlPath.append(filePath);
return urlPath.toString();
}catch (Exception ex){
log.error("minio put file error.",ex);
ex.printStackTrace();
throw new RuntimeException("上传文件失败");
}
}
/**
* 删除文件
* @param pathUrl 文件全路径
*/
@Override
public void delete(String pathUrl) {
String key = pathUrl.replace(minIOConfigProperties.getEndpoint(),"");
int index = key.indexOf(separator);
String bucket = key.substring(0,index);
String filePath = key.substring(index+1);
// 删除Objects
RemoveObjectArgs removeObjectArgs = RemoveObjectArgs.builder()
.bucket(bucket).object(filePath).build();
try {
minioClient.removeObject(removeObjectArgs);
} catch (Exception e) {
log.error("minio remove file error. pathUrl:{}",pathUrl);
e.printStackTrace();
}
}
/**
* @param pathUrls 全路径集合
* @throws Exception
* @Description 批量文件删除
*/
@Override
public void deleteBatch(List<String> pathUrls) {
List<DeleteObject> objects = new LinkedList<>();
for (String pathUrl : pathUrls) {
String key = pathUrl.replace(minIOConfigProperties.getEndpoint(),"");
int index = key.indexOf(separator);
// String bucket = key.substring(0,index);
String filePath = key.substring(index+1);
objects.add( new DeleteObject(filePath));
}
try {
minioClient.removeObjects(
RemoveObjectsArgs.builder()
.bucket(minIOConfigProperties.getBucket())
.objects(objects).build());
} catch (Exception e) {
log.error("minio remove file error.");
e.printStackTrace();
}
}
/**
* 下载文件
* @param pathUrl 文件全路径
* @return 文件流
*
*/
@Override
public InputStream downloadFile(String pathUrl) {
String key = pathUrl.replace(minIOConfigProperties.getEndpoint(),"");
int index = key.indexOf(separator);
String bucket = key.substring(0,index);
String filePath = key.substring(index+1);
InputStream inputStream = null;
try {
inputStream = minioClient.getObject(GetObjectArgs.builder().bucket(minIOConfigProperties.getBucket()).object(filePath).build());
} catch (Exception e) {
log.error("minio down file error. pathUrl:{}",pathUrl);
e.printStackTrace();
}
return inputStream;
}
/**
* @param pathUrl 全路径
* @return
* @throws IOException
* @Description 获取文件文本内容
*/
@Override
public String getFileContent(String pathUrl) throws IOException {
return null;
}
}
4.5 加入自动配置
在resources中新建META-INF/spring.factories
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.heima.file.service.impl.OSSAliyunFileStorageService,com.heima.file.service.impl.MinIOFileStorageService
4.6 其他微服务使用
第一,导入file-spring-boot-starter的依赖
第二,在配置中心 share-file.yml
中添加minio所需要的配置
#OSS配置
file:
oss:
bucket-name: <!-- 存储空间 -->
access-key-id: <!-- OSS密钥key -->
access-key-secret: <!-- OSS密钥 -->
endpoint: oss-cn-shanghai.aliyuncs.com
web-site: <!-- OSS访问前缀 -->
proxy-username: aliyun-sdk-java
socket-timeout: 10000
idle-connection-time: 10000
connection-request-timeout: 4000
max-connections: 2048
max-error-retry: 5
white-list: 127.0.0.1
connection-timeout: 10000
prefix: material
# minIO配置
minio:
accessKey: minio
secretKey: minio123
bucket: <!-- minIO中 bucket -->
endpoint: http://${spring.profiles.ip}:9090/
readPath: http://${spring.profiles.ip}:9090/
prefix: article
第三,在对应使用的业务类中注入FileStorageService
,样例如下:
package com.heima.wemedia;
import com.heima.file.service.FileStorageService;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import javax.annotation.Resource;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
@SpringBootTest
@RunWith(SpringRunner.class)
public class MinIoTest {
// 指定MinIo实现
@Resource(name = "minIOFileStorageService")
FileStorageService fileStorageService;
// 不指定 beanName 注入的是OSS的实现
@Autowired
FileStorageService fileStorageService2;
@Test
public void uploadToMinIo() throws FileNotFoundException {
System.out.println(fileStorageService);
System.out.println(fileStorageService2);
// 准备好一个静态页
FileInputStream fileInputStream = new FileInputStream("D://list.html");
// 将静态页上传到minIO文件服务器中 文件名称 文件类型 文件流
fileStorageService.store("test","list.html","text/html",fileInputStream);
}
}
上传成功并成功预览:
永久有效链接配置
1.Docker 获取 minio/mc容器
docker pull minio/mc:RELEASE.2021-06-13T17-48-22Z
2.Docker 启动 minio/mc容器
docker run -it --entrypoint=/bin/sh minio/mc:RELEASE.2021-06-13T17-48-22Z
启动之后 会直接 进入 sh-4.4#
输入mc version 查看版本信息
3.minio/mc 绑定 minio server服务
mc config host add <ALIAS> <YOUR-S3-ENDPOINT> <YOUR-ACCESS-KEY> <YOUR-SECRET-KEY> [--api API-SIGNATURE]
mc config host add minio http://IP地址:9000 admin adminminio --api S3v4
解释:
- ALIAS: 别名就是给你的云存储服务起了一个短点的外号。
- S3 endpoint,access key和secret key是你的云存储服务提供的。
- endpoint http://ip:9000
- access key、secret key 到这里大家肯定都知道啦。
- API签名是可选参数,默认情况下,它被设置为"S3v4"。
比如:
# 从MinIO服务获得URL、access key和secret key。
mc config host add minio http://192.168.200.130:9000 minio minio123 --api S3v4
绑定成功
4.mc命令
查看所用的桶
mc ls minio
5.设置桶或者目录的访问权限为public(设置公开访问 永久访问链接)
//将minio中,test-bucket存储桶下的public目录设置为公开可访问
mc policy set public minio/test-bucket/public
//将minio中,test-bucket2存储桶设置为公开可访问
mc policy set public minio/test-bucket2
mc policy set public minio/test
设置成功
http://IP地址:9000/test/demo.png