feat: allow multiple roots

This commit is contained in:
Max Richter
2025-10-03 21:41:39 +02:00
parent 3986e22dd3
commit ab81c980b5
3 changed files with 120 additions and 22 deletions

View File

@@ -10,6 +10,7 @@
- path: "_type" - path: "_type"
codec: const codec: const
value: Recipe value: Recipe
- path: link
- path: image - path: image
- path: author._type - path: author._type
codec: const codec: const

View File

@@ -6,31 +6,48 @@ import (
"net/http" "net/http"
"os" "os"
"path/filepath" "path/filepath"
"strings"
"git.max-richter.dev/max/marka/server-new/internal/adapters" "git.max-richter.dev/max/marka/server-new/internal/adapters"
"git.max-richter.dev/max/marka/server-new/internal/handler" "git.max-richter.dev/max/marka/server-new/internal/handler"
) )
type multi []string
func (m *multi) String() string { return strings.Join(*m, ",") }
func (m *multi) Set(v string) error {
*m = append(*m, v)
return nil
}
func main() { func main() {
root := flag.String("root", ".", "filesystem root to serve") var roots multi
flag.Var(&roots, "root", "repeatable; specify multiple -root flags")
addr := flag.String("addr", ":8080", "listen address") addr := flag.String("addr", ":8080", "listen address")
flag.Parse() flag.Parse()
absRoot, err := filepath.Abs(*root) if len(roots) == 0 {
must(err) log.Fatal("at least one -root flag must be specified")
info, err := os.Stat(absRoot)
must(err)
if !info.IsDir() {
log.Fatal("root is not a directory")
} }
fsAdapter, err := adapters.NewLocalFsAdapter(absRoot) absRoots := make([]string, len(roots))
for i, r := range roots {
abs, err := filepath.Abs(r)
must(err)
info, err := os.Stat(abs)
must(err)
if !info.IsDir() {
log.Fatalf("root %s is not a directory", r)
}
absRoots[i] = abs
}
fsAdapter, err := adapters.NewLocalFsAdapter(absRoots)
must(err) must(err)
http.Handle("/", handler.NewHandler(fsAdapter)) http.Handle("/", handler.NewHandler(fsAdapter))
log.Printf("listening on %s, root=%s", *addr, absRoot) log.Printf("listening on %s, roots=%s", *addr, strings.Join(absRoots, ", "))
log.Fatal(http.ListenAndServe(*addr, nil)) log.Fatal(http.ListenAndServe(*addr, nil))
} }

View File

@@ -1,16 +1,22 @@
package adapters package adapters
import ( import (
"errors"
"os" "os"
"path/filepath" "path/filepath"
"strings"
"time"
) )
type LocalFsAdapter struct { type LocalFsAdapter struct {
root string roots []string
} }
func (l LocalFsAdapter) readDir(path string) (FsResponse, error) { func (l LocalFsAdapter) readDir(path string, root string) (FsResponse, error) {
dirInfo, _ := os.Stat(path) dirInfo, err := os.Stat(path)
if err != nil {
return FsResponse{}, err
}
entries, err := os.ReadDir(path) entries, err := os.ReadDir(path)
if err != nil { if err != nil {
@@ -29,20 +35,31 @@ func (l LocalFsAdapter) readDir(path string) (FsResponse, error) {
out = append(out, FsDirEntry{ out = append(out, FsDirEntry{
Name: e.Name(), Name: e.Name(),
Type: entryType, Type: entryType,
IsDir: e.IsDir(),
ModTime: info.ModTime(), ModTime: info.ModTime(),
}) })
} }
relPath, err := filepath.Rel(root, path)
if err != nil {
return FsResponse{}, err
}
if relPath == "." {
relPath = ""
}
responsePath := "/" + filepath.ToSlash(filepath.Join(filepath.Base(root), relPath))
return FsResponse{ return FsResponse{
Dir: &FsDir{ Dir: &FsDir{
Files: out, Files: out,
Name: ResponsePath(l.root, path), Name: responsePath,
ModTime: dirInfo.ModTime(), ModTime: dirInfo.ModTime(),
}, },
}, nil }, nil
} }
func (l LocalFsAdapter) readFile(path string) (FsResponse, error) { func (l LocalFsAdapter) readFile(path string, root string) (FsResponse, error) {
fi, err := os.Stat(path) fi, err := os.Stat(path)
if err != nil { if err != nil {
return FsResponse{}, err return FsResponse{}, err
@@ -53,9 +70,16 @@ func (l LocalFsAdapter) readFile(path string) (FsResponse, error) {
return FsResponse{}, err return FsResponse{}, err
} }
relPath, err := filepath.Rel(root, path)
if err != nil {
return FsResponse{}, err
}
responsePath := "/" + filepath.ToSlash(filepath.Join(filepath.Base(root), relPath))
return FsResponse{ return FsResponse{
File: &FsFile{ File: &FsFile{
Name: ResponsePath(l.root, path), Name: responsePath,
Type: contentTypeFor(path), Type: contentTypeFor(path),
ModTime: fi.ModTime(), ModTime: fi.ModTime(),
Content: data, Content: data,
@@ -64,11 +88,66 @@ func (l LocalFsAdapter) readFile(path string) (FsResponse, error) {
} }
func (l LocalFsAdapter) Read(path string) (FsResponse, error) { func (l LocalFsAdapter) Read(path string) (FsResponse, error) {
cleanRel, err := SafeRel(l.root, path) if path == "/" {
entries := make([]FsDirEntry, 0, len(l.roots))
var latestModTime time.Time
for _, r := range l.roots {
info, err := os.Stat(r)
if err != nil {
continue
}
entries = append(entries, FsDirEntry{
Name: filepath.Base(r),
Type: "dir",
IsDir: true,
ModTime: info.ModTime(),
})
if info.ModTime().After(latestModTime) {
latestModTime = info.ModTime()
}
}
return FsResponse{
Dir: &FsDir{
Files: entries,
Name: "/",
ModTime: latestModTime,
},
}, nil
}
pathParts := strings.Split(strings.Trim(path, "/"), "/")
if len(pathParts) == 0 || pathParts[0] == "" {
return FsResponse{}, ErrNotFound
}
rootIdentifier := pathParts[0]
var targetRoot string
for _, r := range l.roots {
if filepath.Base(r) == rootIdentifier {
targetRoot = r
break
}
}
if targetRoot == "" {
return FsResponse{}, ErrNotFound
}
subPath := filepath.Join(pathParts[1:]...)
target := filepath.Join(targetRoot, subPath)
absTarget, err := filepath.Abs(target)
if err != nil { if err != nil {
return FsResponse{}, err return FsResponse{}, err
} }
target := filepath.Join(l.root, filepath.FromSlash(cleanRel)) absRoot, err := filepath.Abs(targetRoot)
if err != nil {
return FsResponse{}, err
}
if !strings.HasPrefix(absTarget, absRoot) {
return FsResponse{}, errors.New("path escapes root")
}
fi, err := os.Stat(target) fi, err := os.Stat(target)
if err != nil { if err != nil {
@@ -80,18 +159,19 @@ func (l LocalFsAdapter) Read(path string) (FsResponse, error) {
} }
if fi.IsDir() { if fi.IsDir() {
return l.readDir(target) return l.readDir(target, targetRoot)
} }
return l.readFile(target) return l.readFile(target, targetRoot)
} }
func (LocalFsAdapter) Write(path string, content []byte) error { func (LocalFsAdapter) Write(path string, content []byte) error {
return nil return nil
} }
func NewLocalFsAdapter(root string) (FileAdapter, error) { func NewLocalFsAdapter(roots []string) (FileAdapter, error) {
return LocalFsAdapter{ return LocalFsAdapter{
root: root, roots: roots,
}, nil }, nil
} }