2019-09-01 00:00:24 +03:00
|
|
|
package memfs
|
|
|
|
|
|
|
|
import (
|
2019-09-01 05:45:22 +03:00
|
|
|
"fmt"
|
2019-09-01 00:00:24 +03:00
|
|
|
"io/ioutil"
|
2019-09-01 05:45:22 +03:00
|
|
|
"os"
|
|
|
|
"strings"
|
2019-09-01 00:00:24 +03:00
|
|
|
|
|
|
|
"github.com/markbates/pkger/fs"
|
|
|
|
"github.com/markbates/pkger/here"
|
|
|
|
"github.com/markbates/pkger/internal/maps"
|
|
|
|
)
|
|
|
|
|
|
|
|
var _ fs.FileSystem = &FS{}
|
|
|
|
|
2019-09-01 06:29:25 +03:00
|
|
|
func WithInfo(fx *FS, infos ...here.Info) {
|
|
|
|
for _, info := range infos {
|
|
|
|
fx.infos.Store(info.ImportPath, info)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-01 00:00:24 +03:00
|
|
|
func New(info here.Info) (*FS, error) {
|
|
|
|
f := &FS{
|
2019-09-01 06:29:25 +03:00
|
|
|
infos: &maps.Infos{},
|
|
|
|
paths: &maps.Paths{
|
|
|
|
Current: info,
|
|
|
|
},
|
2019-09-01 05:47:23 +03:00
|
|
|
files: &maps.Files{},
|
|
|
|
current: info,
|
2019-09-01 00:00:24 +03:00
|
|
|
}
|
|
|
|
return f, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type FS struct {
|
|
|
|
infos *maps.Infos
|
|
|
|
paths *maps.Paths
|
|
|
|
files *maps.Files
|
|
|
|
current here.Info
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *FS) Current() (here.Info, error) {
|
|
|
|
return f.current, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *FS) Info(p string) (here.Info, error) {
|
|
|
|
info, ok := f.infos.Load(p)
|
2019-09-01 06:29:25 +03:00
|
|
|
if !ok {
|
|
|
|
return info, fmt.Errorf("no such package %q", p)
|
2019-09-01 00:00:24 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return info, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *FS) Parse(p string) (fs.Path, error) {
|
|
|
|
return f.paths.Parse(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fx *FS) ReadFile(s string) ([]byte, error) {
|
|
|
|
f, err := fx.Open(s)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
return ioutil.ReadAll(f)
|
|
|
|
}
|
2019-09-01 05:45:22 +03:00
|
|
|
|
|
|
|
func (fx *FS) Remove(name string) error {
|
|
|
|
pt, err := fx.Parse(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := fx.files.Load(pt); !ok {
|
|
|
|
return &os.PathError{"remove", pt.String(), fmt.Errorf("no such file or directory")}
|
|
|
|
}
|
|
|
|
|
|
|
|
fx.files.Delete(pt)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fx *FS) RemoveAll(name string) error {
|
|
|
|
pt, err := fx.Parse(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return fx.Walk("/", func(path string, info os.FileInfo, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.HasPrefix(path, pt.String()) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
ph, err := fx.Parse(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fx.files.Delete(ph)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if _, ok := fx.files.Load(pt); !ok {
|
|
|
|
return &os.PathError{"remove", pt.String(), fmt.Errorf("no such file or directory")}
|
|
|
|
}
|
|
|
|
|
|
|
|
fx.files.Delete(pt)
|
|
|
|
return nil
|
|
|
|
}
|