2024-11-17 16:00:48 +01:00
|
|
|
package fsys
|
|
|
|
|
|
|
|
import (
|
2024-11-18 22:34:07 +01:00
|
|
|
"bytes"
|
|
|
|
"context"
|
2024-11-17 16:00:48 +01:00
|
|
|
"errors"
|
|
|
|
"github.com/minio/minio-go/v7"
|
|
|
|
"github.com/minio/minio-go/v7/pkg/credentials"
|
2024-11-18 22:39:35 +01:00
|
|
|
"io"
|
2024-11-18 22:48:49 +01:00
|
|
|
"io/fs"
|
2024-11-18 22:34:07 +01:00
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2024-11-17 16:00:48 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ErrInvalidStorageType = errors.New("invalid storage type")
|
|
|
|
ErrInvalidPath = errors.New("invalid path")
|
|
|
|
ErrFileNotFound = errors.New("file not found")
|
|
|
|
)
|
|
|
|
|
|
|
|
type Storage struct {
|
2024-11-18 23:00:58 +01:00
|
|
|
config Config
|
2024-11-17 16:00:48 +01:00
|
|
|
s3Client *minio.Client
|
2024-11-18 22:34:07 +01:00
|
|
|
ctx context.Context
|
2024-11-17 16:00:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type Config struct {
|
2024-11-18 23:00:58 +01:00
|
|
|
// Type of storage, can be "local" or "s3"
|
2024-11-17 16:00:48 +01:00
|
|
|
Type string
|
|
|
|
|
|
|
|
Path string
|
|
|
|
|
|
|
|
S3Endpoint string
|
|
|
|
S3Location string
|
|
|
|
S3Secure bool
|
|
|
|
|
|
|
|
S3AccessID string
|
|
|
|
S3AccessKey string
|
|
|
|
|
|
|
|
S3BucketName string
|
|
|
|
}
|
|
|
|
|
2024-11-18 23:00:58 +01:00
|
|
|
// New creates a new storage interface
|
|
|
|
// It takes Config as its parameter.
|
|
|
|
// The function returns the Storage interface, and an error, if any.
|
|
|
|
func New(config Config) (*Storage, error) {
|
2024-11-17 16:00:48 +01:00
|
|
|
newStorage := new(Storage)
|
|
|
|
newStorage.config = config
|
2024-11-18 22:34:07 +01:00
|
|
|
newStorage.ctx = context.Background()
|
2024-11-17 16:00:48 +01:00
|
|
|
|
|
|
|
switch config.Type {
|
|
|
|
case "minio", "s3":
|
|
|
|
s3Client, err := minio.New(config.S3Endpoint, &minio.Options{
|
|
|
|
Creds: credentials.NewStaticV4(config.S3AccessID, config.S3AccessKey, ""),
|
|
|
|
Secure: config.S3Secure,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-11-18 22:34:07 +01:00
|
|
|
|
|
|
|
bucketExists, err := s3Client.BucketExists(newStorage.ctx, config.S3BucketName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !bucketExists {
|
|
|
|
if err := s3Client.MakeBucket(newStorage.ctx, config.S3BucketName, minio.MakeBucketOptions{}); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-17 16:00:48 +01:00
|
|
|
newStorage.s3Client = s3Client
|
|
|
|
case "local":
|
2024-11-18 22:48:49 +01:00
|
|
|
if statInfo, err := os.Stat(config.Path); err != nil && errors.Is(err, fs.ErrNotExist) {
|
|
|
|
if err := os.MkdirAll(config.Path, 0740); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else if err != nil {
|
2024-11-18 22:34:07 +01:00
|
|
|
return nil, err
|
2024-11-18 22:48:49 +01:00
|
|
|
} else if statInfo.IsDir() {
|
|
|
|
return nil, ErrInvalidPath
|
2024-11-18 22:34:07 +01:00
|
|
|
}
|
2024-11-17 16:00:48 +01:00
|
|
|
default:
|
|
|
|
return nil, ErrInvalidStorageType
|
|
|
|
}
|
|
|
|
|
|
|
|
return newStorage, nil
|
|
|
|
}
|
2024-11-18 22:34:07 +01:00
|
|
|
|
2024-11-18 23:00:58 +01:00
|
|
|
// Open opens a file within the Storage interface.
|
|
|
|
// It takes a file name as its parameter.
|
|
|
|
// The function returns a File and an error, if any.
|
2024-11-18 22:34:07 +01:00
|
|
|
func (s *Storage) Open(name string) (*File, error) {
|
|
|
|
returnFile := new(File)
|
|
|
|
returnFile.storage = s
|
|
|
|
|
|
|
|
if s.s3Client != nil {
|
|
|
|
object, err := s.s3Client.GetObject(s.ctx, s.config.S3BucketName, name, minio.GetObjectOptions{})
|
|
|
|
if err != nil {
|
|
|
|
errResp := minio.ToErrorResponse(err)
|
|
|
|
if errResp.StatusCode == http.StatusNotFound {
|
|
|
|
return nil, ErrFileNotFound
|
|
|
|
} else {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
objectStat, err := object.Stat()
|
|
|
|
if err != nil {
|
|
|
|
errResp := minio.ToErrorResponse(err)
|
|
|
|
if errResp.StatusCode == http.StatusNotFound {
|
|
|
|
return nil, ErrFileNotFound
|
|
|
|
} else {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
returnFile.Name = objectStat.Key
|
|
|
|
returnFile.file = object
|
|
|
|
} else {
|
|
|
|
if _, err := os.Stat(filepath.Join(s.config.Path, name)); err != nil && os.IsNotExist(err) {
|
|
|
|
return nil, ErrFileNotFound
|
|
|
|
} else if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
file, err := os.Open(filepath.Join(s.config.Path, name))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
returnFile.file = file
|
|
|
|
|
|
|
|
cutStr, _ := strings.CutPrefix(file.Name(), filepath.Clean(s.config.Path)+"/")
|
|
|
|
|
|
|
|
returnFile.Name = cutStr
|
|
|
|
}
|
|
|
|
|
|
|
|
return returnFile, nil
|
|
|
|
}
|
|
|
|
|
2024-11-18 23:00:58 +01:00
|
|
|
// Read opens a file within the Storage interfaces and reads the contents.
|
|
|
|
// It takes a file name as its parameter.
|
|
|
|
// The function returns a []byte, or an error, if any.
|
2024-11-18 22:34:07 +01:00
|
|
|
func (s *Storage) Read(name string) ([]byte, error) {
|
|
|
|
if s.s3Client != nil {
|
|
|
|
object, err := s.s3Client.GetObject(s.ctx, s.config.S3BucketName, name, minio.GetObjectOptions{})
|
|
|
|
if err != nil {
|
|
|
|
errResp := minio.ToErrorResponse(err)
|
|
|
|
if errResp.StatusCode == http.StatusNotFound {
|
|
|
|
return nil, ErrFileNotFound
|
|
|
|
} else {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-18 22:39:35 +01:00
|
|
|
objectBytes, err := io.ReadAll(object)
|
|
|
|
if err != nil {
|
2024-11-18 22:34:07 +01:00
|
|
|
return nil, err
|
|
|
|
}
|
2024-11-18 22:39:35 +01:00
|
|
|
|
2024-11-18 22:34:07 +01:00
|
|
|
return objectBytes, nil
|
|
|
|
} else {
|
|
|
|
return os.ReadFile(filepath.Join(s.config.Path, name))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type WriteOptions struct {
|
|
|
|
S3Tags map[string]string
|
|
|
|
}
|
|
|
|
|
2024-11-18 23:00:58 +01:00
|
|
|
// Write writes to a file within the Storage interface.
|
|
|
|
// It takes a name, a []byte, and WriteOptions as its parameters.
|
|
|
|
// The function returns an error, if any.
|
2024-11-18 22:34:07 +01:00
|
|
|
func (s *Storage) Write(name string, data []byte, opts WriteOptions) error {
|
|
|
|
if s.s3Client != nil {
|
|
|
|
if _, err := s.s3Client.PutObject(s.ctx, s.config.S3BucketName, name, bytes.NewReader(data), int64(len(data)), minio.PutObjectOptions{
|
|
|
|
UserTags: opts.S3Tags,
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
} else {
|
2024-11-18 23:04:50 +01:00
|
|
|
return os.WriteFile(filepath.Join(s.config.Path, name), data, 0600)
|
2024-11-18 22:34:07 +01:00
|
|
|
}
|
|
|
|
}
|
2024-11-18 22:48:49 +01:00
|
|
|
|
2024-11-18 23:00:58 +01:00
|
|
|
// Delete deleted a file within the storage interface.
|
|
|
|
// It takes a file name as its parameter.
|
|
|
|
// The function returns an error, if any.
|
2024-11-18 22:48:49 +01:00
|
|
|
func (s *Storage) Delete(name string) error {
|
|
|
|
if s.s3Client != nil {
|
|
|
|
if err := s.s3Client.RemoveObject(s.ctx, s.config.S3BucketName, name, minio.RemoveObjectOptions{}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
} else {
|
|
|
|
return os.Remove(filepath.Join(s.config.Path, name))
|
|
|
|
}
|
|
|
|
}
|