2022-07-07 00:19:05 +03:00
|
|
|
package filesystem
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
2022-08-17 22:29:11 +03:00
|
|
|
"image"
|
2022-07-07 00:19:05 +03:00
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2022-08-17 22:29:11 +03:00
|
|
|
"regexp"
|
2022-07-07 00:19:05 +03:00
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
|
|
"github.com/aws/aws-sdk-go/aws/credentials"
|
|
|
|
"github.com/aws/aws-sdk-go/aws/session"
|
|
|
|
"github.com/disintegration/imaging"
|
2022-09-05 15:46:40 +03:00
|
|
|
"github.com/gabriel-vasile/mimetype"
|
2022-07-21 12:56:17 +03:00
|
|
|
"github.com/pocketbase/pocketbase/tools/list"
|
2022-07-07 00:19:05 +03:00
|
|
|
"gocloud.dev/blob"
|
|
|
|
"gocloud.dev/blob/fileblob"
|
|
|
|
"gocloud.dev/blob/s3blob"
|
|
|
|
)
|
|
|
|
|
|
|
|
type System struct {
|
|
|
|
ctx context.Context
|
|
|
|
bucket *blob.Bucket
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewS3 initializes an S3 filesystem instance.
|
|
|
|
//
|
|
|
|
// NB! Make sure to call `Close()` after you are done working with it.
|
|
|
|
func NewS3(
|
|
|
|
bucketName string,
|
|
|
|
region string,
|
|
|
|
endpoint string,
|
|
|
|
accessKey string,
|
|
|
|
secretKey string,
|
2022-07-19 10:45:38 +03:00
|
|
|
s3ForcePathStyle bool,
|
2022-07-07 00:19:05 +03:00
|
|
|
) (*System, error) {
|
|
|
|
ctx := context.Background() // default context
|
|
|
|
|
|
|
|
cred := credentials.NewStaticCredentials(accessKey, secretKey, "")
|
|
|
|
|
|
|
|
sess, err := session.NewSession(&aws.Config{
|
2022-07-19 10:45:38 +03:00
|
|
|
Region: aws.String(region),
|
|
|
|
Endpoint: aws.String(endpoint),
|
|
|
|
Credentials: cred,
|
|
|
|
S3ForcePathStyle: aws.Bool(s3ForcePathStyle),
|
2022-07-07 00:19:05 +03:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
bucket, err := s3blob.OpenBucket(ctx, sess, bucketName, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &System{ctx: ctx, bucket: bucket}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewLocal initializes a new local filesystem instance.
|
|
|
|
//
|
|
|
|
// NB! Make sure to call `Close()` after you are done working with it.
|
|
|
|
func NewLocal(dirPath string) (*System, error) {
|
|
|
|
ctx := context.Background() // default context
|
|
|
|
|
|
|
|
// makes sure that the directory exist
|
|
|
|
if err := os.MkdirAll(dirPath, os.ModePerm); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
bucket, err := fileblob.OpenBucket(dirPath, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &System{ctx: ctx, bucket: bucket}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close releases any resources used for the related filesystem.
|
|
|
|
func (s *System) Close() error {
|
|
|
|
return s.bucket.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exists checks if file with fileKey path exists or not.
|
|
|
|
func (s *System) Exists(fileKey string) (bool, error) {
|
|
|
|
return s.bucket.Exists(s.ctx, fileKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attributes returns the attributes for the file with fileKey path.
|
|
|
|
func (s *System) Attributes(fileKey string) (*blob.Attributes, error) {
|
|
|
|
return s.bucket.Attributes(s.ctx, fileKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Upload writes content into the fileKey location.
|
|
|
|
func (s *System) Upload(content []byte, fileKey string) error {
|
2022-09-05 15:46:40 +03:00
|
|
|
opts := &blob.WriterOptions{
|
|
|
|
ContentType: mimetype.Detect(content).String(),
|
|
|
|
}
|
|
|
|
|
|
|
|
w, writerErr := s.bucket.NewWriter(s.ctx, fileKey, opts)
|
2022-07-07 00:19:05 +03:00
|
|
|
if writerErr != nil {
|
|
|
|
return writerErr
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := w.Write(content); err != nil {
|
|
|
|
w.Close()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return w.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete deletes stored file at fileKey location.
|
|
|
|
func (s *System) Delete(fileKey string) error {
|
|
|
|
return s.bucket.Delete(s.ctx, fileKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeletePrefix deletes everything starting with the specified prefix.
|
|
|
|
func (s *System) DeletePrefix(prefix string) []error {
|
|
|
|
failed := []error{}
|
|
|
|
|
|
|
|
if prefix == "" {
|
|
|
|
failed = append(failed, errors.New("Prefix mustn't be empty."))
|
|
|
|
return failed
|
|
|
|
}
|
|
|
|
|
|
|
|
dirsMap := map[string]struct{}{}
|
|
|
|
dirsMap[prefix] = struct{}{}
|
|
|
|
|
|
|
|
opts := blob.ListOptions{
|
|
|
|
Prefix: prefix,
|
|
|
|
}
|
|
|
|
|
2022-07-09 22:17:41 +08:00
|
|
|
// delete all files with the prefix
|
2022-07-07 00:19:05 +03:00
|
|
|
// ---
|
|
|
|
iter := s.bucket.List(&opts)
|
|
|
|
for {
|
|
|
|
obj, err := iter.Next(s.ctx)
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
failed = append(failed, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.Delete(obj.Key); err != nil {
|
|
|
|
failed = append(failed, err)
|
|
|
|
} else {
|
|
|
|
dirsMap[filepath.Dir(obj.Key)] = struct{}{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// ---
|
|
|
|
|
|
|
|
// try to delete the empty remaining dir objects
|
|
|
|
// (this operation usually is optional and there is no need to strictly check the result)
|
|
|
|
// ---
|
|
|
|
// fill dirs slice
|
2022-07-18 01:03:09 +03:00
|
|
|
dirs := make([]string, 0, len(dirsMap))
|
2022-07-07 00:19:05 +03:00
|
|
|
for d := range dirsMap {
|
|
|
|
dirs = append(dirs, d)
|
|
|
|
}
|
|
|
|
|
|
|
|
// sort the child dirs first, aka. ["a/b/c", "a/b", "a"]
|
|
|
|
sort.SliceStable(dirs, func(i, j int) bool {
|
|
|
|
return len(strings.Split(dirs[i], "/")) > len(strings.Split(dirs[j], "/"))
|
|
|
|
})
|
|
|
|
|
|
|
|
// delete dirs
|
|
|
|
for _, d := range dirs {
|
|
|
|
if d != "" {
|
|
|
|
s.Delete(d)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// ---
|
|
|
|
|
|
|
|
return failed
|
|
|
|
}
|
|
|
|
|
2022-07-21 12:56:17 +03:00
|
|
|
var inlineServeContentTypes = []string{
|
2022-08-11 20:09:26 +03:00
|
|
|
// image
|
|
|
|
"image/png", "image/jpg", "image/jpeg", "image/gif", "image/webp", "image/x-icon", "image/bmp",
|
|
|
|
// video
|
|
|
|
"video/webm", "video/mp4", "video/3gpp", "video/quicktime", "video/x-ms-wmv",
|
|
|
|
// audio
|
2022-09-05 15:46:40 +03:00
|
|
|
"audio/basic", "audio/aiff", "audio/mpeg", "audio/midi", "audio/mp3", "audio/wave",
|
|
|
|
"audio/wav", "audio/x-wav", "audio/x-mpeg", "audio/x-m4a", "audio/aac",
|
2022-08-11 20:09:26 +03:00
|
|
|
// document
|
2022-09-05 15:46:40 +03:00
|
|
|
"application/pdf", "application/x-pdf",
|
2022-07-21 12:56:17 +03:00
|
|
|
}
|
|
|
|
|
2022-07-07 00:19:05 +03:00
|
|
|
// Serve serves the file at fileKey location to an HTTP response.
|
|
|
|
func (s *System) Serve(response http.ResponseWriter, fileKey string, name string) error {
|
|
|
|
r, readErr := s.bucket.NewReader(s.ctx, fileKey, nil)
|
|
|
|
if readErr != nil {
|
|
|
|
return readErr
|
|
|
|
}
|
|
|
|
defer r.Close()
|
|
|
|
|
2022-07-21 12:56:17 +03:00
|
|
|
disposition := "attachment"
|
|
|
|
realContentType := r.ContentType()
|
|
|
|
if list.ExistInSlice(realContentType, inlineServeContentTypes) {
|
|
|
|
disposition = "inline"
|
|
|
|
}
|
|
|
|
|
2022-09-05 15:46:40 +03:00
|
|
|
// make an exception for svg and force a custom content type
|
|
|
|
// to send in the response so that it can be loaded in an img tag
|
2022-07-21 12:56:17 +03:00
|
|
|
// (see https://github.com/whatwg/mimesniff/issues/7)
|
|
|
|
extContentType := realContentType
|
2022-09-05 15:46:40 +03:00
|
|
|
if extContentType != "image/svg+xml" && filepath.Ext(name) == ".svg" {
|
2022-07-21 12:56:17 +03:00
|
|
|
extContentType = "image/svg+xml"
|
|
|
|
}
|
|
|
|
|
|
|
|
response.Header().Set("Content-Disposition", disposition+"; filename="+name)
|
|
|
|
response.Header().Set("Content-Type", extContentType)
|
2022-07-07 00:19:05 +03:00
|
|
|
response.Header().Set("Content-Length", strconv.FormatInt(r.Size(), 10))
|
2022-07-21 12:56:17 +03:00
|
|
|
response.Header().Set("Content-Security-Policy", "default-src 'none'; style-src 'unsafe-inline'; sandbox")
|
2022-07-10 20:53:24 +03:00
|
|
|
|
|
|
|
// All HTTP date/time stamps MUST be represented in Greenwich Mean Time (GMT)
|
|
|
|
// (see https://www.w3.org/Protocols/rfc2616/rfc2616-sec3.html#sec3.3.1)
|
|
|
|
//
|
|
|
|
// NB! time.LoadLocation may fail on non-Unix systems (see https://github.com/pocketbase/pocketbase/issues/45)
|
|
|
|
location, locationErr := time.LoadLocation("GMT")
|
|
|
|
if locationErr == nil {
|
|
|
|
response.Header().Set("Last-Modified", r.ModTime().In(location).Format("Mon, 02 Jan 06 15:04:05 MST"))
|
|
|
|
}
|
2022-07-07 00:19:05 +03:00
|
|
|
|
|
|
|
// copy from the read range to response.
|
|
|
|
_, err := io.Copy(response, r)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-08-17 22:29:11 +03:00
|
|
|
var ThumbSizeRegex = regexp.MustCompile(`^(\d+)x(\d+)(t|b|f)?$`)
|
|
|
|
|
2022-07-07 00:19:05 +03:00
|
|
|
// CreateThumb creates a new thumb image for the file at originalKey location.
|
|
|
|
// The new thumb file is stored at thumbKey location.
|
|
|
|
//
|
2022-08-17 22:29:11 +03:00
|
|
|
// thumbSize is in the format:
|
|
|
|
// - 0xH (eg. 0x100) - resize to H height preserving the aspect ratio
|
|
|
|
// - Wx0 (eg. 300x0) - resize to W width preserving the aspect ratio
|
|
|
|
// - WxH (eg. 300x100) - resize and crop to WxH viewbox (from center)
|
|
|
|
// - WxHt (eg. 300x100t) - resize and crop to WxH viewbox (from top)
|
|
|
|
// - WxHb (eg. 300x100b) - resize and crop to WxH viewbox (from bottom)
|
|
|
|
// - WxHf (eg. 300x100f) - fit inside a WxH viewbox (without cropping)
|
|
|
|
func (s *System) CreateThumb(originalKey string, thumbKey, thumbSize string) error {
|
|
|
|
sizeParts := ThumbSizeRegex.FindStringSubmatch(thumbSize)
|
|
|
|
if len(sizeParts) != 4 {
|
|
|
|
return errors.New("Thumb size must be in WxH, WxHt, WxHb or WxHf format.")
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
|
2022-08-17 22:29:11 +03:00
|
|
|
width, _ := strconv.Atoi(sizeParts[1])
|
|
|
|
height, _ := strconv.Atoi(sizeParts[2])
|
|
|
|
resizeType := sizeParts[3]
|
|
|
|
|
|
|
|
if width == 0 && height == 0 {
|
|
|
|
return errors.New("Thumb width and height cannot be zero at the same time.")
|
|
|
|
}
|
2022-07-07 00:19:05 +03:00
|
|
|
|
|
|
|
// fetch the original
|
|
|
|
r, readErr := s.bucket.NewReader(s.ctx, originalKey, nil)
|
|
|
|
if readErr != nil {
|
|
|
|
return readErr
|
|
|
|
}
|
|
|
|
defer r.Close()
|
|
|
|
|
|
|
|
// create imaging object from the origial reader
|
|
|
|
img, decodeErr := imaging.Decode(r)
|
|
|
|
if decodeErr != nil {
|
|
|
|
return decodeErr
|
|
|
|
}
|
|
|
|
|
2022-08-17 22:29:11 +03:00
|
|
|
var thumbImg *image.NRGBA
|
|
|
|
|
|
|
|
if width == 0 || height == 0 {
|
|
|
|
// force resize preserving aspect ratio
|
|
|
|
thumbImg = imaging.Resize(img, width, height, imaging.CatmullRom)
|
|
|
|
} else {
|
|
|
|
switch resizeType {
|
|
|
|
case "f":
|
|
|
|
// fit
|
|
|
|
thumbImg = imaging.Fit(img, width, height, imaging.CatmullRom)
|
|
|
|
case "t":
|
|
|
|
// fill and crop from top
|
|
|
|
thumbImg = imaging.Fill(img, width, height, imaging.Top, imaging.CatmullRom)
|
|
|
|
case "b":
|
|
|
|
// fill and crop from bottom
|
|
|
|
thumbImg = imaging.Fill(img, width, height, imaging.Bottom, imaging.CatmullRom)
|
|
|
|
default:
|
|
|
|
// fill and crop from center
|
|
|
|
thumbImg = imaging.Fill(img, width, height, imaging.Center, imaging.CatmullRom)
|
|
|
|
}
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// open a thumb storage writer (aka. prepare for upload)
|
|
|
|
w, writerErr := s.bucket.NewWriter(s.ctx, thumbKey, nil)
|
|
|
|
if writerErr != nil {
|
|
|
|
return writerErr
|
|
|
|
}
|
|
|
|
|
|
|
|
// thumb encode (aka. upload)
|
|
|
|
if err := imaging.Encode(w, thumbImg, imaging.PNG); err != nil {
|
|
|
|
w.Close()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for close errors to ensure that the thumb was really saved
|
|
|
|
return w.Close()
|
|
|
|
}
|