pkger/index.go

192 lines
3.2 KiB
Go

package pkger
import (
"encoding/json"
"fmt"
"os"
"path/filepath"
"strings"
"sync"
"time"
"github.com/markbates/pkger/here"
"github.com/markbates/pkger/internal/debug"
)
type index struct {
Files *filesMap `json:"files"`
Infos *infosMap `json:"infos"`
Paths *pathsMap `json:"paths"`
Current here.Info `json:"current"`
once *sync.Once
}
func (i *index) debug(key, format string, args ...interface{}) {
s := fmt.Sprintf(format, args...)
debug.Debug("[*index|%s|%s] %s", i.Current.ImportPath, key, s)
}
func (i *index) Parse(p string) (Path, error) {
i.debug("Parse", p)
pt, ok := i.Paths.Load(p)
if ok {
return pt, nil
}
if len(p) == 0 {
return build(p, "", "")
}
res := pathrx.FindAllStringSubmatch(p, -1)
if len(res) == 0 {
return pt, fmt.Errorf("could not parse %q", p)
}
matches := res[0]
if len(matches) != 4 {
return pt, fmt.Errorf("could not parse %q", p)
}
return build(p, matches[1], matches[3])
}
func (i *index) Info(p string) (here.Info, error) {
info, ok := i.Infos.Load(p)
if ok {
return info, nil
}
info, err := here.Package(p)
if err != nil {
return info, err
}
i.Infos.Store(p, info)
return info, nil
}
func (i *index) Stat() (here.Info, error) {
var err error
i.once.Do(func() {
if i.Current.IsZero() {
i.Current, err = here.Current()
}
})
return i.Current, err
}
func (i *index) Create(pt Path) (*File, error) {
her, err := Info(pt.Pkg)
if err != nil {
return nil, err
}
f := &File{
path: pt,
her: her,
info: &FileInfo{
name: strings.TrimPrefix(pt.Name, "/"),
mode: 0666,
modTime: time.Now(),
virtual: true,
},
}
if i.Files == nil {
i.Files = &filesMap{}
}
i.Files.Store(pt, f)
dir := Path{
Pkg: pt.Pkg,
Name: filepath.Dir(pt.Name),
}
i.MkdirAll(dir, 0644)
return f, nil
}
func (i *index) UnmarshalJSON(b []byte) error {
i.once = &sync.Once{}
m := map[string]json.RawMessage{}
if err := json.Unmarshal(b, &m); err != nil {
return err
}
infos, ok := m["infos"]
if !ok {
return fmt.Errorf("missing infos")
}
i.Infos = &infosMap{}
if err := json.Unmarshal(infos, i.Infos); err != nil {
return err
}
files, ok := m["files"]
if !ok {
return fmt.Errorf("missing files")
}
i.Files = &filesMap{}
if err := json.Unmarshal(files, i.Files); err != nil {
return err
}
paths, ok := m["paths"]
if !ok {
return fmt.Errorf("missing paths")
}
i.Paths = &pathsMap{}
if err := json.Unmarshal(paths, i.Paths); err != nil {
return err
}
current, ok := m["current"]
if !ok {
return fmt.Errorf("missing current")
}
if err := json.Unmarshal(current, &i.Current); err != nil {
return err
}
i.debug("UnmarshalJSON", "%v", i)
return nil
}
func (i index) openDisk(pt Path) (*File, error) {
i.debug("openDisk", pt.String())
info, err := Info(pt.Pkg)
if err != nil {
return nil, err
}
fp := info.Dir
if len(pt.Name) > 0 {
fp = filepath.Join(fp, pt.Name)
}
fi, err := os.Stat(fp)
if err != nil {
return nil, err
}
f := &File{
info: WithName(pt.Name, NewFileInfo(fi)),
her: info,
path: pt,
}
return f, nil
}
func newIndex() *index {
return &index{
Files: &filesMap{},
Infos: &infosMap{},
Paths: &pathsMap{},
once: &sync.Once{},
}
}
var rootIndex = func() *index {
i := newIndex()
return i
}()