filebrowser/http/preview.go

158 lines
3.9 KiB
Go
Raw Normal View History

//go:generate go-enum --sql --marshal --names --file $GOFILE
package http
import (
2020-07-28 01:01:02 +08:00
"bytes"
2020-07-23 08:41:19 +08:00
"context"
2024-04-02 00:24:06 +08:00
"errors"
"fmt"
2020-07-23 08:41:19 +08:00
"io"
"net/http"
"github.com/gorilla/mux"
"github.com/filebrowser/filebrowser/v2/files"
2020-07-23 08:41:19 +08:00
"github.com/filebrowser/filebrowser/v2/img"
)
/*
ENUM(
thumb
big
)
*/
type PreviewSize int
2020-07-23 08:41:19 +08:00
type ImgService interface {
FormatFromExtension(ext string) (img.Format, error)
2020-07-25 02:08:26 +08:00
Resize(ctx context.Context, in io.Reader, width, height int, out io.Writer, options ...img.Option) error
2020-07-23 08:41:19 +08:00
}
2020-07-28 01:01:02 +08:00
type FileCache interface {
Store(ctx context.Context, key string, value []byte) error
Load(ctx context.Context, key string) ([]byte, bool, error)
Delete(ctx context.Context, key string) error
2020-07-28 01:01:02 +08:00
}
func previewHandler(imgSvc ImgService, fileCache FileCache, enableThumbnails, resizePreview bool) handleFunc {
2020-07-23 08:41:19 +08:00
return withUser(func(w http.ResponseWriter, r *http.Request, d *data) (int, error) {
if !d.user.Perm.Download {
return http.StatusAccepted, nil
}
vars := mux.Vars(r)
previewSize, err := ParsePreviewSize(vars["size"])
if err != nil {
return http.StatusBadRequest, err
2020-07-23 08:41:19 +08:00
}
2024-04-02 00:24:06 +08:00
file, err := files.NewFileInfo(&files.FileOptions{
Fs: d.user.Fs,
Path: "/" + vars["path"],
Modify: d.user.Perm.Modify,
Expand: true,
ReadHeader: d.server.TypeDetectionByHeader,
Checker: d,
2020-07-23 08:41:19 +08:00
})
if err != nil {
return errToStatus(err), err
}
2020-07-23 08:41:19 +08:00
setContentDisposition(w, r, file)
2020-07-23 08:41:19 +08:00
switch file.Type {
case "image":
return handleImagePreview(w, r, imgSvc, fileCache, file, previewSize, enableThumbnails, resizePreview)
2020-07-23 08:41:19 +08:00
default:
return http.StatusNotImplemented, fmt.Errorf("can't create preview for %s type", file.Type)
}
})
}
func handleImagePreview(
w http.ResponseWriter,
r *http.Request,
imgSvc ImgService,
fileCache FileCache,
file *files.FileInfo,
previewSize PreviewSize,
enableThumbnails, resizePreview bool,
) (int, error) {
if (previewSize == PreviewSizeBig && !resizePreview) ||
(previewSize == PreviewSizeThumb && !enableThumbnails) {
return rawFileHandler(w, r, file)
}
2021-04-19 21:16:48 +08:00
format, err := imgSvc.FormatFromExtension(file.Extension)
2021-04-19 21:16:48 +08:00
// Unsupported extensions directly return the raw data
2024-04-02 00:24:06 +08:00
if errors.Is(err, img.ErrUnsupportedFormat) || format == img.FormatGif {
2021-04-19 21:16:48 +08:00
return rawFileHandler(w, r, file)
}
if err != nil {
return errToStatus(err), err
}
cacheKey := previewCacheKey(file, previewSize)
2021-08-06 20:31:39 +08:00
resizedImage, ok, err := fileCache.Load(r.Context(), cacheKey)
2020-07-28 01:01:02 +08:00
if err != nil {
return errToStatus(err), err
}
2021-08-06 20:31:39 +08:00
if !ok {
resizedImage, err = createPreview(imgSvc, fileCache, file, previewSize)
2021-08-06 20:31:39 +08:00
if err != nil {
return errToStatus(err), err
}
}
2021-04-19 21:16:48 +08:00
2021-08-06 20:31:39 +08:00
w.Header().Set("Cache-Control", "private")
http.ServeContent(w, r, file.Name, file.ModTime, bytes.NewReader(resizedImage))
2021-04-19 21:16:48 +08:00
return 0, nil
}
func createPreview(imgSvc ImgService, fileCache FileCache,
file *files.FileInfo, previewSize PreviewSize) ([]byte, error) {
2021-04-19 21:16:48 +08:00
fd, err := file.Fs.Open(file.Path)
if err != nil {
return nil, err
}
defer fd.Close()
2020-07-23 08:41:19 +08:00
var (
width int
height int
options []img.Option
)
switch {
case previewSize == PreviewSizeBig:
2020-07-23 08:41:19 +08:00
width = 1080
height = 1080
options = append(options, img.WithMode(img.ResizeModeFit), img.WithQuality(img.QualityMedium))
case previewSize == PreviewSizeThumb:
2022-01-16 02:20:13 +08:00
width = 256
height = 256
options = append(options, img.WithMode(img.ResizeModeFill), img.WithQuality(img.QualityLow), img.WithFormat(img.FormatJpeg))
default:
2021-04-19 21:16:48 +08:00
return nil, img.ErrUnsupportedFormat
}
2020-07-28 01:01:02 +08:00
buf := &bytes.Buffer{}
if err := imgSvc.Resize(context.Background(), fd, width, height, buf, options...); err != nil {
2021-04-19 21:16:48 +08:00
return nil, err
}
2020-07-28 01:01:02 +08:00
go func() {
cacheKey := previewCacheKey(file, previewSize)
2020-07-28 01:01:02 +08:00
if err := fileCache.Store(context.Background(), cacheKey, buf.Bytes()); err != nil {
fmt.Printf("failed to cache resized image: %v", err)
}
}()
2021-08-06 20:31:39 +08:00
return buf.Bytes(), nil
}
func previewCacheKey(f *files.FileInfo, previewSize PreviewSize) string {
return fmt.Sprintf("%x%x%x", f.RealPath(), f.ModTime.Unix(), previewSize)
}