package seekable 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/seekable/internal/gitdir" "gopkg.in/src-d/go-git.v4/storage/seekable/internal/index" "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 } 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) } func buildIndexFromIdxfile(fs fs.FS, path string) (index.Index, error) { f, err := fs.Open(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") } // 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 } f, err := fs.Open(path) 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 } 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) } return head, nil }