2019-09-03 18:29:28 +03:00
|
|
|
package mem
|
2019-09-01 00:00:24 +03:00
|
|
|
|
|
|
|
import (
|
2019-09-01 05:45:22 +03:00
|
|
|
"fmt"
|
|
|
|
"os"
|
2019-09-03 22:18:31 +03:00
|
|
|
"path/filepath"
|
2019-09-01 05:45:22 +03:00
|
|
|
"strings"
|
2019-09-03 22:18:31 +03:00
|
|
|
"time"
|
2019-09-01 00:00:24 +03:00
|
|
|
|
|
|
|
"github.com/markbates/pkger/here"
|
|
|
|
"github.com/markbates/pkger/internal/maps"
|
2019-09-02 01:02:45 +03:00
|
|
|
"github.com/markbates/pkger/pkging"
|
2019-09-01 00:00:24 +03:00
|
|
|
)
|
|
|
|
|
2019-09-03 18:29:28 +03:00
|
|
|
var _ pkging.Pkger = &Pkger{}
|
2019-09-01 00:00:24 +03:00
|
|
|
|
2019-09-03 18:29:28 +03:00
|
|
|
func WithInfo(fx *Pkger, infos ...here.Info) {
|
2019-09-01 06:29:25 +03:00
|
|
|
for _, info := range infos {
|
|
|
|
fx.infos.Store(info.ImportPath, info)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-03 18:29:28 +03:00
|
|
|
func New(info here.Info) (*Pkger, error) {
|
|
|
|
f := &Pkger{
|
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-03 18:29:28 +03:00
|
|
|
type Pkger struct {
|
2019-09-01 00:00:24 +03:00
|
|
|
infos *maps.Infos
|
|
|
|
paths *maps.Paths
|
|
|
|
files *maps.Files
|
|
|
|
current here.Info
|
|
|
|
}
|
|
|
|
|
2019-09-03 18:29:28 +03:00
|
|
|
func (f *Pkger) 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-03 18:29:28 +03:00
|
|
|
func (f *Pkger) AbsPath(pt pkging.Path) (string, error) {
|
2019-09-01 23:18:39 +03:00
|
|
|
return pt.String(), nil
|
|
|
|
}
|
|
|
|
|
2019-09-03 18:29:28 +03:00
|
|
|
func (f *Pkger) Current() (here.Info, error) {
|
2019-09-01 00:00:24 +03:00
|
|
|
return f.current, nil
|
|
|
|
}
|
|
|
|
|
2019-09-03 18:29:28 +03:00
|
|
|
func (f *Pkger) 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-03 18:29:28 +03:00
|
|
|
func (f *Pkger) Parse(p string) (pkging.Path, error) {
|
2019-09-01 00:00:24 +03:00
|
|
|
return f.paths.Parse(p)
|
|
|
|
}
|
|
|
|
|
2019-09-03 18:29:28 +03:00
|
|
|
func (fx *Pkger) 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-03 18:29:28 +03:00
|
|
|
func (fx *Pkger) RemoveAll(name string) error {
|
2019-09-01 05:45:22 +03:00
|
|
|
pt, err := fx.Parse(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-09-03 19:41:21 +03:00
|
|
|
fx.files.Range(func(key pkging.Path, file pkging.File) bool {
|
|
|
|
if strings.HasPrefix(key.Name, pt.Name) {
|
|
|
|
fx.files.Delete(key)
|
2019-09-01 05:45:22 +03:00
|
|
|
}
|
2019-09-03 19:41:21 +03:00
|
|
|
return true
|
2019-09-01 05:45:22 +03:00
|
|
|
})
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2019-09-03 22:18:31 +03:00
|
|
|
|
|
|
|
func (fx *Pkger) Create(name string) (pkging.File, error) {
|
|
|
|
pt, err := fx.Parse(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
her, err := fx.Info(pt.Pkg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := fx.Stat(filepath.Dir(pt.Name)); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
f := &File{
|
|
|
|
path: pt,
|
|
|
|
her: her,
|
|
|
|
info: &pkging.FileInfo{
|
|
|
|
Details: pkging.Details{
|
|
|
|
Name: pt.Name,
|
|
|
|
Mode: 0644,
|
|
|
|
ModTime: pkging.ModTime(time.Now()),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
pkging: fx,
|
|
|
|
}
|
|
|
|
|
|
|
|
fx.files.Store(pt, f)
|
|
|
|
|
|
|
|
return f, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fx *Pkger) MkdirAll(p string, perm os.FileMode) error {
|
|
|
|
path, err := fx.Parse(p)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
root := path.Name
|
|
|
|
|
|
|
|
cur, err := fx.Current()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for root != "" {
|
|
|
|
pt := pkging.Path{
|
|
|
|
Pkg: path.Pkg,
|
|
|
|
Name: root,
|
|
|
|
}
|
|
|
|
if _, ok := fx.files.Load(pt); ok {
|
|
|
|
root = filepath.Dir(root)
|
|
|
|
if root == "/" || root == "\\" {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
f := &File{
|
|
|
|
pkging: fx,
|
|
|
|
path: pt,
|
|
|
|
her: cur,
|
|
|
|
info: &pkging.FileInfo{
|
|
|
|
Details: pkging.Details{
|
|
|
|
Name: pt.Name,
|
|
|
|
Mode: perm,
|
|
|
|
ModTime: pkging.ModTime(time.Now()),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
f.info.Details.IsDir = true
|
|
|
|
f.info.Details.Mode = perm
|
|
|
|
if err := f.Close(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fx.files.Store(pt, f)
|
|
|
|
root = filepath.Dir(root)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fx *Pkger) Open(name string) (pkging.File, error) {
|
|
|
|
pt, err := fx.Parse(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
fl, ok := fx.files.Load(pt)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("could not open %s", name)
|
|
|
|
}
|
|
|
|
f, ok := fl.(*File)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("could not open %s", name)
|
|
|
|
}
|
|
|
|
nf := &File{
|
|
|
|
pkging: fx,
|
|
|
|
info: pkging.WithName(f.info.Name(), f.info),
|
|
|
|
path: f.path,
|
|
|
|
data: f.data,
|
|
|
|
her: f.her,
|
|
|
|
}
|
|
|
|
|
|
|
|
return nf, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fx *Pkger) Stat(name string) (os.FileInfo, error) {
|
|
|
|
pt, err := fx.Parse(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
f, ok := fx.files.Load(pt)
|
|
|
|
if ok {
|
|
|
|
return f.Stat()
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("could not stat %s", pt)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *Pkger) Walk(p string, wf filepath.WalkFunc) error {
|
|
|
|
keys := f.files.Keys()
|
|
|
|
|
|
|
|
pt, err := f.Parse(p)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
skip := "!"
|
|
|
|
|
|
|
|
for _, k := range keys {
|
|
|
|
if !strings.HasPrefix(k.Name, pt.Name) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if strings.HasPrefix(k.Name, skip) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
fl, ok := f.files.Load(k)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("could not find %s", k)
|
|
|
|
}
|
|
|
|
fi, err := fl.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
fi = pkging.WithName(strings.TrimPrefix(k.Name, pt.Name), fi)
|
|
|
|
err = wf(k.String(), fi, nil)
|
|
|
|
if err == filepath.SkipDir {
|
|
|
|
|
|
|
|
skip = k.Name
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|