package dotgit
import (
"errors"
"os"
"strings"
"gopkg.in/src-d/go-git.v4/core"
"gopkg.in/src-d/go-git.v4/utils/fs"
)
const (
suffix = ".git"
packedRefsPath = "packed-refs"
configPath = "config"
)
var (
// ErrNotFound is returned by New when the path is not found.
ErrNotFound = errors.New("path not found")
// ErrIdxNotFound is returned by Idxfile when the idx file is not found
ErrIdxNotFound = errors.New("idx file not found")
// ErrPackfileNotFound is returned by Packfile when the packfile is not found
ErrPackfileNotFound = errors.New("packfile not found")
// ErrObjfileNotFound is returned by Objectfile when the objectffile is not found
ErrObjfileNotFound = errors.New("object file not found")
// ErrConfigNotFound is returned by Config when the config is not found
ErrConfigNotFound = errors.New("config file not found")
)
// The DotGit type represents a local git repository on disk. This
// type is not zero-value-safe, use the New function to initialize it.
type DotGit struct {
fs fs.FS
path string
}
// New returns a DotGit value ready to be used. The path argument must
// be the absolute path of a git repository directory (e.g.
// "/foo/bar/.git").
func New(fs fs.FS, path string) (*DotGit, error) {
d := &DotGit{fs: fs, path: path}
if _, err := fs.Stat(path); err != nil {
if os.IsNotExist(err) {
return nil, ErrNotFound
}
return nil, err
}
return d, nil
}
// Refs scans the git directory collecting references, which it returns.
// Symbolic references are resolved and included in the output.
func (d *DotGit) Refs() ([]*core.Reference, error) {
var refs []*core.Reference
if err := d.addRefsFromPackedRefs(&refs); err != nil {
return nil, err
}
if err := d.addRefsFromRefDir(&refs); err != nil {
return nil, err
}
if err := d.addRefFromHEAD(&refs); err != nil {
return nil, err
}
return refs, nil
}
// Packfile returns the path of the packfile (really, it returns the
// path of the first file in the "objects/pack/" directory with a
// ".pack" extension.
func (d *DotGit) Packfile() (fs.FS, string, error) {
packDir := d.fs.Join(d.path, "objects", "pack")
files, err := d.fs.ReadDir(packDir)
if err != nil {
if os.IsNotExist(err) {
return nil, "", ErrPackfileNotFound
}
return nil, "", err
}
for _, f := range files {
if strings.HasSuffix(f.Name(), ".pack") {
return d.fs, d.fs.Join(packDir, f.Name()), nil
}
}
return nil, "", ErrPackfileNotFound
}
// Idxfile returns the path of the idx file (really, it returns the
// path of the first file in the "objects/pack/" directory with an
// ".idx" extension.
func (d *DotGit) Idxfile() (fs.FS, string, error) {
packDir := d.fs.Join(d.path, "objects", "pack")
files, err := d.fs.ReadDir(packDir)
if err != nil {
if os.IsNotExist(err) {
return nil, "", ErrIdxNotFound
}
return nil, "", err
}
for _, f := range files {
if strings.HasSuffix(f.Name(), ".idx") {
return d.fs, d.fs.Join(packDir, f.Name()), nil
}
}
return nil, "", ErrIdxNotFound
}
// Config returns the path of the config file
func (d *DotGit) Config() (fs.FS, string, error) {
configFile := d.fs.Join(d.path, configPath)
if _, err := d.fs.Stat(configFile); err != nil {
if os.IsNotExist(err) {
return nil, "", ErrNotFound
}
return nil, "", err
}
return d.fs, configFile, nil
}
// Objectfiles returns a slice with the hashes of objects found under the
// .git/objects/ directory.
func (dg *DotGit) Objectfiles() (fs.FS, []core.Hash, error) {
objsDir := dg.fs.Join(dg.path, "objects")
files, err := dg.fs.ReadDir(objsDir)
if err != nil {
if os.IsNotExist(err) {
return nil, nil, ErrObjfileNotFound
}
return nil, nil, err
}
var objects []core.Hash
for _, f := range files {
if f.IsDir() && len(f.Name()) == 2 && isHex(f.Name()) {
objDir := f.Name()
d, err := dg.fs.ReadDir(dg.fs.Join(objsDir, objDir))
if err != nil {
return nil, nil, err
}
for _, o := range d {
objects = append(objects, core.NewHash(objDir+o.Name()))
}
}
}
return dg.fs, objects, nil
}
func isHex(s string) bool {
for _, b := range []byte(s) {
if isNum(b) {
continue
}
if isHexAlpha(b) {
continue
}
return false
}
return true
}
func isNum(b byte) bool {
return b >= '0' && b <= '9'
}
func isHexAlpha(b byte) bool {
return b >= 'a' && b <= 'f' || b >= 'A' && b <= 'F'
}
// Objectfile returns the path of the object file for a given hash
// *if the file exists*, otherwise returns an ErrObjfileNotFound error.
func (d *DotGit) Objectfile(h core.Hash) (fs.FS, string, error) {
hash := h.String()
objFile := d.fs.Join(d.path, "objects", hash[0:2], hash[2:40])
if _, err := d.fs.Stat(objFile); err != nil {
if os.IsNotExist(err) {
return nil, "", ErrObjfileNotFound
}
return nil, "", err
}
return d.fs, objFile, nil
}