octopus/pkg/storage/storage_minio.go

104 lines
3.1 KiB
Go
Raw Permalink Normal View History

2023-03-23 13:52:44 +08:00
package storage
import (
"context"
"fmt"
"net/url"
"time"
"github.com/minio/minio-go/v7"
"github.com/minio/minio-go/v7/pkg/credentials"
)
2023-03-23 22:32:19 +08:00
type Storage struct {
2023-03-23 13:52:44 +08:00
client *minio.Client
endpoint string
accessKey string
accessSecret string
secure bool
bucket string
}
2023-03-23 22:32:19 +08:00
func newStorage(endpoint, accessKey, accessSecret string, secure bool, bucket string) (*Storage, error) {
2023-03-23 13:52:44 +08:00
client, err := minio.New(endpoint, &minio.Options{
Creds: credentials.NewStaticV4(accessKey, accessSecret, ""),
Secure: secure,
})
if err != nil {
return nil, err
}
2023-03-23 22:32:19 +08:00
return &Storage{
2023-03-23 13:52:44 +08:00
client: client,
endpoint: endpoint,
accessKey: accessKey,
accessSecret: accessSecret,
secure: secure,
bucket: bucket,
}, nil
}
// PresignedPutObject implements ObjectStorage
2023-03-23 22:32:19 +08:00
func (m *Storage) PresignedPutObject(ctx context.Context, objectName string, expires time.Duration) (u *url.URL, err error) {
2023-03-23 13:52:44 +08:00
return m.client.PresignedPutObject(ctx, m.bucket, objectName, expires)
}
// PresignedGetObject implements ObjectStorage
2023-03-23 22:54:18 +08:00
func (m *Storage) PresignedGetObject(ctx context.Context, objectName string, expires time.Duration, fileName string) (u *url.URL, err error) {
p := url.Values{
"response-content-disposition": []string{"attachment; filename*=utf-8''" + url.QueryEscape("真的吗.txt")},
}
return m.client.PresignedGetObject(ctx, m.bucket, objectName, expires, p)
2023-03-23 13:52:44 +08:00
}
// MoveObject implements ObjectStorage
2023-03-23 22:32:19 +08:00
func (m *Storage) MoveObject(ctx context.Context, srcObject, dstObject string) (err error) {
2023-03-23 13:52:44 +08:00
dst := minio.CopyDestOptions{Bucket: m.bucket, Object: dstObject}
src := minio.CopySrcOptions{Bucket: m.bucket, Object: srcObject}
if _, err := m.client.CopyObject(ctx, dst, src); err != nil {
return fmt.Errorf("move object failed while copy: %w", err)
2023-03-23 13:52:44 +08:00
}
if err := m.client.RemoveObject(ctx, m.bucket, srcObject, minio.RemoveObjectOptions{}); err != nil {
return fmt.Errorf("move object failed while remove: %w", err)
2023-03-23 13:52:44 +08:00
}
return nil
}
// ObjectExists implements ObjectStorage
2023-03-23 22:32:19 +08:00
func (m *Storage) ObjectStats(ctx context.Context, objectName string) (stat *minio.ObjectInfo, exists bool, err error) {
obj, err := m.client.StatObject(ctx, m.bucket, objectName, minio.StatObjectOptions{})
2023-03-23 13:52:44 +08:00
if err != nil {
switch minio.ToErrorResponse(err).Code {
2023-03-23 13:52:44 +08:00
case "NoSuchKey", "NoSuchBucket":
return nil, false, nil
2023-03-23 13:52:44 +08:00
default:
return nil, false, err
2023-03-23 13:52:44 +08:00
}
}
return &obj, true, nil
}
// MoveObject implements ObjectStorage
2023-03-23 22:32:19 +08:00
func (m *Storage) DeleteObject(ctx context.Context, objectName string) (err error) {
if err := m.client.RemoveObject(ctx, m.bucket, objectName, minio.RemoveObjectOptions{}); err != nil {
return fmt.Errorf("move object failed while remove: %w", err)
}
return nil
2023-03-23 13:52:44 +08:00
}
// MoveObject implements ObjectStorage
2023-03-23 22:32:19 +08:00
func (m *Storage) DeleteObjects(ctx context.Context, objectNames []string) []minio.RemoveObjectError {
2023-03-23 22:28:17 +08:00
// FIXME: why does DeleteObjects not working??
var errors []minio.RemoveObjectError
2023-03-23 22:28:17 +08:00
for _, objectName := range objectNames {
if err := m.DeleteObject(ctx, objectName); err != nil {
errors = append(errors, minio.RemoveObjectError{
ObjectName: objectName,
Err: err,
})
}
}
return errors
}