diff options
Diffstat (limited to 'storage/filesystem/storage.go')
-rw-r--r-- | storage/filesystem/storage.go | 195 |
1 files changed, 19 insertions, 176 deletions
diff --git a/storage/filesystem/storage.go b/storage/filesystem/storage.go index 427de94..8f9c555 100644 --- a/storage/filesystem/storage.go +++ b/storage/filesystem/storage.go @@ -1,203 +1,46 @@ package filesystem import ( - "fmt" - "os" - "strings" - "gopkg.in/src-d/go-git.v4/core" - "gopkg.in/src-d/go-git.v4/formats/packfile" - "gopkg.in/src-d/go-git.v4/storage/filesystem/internal/gitdir" - "gopkg.in/src-d/go-git.v4/storage/filesystem/internal/index" + "gopkg.in/src-d/go-git.v4/storage/filesystem/internal/dotgit" "gopkg.in/src-d/go-git.v4/utils/fs" ) -// ObjectStorage is an implementation of core.ObjectStorage that stores -// data on disk in the standard git format (this is, the .git directory). -// -// Zero values of this type are not safe to use, see the New function below. -// -// Currently only reads are supported, no writting. -// -// Also values from this type are not yet able to track changes on disk, this is, -// Gitdir values will get outdated as soon as repositories change on disk. -type ObjectStorage struct { - dir *gitdir.GitDir - index index.Index -} - -// New returns a new ObjectStorage for the git directory at the specified path. -func New(fs fs.FS, path string) (*ObjectStorage, error) { - s := &ObjectStorage{} - - var err error - s.dir, err = gitdir.New(fs, path) - if err != nil { - return nil, err - } - - s.index, err = buildIndex(s.dir) - - return s, err -} - -func buildIndex(dir *gitdir.GitDir) (index.Index, error) { - fs, idxfile, err := dir.Idxfile() - if err != nil { - if err == gitdir.ErrIdxNotFound { - return buildIndexFromPackfile(dir) - } - return nil, err - } - - return buildIndexFromIdxfile(fs, idxfile) -} - -func buildIndexFromPackfile(dir *gitdir.GitDir) (index.Index, error) { - fs, packfile, err := dir.Packfile() - if err != nil { - return nil, err - } +type Storage struct { + dir *dotgit.DotGit - f, err := fs.Open(packfile) - if err != nil { - return nil, err - } - - defer func() { - errClose := f.Close() - if err == nil { - err = errClose - } - }() - - return index.NewFromPackfile(f) + o *ObjectStorage + r *ReferenceStorage } -func buildIndexFromIdxfile(fs fs.FS, path string) (index.Index, error) { - f, err := fs.Open(path) +func NewStorage(fs fs.FS, path string) (*Storage, error) { + dir, err := dotgit.New(fs, path) if err != nil { return nil, err } - defer func() { - errClose := f.Close() - if err == nil { - err = errClose - } - }() - - return index.NewFromIdx(f) -} - -func (s *ObjectStorage) NewObject() core.Object { - return &core.MemoryObject{} -} - -// Set adds a new object to the storage. As this functionality is not -// yet supported, this method always returns a "not implemented yet" -// error an zero hash. -func (s *ObjectStorage) Set(core.Object) (core.Hash, error) { - return core.ZeroHash, fmt.Errorf("not implemented yet") + return &Storage{dir: dir}, nil } -// Get returns the object with the given hash, by searching for it in -// the packfile. -func (s *ObjectStorage) Get(h core.Hash) (core.Object, error) { - offset, err := s.index.Get(h) - if err != nil { - return nil, err - } - - fs, path, err := s.dir.Packfile() - if err != nil { - return nil, err +func (s *Storage) ObjectStorage() (core.ObjectStorage, error) { + if s.o != nil { + return s.o, nil } - f, err := fs.Open(path) + i, err := buildIndex(s.dir) if err != nil { return nil, err } - defer func() { - errClose := f.Close() - if err == nil { - err = errClose - } - }() - - _, err = f.Seek(offset, os.SEEK_SET) - if err != nil { - return nil, err - } - - r := packfile.NewSeekable(f) - r.HashToOffset = map[core.Hash]int64(s.index) - p := packfile.NewParser(r) - - obj := s.NewObject() - return obj, p.FillObject(obj) -} - -// Iter returns an iterator for all the objects in the packfile with the -// given type. -func (s *ObjectStorage) Iter(t core.ObjectType) (core.ObjectIter, error) { - var objects []core.Object - - for hash := range s.index { - object, err := s.Get(hash) - if err != nil { - return nil, err - } - if object.Type() == t { - objects = append(objects, object) - } - } - - return core.NewObjectSliceIter(objects), nil + s.o = &ObjectStorage{dir: s.dir, index: i} + return s.o, nil } -const ( - headErrPrefix = "cannot get HEAD reference:" - symrefCapability = "symref" - headRefPrefix = "HEAD:" -) - -// Head returns the hash of the HEAD reference -func (s *ObjectStorage) Head() (core.Hash, error) { - cap, err := s.dir.Capabilities() - if err != nil { - return core.ZeroHash, fmt.Errorf("%s %s", headErrPrefix, err) - } - - ok := cap.Supports(symrefCapability) - if !ok { - return core.ZeroHash, - fmt.Errorf("%s symref capability not supported", headErrPrefix) - } - - symrefs := cap.Get(symrefCapability) - var headRef string - for _, ref := range symrefs.Values { - if strings.HasPrefix(ref, headRefPrefix) { - headRef = strings.TrimPrefix(ref, headRefPrefix) - } - } - if headRef == "" { - return core.ZeroHash, fmt.Errorf("%s HEAD reference not found", - headErrPrefix) - } - - refs, err := s.dir.Refs() - if err != nil { - return core.ZeroHash, fmt.Errorf("%s %s", headErrPrefix, err) - } - - head, ok := refs[headRef] - if !ok { - return core.ZeroHash, fmt.Errorf("%s reference %q not found", - headErrPrefix, headRef) +func (s *Storage) ReferenceStorage() (core.ReferenceStorage, error) { + if s.r != nil { + return s.r, nil } - return head, nil + s.r = &ReferenceStorage{dir: s.dir} + return s.r, nil } |