pkger/index.go

186 lines
3.0 KiB
Go

package pkger
import (
"encoding/json"
"fmt"
"os"
"path/filepath"
"strings"
"sync"
"time"
"github.com/gobuffalo/here"
)
type index struct {
Files map[Path]*File
Infos map[string]here.Info
current here.Info
once sync.Once
}
func (i *index) Info(p string) (here.Info, error) {
if info, ok := i.Infos[p]; ok {
return info, nil
}
info, err := here.Cache(p, here.Package)
if err != nil {
return info, err
}
i.Infos[p] = info
return info, nil
}
func (i *index) Current() (here.Info, error) {
if !i.current.IsZero() {
return i.current, nil
}
var err error
i.once.Do(func() {
i.current, err = here.Cache("", func(string) (here.Info, error) {
return 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(),
},
}
i.Files[pt] = f
return f, nil
}
func (i *index) MarshalJSON() ([]byte, error) {
m := map[string]interface{}{}
fm := map[string]json.RawMessage{}
for k, v := range i.Files {
b, err := v.MarshalJSON()
if err != nil {
return b, err
}
fm[k.String()] = b
}
m["files"] = fm
m["infos"] = i.Infos
m["current"] = i.current
return json.Marshal(m)
}
func (i *index) UnmarshalJSON(b []byte) error {
// fmt.Println(string(b))
return nil
}
func (i index) Walk(pt Path, wf WalkFunc) error {
if len(i.Files) > 0 {
for k, v := range i.Files {
if k.Pkg != pt.Pkg {
continue
}
if err := wf(k, v.info); err != nil {
return err
}
}
}
var info here.Info
var err error
if pt.Pkg == "." {
info, err = Current()
if err != nil {
return err
}
pt.Pkg = info.ImportPath
}
if info.IsZero() {
info, err = Info(pt.Pkg)
if err != nil {
return fmt.Errorf("%s: %s", pt, err)
}
}
fp := filepath.Join(info.Dir, pt.Name)
err = filepath.Walk(fp, func(path string, fi os.FileInfo, err error) error {
if err != nil {
return err
}
path = strings.TrimPrefix(path, info.Dir)
pt, err := Parse(fmt.Sprintf("%s:%s", pt.Pkg, path))
if err != nil {
return err
}
return wf(pt, NewFileInfo(fi))
})
return err
}
func (i *index) Open(pt Path) (*File, error) {
f, ok := i.Files[pt]
if !ok {
return i.openDisk(pt)
}
nf := &File{
info: f.info,
path: f.path,
data: f.data,
her: f.her,
}
return nf, nil
}
func (i index) openDisk(pt Path) (*File, error) {
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(strings.TrimPrefix(pt.Name, "/"), NewFileInfo(fi)),
her: info,
path: pt,
}
return f, nil
}
func newIndex() *index {
return &index{
Files: map[Path]*File{},
Infos: map[string]here.Info{},
}
}
var rootIndex = func() *index {
i := newIndex()
return i
}()