2019-09-02 00:58:20 +03:00
|
|
|
package memware
|
2019-09-01 00:00:24 +03:00
|
|
|
|
|
|
|
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"
|
|
|
|
)
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
var _ fs.Warehouse = &Warehouse{}
|
2019-09-01 00:00:24 +03:00
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func WithInfo(fx *Warehouse, infos ...here.Info) {
|
2019-09-01 06:29:25 +03:00
|
|
|
for _, info := range infos {
|
|
|
|
fx.infos.Store(info.ImportPath, info)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func New(info here.Info) (*Warehouse, error) {
|
|
|
|
f := &Warehouse{
|
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
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
type Warehouse struct {
|
2019-09-01 00:00:24 +03:00
|
|
|
infos *maps.Infos
|
|
|
|
paths *maps.Paths
|
|
|
|
files *maps.Files
|
|
|
|
current here.Info
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (f *Warehouse) Abs(p string) (string, error) {
|
2019-09-01 23:18:39 +03:00
|
|
|
pt, err := f.Parse(p)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return f.AbsPath(pt)
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (f *Warehouse) AbsPath(pt fs.Path) (string, error) {
|
2019-09-01 23:18:39 +03:00
|
|
|
return pt.String(), nil
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (f *Warehouse) Current() (here.Info, error) {
|
2019-09-01 00:00:24 +03:00
|
|
|
return f.current, nil
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (f *Warehouse) Info(p string) (here.Info, error) {
|
2019-09-01 00:00:24 +03:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (f *Warehouse) Parse(p string) (fs.Path, error) {
|
2019-09-01 00:00:24 +03:00
|
|
|
return f.paths.Parse(p)
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (fx *Warehouse) ReadFile(s string) ([]byte, error) {
|
2019-09-01 00:00:24 +03:00
|
|
|
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
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (fx *Warehouse) Remove(name string) error {
|
2019-09-01 05:45:22 +03:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-09-02 00:54:05 +03:00
|
|
|
func (fx *Warehouse) RemoveAll(name string) error {
|
2019-09-01 05:45:22 +03:00
|
|
|
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
|
|
|
|
}
|