自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Minio + Docker 搭建屬于自己的OSS存儲服務(wù)

云計(jì)算 云原生
MinIO 是一個基于 Apache License v2.0 開源協(xié)議的對象存儲服務(wù),它兼容亞馬遜 S3 云存儲服務(wù)接口,非常適合于存儲大容量非結(jié)構(gòu)化的數(shù)據(jù),例如圖片、視頻、日志文件、備份數(shù)據(jù)和容器/虛擬機(jī)鏡像等,而一個對象文件可以是任意大小,從幾 KB 到最大 5T 不等。

現(xiàn)在 OSS 服務(wù)算是一個基礎(chǔ)服務(wù)了,很多云服務(wù)廠商都有提供這樣的服務(wù),價格也不貴,松哥自己的 www.javaboy.org 用的就是類似的服務(wù)。

不過對于中小公司來說,除了購買 OSS 服務(wù)之外,也可以自己搭建專業(yè)的文件服務(wù)器,自己搭建專門的文件服務(wù)器的話,曾經(jīng)比較專業(yè)的做法是 FastDFS,松哥之前也專門為之錄過視頻發(fā)在 B 站上,感興趣的小伙伴可以自行查看。不過 FastDFS 搭建比較麻煩,非常容易出錯,所以對各位小伙伴來說多多少少有一點(diǎn)門檻。

松哥在之前的文章錄制的一些項(xiàng)目視頻中,如果涉及到文件上傳,基本上都是保存在項(xiàng)目本地,這種方式比較省事,但是安全性不高。

所以,今天給大伙介紹一個較好的玩意 MinIO,看看這個工具帶給我們什么驚喜。

1. MinIO 簡介

MinIO 是一個基于 Apache License v2.0 開源協(xié)議的對象存儲服務(wù),它兼容亞馬遜 S3 云存儲服務(wù)接口,非常適合于存儲大容量非結(jié)構(gòu)化的數(shù)據(jù),例如圖片、視頻、日志文件、備份數(shù)據(jù)和容器/虛擬機(jī)鏡像等,而一個對象文件可以是任意大小,從幾 KB 到最大 5T 不等。

MinIO 是一個非常輕量的服務(wù),可以很簡單的和其他應(yīng)用的結(jié)合,類似 NodeJS, Redis 或者 MySQL。

簡單來說,可以使用 MinIO 來搭建一個對象存儲服務(wù),而且 MinIO 的 Java 客戶端和亞馬遜的 S3 云存儲服務(wù)客戶端接口兼容,換句話說,你會往 MinIO 上存數(shù)據(jù),就會往 S3 上存數(shù)據(jù)。

MinIO 的特點(diǎn):

  • 兼容 Amazon S3:可以使用 MinIO SDK,MinIO Client,AWS SDK 和 AWS CLI 訪問 MinIO 服務(wù)器。
  • 較強(qiáng)的數(shù)據(jù)保護(hù)能力:MinIO 使用 Minio Erasure Code 來防止硬件故障。
  • 高度可用:MinIO 服務(wù)器可以容忍分布式設(shè)置中高達(dá)(N/2)-1 節(jié)點(diǎn)故障。
  • 支持 Lambda 計(jì)算。
  • 具有加密和防篡改功能:MinIO 為加密數(shù)據(jù)提供了機(jī)密性,完整性和真實(shí)性保證,而且性能開銷微乎其微。使用 AES-256-GCM,ChaCha20-Poly1305 和 AES-CBC 支持服務(wù)器端和客戶端加密。
  • 可對接后端存儲:除了 MinIO 自己的文件系統(tǒng),還支持 DAS、 JBODs、NAS、Google 云存儲和 Azure Blob 存儲。

2. MinIO 安裝

不廢話了,趕緊裝一個體驗(yàn)一把吧。

為了省事,咱們就直接用 docker 來安裝吧,如果你對 docker 還不熟悉,公眾號后臺回復(fù) docker 獲取松哥的 docker 教程。

我們執(zhí)行如下命令,安裝 MinIO:

docker run -p 9000:9000 -p 9001:9001 -d minio/minio server /data --console-address ":9000" --address ":9001"

這個啟動命令中配置了兩個端口:console-address 是后臺管理的網(wǎng)頁端口;address 則是 API 通信端口。以上面的啟動腳本為例,項(xiàng)目啟動成功后,網(wǎng)頁上的訪問端口是 9000,如果我們通過 Java 代碼上傳文件,通信端口則是 9001。

項(xiàng)目啟動成功后,瀏覽器地址欄輸入 http://127.0.0.1:9000/login 即可訪問到 MinIO 的后端頁面:

圖片圖片

默認(rèn)的登錄用戶名和密碼均為 minioadmin。

登錄成功之后,我們首先創(chuàng)建一個 bucket,將來我們上傳的文件都處于 bucket 之中,如下:

圖片圖片

圖片圖片

創(chuàng)建成功之后,我們還需要設(shè)置一下桶的讀取權(quán)限,確保文件將來上傳成功之后可以讀取到,點(diǎn)擊左上角的設(shè)置按鈕進(jìn)行設(shè)置,如下:

圖片圖片

圖片

設(shè)置完成后,接下來我們就可以往這個桶中上傳資源了,如下圖:

圖片圖片

上傳完成后,就可以看到剛剛上傳的文件了:

圖片圖片

上傳成功后,點(diǎn)擊文件,然后點(diǎn)擊右邊的 Share 按鈕會彈出來文件的訪問鏈接,由于我們已經(jīng)設(shè)置了文件可讀,因此可以不用管這里的鏈接有效期了,直接通過路徑的前面部分就可以訪問到剛剛上傳的圖片了,如下:

圖片圖片

圖片圖片

現(xiàn)在文件就可上傳可訪問了。是不是比 FastDFS 容易多了!

不過前面這種安裝方式其實(shí)有點(diǎn)小問題,因?yàn)槲覀儧]有為 docker 容器設(shè)置數(shù)據(jù)卷,所以如果你把 docker 容器不小心刪除了,那么數(shù)據(jù)也就沒了!

所以我們要設(shè)置數(shù)據(jù)卷。

修正后的 docker 腳本如下:

docker run -p 9000:9000 -p 9001:9001 -d --name minio -v /Users/sang/minio/data:/data -v /Users/sang/minio/config:/root/.minio -e "MINIO_ROOT_USER=javaboy" -e "MINIO_ROOT_PASSWORD=123@45678" minio/minio server /data --console-address ":9000" --address ":9001"

主要是加了數(shù)據(jù)卷映射功能,將 MinIO 的數(shù)據(jù)和配置文件映射到宿主機(jī)上,這樣將來即使容器刪除了,數(shù)據(jù)也都還在。

注意上面也自定義了登錄用戶名和密碼。

按照上面的命令,重新創(chuàng)建容器之后,我們也創(chuàng)建一個桶并上傳文件,上傳成功之后,我們就可以在本地對應(yīng)的文件夾看到我們上傳的文件,如下:

圖片圖片

3. 整合 Spring Boot

接下來我們再來看看在 Spring Boot 中如何玩 MinIO。

首先我們創(chuàng)建一個 Spring Boot 項(xiàng)目,引入 Web 依賴。

項(xiàng)目創(chuàng)建成功之后,我們再來手動添加一下 MinIO 的依賴,如下:

<dependency>
    <groupId>io.minio</groupId>
    <artifactId>minio</artifactId>
    <version>8.2.1</version>
</dependency>

接下來我們來配置一下 application.yaml,配置一下文件上傳所需要的基本信息:

minio:
  endpoint: http://localhost:9001
  accessKey: javaboy
  secretKey: 123@45678
  nginxHost: http://local.javaboy.org:9001

這里四個屬性:

  • endpoint:這是 MinIO 的 API 通信地址。
  • accessKey 和 secretKey 是通信的用戶名和密碼,這跟網(wǎng)頁上登錄時候的用戶名密碼一致。
  • nginxHost:這個配置用來生成上傳文件的訪問路徑。對于這個路徑,有的小伙伴可能會有疑問,nginxHost 不就是 endpoint 嗎?為什么還要單獨(dú)配置?因?yàn)閷τ谖募?wù)器而言,我們上傳文件是通過 MinIO,但是訪問的時候不一定通過 MinIO,我們可能會自己搭建一個 Nginx 服務(wù)器,通過 Nginx 服務(wù)器來訪問上傳后的資源,大家知道 Nginx 非常擅長于做這個事情,效率非常高。所以這里的 nginxHost 其實(shí)是指 Nginx 的訪問路徑。

接下來我們提供一個 MinioProperties 來接收這里的四個屬性,如下:

@ConfigurationProperties(prefix = "minio")
public class MinioProperties {
    /**
     * 連接地址
     */
    private String endpoint;
    /**
     * 用戶名
     */
    private String accessKey;
    /**
     * 密碼
     */
    private String secretKey;
    /**
     * 域名
     */
    private String nginxHost;

    public String getEndpoint() {
        return endpoint;
    }

    public void setEndpoint(String endpoint) {
        this.endpoint = endpoint;
    }

    public String getAccessKey() {
        return accessKey;
    }

    public void setAccessKey(String accessKey) {
        this.accessKey = accessKey;
    }

    public String getSecretKey() {
        return secretKey;
    }

    public void setSecretKey(String secretKey) {
        this.secretKey = secretKey;
    }

    public String getNginxHost() {
        return nginxHost;
    }

    public void setNginxHost(String nginxHost) {
        this.nginxHost = nginxHost;
    }
}

將 application.yaml 中相關(guān)的配置注入到這個配置類中來。

接下來我們需要提供一個 MinIOClient,通過這個客戶端工具可以操作 MinIO,如下:

@Configuration
@EnableConfigurationProperties(MinioProperties.class)
public class MinioConfig {

    @Autowired
    private MinioProperties minioProperties;

    /**
     * 獲取MinioClient
     */
    @Bean
    public MinioClient minioClient() {
        return MinioClient.builder()
                .endpoint(minioProperties.getEndpoint())
                .credentials(minioProperties.getAccessKey(), minioProperties.getSecretKey())
                .build();
    }

}

這個也沒啥好說的,傳入通信地址以及用戶名密碼,就可以構(gòu)建出一個 MinioClient 出來。

當(dāng)文件上傳成功之后,我們可以通過 MinIO 去訪問,也可以通過 Nginx 訪問,所以接下來我們就需要提供一個類,來封裝這兩個地址:

public class UploadResponse {
    private String minIoUrl;

    private String nginxUrl;

    public UploadResponse() {
    }

    public UploadResponse(String minIoUrl, String nginxUrl) {
        this.minIoUrl = minIoUrl;
        this.nginxUrl = nginxUrl;
    }

    public String getMinIoUrl() {
        return minIoUrl;
    }

    public void setMinIoUrl(String minIoUrl) {
        this.minIoUrl = minIoUrl;
    }

    public String getNginxUrl() {
        return nginxUrl;
    }

    public void setNginxUrl(String nginxUrl) {
        this.nginxUrl = nginxUrl;
    }
}

再來提供一個 MinIO 文件上傳工具類:

@Component
public class MinioUtil {

    @Autowired
    private MinioProperties minioProperties;

    @Autowired
    private MinioClient client;

    /**
     * 創(chuàng)建bucket
     */
    public void createBucket(String bucketName) throws Exception {
        if (!client.bucketExists(BucketExistsArgs.builder().bucket(bucketName).build())) {
            client.makeBucket(MakeBucketArgs.builder().bucket(bucketName).build());
        }
    }

    /**
     * 上傳文件
     */
    public UploadResponse uploadFile(MultipartFile file, String bucketName) throws Exception {
        //判斷文件是否為空
        if (null == file || 0 == file.getSize()) {
            return null;
        }
        //判斷存儲桶是否存在  不存在則創(chuàng)建
        createBucket(bucketName);
        //文件名
        String originalFilename = file.getOriginalFilename();
        //新的文件名 = 存儲桶文件名_時間戳.后綴名
        assert originalFilename != null;
        SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
        String fileName = bucketName + "_" +
                System.currentTimeMillis() + "_" + format.format(new Date()) + "_" + new Random().nextInt(1000) +
                originalFilename.substring(originalFilename.lastIndexOf("."));
        //開始上傳
        client.putObject(
                PutObjectArgs.builder().bucket(bucketName).object(fileName).stream(
                        file.getInputStream(), file.getSize(), -1)
                        .contentType(file.getContentType())
                        .build());
        String url = minioProperties.getEndpoint() + "/" + bucketName + "/" + fileName;
        String urlHost = minioProperties.getNginxHost() + "/" + bucketName + "/" + fileName;
        return new UploadResponse(url, urlHost);
    }

    /**
     * 獲取全部bucket
     *
     * @return
     */
    public List<Bucket> getAllBuckets() throws Exception {
        return client.listBuckets();
    }

    /**
     * 根據(jù)bucketName獲取信息
     *
     * @param bucketName bucket名稱
     */
    public Optional<Bucket> getBucket(String bucketName) throws IOException, InvalidKeyException, NoSuchAlgorithmException, InsufficientDataException, InvalidResponseException, InternalException, ErrorResponseException, ServerException, XmlParserException, ServerException {
        return client.listBuckets().stream().filter(b -> b.name().equals(bucketName)).findFirst();
    }

    /**
     * 根據(jù)bucketName刪除信息
     *
     * @param bucketName bucket名稱
     */
    public void removeBucket(String bucketName) throws Exception {
        client.removeBucket(RemoveBucketArgs.builder().bucket(bucketName).build());
    }

    /**
     * 獲取?件外鏈
     *
     * @param bucketName bucket名稱
     * @param objectName ?件名稱
     * @param expires    過期時間 <=7
     * @return url
     */
    public String getObjectURL(String bucketName, String objectName, Integer expires) throws Exception {
        return client.getPresignedObjectUrl(GetPresignedObjectUrlArgs.builder().bucket(bucketName).object(objectName).expiry(expires).build());
    }

    /**
     * 獲取?件
     *
     * @param bucketName bucket名稱
     * @param objectName ?件名稱
     * @return ?進(jìn)制流
     */
    public InputStream getObject(String bucketName, String objectName) throws Exception {
        return client.getObject(GetObjectArgs.builder().bucket(bucketName).object(objectName).build());
    }

    /**
     * 上傳?件
     *
     * @param bucketName bucket名稱
     * @param objectName ?件名稱
     * @param stream     ?件流
     * @throws Exception https://docs.minio.io/cn/java-client-api-reference.html#putObject
     */
    public void putObject(String bucketName, String objectName, InputStream stream) throws
            Exception {
        client.putObject(PutObjectArgs.builder().bucket(bucketName).object(objectName).stream(stream, stream.available(), -1).contentType(objectName.substring(objectName.lastIndexOf("."))).build());
    }

    /**
     * 上傳?件
     *
     * @param bucketName  bucket名稱
     * @param objectName  ?件名稱
     * @param stream      ?件流
     * @param size        ??
     * @param contextType 類型
     * @throws Exception https://docs.minio.io/cn/java-client-api-reference.html#putObject
     */
    public void putObject(String bucketName, String objectName, InputStream stream, long
            size, String contextType) throws Exception {
        client.putObject(PutObjectArgs.builder().bucket(bucketName).object(objectName).stream(stream, size, -1).contentType(contextType).build());
    }

    /**
     * 獲取?件信息
     *
     * @param bucketName bucket名稱
     * @param objectName ?件名稱
     * @throws Exception https://docs.minio.io/cn/java-client-api-reference.html#statObject
     */
    public StatObjectResponse getObjectInfo(String bucketName, String objectName) throws Exception {
        return client.statObject(StatObjectArgs.builder().bucket(bucketName).object(objectName).build());
    }

    /**
     * 刪除?件
     *
     * @param bucketName bucket名稱
     * @param objectName ?件名稱
     * @throws Exception https://docs.minio.io/cn/java-client-apireference.html#removeObject
     */
    public void removeObject(String bucketName, String objectName) throws Exception {
        client.removeObject(RemoveObjectArgs.builder().bucket(bucketName).object(objectName).build());
    }
}

都是一些常規(guī)的 API 調(diào)用,我就不逐行解釋了,接下來我們來一個文件上傳接口:

@RestController
public class FileUploadController {
    @Autowired
    MinioUtil minioUtil;

    @PostMapping("/upload")
    public String fileUpload(MultipartFile file) throws Exception {
        UploadResponse bucket01 = minioUtil.uploadFile(file, "bucket01");
        System.out.println("bucket01.getMinIoUrl() = " + bucket01.getMinIoUrl());
        System.out.println("bucket01.getNginxUrl() = " + bucket01.getNginxUrl());
        return bucket01.getMinIoUrl();
    }
}

好啦,大功告成。

接下來啟動 Spring Boot 項(xiàng)目,然后調(diào)用這個接口上傳文件,上傳成功后,控制臺會打印如下信息:

這就表示文件上傳成功了。

4. 配置 nginx

前面提到了 MinIO 可以結(jié)合 Nginx 來使用,那我們這里就來配一配 Nginx 看看。

為了省事,Nginx 我也選擇安裝到 docker 容器中,但是前面安裝 MinIO 時,我們已經(jīng)做了數(shù)據(jù)卷映射,即上傳到 MinIO 的文件實(shí)際上是保存在宿主機(jī)的,所以現(xiàn)在也得給 Nginx 配置數(shù)據(jù)卷,將來讓 Nginx 也去 /Users/sang/minio/data 路徑下查找文件。

Nginx 安裝指令如下:

docker run --name nginx01 -p 8888:80 -v /Users/sang/minio/data:/usr/share/nginx/html:ro -d nginx

這里兩個關(guān)鍵點(diǎn):

  • 設(shè)置 Nginx 端口為 8888。
  • 將 MinIO 映射到宿主機(jī)的數(shù)據(jù)卷,再次掛載到 Nginx 上去。

大家知道,默認(rèn)情況下,當(dāng)我們訪問 Nginx 的時候,Nginx 給我們展示出來的數(shù)據(jù)其實(shí)就是 /usr/share/nginx/html 目錄下的,現(xiàn)在該目錄其實(shí)就相當(dāng)于我宿主機(jī)的 /Users/sang/minio/data 目錄,所以我現(xiàn)在都不用修改 Nginx 的配置了,裝好之后直接使用 Nginx 即可。

好啦,接下來我們修改一下 application.yaml,如下:

minio:
  endpoint: http://localhost:9001
  accessKey: javaboy
  secretKey: 123@45678
  nginxHost: http://local.javaboy.org:8888

改完之后,再次上傳文件,此時打印出來的文件訪問路徑如下:

圖片圖片

現(xiàn)在我們通過這個 Nginx 路徑也能訪問到剛剛上傳的文件了。

責(zé)任編輯:武曉燕 來源: 江南一點(diǎn)雨
相關(guān)推薦

2021-02-05 05:29:51

服務(wù)器GitGogs

2021-12-14 16:15:47

LinuxNginxWeb

2010-10-14 14:31:31

Ubuntu發(fā)行版

2023-11-21 17:36:04

OpenFeignSentinel

2011-01-14 12:02:50

Linux影音

2022-10-09 07:21:21

wordpress數(shù)據(jù)庫mysql

2020-12-09 05:15:53

對象存儲前端

2024-12-02 15:50:42

2018-11-22 09:40:11

Docker服務(wù)器企業(yè)版

2025-04-25 01:30:00

RAGFlowDifyMiniO

2014-08-26 10:05:37

框架開發(fā)AngularJS

2022-09-01 10:46:02

前端組件庫

2020-10-11 21:00:31

開發(fā)存儲服務(wù)技術(shù)

2012-02-27 15:44:12

存儲服務(wù)器寶通

2020-04-02 08:47:04

開發(fā)網(wǎng)站技術(shù)

2021-10-21 10:25:32

Windows7操作系統(tǒng)微軟

2024-09-14 14:09:40

2024-07-31 09:34:59

2023-08-31 22:05:02

SAN環(huán)境存儲

2022-11-09 07:40:18

點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號