mirror of
https://github.com/datarhei/core.git
synced 2025-09-26 20:11:29 +08:00
Add S3 storage support
This commit is contained in:
@@ -6,3 +6,10 @@ type FileInfo struct {
|
||||
Size int64 `json:"size_bytes" jsonschema:"minimum=0" format:"int64"`
|
||||
LastMod int64 `json:"last_modified" jsonschema:"minimum=0" format:"int64"`
|
||||
}
|
||||
|
||||
// FilesystemInfo represents information about a filesystem
|
||||
type FilesystemInfo struct {
|
||||
Name string `json:"name"`
|
||||
Type string `json:"type"`
|
||||
Mount string `json:"mount"`
|
||||
}
|
||||
|
25
http/fs/fs.go
Normal file
25
http/fs/fs.go
Normal file
@@ -0,0 +1,25 @@
|
||||
package fs
|
||||
|
||||
import (
|
||||
"github.com/datarhei/core/v16/http/cache"
|
||||
"github.com/datarhei/core/v16/io/fs"
|
||||
)
|
||||
|
||||
type FS struct {
|
||||
Name string
|
||||
Mountpoint string
|
||||
|
||||
AllowWrite bool
|
||||
|
||||
EnableAuth bool
|
||||
Username string
|
||||
Password string
|
||||
|
||||
DefaultFile string
|
||||
DefaultContentType string
|
||||
Gzip bool
|
||||
|
||||
Filesystem fs.Filesystem
|
||||
|
||||
Cache cache.Cacher
|
||||
}
|
@@ -1,215 +0,0 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
"path/filepath"
|
||||
"sort"
|
||||
|
||||
"github.com/datarhei/core/v16/http/api"
|
||||
"github.com/datarhei/core/v16/http/cache"
|
||||
"github.com/datarhei/core/v16/http/handler"
|
||||
"github.com/datarhei/core/v16/http/handler/util"
|
||||
"github.com/datarhei/core/v16/io/fs"
|
||||
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
// The DiskFSHandler type provides handlers for manipulating a filesystem
|
||||
type DiskFSHandler struct {
|
||||
cache cache.Cacher
|
||||
filesystem fs.Filesystem
|
||||
handler *handler.DiskFSHandler
|
||||
}
|
||||
|
||||
// NewDiskFS return a new DiskFS type. You have to provide a filesystem to act on and optionally
|
||||
// a Cacher where files will be purged from if the Cacher is related to the filesystem.
|
||||
func NewDiskFS(fs fs.Filesystem, cache cache.Cacher) *DiskFSHandler {
|
||||
return &DiskFSHandler{
|
||||
cache: cache,
|
||||
filesystem: fs,
|
||||
handler: handler.NewDiskFS(fs, cache),
|
||||
}
|
||||
}
|
||||
|
||||
// GetFile returns the file at the given path
|
||||
// @Summary Fetch a file from the filesystem
|
||||
// @Description Fetch a file from the filesystem. The contents of that file are returned.
|
||||
// @Tags v16.7.2
|
||||
// @ID diskfs-3-get-file
|
||||
// @Produce application/data
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {file} byte
|
||||
// @Success 301 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/disk/{path} [get]
|
||||
func (h *DiskFSHandler) GetFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
mimeType := c.Response().Header().Get(echo.HeaderContentType)
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
file := h.filesystem.Open(path)
|
||||
if file == nil {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
stat, _ := file.Stat()
|
||||
|
||||
if stat.IsDir() {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
defer file.Close()
|
||||
|
||||
c.Response().Header().Set("Last-Modified", stat.ModTime().UTC().Format("Mon, 02 Jan 2006 15:04:05 GMT"))
|
||||
|
||||
if path, ok := stat.IsLink(); ok {
|
||||
path = filepath.Clean("/" + path)
|
||||
|
||||
if path[0] == '/' {
|
||||
path = path[1:]
|
||||
}
|
||||
|
||||
return c.Redirect(http.StatusMovedPermanently, path)
|
||||
}
|
||||
|
||||
c.Response().Header().Set(echo.HeaderContentType, mimeType)
|
||||
|
||||
if c.Request().Method == "HEAD" {
|
||||
return c.Blob(http.StatusOK, "application/data", nil)
|
||||
}
|
||||
|
||||
return c.Stream(http.StatusOK, "application/data", file)
|
||||
}
|
||||
|
||||
// PutFile adds or overwrites a file at the given path
|
||||
// @Summary Add a file to the filesystem
|
||||
// @Description Writes or overwrites a file on the filesystem
|
||||
// @Tags v16.7.2
|
||||
// @ID diskfs-3-put-file
|
||||
// @Accept application/data
|
||||
// @Produce text/plain
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Param data body []byte true "File data"
|
||||
// @Success 201 {string} string
|
||||
// @Success 204 {string} string
|
||||
// @Failure 507 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/disk/{path} [put]
|
||||
func (h *DiskFSHandler) PutFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
req := c.Request()
|
||||
|
||||
_, created, err := h.filesystem.Store(path, req.Body)
|
||||
if err != nil {
|
||||
return api.Err(http.StatusBadRequest, "%s", err)
|
||||
}
|
||||
|
||||
if h.cache != nil {
|
||||
h.cache.Delete(path)
|
||||
}
|
||||
|
||||
c.Response().Header().Set("Content-Location", req.URL.RequestURI())
|
||||
|
||||
if created {
|
||||
return c.String(http.StatusCreated, path)
|
||||
}
|
||||
|
||||
return c.NoContent(http.StatusNoContent)
|
||||
}
|
||||
|
||||
// DeleteFile removes a file from the filesystem
|
||||
// @Summary Remove a file from the filesystem
|
||||
// @Description Remove a file from the filesystem
|
||||
// @Tags v16.7.2
|
||||
// @ID diskfs-3-delete-file
|
||||
// @Produce text/plain
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/disk/{path} [delete]
|
||||
func (h *DiskFSHandler) DeleteFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
size := h.filesystem.Delete(path)
|
||||
|
||||
if size < 0 {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
if h.cache != nil {
|
||||
h.cache.Delete(path)
|
||||
}
|
||||
|
||||
return c.String(http.StatusOK, "OK")
|
||||
}
|
||||
|
||||
// ListFiles lists all files on the filesystem
|
||||
// @Summary List all files on the filesystem
|
||||
// @Description List all files on the filesystem. The listing can be ordered by name, size, or date of last modification in ascending or descending order.
|
||||
// @Tags v16.7.2
|
||||
// @ID diskfs-3-list-files
|
||||
// @Produce json
|
||||
// @Param glob query string false "glob pattern for file names"
|
||||
// @Param sort query string false "none, name, size, lastmod"
|
||||
// @Param order query string false "asc, desc"
|
||||
// @Success 200 {array} api.FileInfo
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/disk [get]
|
||||
func (h *DiskFSHandler) ListFiles(c echo.Context) error {
|
||||
pattern := util.DefaultQuery(c, "glob", "")
|
||||
sortby := util.DefaultQuery(c, "sort", "none")
|
||||
order := util.DefaultQuery(c, "order", "asc")
|
||||
|
||||
files := h.filesystem.List(pattern)
|
||||
|
||||
var sortFunc func(i, j int) bool
|
||||
|
||||
switch sortby {
|
||||
case "name":
|
||||
if order == "desc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].Name() > files[j].Name() }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].Name() < files[j].Name() }
|
||||
}
|
||||
case "size":
|
||||
if order == "desc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].Size() > files[j].Size() }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].Size() < files[j].Size() }
|
||||
}
|
||||
default:
|
||||
if order == "asc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].ModTime().Before(files[j].ModTime()) }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].ModTime().After(files[j].ModTime()) }
|
||||
}
|
||||
}
|
||||
|
||||
sort.Slice(files, sortFunc)
|
||||
|
||||
fileinfos := []api.FileInfo{}
|
||||
|
||||
for _, f := range files {
|
||||
if f.IsDir() {
|
||||
continue
|
||||
}
|
||||
|
||||
fileinfos = append(fileinfos, api.FileInfo{
|
||||
Name: f.Name(),
|
||||
Size: f.Size(),
|
||||
LastMod: f.ModTime().Unix(),
|
||||
})
|
||||
}
|
||||
|
||||
return c.JSON(http.StatusOK, fileinfos)
|
||||
}
|
146
http/handler/api/filesystems.go
Normal file
146
http/handler/api/filesystems.go
Normal file
@@ -0,0 +1,146 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
|
||||
"github.com/datarhei/core/v16/http/api"
|
||||
"github.com/datarhei/core/v16/http/handler"
|
||||
"github.com/datarhei/core/v16/http/handler/util"
|
||||
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
type FSConfig struct {
|
||||
Type string
|
||||
Mountpoint string
|
||||
Handler *handler.FSHandler
|
||||
}
|
||||
|
||||
// The FSHandler type provides handlers for manipulating a filesystem
|
||||
type FSHandler struct {
|
||||
filesystems map[string]FSConfig
|
||||
}
|
||||
|
||||
// NewFS return a new FSHanlder type. You have to provide a filesystem to act on.
|
||||
func NewFS(filesystems map[string]FSConfig) *FSHandler {
|
||||
return &FSHandler{
|
||||
filesystems: filesystems,
|
||||
}
|
||||
}
|
||||
|
||||
// GetFileAPI returns the file at the given path
|
||||
// @Summary Fetch a file from a filesystem
|
||||
// @Description Fetch a file from a filesystem
|
||||
// @ID filesystem-3-get-file
|
||||
// @Produce application/data
|
||||
// @Produce json
|
||||
// @Param name path string true "Name of the filesystem"
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {file} byte
|
||||
// @Success 301 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/{name}/{path} [get]
|
||||
func (h *FSHandler) GetFile(c echo.Context) error {
|
||||
name := util.PathParam(c, "name")
|
||||
|
||||
config, ok := h.filesystems[name]
|
||||
if !ok {
|
||||
return api.Err(http.StatusNotFound, "File not found", "unknown filesystem: %s", name)
|
||||
}
|
||||
|
||||
return config.Handler.GetFile(c)
|
||||
}
|
||||
|
||||
// PutFileAPI adds or overwrites a file at the given path
|
||||
// @Summary Add a file to a filesystem
|
||||
// @Description Writes or overwrites a file on a filesystem
|
||||
// @ID filesystem-3-put-file
|
||||
// @Accept application/data
|
||||
// @Produce text/plain
|
||||
// @Produce json
|
||||
// @Param name path string true "Name of the filesystem"
|
||||
// @Param path path string true "Path to file"
|
||||
// @Param data body []byte true "File data"
|
||||
// @Success 201 {string} string
|
||||
// @Success 204 {string} string
|
||||
// @Failure 507 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/{name}/{path} [put]
|
||||
func (h *FSHandler) PutFile(c echo.Context) error {
|
||||
name := util.PathParam(c, "name")
|
||||
|
||||
config, ok := h.filesystems[name]
|
||||
if !ok {
|
||||
return api.Err(http.StatusNotFound, "File not found", "unknown filesystem: %s", name)
|
||||
}
|
||||
|
||||
return config.Handler.PutFile(c)
|
||||
}
|
||||
|
||||
// DeleteFileAPI removes a file from a filesystem
|
||||
// @Summary Remove a file from a filesystem
|
||||
// @Description Remove a file from a filesystem
|
||||
// @ID filesystem-3-delete-file
|
||||
// @Produce text/plain
|
||||
// @Param name path string true "Name of the filesystem"
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/{name}/{path} [delete]
|
||||
func (h *FSHandler) DeleteFile(c echo.Context) error {
|
||||
name := util.PathParam(c, "name")
|
||||
|
||||
config, ok := h.filesystems[name]
|
||||
if !ok {
|
||||
return api.Err(http.StatusNotFound, "File not found", "unknown filesystem: %s", name)
|
||||
}
|
||||
|
||||
return config.Handler.DeleteFile(c)
|
||||
}
|
||||
|
||||
// ListFiles lists all files on a filesystem
|
||||
// @Summary List all files on a filesystem
|
||||
// @Description List all files on a filesystem. The listing can be ordered by name, size, or date of last modification in ascending or descending order.
|
||||
// @ID filesystem-3-list-files
|
||||
// @Produce json
|
||||
// @Param name path string true "Name of the filesystem"
|
||||
// @Param glob query string false "glob pattern for file names"
|
||||
// @Param sort query string false "none, name, size, lastmod"
|
||||
// @Param order query string false "asc, desc"
|
||||
// @Success 200 {array} api.FileInfo
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/{name} [get]
|
||||
func (h *FSHandler) ListFiles(c echo.Context) error {
|
||||
name := util.PathParam(c, "name")
|
||||
|
||||
config, ok := h.filesystems[name]
|
||||
if !ok {
|
||||
return api.Err(http.StatusNotFound, "File not found", "unknown filesystem: %s", name)
|
||||
}
|
||||
|
||||
return config.Handler.ListFiles(c)
|
||||
}
|
||||
|
||||
// List lists all registered filesystems
|
||||
// @Summary List all registered filesystems
|
||||
// @Description Listall registered filesystems
|
||||
// @ID filesystem-3-list
|
||||
// @Produce json
|
||||
// @Success 200 {array} api.FilesystemInfo
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs [get]
|
||||
func (h *FSHandler) List(c echo.Context) error {
|
||||
fss := []api.FilesystemInfo{}
|
||||
|
||||
for name, config := range h.filesystems {
|
||||
fss = append(fss, api.FilesystemInfo{
|
||||
Name: name,
|
||||
Type: config.Type,
|
||||
Mount: config.Mountpoint,
|
||||
})
|
||||
}
|
||||
|
||||
return c.JSON(http.StatusOK, fss)
|
||||
}
|
@@ -1,177 +0,0 @@
|
||||
package api
|
||||
|
||||
import (
|
||||
"io"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"sort"
|
||||
|
||||
"github.com/datarhei/core/v16/http/api"
|
||||
"github.com/datarhei/core/v16/http/handler"
|
||||
"github.com/datarhei/core/v16/http/handler/util"
|
||||
"github.com/datarhei/core/v16/io/fs"
|
||||
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
// The MemFSHandler type provides handlers for manipulating a filesystem
|
||||
type MemFSHandler struct {
|
||||
filesystem fs.Filesystem
|
||||
handler *handler.MemFSHandler
|
||||
}
|
||||
|
||||
// NewMemFS return a new MemFS type. You have to provide a filesystem to act on.
|
||||
func NewMemFS(fs fs.Filesystem) *MemFSHandler {
|
||||
return &MemFSHandler{
|
||||
filesystem: fs,
|
||||
handler: handler.NewMemFS(fs),
|
||||
}
|
||||
}
|
||||
|
||||
// GetFileAPI returns the file at the given path
|
||||
// @Summary Fetch a file from the memory filesystem
|
||||
// @Description Fetch a file from the memory filesystem
|
||||
// @Tags v16.7.2
|
||||
// @ID memfs-3-get-file
|
||||
// @Produce application/data
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {file} byte
|
||||
// @Success 301 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/mem/{path} [get]
|
||||
func (h *MemFSHandler) GetFile(c echo.Context) error {
|
||||
return h.handler.GetFile(c)
|
||||
}
|
||||
|
||||
// PutFileAPI adds or overwrites a file at the given path
|
||||
// @Summary Add a file to the memory filesystem
|
||||
// @Description Writes or overwrites a file on the memory filesystem
|
||||
// @Tags v16.7.2
|
||||
// @ID memfs-3-put-file
|
||||
// @Accept application/data
|
||||
// @Produce text/plain
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Param data body []byte true "File data"
|
||||
// @Success 201 {string} string
|
||||
// @Success 204 {string} string
|
||||
// @Failure 507 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/mem/{path} [put]
|
||||
func (h *MemFSHandler) PutFile(c echo.Context) error {
|
||||
return h.handler.PutFile(c)
|
||||
}
|
||||
|
||||
// DeleteFileAPI removes a file from the filesystem
|
||||
// @Summary Remove a file from the memory filesystem
|
||||
// @Description Remove a file from the memory filesystem
|
||||
// @Tags v16.7.2
|
||||
// @ID memfs-3-delete-file
|
||||
// @Produce text/plain
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/mem/{path} [delete]
|
||||
func (h *MemFSHandler) DeleteFile(c echo.Context) error {
|
||||
return h.handler.DeleteFile(c)
|
||||
}
|
||||
|
||||
// PatchFile creates a symbolic link to a file in the filesystem
|
||||
// @Summary Create a link to a file in the memory filesystem
|
||||
// @Description Create a link to a file in the memory filesystem. The file linked to has to exist.
|
||||
// @Tags v16.7.2
|
||||
// @ID memfs-3-patch
|
||||
// @Accept application/data
|
||||
// @Produce text/plain
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Param url body string true "Path to the file to link to"
|
||||
// @Success 201 {string} string
|
||||
// @Failure 400 {object} api.Error
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/mem/{path} [patch]
|
||||
func (h *MemFSHandler) PatchFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
req := c.Request()
|
||||
|
||||
body, err := io.ReadAll(req.Body)
|
||||
if err != nil {
|
||||
return api.Err(http.StatusBadRequest, "Failed reading request body", "%s", err)
|
||||
}
|
||||
|
||||
u, err := url.Parse(string(body))
|
||||
if err != nil {
|
||||
return api.Err(http.StatusBadRequest, "Body doesn't contain a valid path", "%s", err)
|
||||
}
|
||||
|
||||
if err := h.filesystem.Symlink(u.Path, path); err != nil {
|
||||
return api.Err(http.StatusBadRequest, "Failed to create symlink", "%s", err)
|
||||
}
|
||||
|
||||
c.Response().Header().Set("Content-Location", req.URL.RequestURI())
|
||||
|
||||
return c.String(http.StatusCreated, "")
|
||||
}
|
||||
|
||||
// ListFiles lists all files on the filesystem
|
||||
// @Summary List all files on the memory filesystem
|
||||
// @Description List all files on the memory filesystem. The listing can be ordered by name, size, or date of last modification in ascending or descending order.
|
||||
// @Tags v16.7.2
|
||||
// @ID memfs-3-list-files
|
||||
// @Produce json
|
||||
// @Param glob query string false "glob pattern for file names"
|
||||
// @Param sort query string false "none, name, size, lastmod"
|
||||
// @Param order query string false "asc, desc"
|
||||
// @Success 200 {array} api.FileInfo
|
||||
// @Security ApiKeyAuth
|
||||
// @Router /api/v3/fs/mem [get]
|
||||
func (h *MemFSHandler) ListFiles(c echo.Context) error {
|
||||
pattern := util.DefaultQuery(c, "glob", "")
|
||||
sortby := util.DefaultQuery(c, "sort", "none")
|
||||
order := util.DefaultQuery(c, "order", "asc")
|
||||
|
||||
files := h.filesystem.List(pattern)
|
||||
|
||||
var sortFunc func(i, j int) bool
|
||||
|
||||
switch sortby {
|
||||
case "name":
|
||||
if order == "desc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].Name() > files[j].Name() }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].Name() < files[j].Name() }
|
||||
}
|
||||
case "size":
|
||||
if order == "desc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].Size() > files[j].Size() }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].Size() < files[j].Size() }
|
||||
}
|
||||
default:
|
||||
if order == "asc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].ModTime().Before(files[j].ModTime()) }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].ModTime().After(files[j].ModTime()) }
|
||||
}
|
||||
}
|
||||
|
||||
sort.Slice(files, sortFunc)
|
||||
|
||||
var fileinfos []api.FileInfo = make([]api.FileInfo, len(files))
|
||||
|
||||
for i, f := range files {
|
||||
fileinfos[i] = api.FileInfo{
|
||||
Name: f.Name(),
|
||||
Size: f.Size(),
|
||||
LastMod: f.ModTime().Unix(),
|
||||
}
|
||||
}
|
||||
|
||||
return c.JSON(http.StatusOK, fileinfos)
|
||||
}
|
@@ -1,88 +0,0 @@
|
||||
package handler
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
"path/filepath"
|
||||
|
||||
"github.com/datarhei/core/v16/http/api"
|
||||
"github.com/datarhei/core/v16/http/cache"
|
||||
"github.com/datarhei/core/v16/http/handler/util"
|
||||
"github.com/datarhei/core/v16/io/fs"
|
||||
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
// The DiskFSHandler type provides handlers for manipulating a filesystem
|
||||
type DiskFSHandler struct {
|
||||
cache cache.Cacher
|
||||
filesystem fs.Filesystem
|
||||
}
|
||||
|
||||
// NewDiskFS return a new DiskFS type. You have to provide a filesystem to act on and optionally
|
||||
// a Cacher where files will be purged from if the Cacher is related to the filesystem.
|
||||
func NewDiskFS(fs fs.Filesystem, cache cache.Cacher) *DiskFSHandler {
|
||||
return &DiskFSHandler{
|
||||
cache: cache,
|
||||
filesystem: fs,
|
||||
}
|
||||
}
|
||||
|
||||
// GetFile returns the file at the given path
|
||||
// @Summary Fetch a file from the filesystem
|
||||
// @Description Fetch a file from the filesystem. If the file is a directory, a index.html is returned, if it exists.
|
||||
// @ID diskfs-get-file
|
||||
// @Produce application/data
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {file} byte
|
||||
// @Success 301 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Router /{path} [get]
|
||||
func (h *DiskFSHandler) GetFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
mimeType := c.Response().Header().Get(echo.HeaderContentType)
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
file := h.filesystem.Open(path)
|
||||
if file == nil {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
stat, _ := file.Stat()
|
||||
|
||||
if stat.IsDir() {
|
||||
path = filepath.Join(path, "index.html")
|
||||
|
||||
file.Close()
|
||||
|
||||
file = h.filesystem.Open(path)
|
||||
if file == nil {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
stat, _ = file.Stat()
|
||||
}
|
||||
|
||||
defer file.Close()
|
||||
|
||||
c.Response().Header().Set("Last-Modified", stat.ModTime().UTC().Format("Mon, 02 Jan 2006 15:04:05 GMT"))
|
||||
|
||||
if path, ok := stat.IsLink(); ok {
|
||||
path = filepath.Clean("/" + path)
|
||||
|
||||
if path[0] == '/' {
|
||||
path = path[1:]
|
||||
}
|
||||
|
||||
return c.Redirect(http.StatusMovedPermanently, path)
|
||||
}
|
||||
|
||||
c.Response().Header().Set(echo.HeaderContentType, mimeType)
|
||||
|
||||
if c.Request().Method == "HEAD" {
|
||||
return c.Blob(http.StatusOK, "application/data", nil)
|
||||
}
|
||||
|
||||
return c.Stream(http.StatusOK, "application/data", file)
|
||||
}
|
164
http/handler/filesystem.go
Normal file
164
http/handler/filesystem.go
Normal file
@@ -0,0 +1,164 @@
|
||||
package handler
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
"path/filepath"
|
||||
"sort"
|
||||
|
||||
"github.com/datarhei/core/v16/http/api"
|
||||
"github.com/datarhei/core/v16/http/fs"
|
||||
"github.com/datarhei/core/v16/http/handler/util"
|
||||
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
// The FSHandler type provides handlers for manipulating a filesystem
|
||||
type FSHandler struct {
|
||||
fs fs.FS
|
||||
}
|
||||
|
||||
// NewFS return a new FSHandler type. You have to provide a filesystem to act on.
|
||||
func NewFS(fs fs.FS) *FSHandler {
|
||||
return &FSHandler{
|
||||
fs: fs,
|
||||
}
|
||||
}
|
||||
|
||||
func (h *FSHandler) GetFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
mimeType := c.Response().Header().Get(echo.HeaderContentType)
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
file := h.fs.Filesystem.Open(path)
|
||||
if file == nil {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
stat, _ := file.Stat()
|
||||
|
||||
if len(h.fs.DefaultFile) != 0 {
|
||||
if stat.IsDir() {
|
||||
path = filepath.Join(path, h.fs.DefaultFile)
|
||||
|
||||
file.Close()
|
||||
|
||||
file = h.fs.Filesystem.Open(path)
|
||||
if file == nil {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
stat, _ = file.Stat()
|
||||
}
|
||||
}
|
||||
|
||||
defer file.Close()
|
||||
|
||||
c.Response().Header().Set("Last-Modified", stat.ModTime().UTC().Format("Mon, 02 Jan 2006 15:04:05 GMT"))
|
||||
|
||||
if path, ok := stat.IsLink(); ok {
|
||||
path = filepath.Clean("/" + path)
|
||||
|
||||
if path[0] == '/' {
|
||||
path = path[1:]
|
||||
}
|
||||
|
||||
return c.Redirect(http.StatusMovedPermanently, path)
|
||||
}
|
||||
|
||||
c.Response().Header().Set(echo.HeaderContentType, mimeType)
|
||||
|
||||
if c.Request().Method == "HEAD" {
|
||||
return c.Blob(http.StatusOK, "application/data", nil)
|
||||
}
|
||||
|
||||
return c.Stream(http.StatusOK, "application/data", file)
|
||||
}
|
||||
|
||||
func (h *FSHandler) PutFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
req := c.Request()
|
||||
|
||||
_, created, err := h.fs.Filesystem.Store(path, req.Body)
|
||||
if err != nil {
|
||||
return api.Err(http.StatusBadRequest, "%s", err)
|
||||
}
|
||||
|
||||
if h.fs.Cache != nil {
|
||||
h.fs.Cache.Delete(path)
|
||||
}
|
||||
|
||||
c.Response().Header().Set("Content-Location", req.URL.RequestURI())
|
||||
|
||||
if created {
|
||||
return c.String(http.StatusCreated, "")
|
||||
}
|
||||
|
||||
return c.NoContent(http.StatusNoContent)
|
||||
}
|
||||
|
||||
func (h *FSHandler) DeleteFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
size := h.fs.Filesystem.Delete(path)
|
||||
|
||||
if size < 0 {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
if h.fs.Cache != nil {
|
||||
h.fs.Cache.Delete(path)
|
||||
}
|
||||
|
||||
return c.String(http.StatusOK, "Deleted: "+path)
|
||||
}
|
||||
|
||||
func (h *FSHandler) ListFiles(c echo.Context) error {
|
||||
pattern := util.DefaultQuery(c, "glob", "")
|
||||
sortby := util.DefaultQuery(c, "sort", "none")
|
||||
order := util.DefaultQuery(c, "order", "asc")
|
||||
|
||||
files := h.fs.Filesystem.List(pattern)
|
||||
|
||||
var sortFunc func(i, j int) bool
|
||||
|
||||
switch sortby {
|
||||
case "name":
|
||||
if order == "desc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].Name() > files[j].Name() }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].Name() < files[j].Name() }
|
||||
}
|
||||
case "size":
|
||||
if order == "desc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].Size() > files[j].Size() }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].Size() < files[j].Size() }
|
||||
}
|
||||
default:
|
||||
if order == "asc" {
|
||||
sortFunc = func(i, j int) bool { return files[i].ModTime().Before(files[j].ModTime()) }
|
||||
} else {
|
||||
sortFunc = func(i, j int) bool { return files[i].ModTime().After(files[j].ModTime()) }
|
||||
}
|
||||
}
|
||||
|
||||
sort.Slice(files, sortFunc)
|
||||
|
||||
var fileinfos []api.FileInfo = make([]api.FileInfo, len(files))
|
||||
|
||||
for i, f := range files {
|
||||
fileinfos[i] = api.FileInfo{
|
||||
Name: f.Name(),
|
||||
Size: f.Size(),
|
||||
LastMod: f.ModTime().Unix(),
|
||||
}
|
||||
}
|
||||
|
||||
return c.JSON(http.StatusOK, fileinfos)
|
||||
}
|
@@ -1,130 +0,0 @@
|
||||
package handler
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
"path/filepath"
|
||||
|
||||
"github.com/datarhei/core/v16/http/api"
|
||||
"github.com/datarhei/core/v16/http/handler/util"
|
||||
"github.com/datarhei/core/v16/io/fs"
|
||||
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
// The MemFSHandler type provides handlers for manipulating a filesystem
|
||||
type MemFSHandler struct {
|
||||
filesystem fs.Filesystem
|
||||
}
|
||||
|
||||
// NewMemFS return a new MemFS type. You have to provide a filesystem to act on.
|
||||
func NewMemFS(fs fs.Filesystem) *MemFSHandler {
|
||||
return &MemFSHandler{
|
||||
filesystem: fs,
|
||||
}
|
||||
}
|
||||
|
||||
// GetFile returns the file at the given path
|
||||
// @Summary Fetch a file from the memory filesystem
|
||||
// @Description Fetch a file from the memory filesystem
|
||||
// @ID memfs-get-file
|
||||
// @Produce application/data
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {file} byte
|
||||
// @Success 301 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Router /memfs/{path} [get]
|
||||
func (h *MemFSHandler) GetFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
mimeType := c.Response().Header().Get(echo.HeaderContentType)
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
file := h.filesystem.Open(path)
|
||||
if file == nil {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
defer file.Close()
|
||||
|
||||
stat, _ := file.Stat()
|
||||
|
||||
c.Response().Header().Set("Last-Modified", stat.ModTime().UTC().Format("Mon, 02 Jan 2006 15:04:05 GMT"))
|
||||
|
||||
if path, ok := stat.IsLink(); ok {
|
||||
path = filepath.Clean("/" + path)
|
||||
|
||||
if path[0] == '/' {
|
||||
path = path[1:]
|
||||
}
|
||||
|
||||
return c.Redirect(http.StatusMovedPermanently, path)
|
||||
}
|
||||
|
||||
c.Response().Header().Set(echo.HeaderContentType, mimeType)
|
||||
|
||||
if c.Request().Method == "HEAD" {
|
||||
return c.Blob(http.StatusOK, "application/data", nil)
|
||||
}
|
||||
|
||||
return c.Stream(http.StatusOK, "application/data", file)
|
||||
}
|
||||
|
||||
// PutFile adds or overwrites a file at the given path
|
||||
// @Summary Add a file to the memory filesystem
|
||||
// @Description Writes or overwrites a file on the memory filesystem
|
||||
// @ID memfs-put-file
|
||||
// @Accept application/data
|
||||
// @Produce text/plain
|
||||
// @Produce json
|
||||
// @Param path path string true "Path to file"
|
||||
// @Param data body []byte true "File data"
|
||||
// @Success 201 {string} string
|
||||
// @Success 204 {string} string
|
||||
// @Failure 507 {object} api.Error
|
||||
// @Security BasicAuth
|
||||
// @Router /memfs/{path} [put]
|
||||
func (h *MemFSHandler) PutFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
req := c.Request()
|
||||
|
||||
_, created, err := h.filesystem.Store(path, req.Body)
|
||||
if err != nil {
|
||||
return api.Err(http.StatusBadRequest, "%s", err)
|
||||
}
|
||||
|
||||
c.Response().Header().Set("Content-Location", req.URL.RequestURI())
|
||||
|
||||
if created {
|
||||
return c.String(http.StatusCreated, "")
|
||||
}
|
||||
|
||||
return c.NoContent(http.StatusNoContent)
|
||||
}
|
||||
|
||||
// DeleteFile removes a file from the filesystem
|
||||
// @Summary Remove a file from the memory filesystem
|
||||
// @Description Remove a file from the memory filesystem
|
||||
// @ID memfs-delete-file
|
||||
// @Produce text/plain
|
||||
// @Param path path string true "Path to file"
|
||||
// @Success 200 {string} string
|
||||
// @Failure 404 {object} api.Error
|
||||
// @Security BasicAuth
|
||||
// @Router /memfs/{path} [delete]
|
||||
func (h *MemFSHandler) DeleteFile(c echo.Context) error {
|
||||
path := util.PathWildcardParam(c)
|
||||
|
||||
c.Response().Header().Del(echo.HeaderContentType)
|
||||
|
||||
size := h.filesystem.Delete(path)
|
||||
|
||||
if size < 0 {
|
||||
return api.Err(http.StatusNotFound, "File not found", path)
|
||||
}
|
||||
|
||||
return c.String(http.StatusOK, "Deleted: "+path)
|
||||
}
|
237
http/server.go
237
http/server.go
@@ -29,19 +29,20 @@
|
||||
package http
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
"strings"
|
||||
|
||||
cfgstore "github.com/datarhei/core/v16/config/store"
|
||||
"github.com/datarhei/core/v16/http/cache"
|
||||
"github.com/datarhei/core/v16/http/errorhandler"
|
||||
"github.com/datarhei/core/v16/http/fs"
|
||||
"github.com/datarhei/core/v16/http/graph/resolver"
|
||||
"github.com/datarhei/core/v16/http/handler"
|
||||
api "github.com/datarhei/core/v16/http/handler/api"
|
||||
"github.com/datarhei/core/v16/http/jwt"
|
||||
"github.com/datarhei/core/v16/http/router"
|
||||
"github.com/datarhei/core/v16/http/validator"
|
||||
"github.com/datarhei/core/v16/io/fs"
|
||||
"github.com/datarhei/core/v16/log"
|
||||
"github.com/datarhei/core/v16/monitor"
|
||||
"github.com/datarhei/core/v16/net"
|
||||
@@ -79,8 +80,7 @@ type Config struct {
|
||||
Metrics monitor.HistoryReader
|
||||
Prometheus prometheus.Reader
|
||||
MimeTypesFile string
|
||||
DiskFS fs.Filesystem
|
||||
MemFS MemFSConfig
|
||||
Filesystems []fs.FS
|
||||
IPLimiter net.IPLimiter
|
||||
Profiling bool
|
||||
Cors CorsConfig
|
||||
@@ -94,13 +94,6 @@ type Config struct {
|
||||
ReadOnly bool
|
||||
}
|
||||
|
||||
type MemFSConfig struct {
|
||||
EnableAuth bool
|
||||
Username string
|
||||
Password string
|
||||
Filesystem fs.Filesystem
|
||||
}
|
||||
|
||||
type CorsConfig struct {
|
||||
Origins []string
|
||||
}
|
||||
@@ -114,8 +107,6 @@ type server struct {
|
||||
|
||||
handler struct {
|
||||
about *api.AboutHandler
|
||||
memfs *handler.MemFSHandler
|
||||
diskfs *handler.DiskFSHandler
|
||||
prometheus *handler.PrometheusHandler
|
||||
profiling *handler.ProfilingHandler
|
||||
ping *handler.PingHandler
|
||||
@@ -127,8 +118,6 @@ type server struct {
|
||||
log *api.LogHandler
|
||||
restream *api.RestreamHandler
|
||||
playout *api.PlayoutHandler
|
||||
memfs *api.MemFSHandler
|
||||
diskfs *api.DiskFSHandler
|
||||
rtmp *api.RTMPHandler
|
||||
srt *api.SRTHandler
|
||||
config *api.ConfigHandler
|
||||
@@ -148,18 +137,12 @@ type server struct {
|
||||
hlsrewrite echo.MiddlewareFunc
|
||||
}
|
||||
|
||||
memfs struct {
|
||||
enableAuth bool
|
||||
username string
|
||||
password string
|
||||
}
|
||||
|
||||
diskfs fs.Filesystem
|
||||
|
||||
gzip struct {
|
||||
mimetypes []string
|
||||
}
|
||||
|
||||
filesystems map[string]*filesystem
|
||||
|
||||
router *echo.Echo
|
||||
mimeTypesFile string
|
||||
profiling bool
|
||||
@@ -167,32 +150,62 @@ type server struct {
|
||||
readOnly bool
|
||||
}
|
||||
|
||||
type filesystem struct {
|
||||
fs.FS
|
||||
|
||||
handler *handler.FSHandler
|
||||
}
|
||||
|
||||
func NewServer(config Config) (Server, error) {
|
||||
s := &server{
|
||||
logger: config.Logger,
|
||||
mimeTypesFile: config.MimeTypesFile,
|
||||
profiling: config.Profiling,
|
||||
diskfs: config.DiskFS,
|
||||
readOnly: config.ReadOnly,
|
||||
}
|
||||
|
||||
s.v3handler.diskfs = api.NewDiskFS(
|
||||
config.DiskFS,
|
||||
config.Cache,
|
||||
)
|
||||
s.filesystems = map[string]*filesystem{}
|
||||
|
||||
s.handler.diskfs = handler.NewDiskFS(
|
||||
config.DiskFS,
|
||||
config.Cache,
|
||||
)
|
||||
corsPrefixes := map[string][]string{
|
||||
"/api": {"*"},
|
||||
}
|
||||
|
||||
s.middleware.hlsrewrite = mwhlsrewrite.NewHLSRewriteWithConfig(mwhlsrewrite.HLSRewriteConfig{
|
||||
PathPrefix: config.DiskFS.Base(),
|
||||
})
|
||||
for _, fs := range config.Filesystems {
|
||||
if _, ok := s.filesystems[fs.Name]; ok {
|
||||
return nil, fmt.Errorf("the filesystem name '%s' is already in use", fs.Name)
|
||||
}
|
||||
|
||||
s.memfs.enableAuth = config.MemFS.EnableAuth
|
||||
s.memfs.username = config.MemFS.Username
|
||||
s.memfs.password = config.MemFS.Password
|
||||
if !strings.HasPrefix(fs.Mountpoint, "/") {
|
||||
fs.Mountpoint = "/" + fs.Mountpoint
|
||||
}
|
||||
|
||||
if !strings.HasSuffix(fs.Mountpoint, "/") {
|
||||
fs.Mountpoint = strings.TrimSuffix(fs.Mountpoint, "/")
|
||||
}
|
||||
|
||||
if _, ok := corsPrefixes[fs.Mountpoint]; ok {
|
||||
return nil, fmt.Errorf("the mount point '%s' is already in use (%s)", fs.Mountpoint, fs.Name)
|
||||
}
|
||||
|
||||
corsPrefixes[fs.Mountpoint] = config.Cors.Origins
|
||||
|
||||
filesystem := &filesystem{
|
||||
FS: fs,
|
||||
handler: handler.NewFS(fs),
|
||||
}
|
||||
|
||||
s.filesystems[filesystem.Name] = filesystem
|
||||
|
||||
if fs.Filesystem.Type() == "disk" {
|
||||
s.middleware.hlsrewrite = mwhlsrewrite.NewHLSRewriteWithConfig(mwhlsrewrite.HLSRewriteConfig{
|
||||
PathPrefix: fs.Filesystem.Base(),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
if _, ok := corsPrefixes["/"]; !ok {
|
||||
return nil, fmt.Errorf("one filesystem must be mounted at /")
|
||||
}
|
||||
|
||||
if config.Logger == nil {
|
||||
s.logger = log.New("HTTP")
|
||||
@@ -224,16 +237,6 @@ func NewServer(config Config) (Server, error) {
|
||||
)
|
||||
}
|
||||
|
||||
if config.MemFS.Filesystem != nil {
|
||||
s.v3handler.memfs = api.NewMemFS(
|
||||
config.MemFS.Filesystem,
|
||||
)
|
||||
|
||||
s.handler.memfs = handler.NewMemFS(
|
||||
config.MemFS.Filesystem,
|
||||
)
|
||||
}
|
||||
|
||||
if config.Prometheus != nil {
|
||||
s.handler.prometheus = handler.NewPrometheus(
|
||||
config.Prometheus.HTTPHandler(),
|
||||
@@ -292,12 +295,6 @@ func NewServer(config Config) (Server, error) {
|
||||
Logger: s.logger,
|
||||
})
|
||||
|
||||
if config.Cache != nil {
|
||||
s.middleware.cache = mwcache.NewWithConfig(mwcache.Config{
|
||||
Cache: config.Cache,
|
||||
})
|
||||
}
|
||||
|
||||
s.v3handler.widget = api.NewWidget(api.WidgetConfig{
|
||||
Restream: config.Restream,
|
||||
Registry: config.Sessions,
|
||||
@@ -308,11 +305,7 @@ func NewServer(config Config) (Server, error) {
|
||||
})
|
||||
|
||||
if middleware, err := mwcors.NewWithConfig(mwcors.Config{
|
||||
Prefixes: map[string][]string{
|
||||
"/": config.Cors.Origins,
|
||||
"/api": {"*"},
|
||||
"/memfs": config.Cors.Origins,
|
||||
},
|
||||
Prefixes: corsPrefixes,
|
||||
}); err != nil {
|
||||
return nil, err
|
||||
} else {
|
||||
@@ -437,65 +430,58 @@ func (s *server) setRoutes() {
|
||||
doc.Use(gzipMiddleware)
|
||||
doc.GET("", echoSwagger.WrapHandler)
|
||||
|
||||
// Serve static data
|
||||
fs := s.router.Group("/*")
|
||||
fs.Use(mwmime.NewWithConfig(mwmime.Config{
|
||||
MimeTypesFile: s.mimeTypesFile,
|
||||
DefaultContentType: "text/html",
|
||||
}))
|
||||
fs.Use(mwgzip.NewWithConfig(mwgzip.Config{
|
||||
Level: mwgzip.BestSpeed,
|
||||
MinLength: 1000,
|
||||
Skipper: mwgzip.ContentTypeSkipper(s.gzip.mimetypes),
|
||||
}))
|
||||
if s.middleware.cache != nil {
|
||||
fs.Use(s.middleware.cache)
|
||||
}
|
||||
fs.Use(s.middleware.hlsrewrite)
|
||||
if s.middleware.session != nil {
|
||||
fs.Use(s.middleware.session)
|
||||
}
|
||||
// Mount filesystems
|
||||
for _, filesystem := range s.filesystems {
|
||||
// Define a local variable because later in the loop we have a closure
|
||||
filesystem := filesystem
|
||||
|
||||
fs.GET("", s.handler.diskfs.GetFile)
|
||||
fs.HEAD("", s.handler.diskfs.GetFile)
|
||||
|
||||
// Memory FS
|
||||
if s.handler.memfs != nil {
|
||||
memfs := s.router.Group("/memfs/*")
|
||||
memfs.Use(mwmime.NewWithConfig(mwmime.Config{
|
||||
MimeTypesFile: s.mimeTypesFile,
|
||||
DefaultContentType: "application/data",
|
||||
}))
|
||||
memfs.Use(mwgzip.NewWithConfig(mwgzip.Config{
|
||||
Level: mwgzip.BestSpeed,
|
||||
MinLength: 1000,
|
||||
Skipper: mwgzip.ContentTypeSkipper(s.gzip.mimetypes),
|
||||
}))
|
||||
if s.middleware.session != nil {
|
||||
memfs.Use(s.middleware.session)
|
||||
mountpoint := filesystem.Mountpoint + "/*"
|
||||
if filesystem.Mountpoint == "/" {
|
||||
mountpoint = "/*"
|
||||
}
|
||||
|
||||
memfs.HEAD("", s.handler.memfs.GetFile)
|
||||
memfs.GET("", s.handler.memfs.GetFile)
|
||||
fs := s.router.Group(mountpoint)
|
||||
fs.Use(mwmime.NewWithConfig(mwmime.Config{
|
||||
MimeTypesFile: s.mimeTypesFile,
|
||||
DefaultContentType: filesystem.DefaultContentType,
|
||||
}))
|
||||
|
||||
var authmw echo.MiddlewareFunc
|
||||
if filesystem.Gzip {
|
||||
fs.Use(mwgzip.NewWithConfig(mwgzip.Config{
|
||||
Skipper: mwgzip.ContentTypeSkipper(s.gzip.mimetypes),
|
||||
Level: mwgzip.BestSpeed,
|
||||
MinLength: 1000,
|
||||
}))
|
||||
}
|
||||
|
||||
if s.memfs.enableAuth {
|
||||
authmw = middleware.BasicAuth(func(username, password string, c echo.Context) (bool, error) {
|
||||
if username == s.memfs.username && password == s.memfs.password {
|
||||
return true, nil
|
||||
}
|
||||
|
||||
return false, nil
|
||||
if filesystem.Cache != nil {
|
||||
mwcache := mwcache.NewWithConfig(mwcache.Config{
|
||||
Cache: filesystem.Cache,
|
||||
})
|
||||
fs.Use(mwcache)
|
||||
}
|
||||
|
||||
memfs.POST("", s.handler.memfs.PutFile, authmw)
|
||||
memfs.PUT("", s.handler.memfs.PutFile, authmw)
|
||||
memfs.DELETE("", s.handler.memfs.DeleteFile, authmw)
|
||||
} else {
|
||||
memfs.POST("", s.handler.memfs.PutFile)
|
||||
memfs.PUT("", s.handler.memfs.PutFile)
|
||||
memfs.DELETE("", s.handler.memfs.DeleteFile)
|
||||
fs.GET("", filesystem.handler.GetFile)
|
||||
fs.HEAD("", filesystem.handler.GetFile)
|
||||
|
||||
if filesystem.AllowWrite {
|
||||
if filesystem.EnableAuth {
|
||||
authmw := middleware.BasicAuth(func(username, password string, c echo.Context) (bool, error) {
|
||||
if username == filesystem.Username && password == filesystem.Password {
|
||||
return true, nil
|
||||
}
|
||||
|
||||
return false, nil
|
||||
})
|
||||
|
||||
fs.POST("", filesystem.handler.PutFile, authmw)
|
||||
fs.PUT("", filesystem.handler.PutFile, authmw)
|
||||
fs.DELETE("", filesystem.handler.DeleteFile, authmw)
|
||||
} else {
|
||||
fs.POST("", filesystem.handler.PutFile)
|
||||
fs.PUT("", filesystem.handler.PutFile)
|
||||
fs.DELETE("", filesystem.handler.DeleteFile)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -593,32 +579,33 @@ func (s *server) setRoutesV3(v3 *echo.Group) {
|
||||
}
|
||||
}
|
||||
|
||||
// v3 Memory FS
|
||||
if s.v3handler.memfs != nil {
|
||||
v3.GET("/fs/mem", s.v3handler.memfs.ListFiles)
|
||||
v3.GET("/fs/mem/*", s.v3handler.memfs.GetFile)
|
||||
|
||||
if !s.readOnly {
|
||||
v3.DELETE("/fs/mem/*", s.v3handler.memfs.DeleteFile)
|
||||
v3.PUT("/fs/mem/*", s.v3handler.memfs.PutFile)
|
||||
v3.PATCH("/fs/mem/*", s.v3handler.memfs.PatchFile)
|
||||
// v3 Filesystems
|
||||
fshandlers := map[string]api.FSConfig{}
|
||||
for _, fs := range s.filesystems {
|
||||
fshandlers[fs.Name] = api.FSConfig{
|
||||
Type: fs.Filesystem.Type(),
|
||||
Mountpoint: fs.Mountpoint,
|
||||
Handler: fs.handler,
|
||||
}
|
||||
}
|
||||
|
||||
// v3 Disk FS
|
||||
v3.GET("/fs/disk", s.v3handler.diskfs.ListFiles)
|
||||
v3.GET("/fs/disk/*", s.v3handler.diskfs.GetFile, mwmime.NewWithConfig(mwmime.Config{
|
||||
handler := api.NewFS(fshandlers)
|
||||
|
||||
v3.GET("/fs", handler.List)
|
||||
|
||||
v3.GET("/fs/:name", handler.ListFiles)
|
||||
v3.GET("/fs/:name/*", handler.GetFile, mwmime.NewWithConfig(mwmime.Config{
|
||||
MimeTypesFile: s.mimeTypesFile,
|
||||
DefaultContentType: "application/data",
|
||||
}))
|
||||
v3.HEAD("/fs/disk/*", s.v3handler.diskfs.GetFile, mwmime.NewWithConfig(mwmime.Config{
|
||||
v3.HEAD("/fs/:name/*", handler.GetFile, mwmime.NewWithConfig(mwmime.Config{
|
||||
MimeTypesFile: s.mimeTypesFile,
|
||||
DefaultContentType: "application/data",
|
||||
}))
|
||||
|
||||
if !s.readOnly {
|
||||
v3.PUT("/fs/disk/*", s.v3handler.diskfs.PutFile)
|
||||
v3.DELETE("/fs/disk/*", s.v3handler.diskfs.DeleteFile)
|
||||
v3.PUT("/fs/:name/*", handler.PutFile)
|
||||
v3.DELETE("/fs/:name/*", handler.DeleteFile)
|
||||
}
|
||||
|
||||
// v3 RTMP
|
||||
|
Reference in New Issue
Block a user