多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
@[TOC](Minino安装) # 本文主要讲Minio的安装和使用 在我们的项目通常要使用对象存储,我们可选的对对象存储有很多,例如阿里的oss、七牛云存储.....但是这些或多或少都是收费的,今天我们就分享一个免费的对象存储Minio。 ## Minio MinIO 是一个基于Apache License v2.0开源协议的对象存储服务。它兼容亚马逊S3云存储服务接口,非常适合于存储大容量非结构化的数据,例如图片、视频、日志文件、备份数据和容器/虚拟机镜像等,而一个对象文件可以是任意大小,从几kb到最大5T不等。 MinIO是一个非常轻量的服务,可以很简单的和其他应用的结合,类似 NodeJS, Redis 或者 MySQL。 ## 下载安装 本文主要讲的是Linux的安装和JAVA的使用 ```bash wget https://dl.minio.io/server/minio/release/linux-amd64/minio ``` ## 修改文件权限 ```bash chmod +x minio ``` ## 运行查看appkey secretkey ```bash ./minio server /data ``` 出现如下图则表示启动成功,使用ip:9000访问可以出现图形化界面则表示启动成功。 ![minio启动](https://img-blog.csdnimg.cn/20201112135848395.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dhbmdkaW5nZmVuZzUxNDE=,size_16,color_FFFFFF,t_70#pic_center) ## 修改用户名、密码、文件夹地址 ```bash export MINIO_ACCESS_KEY=admin export MINIO_SECRET_KEY=123456 export MINIO_VOLUMES="/app/file/minio" ``` ## 后台运行 ```bash nohup /home/minio server /home/data > /home/log/minio.log 2>&1 & ``` ## Minio使用 ### 添加自定义配置Minio服务配置项 ```java @Getter @Setter @Component @ConfigurationProperties(prefix = GlobalsConstants.MINIO_PREFIX) public class MinioConfig { @ApiModelProperty("endPoint是一个URL,域名,IPv4或者IPv6地址") private String endpoint; @ApiModelProperty("TCP/IP端口号") private int port; @ApiModelProperty("accessKey类似于用户ID,用于唯一标识你的账户") private String accessKey; @ApiModelProperty("secretKey是你账户的密码") private String secretKey; @ApiModelProperty("如果是true,则用的是https而不是http,默认值是true") private Boolean secure; @ApiModelProperty("默认存储桶") private String bucketName; @Bean @SneakyThrows public MinioClient getMinioClient() { MinioClient minioClient = new MinioClient(endpoint, port, accessKey, secretKey,secure); return minioClient; } } ``` ### 添加Minio 工具类 方便直接调用 ```java import com.spark.platform.common.base.utils.SpringContextHolder; import io.minio.CopyObjectArgs; import io.minio.CopySource; import io.minio.MinioClient; import io.minio.ObjectStat; import io.minio.PutObjectOptions; import io.minio.Result; import io.minio.errors.ErrorResponseException; import io.minio.errors.InvalidExpiresRangeException; import io.minio.messages.Bucket; import io.minio.messages.DeleteError; import io.minio.messages.Item; import lombok.SneakyThrows; import org.springframework.web.multipart.MultipartFile; import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletResponse; import java.io.InputStream; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.List; /** * @ProjectName: spark-platform * @Package: com.spark.platform.file.biz.utils * @ClassName: MinioUtil * @Author: wangdingfeng * @Description: Minio工具类 * @Date: 2020/11/6 13:19 * @Version: 1.0 */ public class MinioUtil { /** * 静态内部类,延迟加载,懒汉式,线程安全的单例模式 */ private static final class Static { private static MinioClient minioClient = SpringContextHolder.getBean(MinioClient.class); } /** * 失效时间 7天 */ public static final int DEFAULT_EXPIRY_TIME = 7 * 24 * 3600; /** * 检查存储桶是否存在 * * @param bucketName 存储桶名称 * @return */ @SneakyThrows public static boolean bucketExists(String bucketName) { boolean flag = false; flag = Static.minioClient.bucketExists(bucketName); if (flag) { return true; } return false; } /** * 创建存储桶 * * @param bucketName 存储桶名称 */ @SneakyThrows public static boolean makeBucket(String bucketName) { boolean flag = bucketExists(bucketName); if (!flag) { Static.minioClient.makeBucket(bucketName); return true; } else { return false; } } /** * 列出所有存储桶名称 * * @return */ @SneakyThrows public static List<String> listBucketNames() { List<Bucket> bucketList = listBuckets(); List<String> bucketListName = new ArrayList<>(); for (Bucket bucket : bucketList) { bucketListName.add(bucket.name()); } return bucketListName; } /** * 列出所有存储桶 * * @return */ @SneakyThrows public static List<Bucket> listBuckets() { return Static.minioClient.listBuckets(); } /** * 删除存储桶 * * @param bucketName 存储桶名称 * @return */ @SneakyThrows public static boolean removeBucket(String bucketName) { boolean flag = bucketExists(bucketName); if (flag) { Iterable<Result<Item>> myObjects = listObjects(bucketName); for (Result<Item> result : myObjects) { Item item = result.get(); // 有对象文件,则删除失败 if (item.size() > 0) { return false; } } // 删除存储桶,注意,只有存储桶为空时才能删除成功。 Static.minioClient.removeBucket(bucketName); flag = bucketExists(bucketName); if (!flag) { return true; } } return false; } /** * 列出存储桶中的所有对象名称 * * @param bucketName 存储桶名称 * @return */ @SneakyThrows public static List<String> listObjectNames(String bucketName) { List<String> listObjectNames = new ArrayList<>(); boolean flag = bucketExists(bucketName); if (flag) { Iterable<Result<Item>> myObjects = listObjects(bucketName); for (Result<Item> result : myObjects) { Item item = result.get(); listObjectNames.add(item.objectName()); } } return listObjectNames; } /** * 列出存储桶中的所有对象 * * @param bucketName 存储桶名称 * @return */ @SneakyThrows public static Iterable<Result<Item>> listObjects(String bucketName) { boolean flag = bucketExists(bucketName); if (flag) { return Static.minioClient.listObjects(bucketName); } return null; } /** * 通过文件上传到对象 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @param fileName File name * @return */ @SneakyThrows public static boolean putObject(String bucketName, String objectName, String fileName) { boolean flag = bucketExists(bucketName); if (flag) { Static.minioClient.putObject(bucketName, objectName, fileName, null); ObjectStat statObject = statObject(bucketName, objectName); if (statObject != null && statObject.length() > 0) { return true; } } return false; } /** * 文件上传 * * @param bucketName * @param multipartFile */ @SneakyThrows public static void putObject(String bucketName, MultipartFile multipartFile, String filename) { PutObjectOptions putObjectOptions = new PutObjectOptions(multipartFile.getSize(), PutObjectOptions.MIN_MULTIPART_SIZE); putObjectOptions.setContentType(multipartFile.getContentType()); Static.minioClient.putObject(bucketName, filename, multipartFile.getInputStream(), putObjectOptions); } /** * 通过InputStream上传对象 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @param stream 要上传的流 * @return */ @SneakyThrows public static boolean putObject(String bucketName, String objectName, InputStream stream) { boolean flag = bucketExists(bucketName); if (flag) { Static.minioClient.putObject(bucketName, objectName, stream, new PutObjectOptions(stream.available(), -1)); ObjectStat statObject = statObject(bucketName, objectName); if (statObject != null && statObject.length() > 0) { return true; } } return false; } /** * 上传文件返回地址 * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @param stream * @return */ @SneakyThrows public static String putObjectUrl(String bucketName, String objectName, InputStream stream){ boolean flag = putObject(bucketName,objectName,stream); if(!flag){ return ""; } return getObjectUrl(bucketName,objectName); } /** * 以流的形式获取一个文件对象 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @return */ @SneakyThrows public static InputStream getObject(String bucketName, String objectName) { boolean flag = bucketExists(bucketName); if (flag) { ObjectStat statObject = statObject(bucketName, objectName); if (statObject != null && statObject.length() > 0) { InputStream stream = Static.minioClient.getObject(bucketName, objectName); return stream; } } return null; } /** * 以流的形式获取一个文件对象(断点下载) * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @param offset 起始字节的位置 * @param length 要读取的长度 (可选,如果无值则代表读到文件结尾) * @return */ @SneakyThrows public static InputStream getObject(String bucketName, String objectName, long offset, Long length) { boolean flag = bucketExists(bucketName); if (flag) { ObjectStat statObject = statObject(bucketName, objectName); if (statObject != null && statObject.length() > 0) { InputStream stream = Static.minioClient.getObject(bucketName, objectName, offset, length); return stream; } } return null; } /** * 下载并将文件保存到本地 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @param fileName File name * @return */ @SneakyThrows public static boolean getObject(String bucketName, String objectName, String fileName) { boolean flag = bucketExists(bucketName); if (flag) { ObjectStat statObject = statObject(bucketName, objectName); if (statObject != null && statObject.length() > 0) { Static.minioClient.getObject(bucketName, objectName, fileName); return true; } } return false; } /** * 删除一个对象 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 */ @SneakyThrows public static boolean removeObject(String bucketName, String objectName) { boolean flag = bucketExists(bucketName); if (flag) { Static.minioClient.removeObject(bucketName, objectName); return true; } return false; } /** * 删除指定桶的多个文件对象,返回删除错误的对象列表,全部删除成功,返回空列表 * * @param bucketName 存储桶名称 * @param objectNames 含有要删除的多个object名称的迭代器对象 * @return */ @SneakyThrows public static List<String> removeObject(String bucketName, List<String> objectNames) { List<String> deleteErrorNames = new ArrayList<>(); boolean flag = bucketExists(bucketName); if (flag) { Iterable<Result<DeleteError>> results = Static.minioClient.removeObjects(bucketName, objectNames); for (Result<DeleteError> result : results) { DeleteError error = result.get(); deleteErrorNames.add(error.objectName()); } } return deleteErrorNames; } /** * 生成一个给HTTP GET请求用的presigned URL。 * 浏览器/移动端的客户端可以用这个URL进行下载,即使其所在的存储桶是私有的。这个presigned URL可以设置一个失效时间,默认值是7天。 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @param expires 失效时间(以秒为单位),默认是7天,不得大于七天 * @return */ @SneakyThrows public static String presignedGetObject(String bucketName, String objectName, Integer expires) { boolean flag = bucketExists(bucketName); String url = ""; if (flag) { if (expires < 1 || expires > DEFAULT_EXPIRY_TIME) { throw new InvalidExpiresRangeException(expires, "expires must be in range of 1 to " + DEFAULT_EXPIRY_TIME); } url = Static.minioClient.presignedGetObject(bucketName, objectName, expires); } return url; } /** * 生成一个给HTTP PUT请求用的presigned URL。 * 浏览器/移动端的客户端可以用这个URL进行上传,即使其所在的存储桶是私有的。这个presigned URL可以设置一个失效时间,默认值是7天。 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @param expires 失效时间(以秒为单位),默认是7天,不得大于七天 * @return */ @SneakyThrows public static String presignedPutObject(String bucketName, String objectName, Integer expires) { boolean flag = bucketExists(bucketName); String url = ""; if (flag) { if (expires < 1 || expires > DEFAULT_EXPIRY_TIME) { throw new InvalidExpiresRangeException(expires, "expires must be in range of 1 to " + DEFAULT_EXPIRY_TIME); } url = Static.minioClient.presignedPutObject(bucketName, objectName, expires); } return url; } /** * 获取对象的元数据 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @return */ @SneakyThrows public static ObjectStat statObject(String bucketName, String objectName) { boolean flag = bucketExists(bucketName); if (flag) { ObjectStat statObject = Static.minioClient.statObject(bucketName, objectName); return statObject; } return null; } @SneakyThrows public static void copy(String bucketName, String objectName, String destBucketName, String destObjectName){ boolean flag = bucketExists(bucketName); boolean destFlag = bucketExists(destBucketName); if (flag && destFlag) { CopySource source = CopySource.builder().bucket(bucketName).object(objectName).build(); Static.minioClient.copyObject(CopyObjectArgs.builder().bucket(destBucketName).object(destObjectName).source(source).build()); } } /** * 文件访问路径 * * @param bucketName 存储桶名称 * @param objectName 存储桶里的对象名称 * @return */ @SneakyThrows public static String getObjectUrl(String bucketName, String objectName) { boolean flag = bucketExists(bucketName); String url = ""; if (flag) { url = Static.minioClient.getObjectUrl(bucketName, objectName); } return url; } /** * 下载文件 * @param bucketName 存储桶名称 * @param fileName 文件名称 * @param originalName 原始名称 * @param response */ public static void downloadFile(String bucketName, String fileName, String originalName, HttpServletResponse response) { try { InputStream file = Static.minioClient.getObject(bucketName, fileName); String filename = new String(originalName.getBytes("ISO8859-1"), StandardCharsets.UTF_8); response.setHeader("Content-Disposition", "attachment;filename=" + filename); ServletOutputStream servletOutputStream = response.getOutputStream(); int len; byte[] buffer = new byte[1024]; while ((len = file.read(buffer)) > 0) { servletOutputStream.write(buffer, 0, len); } servletOutputStream.flush(); file.close(); servletOutputStream.close(); } catch (ErrorResponseException e) { e.printStackTrace(); } catch (Exception e) { e.printStackTrace(); } } } ```