You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
paopao-ce/internal/dao/storage/minio.go

188 lines
5.3 KiB

// Copyright 2022 ROC. All rights reserved.
// Use of this source code is governed by a MIT style
// license that can be found in the LICENSE file.
package storage
import (
"context"
"io"
"net/url"
"strings"
"time"
"github.com/Masterminds/semver/v3"
"github.com/minio/minio-go/v7"
"github.com/rocboss/paopao-ce/internal/core"
"github.com/sirupsen/logrus"
)
var (
_ core.ObjectStorageService = (*minioServant)(nil)
_ core.OssCreateService = (*minioCreateServant)(nil)
_ core.OssCreateService = (*minioCreateRetentionServant)(nil)
_ core.OssCreateService = (*minioCreateTempDirServant)(nil)
_ core.VersionInfo = (*minioServant)(nil)
)
type minioCreateServant struct {
client *minio.Client
bucket string
domain string
}
type minioCreateRetentionServant struct {
client *minio.Client
bucket string
domain string
retainInDays time.Duration
retainUntilDate time.Time
}
type minioCreateTempDirServant struct {
client *minio.Client
bucket string
domain string
tempDir string
}
type minioServant struct {
core.OssCreateService
client *minio.Client
bucket string
domain string
}
type s3Servant = minioServant
func (s *minioCreateServant) PutObject(objectKey string, reader io.Reader, objectSize int64, contentType string, _persistance bool) (string, error) {
opts := minio.PutObjectOptions{ContentType: contentType}
_, err := s.client.PutObject(context.Background(), s.bucket, objectKey, reader, objectSize, opts)
if err != nil {
return "", err
}
return s.domain + objectKey, nil
}
func (s *minioCreateServant) PersistObject(_objectKey string) error {
// empty
return nil
}
func (s *minioCreateRetentionServant) PutObject(objectKey string, reader io.Reader, objectSize int64, contentType string, persistance bool) (string, error) {
opts := minio.PutObjectOptions{ContentType: contentType}
if !persistance {
opts.Mode = minio.Governance
opts.RetainUntilDate = time.Now().Add(s.retainInDays)
}
_, err := s.client.PutObject(context.Background(), s.bucket, objectKey, reader, objectSize, opts)
if err != nil {
return "", err
}
return s.domain + objectKey, nil
}
func (s *minioCreateRetentionServant) PersistObject(objectKey string) error {
retentionMode := minio.Governance
return s.client.PutObjectRetention(context.Background(), s.bucket, objectKey, minio.PutObjectRetentionOptions{
Mode: &retentionMode,
RetainUntilDate: &s.retainUntilDate,
})
}
func (s *minioCreateTempDirServant) PutObject(objectKey string, reader io.Reader, objectSize int64, contentType string, persistance bool) (string, error) {
opts := minio.PutObjectOptions{ContentType: contentType}
objectName := objectKey
if !persistance {
objectName = s.tempDir + objectKey
}
_, err := s.client.PutObject(context.Background(), s.bucket, objectName, reader, objectSize, opts)
if err != nil {
return "", err
}
return s.domain + objectKey, nil
}
func (s *minioCreateTempDirServant) PersistObject(objectKey string) error {
_, err := s.client.StatObject(context.Background(), s.bucket, objectKey, minio.StatObjectOptions{})
if err == nil {
logrus.Debugf("object exist so do nothing objectKey: %s", objectKey)
return nil
}
tmpObjKey := s.tempDir + objectKey
src := minio.CopySrcOptions{
Bucket: s.bucket,
Object: tmpObjKey,
}
dst := minio.CopyDestOptions{
Bucket: s.bucket,
Object: objectKey,
}
if _, err = s.client.CopyObject(context.Background(), dst, src); err != nil {
return err
}
return s.client.RemoveObject(context.Background(), s.bucket, tmpObjKey, minio.RemoveObjectOptions{ForceDelete: true})
}
func (s *minioServant) DeleteObject(objectKey string) error {
return s.client.RemoveObject(context.Background(), s.bucket, objectKey, minio.RemoveObjectOptions{ForceDelete: true})
}
func (s *minioServant) DeleteObjects(objectKeys []string) (err error) {
objectsCh := make(chan minio.ObjectInfo, len(objectKeys))
resCh := s.client.RemoveObjects(context.Background(), s.bucket, objectsCh, minio.RemoveObjectsOptions{})
for _, objectKey := range objectKeys {
objectsCh <- minio.ObjectInfo{
Key: objectKey,
}
}
// 记得一定要close否则会被卡死退出不了函数造成资源泄露
close(objectsCh)
// 宽松处理所有错误,只记录最后一次发生的错误
for result := range resCh {
if result.Err != nil {
err = result.Err
}
}
return
}
func (s *minioServant) IsObjectExist(objectKey string) (bool, error) {
_, err := s.client.StatObject(context.Background(), s.bucket, objectKey, minio.StatObjectOptions{})
if err != nil {
return false, err
}
return true, nil
}
func (s *minioServant) SignURL(objectKey string, expiredInSec int64) (string, error) {
// TODO: Set request parameters for content-disposition.
reqParams := make(url.Values)
signedURL, err := s.client.PresignedGetObject(context.Background(), s.bucket, objectKey, time.Duration(expiredInSec)*time.Second, reqParams)
if err != nil {
return "", err
}
return signedURL.String(), nil
}
func (s *minioServant) ObjectURL(objetKey string) string {
return s.domain + objetKey
}
func (s *minioServant) ObjectKey(objectUrl string) string {
return strings.Replace(objectUrl, s.domain, "", -1)
}
func (s *minioServant) Name() string {
return "MinIO"
}
func (s *minioServant) Version() *semver.Version {
return semver.MustParse("v0.2.0")
}