mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-05 05:27:37 +03:00
* Use new embed functionality for serving UI assets * Use new embed functionality for serving resources. Remove dependency on go-bindata * Remove Go 1.15
117 lines
2.8 KiB
Go
117 lines
2.8 KiB
Go
package utils
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"sort"
|
|
)
|
|
|
|
// NewMergeFS returns a simple implementation of a merged FS (http.FileSystem), that can combine a base FS with a
|
|
// overlay FS. The semantics are:
|
|
// - Files from the overlay FS will override file s with the same name in the base FS
|
|
// - Directories are combined, with priority for the overlay FS over the base FS for files with matching names
|
|
// TODO Replace http.FileSystem with new fs.FS interface
|
|
func NewMergeFS(base, overlay http.FileSystem) http.FileSystem {
|
|
return &mergeFS{
|
|
base: base,
|
|
overlay: overlay,
|
|
}
|
|
}
|
|
|
|
type mergeFS struct {
|
|
base http.FileSystem
|
|
overlay http.FileSystem
|
|
}
|
|
|
|
func (m mergeFS) Open(name string) (http.File, error) {
|
|
f, err := m.overlay.Open(name)
|
|
if err != nil {
|
|
return m.base.Open(name)
|
|
}
|
|
|
|
info, err := f.Stat()
|
|
if err != nil {
|
|
_ = f.Close()
|
|
return m.base.Open(name)
|
|
}
|
|
|
|
if !info.IsDir() {
|
|
return f, nil
|
|
}
|
|
|
|
baseDir, _ := m.base.Open(name)
|
|
defer func() {
|
|
_ = baseDir.Close()
|
|
_ = f.Close()
|
|
}()
|
|
return m.mergeDirs(name, info, baseDir, f)
|
|
}
|
|
|
|
func (m mergeFS) mergeDirs(name string, info os.FileInfo, baseDir http.File, overlayDir http.File) (http.File, error) {
|
|
merged := map[string]os.FileInfo{}
|
|
|
|
baseFiles, err := baseDir.Readdir(-1)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
sort.Slice(baseFiles, func(i, j int) bool { return baseFiles[i].Name() < baseFiles[j].Name() })
|
|
|
|
overlayFiles, err := overlayDir.Readdir(-1)
|
|
if err != nil {
|
|
overlayFiles = nil
|
|
}
|
|
sort.Slice(overlayFiles, func(i, j int) bool { return overlayFiles[i].Name() < overlayFiles[j].Name() })
|
|
|
|
for _, f := range baseFiles {
|
|
merged[f.Name()] = f
|
|
}
|
|
for _, f := range overlayFiles {
|
|
merged[f.Name()] = f
|
|
}
|
|
|
|
var entries []os.FileInfo
|
|
for _, i := range merged {
|
|
entries = append(entries, i)
|
|
}
|
|
|
|
sort.Slice(entries, func(i, j int) bool { return entries[i].Name() < entries[j].Name() })
|
|
return &mergedDir{
|
|
name: name,
|
|
info: info,
|
|
entries: entries,
|
|
}, nil
|
|
}
|
|
|
|
type mergedDir struct {
|
|
name string
|
|
info os.FileInfo
|
|
entries []os.FileInfo
|
|
pos int
|
|
}
|
|
|
|
func (d *mergedDir) Readdir(count int) ([]os.FileInfo, error) {
|
|
if d.pos >= len(d.entries) && count > 0 {
|
|
return nil, io.EOF
|
|
}
|
|
if count <= 0 || count > len(d.entries)-d.pos {
|
|
count = len(d.entries) - d.pos
|
|
}
|
|
e := d.entries[d.pos : d.pos+count]
|
|
d.pos += count
|
|
return e, nil
|
|
}
|
|
|
|
func (d *mergedDir) Close() error { return nil }
|
|
func (d *mergedDir) Stat() (os.FileInfo, error) { return d.info, nil }
|
|
func (d *mergedDir) Read(p []byte) (n int, err error) {
|
|
return 0, fmt.Errorf("cannot Read from directory %s", d.name)
|
|
}
|
|
func (d *mergedDir) Seek(offset int64, whence int) (int64, error) {
|
|
if offset == 0 && whence == io.SeekStart {
|
|
d.pos = 0
|
|
return 0, nil
|
|
}
|
|
return 0, fmt.Errorf("unsupported Seek in directory %s", d.name)
|
|
}
|