176 lines
5.4 KiB
Go
176 lines
5.4 KiB
Go
package services
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"mime/multipart"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"git.solsynth.dev/hypernet/paperclip/pkg/internal/database"
|
|
"git.solsynth.dev/hypernet/paperclip/pkg/internal/models"
|
|
"github.com/gofiber/fiber/v2"
|
|
jsoniter "github.com/json-iterator/go"
|
|
"github.com/minio/minio-go/v7"
|
|
"github.com/minio/minio-go/v7/pkg/credentials"
|
|
"github.com/spf13/viper"
|
|
)
|
|
|
|
func UploadFileToTemporary(ctx *fiber.Ctx, file *multipart.FileHeader, meta models.Attachment) error {
|
|
destMap := viper.GetStringMap(fmt.Sprintf("destinations.%d", meta.Destination))
|
|
|
|
var dest models.BaseDestination
|
|
rawDest, _ := jsoniter.Marshal(destMap)
|
|
_ = jsoniter.Unmarshal(rawDest, &dest)
|
|
|
|
switch dest.Type {
|
|
case models.DestinationTypeLocal:
|
|
var destConfigured models.LocalDestination
|
|
_ = jsoniter.Unmarshal(rawDest, &destConfigured)
|
|
return ctx.SaveFile(file, filepath.Join(destConfigured.Path, meta.Uuid))
|
|
default:
|
|
return fmt.Errorf("invalid destination: unsupported protocol %s", dest.Type)
|
|
}
|
|
}
|
|
|
|
func UploadChunkToTemporary(ctx *fiber.Ctx, cid string, file *multipart.FileHeader, meta models.Attachment) error {
|
|
destMap := viper.GetStringMap("destinations.0")
|
|
|
|
var dest models.BaseDestination
|
|
rawDest, _ := jsoniter.Marshal(destMap)
|
|
_ = jsoniter.Unmarshal(rawDest, &dest)
|
|
|
|
switch dest.Type {
|
|
case models.DestinationTypeLocal:
|
|
var destConfigured models.LocalDestination
|
|
_ = jsoniter.Unmarshal(rawDest, &destConfigured)
|
|
tempPath := filepath.Join(destConfigured.Path, fmt.Sprintf("%s.part%s.partial", meta.Uuid, cid))
|
|
destPath := filepath.Join(destConfigured.Path, fmt.Sprintf("%s.part%s", meta.Uuid, cid))
|
|
if err := ctx.SaveFile(file, tempPath); err != nil {
|
|
return err
|
|
}
|
|
return os.Rename(tempPath, destPath)
|
|
default:
|
|
return fmt.Errorf("invalid destination: unsupported protocol %s", dest.Type)
|
|
}
|
|
}
|
|
|
|
func UploadChunkToTemporaryWithRaw(ctx *fiber.Ctx, cid string, raw []byte, meta models.Attachment) error {
|
|
destMap := viper.GetStringMap("destinations.0")
|
|
|
|
var dest models.BaseDestination
|
|
rawDest, _ := jsoniter.Marshal(destMap)
|
|
_ = jsoniter.Unmarshal(rawDest, &dest)
|
|
|
|
switch dest.Type {
|
|
case models.DestinationTypeLocal:
|
|
var destConfigured models.LocalDestination
|
|
_ = jsoniter.Unmarshal(rawDest, &destConfigured)
|
|
tempPath := filepath.Join(destConfigured.Path, fmt.Sprintf("%s.part%s.partial", meta.Uuid, cid))
|
|
destPath := filepath.Join(destConfigured.Path, fmt.Sprintf("%s.part%s", meta.Uuid, cid))
|
|
if err := os.WriteFile(tempPath, raw, 0644); err != nil {
|
|
return err
|
|
}
|
|
return os.Rename(tempPath, destPath)
|
|
default:
|
|
return fmt.Errorf("invalid destination: unsupported protocol %s", dest.Type)
|
|
}
|
|
}
|
|
|
|
func CheckChunkExistsInTemporary(meta models.Attachment, cid string) bool {
|
|
destMap := viper.GetStringMap("destinations.0")
|
|
|
|
var dest models.LocalDestination
|
|
rawDest, _ := jsoniter.Marshal(destMap)
|
|
_ = jsoniter.Unmarshal(rawDest, &dest)
|
|
|
|
path := filepath.Join(dest.Path, fmt.Sprintf("%s.part%s", meta.Uuid, cid))
|
|
if _, err := os.Stat(path); errors.Is(err, os.ErrNotExist) {
|
|
return false
|
|
} else {
|
|
return true
|
|
}
|
|
}
|
|
|
|
func ReUploadFileToPermanent(meta models.Attachment, dst int) error {
|
|
if dst == models.AttachmentDstTemporary || meta.Destination != dst {
|
|
return fmt.Errorf("destnation cannot be reversed temporary or the same as the original")
|
|
}
|
|
if meta.Destination != models.AttachmentDstTemporary {
|
|
return fmt.Errorf("attachment isn't in temporary storage, unable to process")
|
|
}
|
|
|
|
meta.Destination = 0
|
|
|
|
destMap := viper.GetStringMap("destinations.permanent")
|
|
|
|
var dest models.BaseDestination
|
|
rawDest, _ := jsoniter.Marshal(destMap)
|
|
_ = jsoniter.Unmarshal(rawDest, &dest)
|
|
|
|
prevDestMap := viper.GetStringMap("destinations.0")
|
|
|
|
// Currently, the temporary destination only supports the local.
|
|
// So we can do this.
|
|
var prevDest models.LocalDestination
|
|
prevRawDest, _ := jsoniter.Marshal(prevDestMap)
|
|
_ = jsoniter.Unmarshal(prevRawDest, &prevDest)
|
|
|
|
inDst := filepath.Join(prevDest.Path, meta.Uuid)
|
|
|
|
switch dest.Type {
|
|
case models.DestinationTypeLocal:
|
|
var destConfigured models.LocalDestination
|
|
_ = jsoniter.Unmarshal(rawDest, &destConfigured)
|
|
|
|
in, err := os.Open(inDst)
|
|
if err != nil {
|
|
return fmt.Errorf("unable to open file in temporary storage: %v", err)
|
|
}
|
|
defer in.Close()
|
|
|
|
out, err := os.Create(filepath.Join(destConfigured.Path, meta.Uuid))
|
|
if err != nil {
|
|
return fmt.Errorf("unable to open dest file: %v", err)
|
|
}
|
|
defer out.Close()
|
|
|
|
_, err = io.Copy(out, in)
|
|
if err != nil {
|
|
return fmt.Errorf("unable to copy data to dest file: %v", err)
|
|
}
|
|
|
|
database.C.Save(&meta)
|
|
CacheAttachment(meta)
|
|
return nil
|
|
case models.DestinationTypeS3:
|
|
var destConfigured models.S3Destination
|
|
_ = jsoniter.Unmarshal(rawDest, &destConfigured)
|
|
|
|
client, err := minio.New(destConfigured.Endpoint, &minio.Options{
|
|
Creds: credentials.NewStaticV4(destConfigured.SecretID, destConfigured.SecretKey, ""),
|
|
Secure: destConfigured.EnableSSL,
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("unable to configure s3 client: %v", err)
|
|
}
|
|
|
|
_, err = client.FPutObject(context.Background(), destConfigured.Bucket, filepath.Join(destConfigured.Path, meta.Uuid), inDst, minio.PutObjectOptions{
|
|
ContentType: meta.MimeType,
|
|
SendContentMd5: false,
|
|
DisableContentSha256: true,
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("unable to upload file to s3: %v", err)
|
|
}
|
|
|
|
database.C.Save(&meta)
|
|
CacheAttachment(meta)
|
|
return nil
|
|
default:
|
|
return fmt.Errorf("invalid destination: unsupported protocol %s", dest.Type)
|
|
}
|
|
}
|