当前位置:   article > 正文

MinIO文件服务器,从安装到使用_minio安装

minio安装

以往的项目,用的比较多的OSS服务是腾讯云和阿里云的存储服务,不过从去年到今年,最近的几个项目,普遍要使用Minio,所以我在开发服务器和测试服务器上都装上了minio

一、首先minio的安装

MInIO的安装有很多方法、单实例的、集群分布式的、docker部署的、支持k8s的,我们使用最简单的一种安装方式:linux单节点安装。

如果希望对MinIO有深入的掌握,访问MinIO官网:https://min.io

1.1 安装工作

首先在/root目录下创建一个minio文件夹

  1. cd /root
  2. mkdir minio
  3. cd minio
  4. wget https://dl.min.io/server/minio/release/linux-amd64/minio

等待下载完成后:

在当前minio目录下,会出现一个minio目录

  1. chmod +x minio
  2. # 创建minio文件存储目录及日志目录
  3. mkdir -p /root/data/minio;
  4. mkdir -p /root/logs/minio;

然后在 /root/minio/目录下,新建一个run.sh并编辑以下内容

vim run.sh,然后将以下内容保存到run.sh,并为其赋予执行权限chmod u+x run.sh

  1. #!/bin/bash
  2. export MINIO_ROOT_USER=root
  3. export MINIO_ROOT_PASSWORD=xiaomifeng1010
  4. # nohup启动服务 指定文件存放路径 /root/data 还有设置日志文件路径 /root/minio/log
  5. nohup ./minio server --address :9002 --console-address :9001 /root/data/minio > /root/logs/minio/minio.log 2>&1 &

注意:以前的老版本minio的配置中,配置用户名和密码时,是这两个参数:

MINIO_ACCESS_KEY 和MINIO_SECRET_KEY

而现在比较新的版本的minio,需要替换成MINIO_ROOT_USER和MINIO_ROOT_PASSWORD

并且用户名和密码都是由长度限制的,用户名长度不能小于3,密码不能小于8个字符

当然,minio安装完成,以及配置完成后,启动项目的run.sh的时候,如果你设置的用户名和密码长度不够,会有警告提示的,还有就是如果你的配置参数写成以前的旧版本的参数,也会提示你替换成对应的MINIO_ROOT_USER和MINIO_ROOT_PASSWORD

--address :9002 --console-address :9001 是配置端口,默认minio端口是9000,如果9000端口被占用了,那就加上这一串配置,端口号的冒号之前不需要特意写出ip,当然如果你的ip的动态变化的,而不是静态的话,前边的ip不用写上,当然最好是用静态的ip

然后启动minio

  1. # 启动minio服务
  2. bash run.sh
  3. # 查看日志
  4. tail -200f /root/logs/minio/minio.log

 ​​

然后会有日志打印信息,然后可以看到minio服务器地址,和控制台信息地址

然后在浏览器中访问地址

http://ip:9002,输入这个地址后会重定向到控制台登录地址http://ip:9001/login

然后在登录界面输入用户名和密码即可登录

   然后登录后进入首页,创建一个bucket

  创建一个xxx-test,区分开发环境和测试环境,当然我在开发环境服务器上创建的是xxx-dev,由于当前在开发阶段,所以项目中我也配置的是xxx-dev这个桶

 接下来我以开发环境中的桶演示minio文件的上传和下载等

二,springboot项目整合MinIO的javaSDK

2.1 项目的pom文件中引入minio依赖

  1. <properties>
  2. <maven.compiler.source>8</maven.compiler.source>
  3. <maven.compiler.target>8</maven.compiler.target>
  4. <minio.version>7.1.0</minio.version>
  5. </properties>
  6. <dependencies>
  7. <dependency>
  8. <groupId>io.minio</groupId>
  9. <artifactId>minio</artifactId>
  10. <version>${minio.version}</version>
  11. </dependency>
  12. </dependencies>

2.2 在application-dev.yml文件中配置minio

  1. minio:
  2. endpoint: http://serverip
  3. port: 9002
  4. bucketName: xxx-dev
  5. secure: false
  6. spring:
  7. mvc:
  8. hiddenmethod:
  9. filter:
  10. enabled: true
  11. # 设置文件上传大小限制
  12. servlet:
  13. multipart:
  14. max-file-size: 100MB
  15. max-request-size: 150MB

     由于csdn的限制不允许出现账号和密码相关配置,所以以上的yml配置请参照下边的minio配置类中的属性名,在yml文件中添加上 

2.3 创建minio配置类和工具类

配置类:

  1. package com.xiaomifeng1010.minio.configuration;
  2. import io.minio.MinioClient;
  3. import io.minio.errors.InvalidPortException;
  4. import lombok.Getter;
  5. import lombok.Setter;
  6. import org.springframework.boot.context.properties.ConfigurationProperties;
  7. import org.springframework.context.annotation.Bean;
  8. import org.springframework.context.annotation.Configuration;
  9. import org.springframework.stereotype.Component;
  10. /**
  11. * @author xiaomifeng1010
  12. * @version 1.0
  13. * @Description minio配置
  14. */
  15. @Configuration
  16. @Component
  17. @ConfigurationProperties(prefix = "minio")
  18. @Getter
  19. @Setter
  20. public class MinioConfig {
  21. private String endpoint;
  22. private int port;
  23. // 对应账号(请手动加到yml文件中)
  24. private String accessKey;
  25. // 对应密码(请手动加到yml文件中)
  26. private String secretKey;
  27. private Boolean secure;
  28. private String bucketName;
  29. @Bean
  30. public MinioClient getMinioClient() throws InvalidPortException {
  31. MinioClient minioClient = MinioClient.builder().endpoint(endpoint, port, secure)
  32. .credentials(accessKey, secretKey)
  33. .build();
  34. return minioClient;
  35. }
  36. //
  37. // @Bean(name = "multipartResolver")
  38. // public MultipartResolver multipartResolver(){
  39. // CommonsMultipartResolver resolver = new CommonsMultipartResolver();
  40. // resolver.setDefaultEncoding("UTF-8");
  41. // //resolveLazily属性启用是为了推迟文件解析,以在在UploadAction中捕获文件大小异常
  42. // resolver.setResolveLazily(true);
  43. // resolver.setMaxInMemorySize(40960);
  44. // //上传文件大小 50M 50*1024*1024
  45. // resolver.setMaxUploadSize(50*1024*1024);
  46. // return resolver;
  47. // }
  48. }

如何你要配置ip和port在同一个参数中,不分开,或者是直接配置域名(域名映射了ip和port),那么配置的yml 修改如下:

 把port注释掉,同时配置类也修改一下就可以了:

 工具类:

  1. package com.xiaomifeng.minio.util;
  2. import io.minio.*;
  3. import io.minio.errors.*;
  4. import io.minio.http.Method;
  5. import io.minio.messages.Bucket;
  6. import io.minio.messages.DeleteError;
  7. import io.minio.messages.DeleteObject;
  8. import io.minio.messages.Item;
  9. import lombok.extern.slf4j.Slf4j;
  10. import org.apache.commons.lang.StringUtils;
  11. import org.springframework.beans.factory.annotation.Autowired;
  12. import org.springframework.stereotype.Component;
  13. import org.springframework.web.multipart.MultipartFile;
  14. import javax.servlet.ServletOutputStream;
  15. import javax.servlet.http.HttpServletResponse;
  16. import java.io.IOException;
  17. import java.io.InputStream;
  18. import java.nio.charset.StandardCharsets;
  19. import java.security.InvalidKeyException;
  20. import java.security.NoSuchAlgorithmException;
  21. import java.util.ArrayList;
  22. import java.util.List;
  23. /**
  24. * MinIO 客户端工具类
  25. */
  26. @Component
  27. @Slf4j
  28. public class MinioClientUtils {
  29. @Autowired
  30. private MinioClient minioClient;
  31. private static final int DEFAULT_EXPIRY_TIME = 7 * 24 * 3600;
  32. /**
  33. * 检查存储桶是否存在
  34. *
  35. * @param bucketName 存储桶名称
  36. * @return boolean
  37. */
  38. public boolean bucketExists(String bucketName) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  39. boolean flag = false;
  40. flag = minioClient.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build());
  41. if (flag) {
  42. return true;
  43. }
  44. return false;
  45. }
  46. /**
  47. * 创建存储桶
  48. *
  49. * @param bucketName 存储桶名称
  50. */
  51. public boolean makeBucket(String bucketName) throws IOException, InvalidKeyException, InvalidResponseException, RegionConflictException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  52. boolean flag = bucketExists(bucketName);
  53. if (!flag) {
  54. minioClient.makeBucket(
  55. MakeBucketArgs.builder()
  56. .bucket(bucketName)
  57. .build());
  58. return true;
  59. } else {
  60. return false;
  61. }
  62. }
  63. /**
  64. * 列出所有存储桶名称
  65. *
  66. * @return List<String>
  67. */
  68. public List<String> listBucketNames() throws IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, XmlParserException, InvalidBucketNameException, InsufficientDataException, InternalException {
  69. List<Bucket> bucketList = listBuckets();
  70. List<String> bucketListName = new ArrayList<>();
  71. for (Bucket bucket : bucketList) {
  72. bucketListName.add(bucket.name());
  73. }
  74. return bucketListName;
  75. }
  76. /**
  77. * 列出所有存储桶
  78. *
  79. * @return List<Bucket>
  80. */
  81. public List<Bucket> listBuckets() throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  82. return minioClient.listBuckets();
  83. }
  84. /**
  85. * 删除存储桶
  86. *
  87. * @param bucketName 存储桶名称
  88. * @return boolean
  89. */
  90. public boolean removeBucket(String bucketName) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  91. boolean flag = bucketExists(bucketName);
  92. if (flag) {
  93. Iterable<Result<Item>> myObjects = listObjects(bucketName);
  94. for (Result<Item> result : myObjects) {
  95. Item item = result.get();
  96. // 有对象文件,则删除失败
  97. if (item.size() > 0) {
  98. return false;
  99. }
  100. }
  101. // 删除存储桶,注意,只有存储桶为空时才能删除成功。
  102. minioClient.removeBucket(RemoveBucketArgs.builder().bucket(bucketName).build());
  103. flag = bucketExists(bucketName);
  104. if (!flag) {
  105. return true;
  106. }
  107. }
  108. return false;
  109. }
  110. /**
  111. * 列出存储桶中的所有对象名称
  112. *
  113. * @param bucketName 存储桶名称
  114. * @return List<String>
  115. */
  116. public List<String> listObjectNames(String bucketName) throws XmlParserException, IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, InvalidBucketNameException, InsufficientDataException, InternalException {
  117. List<String> listObjectNames = new ArrayList<>();
  118. boolean flag = bucketExists(bucketName);
  119. if (flag) {
  120. Iterable<Result<Item>> myObjects = listObjects(bucketName);
  121. for (Result<Item> result : myObjects) {
  122. Item item = result.get();
  123. listObjectNames.add(item.objectName());
  124. }
  125. }
  126. return listObjectNames;
  127. }
  128. /**
  129. * 列出存储桶中的所有对象
  130. *
  131. * @param bucketName 存储桶名称
  132. * @return Iterable<Result<Item>>
  133. */
  134. public Iterable<Result<Item>> listObjects(String bucketName) throws XmlParserException, IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, InvalidBucketNameException, InsufficientDataException, InternalException {
  135. boolean flag = bucketExists(bucketName);
  136. if (flag) {
  137. return minioClient.listObjects( ListObjectsArgs.builder().bucket(bucketName).build());
  138. }
  139. return null;
  140. }
  141. /**
  142. * 通过文件上传到对象
  143. *
  144. * @param bucketName 存储桶名称
  145. * @param objectName 存储桶里的对象名称
  146. * @param fileName File name
  147. * @return boolean
  148. */
  149. public boolean uploadObject(String bucketName, String objectName, String fileName) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  150. boolean flag = bucketExists(bucketName);
  151. if (flag) {
  152. minioClient.uploadObject(
  153. UploadObjectArgs.builder()
  154. .bucket(bucketName).object(objectName).filename(fileName).build());
  155. ObjectStat statObject = statObject(bucketName, objectName);
  156. if (statObject != null && statObject.length() > 0) {
  157. return true;
  158. }
  159. }
  160. return false;
  161. }
  162. /**
  163. * 文件上传
  164. *
  165. * @param bucketName 存储捅名称
  166. * @param multipartFile 文件
  167. * @param filename 文件名
  168. */
  169. public void putObject(String bucketName, MultipartFile multipartFile, String filename) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  170. PutObjectOptions putObjectOptions = new PutObjectOptions(multipartFile.getSize(), PutObjectOptions.MIN_MULTIPART_SIZE);
  171. putObjectOptions.setContentType(multipartFile.getContentType());
  172. minioClient.putObject(
  173. PutObjectArgs.builder().bucket(bucketName).object(filename).stream(
  174. multipartFile.getInputStream(), multipartFile.getSize(), -1).contentType(multipartFile.getContentType())
  175. .build());
  176. }
  177. /**
  178. * 通过InputStream上传对象
  179. *
  180. * @param bucketName 存储桶名称
  181. * @param objectName 存储桶里的对象名称
  182. * @param inputStream 要上传的流
  183. * @param contentType 上传的文件类型 例如 video/mp4 image/jpg
  184. * @return boolean
  185. */
  186. public boolean putObject(String bucketName, String objectName, InputStream inputStream,String contentType) throws IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, XmlParserException, InvalidBucketNameException, InsufficientDataException, InternalException {
  187. boolean flag = bucketExists(bucketName);
  188. if (flag) {
  189. minioClient.putObject(PutObjectArgs.builder().bucket(bucketName).object(objectName).stream(
  190. //不清楚文件的大小时,可以传-1,10485760。如果知道大小也可以传入size,partsize。
  191. inputStream, -1, 10485760)
  192. .contentType(contentType)
  193. .build());
  194. ObjectStat statObject = statObject(bucketName, objectName);
  195. if (statObject != null && statObject.length() > 0) {
  196. return true;
  197. }
  198. }
  199. return false;
  200. }
  201. /**
  202. * 以流的形式获取一个文件对象
  203. *
  204. * @param bucketName 存储桶名称
  205. * @param objectName 存储桶里的对象名称
  206. * @return InputStream
  207. */
  208. public InputStream getObject(String bucketName, String objectName) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  209. boolean flag = bucketExists(bucketName);
  210. if (flag) {
  211. ObjectStat statObject = statObject(bucketName, objectName);
  212. if (statObject != null && statObject.length() > 0) {
  213. InputStream stream = minioClient.getObject( GetObjectArgs.builder()
  214. .bucket(bucketName)
  215. .object(objectName)
  216. .build());
  217. return stream;
  218. }
  219. }
  220. return null;
  221. }
  222. /**
  223. * 以流的形式获取一个文件对象(断点下载)
  224. *
  225. * @param bucketName 存储桶名称
  226. * @param objectName 存储桶里的对象名称
  227. * @param offset 起始字节的位置
  228. * @param length 要读取的长度 (可选,如果无值则代表读到文件结尾)
  229. * @return InputStream
  230. */
  231. public InputStream getObject(String bucketName, String objectName, long offset, Long length) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  232. boolean flag = bucketExists(bucketName);
  233. if (flag) {
  234. ObjectStat statObject = statObject(bucketName, objectName);
  235. if (statObject != null && statObject.length() > 0) {
  236. InputStream stream = minioClient.getObject( GetObjectArgs.builder()
  237. .bucket(bucketName)
  238. .object(objectName)
  239. .offset(1024L)
  240. .length(4096L)
  241. .build());
  242. return stream;
  243. }
  244. }
  245. return null;
  246. }
  247. /**
  248. * 下载并将文件保存到本地
  249. *
  250. * @param bucketName 存储桶名称
  251. * @param objectName 存储桶里的对象名称
  252. * @param fileName File name
  253. * @return boolean
  254. */
  255. public boolean downloadObject(String bucketName, String objectName, String fileName) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  256. boolean flag = bucketExists(bucketName);
  257. if (flag) {
  258. ObjectStat statObject = statObject(bucketName, objectName);
  259. if (statObject != null && statObject.length() > 0) {
  260. minioClient.downloadObject(DownloadObjectArgs.builder()
  261. .bucket(bucketName)
  262. .object(objectName)
  263. .filename(fileName)
  264. .build());
  265. return true;
  266. }
  267. }
  268. return false;
  269. }
  270. /**
  271. * 删除一个对象
  272. *
  273. * @param bucketName 存储桶名称
  274. * @param objectName 存储桶里的对象名称
  275. */
  276. public boolean removeObject(String bucketName, String objectName) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  277. boolean flag = bucketExists(bucketName);
  278. if (flag) {
  279. minioClient.removeObject(RemoveObjectArgs.builder().bucket(bucketName).object(objectName).build());
  280. return true;
  281. }
  282. return false;
  283. }
  284. /**
  285. * 删除指定桶的多个文件对象,返回删除错误的对象列表,全部删除成功,返回空列表
  286. *
  287. * @param bucketName 存储桶名称
  288. * @param objectNames 含有要删除的多个object名称的迭代器对象
  289. * @return
  290. * eg:
  291. * List<DeleteObject> objects = new LinkedList<>();
  292. * objects.add(new DeleteObject("my-objectname1"));
  293. * objects.add(new DeleteObject("my-objectname2"));
  294. * objects.add(new DeleteObject("my-objectname3"));
  295. */
  296. public List<String> removeObjects(String bucketName, List<DeleteObject> objectNames) throws IOException, InvalidKeyException, InvalidResponseException, InsufficientDataException, NoSuchAlgorithmException, ServerException, InternalException, XmlParserException, InvalidBucketNameException, ErrorResponseException {
  297. List<String> deleteErrorNames = new ArrayList<>();
  298. boolean flag = bucketExists(bucketName);
  299. if (flag) {
  300. Iterable<Result<DeleteError>> results = minioClient.removeObjects(RemoveObjectsArgs.builder().bucket(bucketName).objects(objectNames).build());
  301. for (Result<DeleteError> result : results) {
  302. DeleteError error = result.get();
  303. deleteErrorNames.add(error.objectName());
  304. }
  305. }
  306. return deleteErrorNames;
  307. }
  308. /**
  309. * 生成一个给HTTP GET请求用的presigned URL。
  310. * 浏览器/移动端的客户端可以用这个URL进行下载,即使其所在的存储桶是私有的。这个presigned URL可以设置一个失效时间,默认值是7天。
  311. *
  312. * @param bucketName 存储桶名称
  313. * @param objectName 存储桶里的对象名称
  314. * @param expires 失效时间(以秒为单位),默认是7天,不得大于七天
  315. * @return
  316. */
  317. public String getPresignedObjectUrl(String bucketName, String objectName, Integer expires) throws InvalidExpiresRangeException, IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, XmlParserException, InvalidBucketNameException, InsufficientDataException, InternalException {
  318. boolean flag = bucketExists(bucketName);
  319. String url = "";
  320. if (flag) {
  321. if (expires < 1 || expires > DEFAULT_EXPIRY_TIME) {
  322. throw new InvalidExpiresRangeException(expires,
  323. "expires must be in range of 1 to " + DEFAULT_EXPIRY_TIME);
  324. }
  325. try {
  326. url = minioClient.getPresignedObjectUrl(GetPresignedObjectUrlArgs.builder()
  327. .method(Method.GET)
  328. .bucket(bucketName)
  329. .object(objectName)
  330. .expiry(expires)//动态参数
  331. // .expiry(24 * 60 * 60)//用秒来计算一天时间有效期
  332. // .expiry(1, TimeUnit.DAYS)//按天传参
  333. // .expiry(1, TimeUnit.HOURS)//按小时传参数
  334. .build());
  335. } catch (ErrorResponseException | InsufficientDataException | InternalException | InvalidBucketNameException | InvalidExpiresRangeException | InvalidKeyException | InvalidResponseException | IOException | NoSuchAlgorithmException | ServerException | XmlParserException e) {
  336. e.printStackTrace();
  337. }
  338. }
  339. return url;
  340. }
  341. /**
  342. * 生成一个给HTTP PUT请求用的presigned URL。
  343. * 浏览器/移动端的客户端可以用这个URL进行上传,即使其所在的存储桶是私有的。这个presigned URL可以设置一个失效时间,默认值是7天。
  344. *
  345. * @param bucketName 存储桶名称
  346. * @param objectName 存储桶里的对象名称
  347. * @param expires 失效时间(以秒为单位),默认是7天,不得大于七天
  348. * @return String
  349. */
  350. public String presignedPutObject(String bucketName, String objectName, Integer expires) throws IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, XmlParserException, InvalidBucketNameException, InsufficientDataException, InternalException {
  351. boolean flag = bucketExists(bucketName);
  352. String url = "";
  353. if (flag) {
  354. if (expires < 1 || expires > DEFAULT_EXPIRY_TIME) {
  355. try {
  356. throw new InvalidExpiresRangeException(expires,
  357. "expires must be in range of 1 to " + DEFAULT_EXPIRY_TIME);
  358. } catch (InvalidExpiresRangeException e) {
  359. e.printStackTrace();
  360. }
  361. }
  362. try {
  363. url = minioClient.getPresignedObjectUrl(GetPresignedObjectUrlArgs.builder()
  364. .method(Method.PUT)
  365. .bucket(bucketName)
  366. .object(objectName)
  367. .expiry(expires)//动态参数
  368. // .expiry(24 * 60 * 60)//用秒来计算一天时间有效期
  369. // .expiry(1, TimeUnit.DAYS)//按天传参
  370. // .expiry(1, TimeUnit.HOURS)//按小时传参数
  371. .build());
  372. } catch (ErrorResponseException | InsufficientDataException e) {
  373. e.printStackTrace();
  374. } catch (InternalException e) {
  375. log.error("InternalException",e);
  376. } catch (InvalidBucketNameException e) {
  377. log.error("InvalidBucketNameException",e);
  378. } catch (InvalidExpiresRangeException e) {
  379. log.error("InvalidExpiresRangeException",e);
  380. } catch (InvalidKeyException e) {
  381. log.error("InvalidKeyException",e);
  382. } catch (InvalidResponseException e) {
  383. log.error("InvalidResponseException",e);
  384. } catch (IOException e) {
  385. log.error("IOException",e);
  386. } catch (NoSuchAlgorithmException e) {
  387. log.error("NoSuchAlgorithmException",e);
  388. } catch (ServerException e) {
  389. log.error("ServerException",e);
  390. } catch (XmlParserException e) {
  391. log.error("XmlParserException",e);
  392. }
  393. }
  394. return url;
  395. }
  396. /**
  397. * 获取对象的元数据
  398. *
  399. * @param bucketName 存储桶名称
  400. * @param objectName 存储桶里的对象名称
  401. * @return
  402. */
  403. public ObjectStat statObject(String bucketName, String objectName) throws IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, XmlParserException, InvalidBucketNameException, InsufficientDataException, InternalException {
  404. boolean flag = bucketExists(bucketName);
  405. if (flag) {
  406. ObjectStat statObject = null;
  407. try {
  408. statObject = minioClient.statObject(StatObjectArgs.builder().bucket(bucketName).object(objectName).build());
  409. } catch (ErrorResponseException e) {
  410. log.error("ErrorResponseException",e);
  411. } catch (InsufficientDataException e) {
  412. log.error("ErrorResponseException",e);
  413. e.printStackTrace();
  414. } catch (InternalException e) {
  415. log.error("InternalException",e);
  416. } catch (InvalidBucketNameException e) {
  417. log.error("InvalidBucketNameException",e);
  418. } catch (InvalidKeyException e) {
  419. log.error("InvalidKeyException",e);
  420. } catch (InvalidResponseException e) {
  421. log.error("InvalidResponseException",e);
  422. } catch (IOException e) {
  423. log.error("IOException",e);
  424. } catch (NoSuchAlgorithmException e) {
  425. log.error("NoSuchAlgorithmException",e);
  426. } catch (ServerException e) {
  427. log.error("ServerException",e);
  428. } catch (XmlParserException e) {
  429. log.error("XmlParserException",e);
  430. }
  431. return statObject;
  432. }
  433. return null;
  434. }
  435. /**
  436. * 文件访问路径
  437. *
  438. * @param bucketName 存储桶名称
  439. * @param objectName 存储桶里的对象名称
  440. * @return String
  441. */
  442. public String getObjectUrl(String bucketName, String objectName) throws IOException, InvalidResponseException, InvalidKeyException, NoSuchAlgorithmException, ServerException, ErrorResponseException, XmlParserException, InvalidBucketNameException, InsufficientDataException, InternalException {
  443. boolean flag = bucketExists(bucketName);
  444. String url = "";
  445. if (flag) {
  446. try {
  447. url = minioClient.getObjectUrl(bucketName, objectName);
  448. } catch (ErrorResponseException e) {
  449. log.error("XmlParserException",e);
  450. } catch (InsufficientDataException e) {
  451. log.error("InsufficientDataException",e);
  452. } catch (InternalException e) {
  453. log.error("InternalException",e);
  454. } catch (InvalidBucketNameException e) {
  455. log.error("InvalidBucketNameException",e);
  456. } catch (InvalidKeyException e) {
  457. log.error("InvalidKeyException",e);
  458. } catch (InvalidResponseException e) {
  459. log.error("InvalidResponseException",e);
  460. } catch (IOException e) {
  461. log.error("IOException",e);
  462. } catch (NoSuchAlgorithmException e) {
  463. log.error("NoSuchAlgorithmException",e);
  464. } catch (ServerException e) {
  465. log.error("ServerException",e);
  466. } catch (XmlParserException e) {
  467. log.error("XmlParserException",e);
  468. }
  469. }
  470. return url;
  471. }
  472. public void downloadFile(String bucketName, String fileName, String originalName, HttpServletResponse response) {
  473. try {
  474. InputStream file = minioClient.getObject(GetObjectArgs.builder()
  475. .bucket(bucketName)
  476. .object(fileName)
  477. .build());
  478. String filename = new String(fileName.getBytes("ISO8859-1"), StandardCharsets.UTF_8);
  479. if (StringUtils.isNotEmpty(originalName)) {
  480. fileName = originalName;
  481. }
  482. response.setHeader("Content-Disposition", "attachment;filename=" + filename);
  483. ServletOutputStream servletOutputStream = response.getOutputStream();
  484. int len;
  485. byte[] buffer = new byte[1024];
  486. while ((len = file.read(buffer)) > 0) {
  487. servletOutputStream.write(buffer, 0, len);
  488. }
  489. servletOutputStream.flush();
  490. file.close();
  491. servletOutputStream.close();
  492. } catch (ErrorResponseException e) {
  493. log.error("ErrorResponseException",e);
  494. } catch (Exception e) {
  495. log.error("Exception",e);
  496. }
  497. }
  498. }

2.3 创建一个数据表,用于保存上传到minio的文件的信息

  1. CREATE TABLE `minio_file` (
  2. `id` bigint(20) NOT NULL COMMENT '文件id',
  3. `original_file_name` varchar(255) COLLATE utf8mb4_bin DEFAULT NULL COMMENT '原始文件名称',
  4. `file_ext_name` varchar(15) COLLATE utf8mb4_bin DEFAULT NULL COMMENT '文件拓展名',
  5. `file_size` bigint(20) DEFAULT NULL COMMENT '文件大小(单位:字节)',
  6. `file_name` varchar(35) COLLATE utf8mb4_bin DEFAULT NULL COMMENT '存入minio时的文件名称',
  7. `mime` varchar(50) COLLATE utf8mb4_bin DEFAULT NULL COMMENT '文件的content-type',
  8. `file_url` varchar(255) COLLATE utf8mb4_bin DEFAULT NULL COMMENT '文件路径',
  9. `is_delete` tinyint(1) DEFAULT NULL COMMENT '是否删除 0 否 1 是',
  10. `create_by` varchar(25) COLLATE utf8mb4_bin DEFAULT NULL,
  11. `create_time` datetime DEFAULT NULL,
  12. `update_by` varchar(25) COLLATE utf8mb4_bin DEFAULT NULL,
  13. `update_time` datetime DEFAULT NULL,
  14. PRIMARY KEY (`id`)
  15. ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin;

然后使用idea自动代码生成插件,生成对应的实体类,mapper和service层

2.4 创建minio上传接口

  1. package com.xiaomifeng1010.module.minio.controller;
  2. import cn.hutool.core.io.FileUtil;
  3. import com.xiaomifeng1010.base.util.ApiResult;
  4. import com.xiaomifeng1010.module.minio.configuration.MinioConfig;
  5. import com.xiaomifeng1010.module.minio.dto.response.MinioResponseDTO;
  6. import com.xiaomifeng1010.module.minio.entity.MinioFile;
  7. import com.xiaomifeng1010.module.util.MinioClientUtils;
  8. import io.swagger.annotations.Api;
  9. import io.swagger.annotations.ApiImplicitParam;
  10. import io.swagger.annotations.ApiOperation;
  11. import lombok.AllArgsConstructor;
  12. import lombok.extern.slf4j.Slf4j;
  13. import org.apache.commons.collections4.CollectionUtils;
  14. import org.apache.commons.lang3.RandomStringUtils;
  15. import org.apache.commons.lang3.math.NumberUtils;
  16. import org.springframework.stereotype.Controller;
  17. import org.springframework.web.bind.annotation.*;
  18. import org.springframework.web.multipart.MultipartFile;
  19. import java.io.FileInputStream;
  20. import java.time.Instant;
  21. import java.util.ArrayList;
  22. import java.util.List;
  23. /**
  24. * @author xiaomifeng1010
  25. * @version 1.0
  26. * @Description minio 文件处理(上传,下载,获取文件地址等)
  27. */
  28. @Controller
  29. @RequestMapping("/fileHandle")
  30. @Slf4j
  31. @AllArgsConstructor
  32. @Api(tags = "文件处理模块")
  33. public class FileHandleController {
  34. private MinioClientUtils minioClientUtils;
  35. private MinioConfig minioConfig;
  36. @PostMapping(value = {"/admin/uploadFile","/web/uploadFile"})
  37. @ResponseBody
  38. @ApiOperation(value = "上传文件,支持批量上传")
  39. @ApiImplicitParam(name = "files",value = "文件对象",dataType = "File")
  40. public ApiResult uploadFile(@RequestParam("files") List<MultipartFile> files) {
  41. log.info(files.toString());
  42. if (CollectionUtils.isEmpty(files)){
  43. return ApiResult.error("未选择文件!");
  44. }
  45. List<MinioResponseDTO> MinioResponseDTOList=new ArrayList<>();
  46. for (MultipartFile file : files) {
  47. String originalFilename = file.getOriginalFilename();
  48. // 获取文件拓展名
  49. String extName = FileUtil.extName(originalFilename);
  50. log.info("文件拓展名:"+extName);
  51. // 生成新的文件名,存入到minio
  52. long millSeconds = Instant.now().toEpochMilli();
  53. String minioFileName=millSeconds+ RandomStringUtils.randomNumeric(12)+"."+extName;
  54. String contentType = file.getContentType();
  55. log.info("文件mime:{}",contentType);
  56. // 返回文件大小,单位字节
  57. long size = file.getSize();
  58. log.info("文件大小:"+size);
  59. try {
  60. String bucketName = minioConfig.getBucketName();
  61. minioClientUtils.putObject(bucketName,file,minioFileName);
  62. String fileUrl = minioClientUtils.getObjectUrl(bucketName, minioFileName);
  63. MinioFile minioFile = new MinioFile();
  64. minioFile.setOriginalFileName(originalFilename);
  65. minioFile.setFileExtName(extName);
  66. minioFile.setFileName(minioFileName);
  67. minioFile.setFileSize(size);
  68. minioFile.setMime(contentType);
  69. minioFile.setIsDelete(NumberUtils.INTEGER_ZERO);
  70. minioFile.setFileUrl(fileUrl);
  71. boolean insert = minioFile.insert();
  72. if (insert) {
  73. MinioResponseDTO minioResponseDTO = new MinioResponseDTO();
  74. minioResponseDTO.setFileId(minioFile.getId());
  75. minioResponseDTO.setOriginalFileName(originalFilename);
  76. minioResponseDTO.setFileUrl(fileUrl);
  77. MinioResponseDTOList.add(minioResponseDTO);
  78. }
  79. } catch (Exception e) {
  80. log.error("上传文件出错:{}",e);
  81. return ApiResult.error("上传文件出错");
  82. }
  83. }
  84. return ApiResult.success(MinioResponseDTOList);
  85. }
  86. /**
  87. * 仅仅用于测试,是否可以正常上传文件
  88. * @return
  89. * @throws Exception
  90. */
  91. @GetMapping("/test")
  92. @ApiOperation(value = "测试minio文件上传")
  93. public ApiResult testPutObject() throws Exception {
  94. FileInputStream fileInputStream = new FileInputStream("C:\\Users\\MSI\\Desktop\\新建文本文档.txt");
  95. boolean bs = minioClientUtils.putObject("xxx-dev", "新建文本文档.txt", fileInputStream, "image/jpg");
  96. log.info("上传成功?"+bs);
  97. return ApiResult.success("上传成功");
  98. }
  99. }

为了在上传文件后,把问价地址返回给前端,所以封装了一个返回数据对象DTO

  1. package com.xiaomifeng1010.module.minio.dto.response;
  2. import lombok.Data;
  3. /**
  4. * @author xiaomifeng1010
  5. * @version 1.0
  6. * @Description
  7. */
  8. @Data
  9. public class MinioResponseDTO {
  10. private Long fileId;
  11. private String fileUrl;
  12. private String originalFileName;
  13. }

然后启动项目在knife4j文档页面测试,或者在postman中测试

 由于设置了文件上传的最大限制,所以超出100兆,会抛出异MaxUploadSizeExceededException

,所以需要在全局异常处理器中处理

  1. package com.xiaomifeng1010.module.exception.handler;
  2. import com.xiaomifeng1010.base.handler.GlobalExceptionHandler;
  3. import com.xiaomifeng1010.base.util.ApiResult;
  4. import com.xiaomifeng1010.base.util.StatusEnum;
  5. import com.xiaomifeng1010.module.exception.EnterpriseBusinessException;
  6. import lombok.extern.slf4j.Slf4j;
  7. import org.apache.commons.lang3.StringUtils;
  8. import org.apache.tomcat.util.http.fileupload.impl.FileSizeLimitExceededException;
  9. import org.springframework.beans.factory.annotation.Value;
  10. import org.springframework.core.Ordered;
  11. import org.springframework.core.annotation.Order;
  12. import org.springframework.http.HttpStatus;
  13. import org.springframework.validation.BindException;
  14. import org.springframework.validation.BindingResult;
  15. import org.springframework.validation.FieldError;
  16. import org.springframework.web.bind.MethodArgumentNotValidException;
  17. import org.springframework.web.bind.annotation.ExceptionHandler;
  18. import org.springframework.web.bind.annotation.RestControllerAdvice;
  19. import org.springframework.web.multipart.MaxUploadSizeExceededException;
  20. import javax.validation.ConstraintViolation;
  21. import javax.validation.ConstraintViolationException;
  22. import java.util.stream.Collectors;
  23. /**
  24. * @author xiaomifeng1010
  25. * @version 1.0
  26. * @Description 全局异常处理器
  27. */
  28. @RestControllerAdvice
  29. @Order(value = Ordered.HIGHEST_PRECEDENCE)
  30. @Slf4j
  31. public class BaseExceptionHandler extends GlobalExceptionHandler {
  32. @Value("${spring.servlet.multipart.max-file-size}")
  33. private String maxFileSize;
  34. @ExceptionHandler(value = MethodArgumentNotValidException.class)
  35. public ApiResult handleValidException(MethodArgumentNotValidException exception) {
  36. BindingResult result = exception.getBindingResult();
  37. String errorMessage = StringUtils.EMPTY;
  38. if (result.hasErrors()) {
  39. errorMessage = result.getFieldErrors().stream().map(FieldError::getDefaultMessage).collect(Collectors.joining(","));
  40. log.error("请求对象[{}]验证错误信息{}", result.getFieldErrors().get(0).getObjectName(), errorMessage);
  41. }
  42. return ApiResult.error(HttpStatus.BAD_REQUEST.value(), errorMessage);
  43. }
  44. @ExceptionHandler(value = BindException.class)
  45. public ApiResult handleValidException(BindException exception) {
  46. BindingResult result = exception.getBindingResult();
  47. String errorMessage = StringUtils.EMPTY;
  48. if (result.hasErrors()) {
  49. errorMessage = result.getFieldErrors().stream().map(FieldError::getDefaultMessage).collect(Collectors.joining(","));
  50. log.error("请求对象[{}]验证错误信息{}", result.getFieldErrors().get(0).getObjectName(), errorMessage);
  51. }
  52. return ApiResult.error(HttpStatus.BAD_REQUEST.value(), errorMessage);
  53. }
  54. @ExceptionHandler(value = {MaxUploadSizeExceededException.class})
  55. public ApiResult handleValidException(MaxUploadSizeExceededException exception) {
  56. long maxUploadSize = exception.getMaxUploadSize();
  57. log.error("允许的最大上传字节为:{}",maxUploadSize);
  58. String errorMessage = "文件大于最大限制"+maxFileSize+",请重新上传";
  59. log.error(errorMessage);
  60. return ApiResult.error(HttpStatus.BAD_REQUEST.value(), errorMessage);
  61. }
  62. @Override
  63. @ExceptionHandler({ConstraintViolationException.class})
  64. public ApiResult handleConstraintValidationException(final ConstraintViolationException ex) {
  65. String errorMessage = ex.getConstraintViolations().stream().map(ConstraintViolation::getMessage).collect(Collectors.joining(","));
  66. return ApiResult.error(HttpStatus.BAD_REQUEST.value(), errorMessage);
  67. }
  68. @ExceptionHandler({EnterpriseBusinessException.class})
  69. public ApiResult handEnterpriseBusinessException(EnterpriseBusinessException e) {
  70. log.error("当前请求出现业务异常!", e);
  71. return ApiResult.error(e.getCode(), e.getMessage());
  72. }
  73. @Override
  74. @ExceptionHandler({Exception.class})
  75. public ApiResult handleUnknownException(final Exception ex) {
  76. if (ex instanceof MaxUploadSizeExceededException || ex instanceof FileSizeLimitExceededException) {
  77. String errorMessage = "文件超出限制,请重新上传";
  78. return ApiResult.error(errorMessage);
  79. }
  80. log.error("==> Error {} detected when request", ex.getMessage(), ex);
  81. return ApiResult.error(StatusEnum.INTERNAL_SERVER_ERROR);
  82. }
  83. }

 接着再postman中上传一个100多兆的sql脚本文件

会提示超出文件最大限制 

声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号