oss

工作中需要用到文件上传,之前使用的是本地文件系统存储方式,后来重构为支持多个存储源的方式,目前支持三种方式:local、seaweedfs、minio

存储介质

seaweedfs

seaweedfs是一款go语言开发的轻量级、高性能的存储服务器。

https://github.com/chrislusf/seaweedfs

# 启动 master
docker run -d \
-p 9333:9333 \
-p 19333:19333 \
-v /home/mining/report-cloud/docker/seaweedfs/master/data:/data \
--name seaweedfs-master \
chrislusf/seaweedfs:1.53 \
master -ip=master # 启动 volume
docker run -d \
-p 8080:8080 \
-p 18080:18080 \
-v /home/mining/report-cloud/docker/seaweedfs/volume01/data:/data \
--name seaweedfs-volume \
--link seaweedfs-master:master \
chrislusf/seaweedfs:1.53 \
volume -max=5 -mserver="master:9333" -port=8080

访问web地址:http://localhost:9333/

简单使用:

# 申请空间
curl http://localhost:9333/dir/assign
#
{"count":1,"fid":"4,017f52110b","url":"127.0.0.1:8080","publicUrl":"localhost:80
80"}
# 推送文件,4,017f52110b表示第4个volume,017f52110b表示文件的唯一标识
curl -F file=@/home/seaweedfs/balance.png http://127.0.0.1:8080/4,017f52110b

minio

MinIO是与Amazon S3 API兼容的高性能对象存储服务器,提供了人性化的管理页面

https://github.com/minio/minio

docker run --name report-minio \
-p 19000:9000 \
-e "MINIO_ACCESS_KEY=qweasdzxc" \
-e "MINIO_SECRET_KEY=1234567890" \
-v /home/mining/report-cloud/docker/minio:/data \
-d minio/minio:latest server /data # web 管理页面
http://localhost:19000/minio/login
# 创建一个 bucket 为 report
# 通过URL直接访问文件需要设置权限,参考下面博客
# https://blog.csdn.net/iKaChu/article/details/105809957
# 访问格式为
http://localhost:19000/bucket名/对象名

Java代码

maven依赖

<!-- oss 存储的依赖 -->
<!-- seaweedfs 依赖 -->
<dependency>
<groupId>org.lokra.seaweedfs</groupId>
<artifactId>seaweedfs-client</artifactId>
<version>0.7.3.RELEASE</version>
</dependency> <!-- minio 依赖 -->
<dependency>
<groupId>io.minio</groupId>
<artifactId>minio</artifactId>
<version>3.0.10</version>
</dependency>

代码

配置类

FileTag 枚举类

用来标记使用哪种存储介质

/**
* 文件上传的tag
*/
public enum FileTag {
TYPE_LOCAL,
TYPE_MINIO ,
TYPE_SEAWEEDFS;
}

FileUploadConfigLocal

本地存储配置类

public class FileUploadConfigLocal {
private String parentPath;
// 省略 get set
}

FileUploadConfigSeaweedfs

seaweedfs存储配置类

public class FileUploadConfigSeaweedfs {
private String host;
private int port = 9333;
private String publicUrl;
private int connectTimeout = 60; // 60 s
// 省略 get set
}

FileUploadConfigMinio

minio存储配置类

public class FileUploadConfigMinio {
private String domain; // url 前缀
private String accesKey;
private String secretKey;
private String bucket; // 必须事先创建好
// 省略 get set
}

FileUploadConfig

spring application.yaml 配置类

import org.springframework.boot.context.properties.ConfigurationProperties;

@ConfigurationProperties( prefix = "oss-config")
public class FileUploadConfig {
FileTag tag;
FileUploadConfigLocal local;
FileUploadConfigSeaweedfs seaweedfs;
FileUploadConfigMinio minio;
// 省略 get set
}

配置文件

application.yaml

# 对象存储配置
oss-config:
tag: TYPE_LOCAL # 使用哪种存储介质
local:
parent-path: D:\dev\2019-06-19\upload
minio:
domain: http://localhost:19000
bucket: report
acces-key: qweasdzxc
secret-key: 1234567890
seaweedfs:
host: localhost
port: 9333
publicUrl: http://localhost:8080
connectTimeout: 3000 # 5min

FileUploadInfoVO文件上传类

public class FileUploadInfoVO {

    private String fileName; // 文件名
private String suffix; // 文件后缀
private String contentType; // 文件类型
private Long size; // 文件大小
private String fid; // 文件唯一ID,同时也是保存在服务器上的文件名
private boolean delete; // 是否删除,默认false
private String description;
private String url; // 完整的url
private String tag; // 标签,比如
private String uploadTime;
private String deleteTime;
// 省略 get set
}

业务逻辑类

FileStorageService 文件存储接口

/**
* 文件存储的接口
*/
public interface FileStorageService {
FileUploadInfoVO upload(InputStream in, FileUploadInfoVO fileUploadInfoVO) throws Exception ;
FileUploadInfoVO upload(byte[] bytes, FileUploadInfoVO fileUploadInfoVO) throws Exception ;
FileUploadInfoVO delete(FileUploadInfoVO fileUploadInfoVO) throws Exception ;
}

本地存储实现

@Service("FileStorageServiceLocal")
public class FileStorageServiceLocal implements FileStorageService { @Autowired
FileUploadConfig fileUploadConfig; @PostConstruct
public void init() {
FileUtils.checkAndCreateDir(fileUploadConfig.getLocal().getParentPath());
} @Override
public FileUploadInfoVO upload(InputStream in, FileUploadInfoVO fileUploadInfoVO) throws Exception {
String filePath = fileUploadConfig.getLocal().getParentPath() + File.separator + fileUploadInfoVO.getFid();
FileUtils.copyFileFromInputStream(in, filePath);
fileUploadInfoVO.setUrl(filePath+"."+fileUploadInfoVO.getSuffix());
return fileUploadInfoVO;
} @Override
public FileUploadInfoVO upload(byte[] bytes, FileUploadInfoVO fileUploadInfoVO) throws Exception {
String filePath = fileUploadConfig.getLocal().getParentPath() + File.separator + fileUploadInfoVO.getFid();
FileUtils.copyFileFromBytes(bytes, filePath);
fileUploadInfoVO.setUrl(filePath);
return fileUploadInfoVO;
} @Override
public FileUploadInfoVO delete(FileUploadInfoVO fileUploadInfoVO) throws Exception {
File file = new File(fileUploadInfoVO.getUrl());
if(file.exists()) {
file.delete();
}
return fileUploadInfoVO;
}
}

Seaweedfs存储实现

@Service("FileStorageServiceSeaweedfs")
public class FileStorageServiceSeaweedfs implements FileStorageService { @Autowired
FileUploadConfig fileUploadConfig; FileTemplate fileTemplate; @PostConstruct
public void init() {
FileUploadConfigSeaweedfs seaweedfs = fileUploadConfig.getSeaweedfs();
FileSource fileSource = new FileSource();
fileSource.setHost(seaweedfs.getHost());
fileSource.setPort(seaweedfs.getPort());
fileSource.setConnectionTimeout(seaweedfs.getConnectTimeout());//5min
try {
fileSource.startup();
} catch (IOException e) {
throw new RuntimeException("创建seaweedfs连接失败,原因是:" + e.getMessage());
}
fileTemplate = new FileTemplate(fileSource.getConnection(), seaweedfs.getPublicUrl());
} @Override
public FileUploadInfoVO upload(InputStream in, FileUploadInfoVO fileUploadInfoVO) throws Exception {
FileHandleStatus fileHandleStatus = fileTemplate.saveFileByStream(fileUploadInfoVO.getFileName(), in, ContentType.create(fileUploadInfoVO.getContentType(), "utf-8"));
fileUploadInfoVO.setFid(fileHandleStatus.getFileId());
fileUploadInfoVO.setUrl(String.format("%s/%s", fileUploadConfig.getSeaweedfs().getPublicUrl(), fileHandleStatus.getFileId()));
return fileUploadInfoVO;
} @Override
public FileUploadInfoVO upload(byte[] bytes, FileUploadInfoVO fileUploadInfoVO) throws Exception {
return upload(new ByteArrayInputStream(bytes), fileUploadInfoVO);
} @Override
public FileUploadInfoVO delete(FileUploadInfoVO fileUploadInfoVO) throws Exception {
fileTemplate.deleteFile(fileUploadInfoVO.getFid());
return fileUploadInfoVO;
}
}

Minio存储实现

@Service("FileStorageServiceMinio")
public class FileStorageServiceMinio implements FileStorageService { @Autowired
FileUploadConfig fileUploadConfig; @Override
public FileUploadInfoVO upload(InputStream in, FileUploadInfoVO fileUploadInfoVO) throws Exception {
FileUploadConfigMinio minio = fileUploadConfig.getMinio();
String domain = minio.getDomain();
String bucket = minio.getBucket();
String fileName = fileUploadInfoVO.getFileName();
MinioClient minioClient = new MinioClient(domain, minio.getAccesKey(), minio.getSecretKey());
minioClient.putObject("report", fileName, in, fileUploadInfoVO.getSize(), fileUploadInfoVO.getContentType());
fileUploadInfoVO.setUrl(String.format("%s/%s/%s", domain, bucket, fileName));
return fileUploadInfoVO;
} @Override
public FileUploadInfoVO upload(byte[] bytes, FileUploadInfoVO fileUploadInfoVO) throws Exception {
return upload(new ByteArrayInputStream(bytes), fileUploadInfoVO);
} @Override
public FileUploadInfoVO delete(FileUploadInfoVO fileUploadInfoVO) throws Exception {
FileUploadConfigMinio minio = fileUploadConfig.getMinio();
MinioClient minioClient = new MinioClient(minio.getDomain(), minio.getAccesKey(), minio.getSecretKey());
minioClient.removeObject(minio.getBucket(), fileUploadInfoVO.getFileName());
return fileUploadInfoVO;
}
}

存储工厂类实现

用来根据 FileTag 枚举类获得对应的具体实现

@Service
public class FileStorageFactory { Map<FileTag, FileStorageService> fileStorageServiceMap; @Qualifier("FileStorageServiceLocal")
@Autowired
FileStorageService fileStorageServiceLocal; @Qualifier("FileStorageServiceSeaweedfs")
@Autowired
FileStorageService fileStorageServiceSeaweedfs; @Qualifier("FileStorageServiceMinio")
@Autowired
FileStorageService fileStorageServiceMinio; @PostConstruct
public void init() {
fileStorageServiceMap = new HashMap<>();
fileStorageServiceMap.put(FileTag.TYPE_LOCAL, fileStorageServiceLocal);
fileStorageServiceMap.put(FileTag.TYPE_SEAWEEDFS, fileStorageServiceSeaweedfs);
fileStorageServiceMap.put(FileTag.TYPE_MINIO, fileStorageServiceMinio);
} public FileStorageService get(FileTag type) {
FileStorageService fileStorageService = fileStorageServiceMap.get(type);
return fileStorageService == null ? fileStorageServiceMap.get(FileTag.TYPE_LOCAL) : fileStorageService;
}
}

业务中实现

这里根据具体的业务,来选择具体的实现逻辑

eg:

@Autowired
private FileStorageFactory fileStorageFactory;
public FileUploadInfoVO put(MultipartFile multipartFile) throws Exception {
FileUploadInfoVO vo = new FileUploadInfoVO();
// 填充一些信息(省略)
// 获得文件上传处理器
FileStorageService fileStorageService = fileStorageFactory.get(FileTag.TYPE_LOCAL);
fileUploadInfoVO = fileStorageService.upload(multipartFile.getInputStream(), vo);
// ... 业务逻辑
}

这样,配置文件中可以使用FileTag来控制文件存储的介质,从而实现一个简单的对象存储服务器。

java实现简单的oss存储的更多相关文章

  1. Java使用阿里云OSS对象存储上传图片

    原 Java使用阿里云OSS对象存储上传图片 2017年03月27日 10:47:28 陌上桑花开花 阅读数 26804更多 分类专栏: 工作案例总结 版权声明:本文为博主原创文章,遵循CC 4.0 ...

  2. Java项目接入阿里云OSS存储

    需求背景 目前公司内部项目所支持的文件云存储方式还是公司内部项目组提供的方案,但在时间的考验之下,弊端显现,尤其是灾备切换过程中需要切换访问地址,这种操作不方便,更可能因为中间过程的失误导致资源不可用 ...

  3. java实现简单的单点登录

    java实现简单的单点登录 摘要:单点登录(SSO)的技术被越来越广泛地运用到各个领域的软件系统当中.本文从业务的角度分析了单点登录的需求和应用领域:从技术本身的角度分析了单点登录技术的内部机制和实现 ...

  4. 编写你的第一个 Java 版 Raft 分布式 KV 存储

    前言 本文旨在讲述如何使用 Java 语言实现基于 Raft 算法的,分布式的,KV 结构的存储项目.该项目的背景是为了深入理解 Raft 算法,从而深刻理解分布式环境下数据强一致性该如何实现:该项目 ...

  5. BerkeleyDB java的简单使用

    关于BerkeleyDB的有点和优点,列在以下 JE offers the following major features: Large database support. JE databases ...

  6. java实现简单窗体小游戏----球球大作战

    java实现简单窗体小游戏----球球大作战需求分析1.分析小球的属性: ​ 坐标.大小.颜色.方向.速度 2.抽象类:Ball ​ 设计类:BallMain—创建窗体 ​ BallJPanel—画小 ...

  7. 用Java实现简单的区块链

    用 Java 实现简单的区块链 1. 概述 本文中,我们将学习区块链技术的基本概念.也将根据概念使用 Java 来实现一个基本的应用程序. 进一步,我们将讨论一些先进的概念以及该技术的实际应用. 2. ...

  8. Java集合简单介绍

    再最前面分享一下我再学习集合时的方法: 1.首先了解各集合的定义和特点 2.集合的构造方法和常用方法(增删改查等) 3.了解集合使用的场景,再什么情况下使用什么类型的集合(关键是集合的特性) 4.了解 ...

  9. 谷粒 | 10 | 阿里云OSS存储对象服务

    阿里云OSS对象存储服务 准备工作 1.在service模块新建子模块service_oss 2.引入pom.xml文件中引入oss服务依赖 <dependencies> <!--a ...

随机推荐

  1. 转帖 支撑4.5亿活跃用户的WhatsApp架构概览

    http://www.csdn.net/article/2014-02-27/2818559-an-overview-at-whatsapp's-19b-architecture/2 写的很好,确实牛 ...

  2. Spring Boot集成H2数据库

    需求 平时学习的时候,涉及到一些连接数据库相关的操作,经常需要初始化本地数据库,比如装个MySQL,初始化一些脚本,比较麻烦,H2是内存数据库,Spring Boot可以在应用启动的时候对H2数据库初 ...

  3. React:Element

    React Elements 是构成React App的最小单位.React中的组件是由一个或多个Elements构成的. 和DOM不同,React Elements 是纯粹的JS对象.利用React ...

  4. Java基础之数据类型

    一.数据类型 基本数据类型介绍 byte 1字节 char 2字节 short 2字节 int 4字节 long 8字节 float 4字节 double 8字节 以上有Java中八大基本类型的7种, ...

  5. DRF认证组件

    1.DRF认证组件之视图注册用法(自定义简单使用) settings.py配置 INSTALLED_APPS = [ 'django.contrib.admin', 'django.contrib.a ...

  6. 《机器学习_07_02_svm_软间隔支持向量机》

    一.简介 上一节介绍了硬间隔支持向量机,它可以在严格线性可分的数据集上工作的很好,但对于非严格线性可分的情况往往就表现很差了,比如: import numpy as np import matplot ...

  7. MongoDB全球云端技术盛会MongoDB.live

    MongoDB全球云端技术盛会MongoDB.live,将于北京时间6月9日22:00正式开启,大会将以在线直播+按需学习相结合的方式,面向全球开发者.架构师等MongoDB 用户和爱好者免费开放,精 ...

  8. LTE常用标识和参数

    1 基本标识 1 .1 IMSI 1.2 IMEI 1.3 MSISDN 1.4 TMSI 1.5 MSRN 2 区域类标识 2.1 GCI 其中 LA是GSM(2g)中的位置区,对应4G中的跟踪区T ...

  9. 01Java核心-冷门知识001-包

    1)导入静态方法和静态域 import 可以加上static关键字,导入静态的方法和静态域. 例如: package com.gail.test; import static java.lang.Sy ...

  10. Linux suid 提权

    SUID (Set owner User ID up on execution) 是给予文件的一个特殊类型的文件权限.在 Linux/Unix中,当一个程序运行的时候, 程序将从登录用户处继承权限.S ...