s3browser-backend/internal/mutations.go

192 lines
4.0 KiB
Go
Raw Normal View History

2021-08-06 23:19:36 +00:00
package s3browser
import (
"context"
"fmt"
2021-08-15 23:40:01 +00:00
"strings"
2021-08-06 23:19:36 +00:00
2021-08-20 19:38:22 +00:00
"github.com/graph-gophers/dataloader"
2021-08-06 23:19:36 +00:00
"github.com/minio/minio-go/v7"
)
func deleteMutation(ctx context.Context, id string) error {
s3Client, ok := ctx.Value("s3Client").(*minio.Client)
if !ok {
return fmt.Errorf("Failed to get s3Client from context")
}
// TODO: it is posible to remove multiple objects with a single call.
// Is it better to batch this?
2021-08-06 23:42:47 +00:00
err := s3Client.RemoveObject(ctx, bucketName, id, minio.RemoveObjectOptions{})
if err != nil {
return err
}
// Invalidate cache
2021-08-14 12:13:49 +00:00
return invalidateCache(ctx, id)
2021-08-06 23:19:36 +00:00
}
func copyMutation(ctx context.Context, src, dest string) (*File, error) {
s3Client, ok := ctx.Value("s3Client").(*minio.Client)
if !ok {
return nil, fmt.Errorf("Failed to get s3Client from context")
}
2021-08-15 23:40:01 +00:00
// Check if dest is a file or a dir
if strings.HasSuffix(dest, "/") {
// create new dest id
// TODO: What if a file with this id already exists?
dest += getFilenameFromID(src)
}
2021-08-06 23:19:36 +00:00
info, err := s3Client.CopyObject(ctx, minio.CopyDestOptions{
Bucket: bucketName,
Object: dest,
}, minio.CopySrcOptions{
Bucket: bucketName,
Object: src,
})
if err != nil {
return nil, err
}
2021-08-06 23:42:47 +00:00
// Invalidate cache
2021-08-14 12:13:49 +00:00
// TODO: check error
invalidateCache(ctx, info.Key)
2021-08-06 23:42:47 +00:00
2021-08-06 23:19:36 +00:00
return &File{
ID: info.Key,
}, nil
}
func moveMutation(ctx context.Context, src, dest string) (*File, error) {
s3Client, ok := ctx.Value("s3Client").(*minio.Client)
if !ok {
return nil, fmt.Errorf("Failed to get s3Client from context")
}
2021-08-16 00:04:44 +00:00
// Check if dest is a file or a dir
if strings.HasSuffix(dest, "/") {
// create new dest id
// TODO: What if a file with this id already exists?
dest += getFilenameFromID(src)
}
2021-08-06 23:19:36 +00:00
// There is no (spoon) move. Only copy and delete
info, err := s3Client.CopyObject(ctx, minio.CopyDestOptions{
Bucket: bucketName,
Object: dest,
}, minio.CopySrcOptions{
Bucket: bucketName,
Object: src,
})
if err != nil {
return nil, err
}
err = deleteMutation(ctx, src)
if err != nil {
return nil, err
}
2021-08-16 12:58:03 +00:00
invalidateCache(ctx, info.Key)
2021-08-06 23:19:36 +00:00
return &File{
ID: info.Key,
}, nil
}
2021-08-16 17:57:48 +00:00
func createDirectory(ctx context.Context, path string) (*Directory, error) {
s3Client, ok := ctx.Value("s3Client").(*minio.Client)
if !ok {
return nil, fmt.Errorf("Failed to get s3Client from context")
}
if !strings.HasSuffix(path, "/") {
path += "/"
}
info, err := s3Client.PutObject(ctx, bucketName, path, strings.NewReader(""), 0, minio.PutObjectOptions{
ContentType: "application/x-directory",
})
if err != nil {
return nil, err
}
2021-08-16 20:16:42 +00:00
// Invalidate cache
// TODO: check error
invalidateCacheForDir(ctx, info.Key)
2021-08-16 17:57:48 +00:00
return &Directory{
ID: info.Key,
}, nil
}
2021-08-20 19:38:22 +00:00
func deleteDirectory(ctx context.Context, path string) error {
s3Client, ok := ctx.Value("s3Client").(*minio.Client)
if !ok {
return fmt.Errorf("Failed to get s3Client from context")
}
loader, ok := ctx.Value("loader").(map[string]*dataloader.Loader)
if !ok {
return fmt.Errorf("Failed to get dataloader from context")
}
if !strings.HasSuffix(path, "/") {
path += "/"
}
// Get all files inside the directory
2021-08-27 20:03:19 +00:00
thunk := loader["listObjectsRecursive"].Load(ctx, dataloader.StringKey(path))
2021-08-20 19:38:22 +00:00
result, err := thunk()
if err != nil {
return err
}
files, ok := result.([]minio.ObjectInfo)
if !ok {
return fmt.Errorf("Failed to get parse result from listObjects")
}
// Delete all child files
err = deleteMultiple(ctx, *s3Client, files)
if err != nil {
return err
}
// If the dir had no children it exists as an object (object with "/" at the end).
// If it exists as an object and had children it will get delete once the last child has been deleted
// If it had no children we have to delete it manualy
// This is at least the behavior when working with minio as s3 backend
// TODO: check if this is normal behavior when working with s3
if len(files) == 0 {
err := s3Client.RemoveObject(ctx, bucketName, path, minio.RemoveObjectOptions{})
if err != nil {
return err
}
}
//Invalidate cache
invalidateCacheForDir(ctx, path)
return nil
}