Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1552 lines
38 KiB
1552 lines
38 KiB
package git |
|
|
|
import ( |
|
"bytes" |
|
"context" |
|
"errors" |
|
"fmt" |
|
"io" |
|
stdioutil "io/ioutil" |
|
"os" |
|
"path" |
|
"path/filepath" |
|
"strings" |
|
"time" |
|
|
|
"golang.org/x/crypto/openpgp" |
|
"gopkg.in/src-d/go-git.v4/config" |
|
"gopkg.in/src-d/go-git.v4/internal/revision" |
|
"gopkg.in/src-d/go-git.v4/plumbing" |
|
"gopkg.in/src-d/go-git.v4/plumbing/cache" |
|
"gopkg.in/src-d/go-git.v4/plumbing/format/packfile" |
|
"gopkg.in/src-d/go-git.v4/plumbing/object" |
|
"gopkg.in/src-d/go-git.v4/plumbing/storer" |
|
"gopkg.in/src-d/go-git.v4/storage" |
|
"gopkg.in/src-d/go-git.v4/storage/filesystem" |
|
"gopkg.in/src-d/go-git.v4/utils/ioutil" |
|
|
|
"gopkg.in/src-d/go-billy.v4" |
|
"gopkg.in/src-d/go-billy.v4/osfs" |
|
) |
|
|
|
// GitDirName this is a special folder where all the git stuff is. |
|
const GitDirName = ".git" |
|
|
|
var ( |
|
// ErrBranchExists an error stating the specified branch already exists |
|
ErrBranchExists = errors.New("branch already exists") |
|
// ErrBranchNotFound an error stating the specified branch does not exist |
|
ErrBranchNotFound = errors.New("branch not found") |
|
// ErrTagExists an error stating the specified tag already exists |
|
ErrTagExists = errors.New("tag already exists") |
|
// ErrTagNotFound an error stating the specified tag does not exist |
|
ErrTagNotFound = errors.New("tag not found") |
|
// ErrFetching is returned when the packfile could not be downloaded |
|
ErrFetching = errors.New("unable to fetch packfile") |
|
|
|
ErrInvalidReference = errors.New("invalid reference, should be a tag or a branch") |
|
ErrRepositoryNotExists = errors.New("repository does not exist") |
|
ErrRepositoryAlreadyExists = errors.New("repository already exists") |
|
ErrRemoteNotFound = errors.New("remote not found") |
|
ErrRemoteExists = errors.New("remote already exists") |
|
ErrAnonymousRemoteName = errors.New("anonymous remote name must be 'anonymous'") |
|
ErrWorktreeNotProvided = errors.New("worktree should be provided") |
|
ErrIsBareRepository = errors.New("worktree not available in a bare repository") |
|
ErrUnableToResolveCommit = errors.New("unable to resolve commit") |
|
ErrPackedObjectsNotSupported = errors.New("Packed objects not supported") |
|
) |
|
|
|
// Repository represents a git repository |
|
type Repository struct { |
|
Storer storage.Storer |
|
|
|
r map[string]*Remote |
|
wt billy.Filesystem |
|
} |
|
|
|
// Init creates an empty git repository, based on the given Storer and worktree. |
|
// The worktree Filesystem is optional, if nil a bare repository is created. If |
|
// the given storer is not empty ErrRepositoryAlreadyExists is returned |
|
func Init(s storage.Storer, worktree billy.Filesystem) (*Repository, error) { |
|
if err := initStorer(s); err != nil { |
|
return nil, err |
|
} |
|
|
|
r := newRepository(s, worktree) |
|
_, err := r.Reference(plumbing.HEAD, false) |
|
switch err { |
|
case plumbing.ErrReferenceNotFound: |
|
case nil: |
|
return nil, ErrRepositoryAlreadyExists |
|
default: |
|
return nil, err |
|
} |
|
|
|
h := plumbing.NewSymbolicReference(plumbing.HEAD, plumbing.Master) |
|
if err := s.SetReference(h); err != nil { |
|
return nil, err |
|
} |
|
|
|
if worktree == nil { |
|
r.setIsBare(true) |
|
return r, nil |
|
} |
|
|
|
return r, setWorktreeAndStoragePaths(r, worktree) |
|
} |
|
|
|
func initStorer(s storer.Storer) error { |
|
i, ok := s.(storer.Initializer) |
|
if !ok { |
|
return nil |
|
} |
|
|
|
return i.Init() |
|
} |
|
|
|
func setWorktreeAndStoragePaths(r *Repository, worktree billy.Filesystem) error { |
|
type fsBased interface { |
|
Filesystem() billy.Filesystem |
|
} |
|
|
|
// .git file is only created if the storage is file based and the file |
|
// system is osfs.OS |
|
fs, isFSBased := r.Storer.(fsBased) |
|
if !isFSBased { |
|
return nil |
|
} |
|
|
|
if err := createDotGitFile(worktree, fs.Filesystem()); err != nil { |
|
return err |
|
} |
|
|
|
return setConfigWorktree(r, worktree, fs.Filesystem()) |
|
} |
|
|
|
func createDotGitFile(worktree, storage billy.Filesystem) error { |
|
path, err := filepath.Rel(worktree.Root(), storage.Root()) |
|
if err != nil { |
|
path = storage.Root() |
|
} |
|
|
|
if path == GitDirName { |
|
// not needed, since the folder is the default place |
|
return nil |
|
} |
|
|
|
f, err := worktree.Create(GitDirName) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
defer f.Close() |
|
_, err = fmt.Fprintf(f, "gitdir: %s\n", path) |
|
return err |
|
} |
|
|
|
func setConfigWorktree(r *Repository, worktree, storage billy.Filesystem) error { |
|
path, err := filepath.Rel(storage.Root(), worktree.Root()) |
|
if err != nil { |
|
path = worktree.Root() |
|
} |
|
|
|
if path == ".." { |
|
// not needed, since the folder is the default place |
|
return nil |
|
} |
|
|
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
cfg.Core.Worktree = path |
|
return r.Storer.SetConfig(cfg) |
|
} |
|
|
|
// Open opens a git repository using the given Storer and worktree filesystem, |
|
// if the given storer is complete empty ErrRepositoryNotExists is returned. |
|
// The worktree can be nil when the repository being opened is bare, if the |
|
// repository is a normal one (not bare) and worktree is nil the err |
|
// ErrWorktreeNotProvided is returned |
|
func Open(s storage.Storer, worktree billy.Filesystem) (*Repository, error) { |
|
_, err := s.Reference(plumbing.HEAD) |
|
if err == plumbing.ErrReferenceNotFound { |
|
return nil, ErrRepositoryNotExists |
|
} |
|
|
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return newRepository(s, worktree), nil |
|
} |
|
|
|
// Clone a repository into the given Storer and worktree Filesystem with the |
|
// given options, if worktree is nil a bare repository is created. If the given |
|
// storer is not empty ErrRepositoryAlreadyExists is returned. |
|
// |
|
// The provided Context must be non-nil. If the context expires before the |
|
// operation is complete, an error is returned. The context only affects to the |
|
// transport operations. |
|
func Clone(s storage.Storer, worktree billy.Filesystem, o *CloneOptions) (*Repository, error) { |
|
return CloneContext(context.Background(), s, worktree, o) |
|
} |
|
|
|
// CloneContext a repository into the given Storer and worktree Filesystem with |
|
// the given options, if worktree is nil a bare repository is created. If the |
|
// given storer is not empty ErrRepositoryAlreadyExists is returned. |
|
// |
|
// The provided Context must be non-nil. If the context expires before the |
|
// operation is complete, an error is returned. The context only affects to the |
|
// transport operations. |
|
func CloneContext( |
|
ctx context.Context, s storage.Storer, worktree billy.Filesystem, o *CloneOptions, |
|
) (*Repository, error) { |
|
r, err := Init(s, worktree) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return r, r.clone(ctx, o) |
|
} |
|
|
|
// PlainInit create an empty git repository at the given path. isBare defines |
|
// if the repository will have worktree (non-bare) or not (bare), if the path |
|
// is not empty ErrRepositoryAlreadyExists is returned. |
|
func PlainInit(path string, isBare bool) (*Repository, error) { |
|
var wt, dot billy.Filesystem |
|
|
|
if isBare { |
|
dot = osfs.New(path) |
|
} else { |
|
wt = osfs.New(path) |
|
dot, _ = wt.Chroot(GitDirName) |
|
} |
|
|
|
s := filesystem.NewStorage(dot, cache.NewObjectLRUDefault()) |
|
|
|
return Init(s, wt) |
|
} |
|
|
|
// PlainOpen opens a git repository from the given path. It detects if the |
|
// repository is bare or a normal one. If the path doesn't contain a valid |
|
// repository ErrRepositoryNotExists is returned |
|
func PlainOpen(path string) (*Repository, error) { |
|
return PlainOpenWithOptions(path, &PlainOpenOptions{}) |
|
} |
|
|
|
// PlainOpenWithOptions opens a git repository from the given path with specific |
|
// options. See PlainOpen for more info. |
|
func PlainOpenWithOptions(path string, o *PlainOpenOptions) (*Repository, error) { |
|
dot, wt, err := dotGitToOSFilesystems(path, o.DetectDotGit) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
if _, err := dot.Stat(""); err != nil { |
|
if os.IsNotExist(err) { |
|
return nil, ErrRepositoryNotExists |
|
} |
|
|
|
return nil, err |
|
} |
|
|
|
s := filesystem.NewStorage(dot, cache.NewObjectLRUDefault()) |
|
|
|
return Open(s, wt) |
|
} |
|
|
|
func dotGitToOSFilesystems(path string, detect bool) (dot, wt billy.Filesystem, err error) { |
|
if path, err = filepath.Abs(path); err != nil { |
|
return nil, nil, err |
|
} |
|
var fs billy.Filesystem |
|
var fi os.FileInfo |
|
for { |
|
fs = osfs.New(path) |
|
fi, err = fs.Stat(GitDirName) |
|
if err == nil { |
|
// no error; stop |
|
break |
|
} |
|
if !os.IsNotExist(err) { |
|
// unknown error; stop |
|
return nil, nil, err |
|
} |
|
if detect { |
|
// try its parent as long as we haven't reached |
|
// the root dir |
|
if dir := filepath.Dir(path); dir != path { |
|
path = dir |
|
continue |
|
} |
|
} |
|
// not detecting via parent dirs and the dir does not exist; |
|
// stop |
|
return fs, nil, nil |
|
} |
|
|
|
if fi.IsDir() { |
|
dot, err = fs.Chroot(GitDirName) |
|
return dot, fs, err |
|
} |
|
|
|
dot, err = dotGitFileToOSFilesystem(path, fs) |
|
if err != nil { |
|
return nil, nil, err |
|
} |
|
|
|
return dot, fs, nil |
|
} |
|
|
|
func dotGitFileToOSFilesystem(path string, fs billy.Filesystem) (bfs billy.Filesystem, err error) { |
|
f, err := fs.Open(GitDirName) |
|
if err != nil { |
|
return nil, err |
|
} |
|
defer ioutil.CheckClose(f, &err) |
|
|
|
b, err := stdioutil.ReadAll(f) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
line := string(b) |
|
const prefix = "gitdir: " |
|
if !strings.HasPrefix(line, prefix) { |
|
return nil, fmt.Errorf(".git file has no %s prefix", prefix) |
|
} |
|
|
|
gitdir := strings.Split(line[len(prefix):], "\n")[0] |
|
gitdir = strings.TrimSpace(gitdir) |
|
if filepath.IsAbs(gitdir) { |
|
return osfs.New(gitdir), nil |
|
} |
|
|
|
return osfs.New(fs.Join(path, gitdir)), nil |
|
} |
|
|
|
// PlainClone a repository into the path with the given options, isBare defines |
|
// if the new repository will be bare or normal. If the path is not empty |
|
// ErrRepositoryAlreadyExists is returned. |
|
// |
|
// TODO(mcuadros): move isBare to CloneOptions in v5 |
|
func PlainClone(path string, isBare bool, o *CloneOptions) (*Repository, error) { |
|
return PlainCloneContext(context.Background(), path, isBare, o) |
|
} |
|
|
|
// PlainCloneContext a repository into the path with the given options, isBare |
|
// defines if the new repository will be bare or normal. If the path is not empty |
|
// ErrRepositoryAlreadyExists is returned. |
|
// |
|
// The provided Context must be non-nil. If the context expires before the |
|
// operation is complete, an error is returned. The context only affects to the |
|
// transport operations. |
|
// |
|
// TODO(mcuadros): move isBare to CloneOptions in v5 |
|
// TODO(smola): refuse upfront to clone on a non-empty directory in v5, see #1027 |
|
func PlainCloneContext(ctx context.Context, path string, isBare bool, o *CloneOptions) (*Repository, error) { |
|
cleanup, cleanupParent, err := checkIfCleanupIsNeeded(path) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
r, err := PlainInit(path, isBare) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
err = r.clone(ctx, o) |
|
if err != nil && err != ErrRepositoryAlreadyExists { |
|
if cleanup { |
|
cleanUpDir(path, cleanupParent) |
|
} |
|
} |
|
|
|
return r, err |
|
} |
|
|
|
func newRepository(s storage.Storer, worktree billy.Filesystem) *Repository { |
|
return &Repository{ |
|
Storer: s, |
|
wt: worktree, |
|
r: make(map[string]*Remote), |
|
} |
|
} |
|
|
|
func checkIfCleanupIsNeeded(path string) (cleanup bool, cleanParent bool, err error) { |
|
fi, err := os.Stat(path) |
|
if err != nil { |
|
if os.IsNotExist(err) { |
|
return true, true, nil |
|
} |
|
|
|
return false, false, err |
|
} |
|
|
|
if !fi.IsDir() { |
|
return false, false, fmt.Errorf("path is not a directory: %s", path) |
|
} |
|
|
|
f, err := os.Open(path) |
|
if err != nil { |
|
return false, false, err |
|
} |
|
|
|
defer ioutil.CheckClose(f, &err) |
|
|
|
_, err = f.Readdirnames(1) |
|
if err == io.EOF { |
|
return true, false, nil |
|
} |
|
|
|
if err != nil { |
|
return false, false, err |
|
} |
|
|
|
return false, false, nil |
|
} |
|
|
|
func cleanUpDir(path string, all bool) error { |
|
if all { |
|
return os.RemoveAll(path) |
|
} |
|
|
|
f, err := os.Open(path) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
defer ioutil.CheckClose(f, &err) |
|
|
|
names, err := f.Readdirnames(-1) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
for _, name := range names { |
|
if err := os.RemoveAll(filepath.Join(path, name)); err != nil { |
|
return err |
|
} |
|
} |
|
|
|
return err |
|
} |
|
|
|
// Config return the repository config |
|
func (r *Repository) Config() (*config.Config, error) { |
|
return r.Storer.Config() |
|
} |
|
|
|
// Remote return a remote if exists |
|
func (r *Repository) Remote(name string) (*Remote, error) { |
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
c, ok := cfg.Remotes[name] |
|
if !ok { |
|
return nil, ErrRemoteNotFound |
|
} |
|
|
|
return newRemote(r.Storer, c), nil |
|
} |
|
|
|
// Remotes returns a list with all the remotes |
|
func (r *Repository) Remotes() ([]*Remote, error) { |
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
remotes := make([]*Remote, len(cfg.Remotes)) |
|
|
|
var i int |
|
for _, c := range cfg.Remotes { |
|
remotes[i] = newRemote(r.Storer, c) |
|
i++ |
|
} |
|
|
|
return remotes, nil |
|
} |
|
|
|
// CreateRemote creates a new remote |
|
func (r *Repository) CreateRemote(c *config.RemoteConfig) (*Remote, error) { |
|
if err := c.Validate(); err != nil { |
|
return nil, err |
|
} |
|
|
|
remote := newRemote(r.Storer, c) |
|
|
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
if _, ok := cfg.Remotes[c.Name]; ok { |
|
return nil, ErrRemoteExists |
|
} |
|
|
|
cfg.Remotes[c.Name] = c |
|
return remote, r.Storer.SetConfig(cfg) |
|
} |
|
|
|
// CreateRemoteAnonymous creates a new anonymous remote. c.Name must be "anonymous". |
|
// It's used like 'git fetch git@github.com:src-d/go-git.git master:master'. |
|
func (r *Repository) CreateRemoteAnonymous(c *config.RemoteConfig) (*Remote, error) { |
|
if err := c.Validate(); err != nil { |
|
return nil, err |
|
} |
|
|
|
if c.Name != "anonymous" { |
|
return nil, ErrAnonymousRemoteName |
|
} |
|
|
|
remote := newRemote(r.Storer, c) |
|
|
|
return remote, nil |
|
} |
|
|
|
// DeleteRemote delete a remote from the repository and delete the config |
|
func (r *Repository) DeleteRemote(name string) error { |
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
if _, ok := cfg.Remotes[name]; !ok { |
|
return ErrRemoteNotFound |
|
} |
|
|
|
delete(cfg.Remotes, name) |
|
return r.Storer.SetConfig(cfg) |
|
} |
|
|
|
// Branch return a Branch if exists |
|
func (r *Repository) Branch(name string) (*config.Branch, error) { |
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
b, ok := cfg.Branches[name] |
|
if !ok { |
|
return nil, ErrBranchNotFound |
|
} |
|
|
|
return b, nil |
|
} |
|
|
|
// CreateBranch creates a new Branch |
|
func (r *Repository) CreateBranch(c *config.Branch) error { |
|
if err := c.Validate(); err != nil { |
|
return err |
|
} |
|
|
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
if _, ok := cfg.Branches[c.Name]; ok { |
|
return ErrBranchExists |
|
} |
|
|
|
cfg.Branches[c.Name] = c |
|
return r.Storer.SetConfig(cfg) |
|
} |
|
|
|
// DeleteBranch delete a Branch from the repository and delete the config |
|
func (r *Repository) DeleteBranch(name string) error { |
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
if _, ok := cfg.Branches[name]; !ok { |
|
return ErrBranchNotFound |
|
} |
|
|
|
delete(cfg.Branches, name) |
|
return r.Storer.SetConfig(cfg) |
|
} |
|
|
|
// CreateTag creates a tag. If opts is included, the tag is an annotated tag, |
|
// otherwise a lightweight tag is created. |
|
func (r *Repository) CreateTag(name string, hash plumbing.Hash, opts *CreateTagOptions) (*plumbing.Reference, error) { |
|
rname := plumbing.ReferenceName(path.Join("refs", "tags", name)) |
|
|
|
_, err := r.Storer.Reference(rname) |
|
switch err { |
|
case nil: |
|
// Tag exists, this is an error |
|
return nil, ErrTagExists |
|
case plumbing.ErrReferenceNotFound: |
|
// Tag missing, available for creation, pass this |
|
default: |
|
// Some other error |
|
return nil, err |
|
} |
|
|
|
var target plumbing.Hash |
|
if opts != nil { |
|
target, err = r.createTagObject(name, hash, opts) |
|
if err != nil { |
|
return nil, err |
|
} |
|
} else { |
|
target = hash |
|
} |
|
|
|
ref := plumbing.NewHashReference(rname, target) |
|
if err = r.Storer.SetReference(ref); err != nil { |
|
return nil, err |
|
} |
|
|
|
return ref, nil |
|
} |
|
|
|
func (r *Repository) createTagObject(name string, hash plumbing.Hash, opts *CreateTagOptions) (plumbing.Hash, error) { |
|
if err := opts.Validate(r, hash); err != nil { |
|
return plumbing.ZeroHash, err |
|
} |
|
|
|
rawobj, err := object.GetObject(r.Storer, hash) |
|
if err != nil { |
|
return plumbing.ZeroHash, err |
|
} |
|
|
|
tag := &object.Tag{ |
|
Name: name, |
|
Tagger: *opts.Tagger, |
|
Message: opts.Message, |
|
TargetType: rawobj.Type(), |
|
Target: hash, |
|
} |
|
|
|
if opts.SignKey != nil { |
|
sig, err := r.buildTagSignature(tag, opts.SignKey) |
|
if err != nil { |
|
return plumbing.ZeroHash, err |
|
} |
|
|
|
tag.PGPSignature = sig |
|
} |
|
|
|
obj := r.Storer.NewEncodedObject() |
|
if err := tag.Encode(obj); err != nil { |
|
return plumbing.ZeroHash, err |
|
} |
|
|
|
return r.Storer.SetEncodedObject(obj) |
|
} |
|
|
|
func (r *Repository) buildTagSignature(tag *object.Tag, signKey *openpgp.Entity) (string, error) { |
|
encoded := &plumbing.MemoryObject{} |
|
if err := tag.Encode(encoded); err != nil { |
|
return "", err |
|
} |
|
|
|
rdr, err := encoded.Reader() |
|
if err != nil { |
|
return "", err |
|
} |
|
|
|
var b bytes.Buffer |
|
if err := openpgp.ArmoredDetachSign(&b, signKey, rdr, nil); err != nil { |
|
return "", err |
|
} |
|
|
|
return b.String(), nil |
|
} |
|
|
|
// Tag returns a tag from the repository. |
|
// |
|
// If you want to check to see if the tag is an annotated tag, you can call |
|
// TagObject on the hash of the reference in ForEach: |
|
// |
|
// ref, err := r.Tag("v0.1.0") |
|
// if err != nil { |
|
// // Handle error |
|
// } |
|
// |
|
// obj, err := r.TagObject(ref.Hash()) |
|
// switch err { |
|
// case nil: |
|
// // Tag object present |
|
// case plumbing.ErrObjectNotFound: |
|
// // Not a tag object |
|
// default: |
|
// // Some other error |
|
// } |
|
// |
|
func (r *Repository) Tag(name string) (*plumbing.Reference, error) { |
|
ref, err := r.Reference(plumbing.ReferenceName(path.Join("refs", "tags", name)), false) |
|
if err != nil { |
|
if err == plumbing.ErrReferenceNotFound { |
|
// Return a friendly error for this one, versus just ReferenceNotFound. |
|
return nil, ErrTagNotFound |
|
} |
|
|
|
return nil, err |
|
} |
|
|
|
return ref, nil |
|
} |
|
|
|
// DeleteTag deletes a tag from the repository. |
|
func (r *Repository) DeleteTag(name string) error { |
|
_, err := r.Tag(name) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
return r.Storer.RemoveReference(plumbing.ReferenceName(path.Join("refs", "tags", name))) |
|
} |
|
|
|
func (r *Repository) resolveToCommitHash(h plumbing.Hash) (plumbing.Hash, error) { |
|
obj, err := r.Storer.EncodedObject(plumbing.AnyObject, h) |
|
if err != nil { |
|
return plumbing.ZeroHash, err |
|
} |
|
switch obj.Type() { |
|
case plumbing.TagObject: |
|
t, err := object.DecodeTag(r.Storer, obj) |
|
if err != nil { |
|
return plumbing.ZeroHash, err |
|
} |
|
return r.resolveToCommitHash(t.Target) |
|
case plumbing.CommitObject: |
|
return h, nil |
|
default: |
|
return plumbing.ZeroHash, ErrUnableToResolveCommit |
|
} |
|
} |
|
|
|
// Clone clones a remote repository |
|
func (r *Repository) clone(ctx context.Context, o *CloneOptions) error { |
|
if err := o.Validate(); err != nil { |
|
return err |
|
} |
|
|
|
c := &config.RemoteConfig{ |
|
Name: o.RemoteName, |
|
URLs: []string{o.URL}, |
|
Fetch: r.cloneRefSpec(o), |
|
} |
|
|
|
if _, err := r.CreateRemote(c); err != nil { |
|
return err |
|
} |
|
|
|
ref, err := r.fetchAndUpdateReferences(ctx, &FetchOptions{ |
|
RefSpecs: c.Fetch, |
|
Depth: o.Depth, |
|
Auth: o.Auth, |
|
Progress: o.Progress, |
|
Tags: o.Tags, |
|
RemoteName: o.RemoteName, |
|
}, o.ReferenceName) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
if r.wt != nil && !o.NoCheckout { |
|
w, err := r.Worktree() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
head, err := r.Head() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
if err := w.Reset(&ResetOptions{ |
|
Mode: MergeReset, |
|
Commit: head.Hash(), |
|
}); err != nil { |
|
return err |
|
} |
|
|
|
if o.RecurseSubmodules != NoRecurseSubmodules { |
|
if err := w.updateSubmodules(&SubmoduleUpdateOptions{ |
|
RecurseSubmodules: o.RecurseSubmodules, |
|
Auth: o.Auth, |
|
}); err != nil { |
|
return err |
|
} |
|
} |
|
} |
|
|
|
if err := r.updateRemoteConfigIfNeeded(o, c, ref); err != nil { |
|
return err |
|
} |
|
|
|
if ref.Name().IsBranch() { |
|
branchRef := ref.Name() |
|
branchName := strings.Split(string(branchRef), "refs/heads/")[1] |
|
|
|
b := &config.Branch{ |
|
Name: branchName, |
|
Merge: branchRef, |
|
} |
|
if o.RemoteName == "" { |
|
b.Remote = "origin" |
|
} else { |
|
b.Remote = o.RemoteName |
|
} |
|
if err := r.CreateBranch(b); err != nil { |
|
return err |
|
} |
|
} |
|
|
|
return nil |
|
} |
|
|
|
const ( |
|
refspecTag = "+refs/tags/%s:refs/tags/%[1]s" |
|
refspecSingleBranch = "+refs/heads/%s:refs/remotes/%s/%[1]s" |
|
refspecSingleBranchHEAD = "+HEAD:refs/remotes/%s/HEAD" |
|
) |
|
|
|
func (r *Repository) cloneRefSpec(o *CloneOptions) []config.RefSpec { |
|
switch { |
|
case o.ReferenceName.IsTag(): |
|
return []config.RefSpec{ |
|
config.RefSpec(fmt.Sprintf(refspecTag, o.ReferenceName.Short())), |
|
} |
|
case o.SingleBranch && o.ReferenceName == plumbing.HEAD: |
|
return []config.RefSpec{ |
|
config.RefSpec(fmt.Sprintf(refspecSingleBranchHEAD, o.RemoteName)), |
|
config.RefSpec(fmt.Sprintf(refspecSingleBranch, plumbing.Master.Short(), o.RemoteName)), |
|
} |
|
case o.SingleBranch: |
|
return []config.RefSpec{ |
|
config.RefSpec(fmt.Sprintf(refspecSingleBranch, o.ReferenceName.Short(), o.RemoteName)), |
|
} |
|
default: |
|
return []config.RefSpec{ |
|
config.RefSpec(fmt.Sprintf(config.DefaultFetchRefSpec, o.RemoteName)), |
|
} |
|
} |
|
} |
|
|
|
func (r *Repository) setIsBare(isBare bool) error { |
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
cfg.Core.IsBare = isBare |
|
return r.Storer.SetConfig(cfg) |
|
} |
|
|
|
func (r *Repository) updateRemoteConfigIfNeeded(o *CloneOptions, c *config.RemoteConfig, head *plumbing.Reference) error { |
|
if !o.SingleBranch { |
|
return nil |
|
} |
|
|
|
c.Fetch = r.cloneRefSpec(o) |
|
|
|
cfg, err := r.Storer.Config() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
cfg.Remotes[c.Name] = c |
|
return r.Storer.SetConfig(cfg) |
|
} |
|
|
|
func (r *Repository) fetchAndUpdateReferences( |
|
ctx context.Context, o *FetchOptions, ref plumbing.ReferenceName, |
|
) (*plumbing.Reference, error) { |
|
|
|
if err := o.Validate(); err != nil { |
|
return nil, err |
|
} |
|
|
|
remote, err := r.Remote(o.RemoteName) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
objsUpdated := true |
|
remoteRefs, err := remote.fetch(ctx, o) |
|
if err == NoErrAlreadyUpToDate { |
|
objsUpdated = false |
|
} else if err == packfile.ErrEmptyPackfile { |
|
return nil, ErrFetching |
|
} else if err != nil { |
|
return nil, err |
|
} |
|
|
|
resolvedRef, err := storer.ResolveReference(remoteRefs, ref) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
refsUpdated, err := r.updateReferences(remote.c.Fetch, resolvedRef) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
if !objsUpdated && !refsUpdated { |
|
return nil, NoErrAlreadyUpToDate |
|
} |
|
|
|
return resolvedRef, nil |
|
} |
|
|
|
func (r *Repository) updateReferences(spec []config.RefSpec, |
|
resolvedRef *plumbing.Reference) (updated bool, err error) { |
|
|
|
if !resolvedRef.Name().IsBranch() { |
|
// Detached HEAD mode |
|
h, err := r.resolveToCommitHash(resolvedRef.Hash()) |
|
if err != nil { |
|
return false, err |
|
} |
|
head := plumbing.NewHashReference(plumbing.HEAD, h) |
|
return updateReferenceStorerIfNeeded(r.Storer, head) |
|
} |
|
|
|
refs := []*plumbing.Reference{ |
|
// Create local reference for the resolved ref |
|
resolvedRef, |
|
// Create local symbolic HEAD |
|
plumbing.NewSymbolicReference(plumbing.HEAD, resolvedRef.Name()), |
|
} |
|
|
|
refs = append(refs, r.calculateRemoteHeadReference(spec, resolvedRef)...) |
|
|
|
for _, ref := range refs { |
|
u, err := updateReferenceStorerIfNeeded(r.Storer, ref) |
|
if err != nil { |
|
return updated, err |
|
} |
|
|
|
if u { |
|
updated = true |
|
} |
|
} |
|
|
|
return |
|
} |
|
|
|
func (r *Repository) calculateRemoteHeadReference(spec []config.RefSpec, |
|
resolvedHead *plumbing.Reference) []*plumbing.Reference { |
|
|
|
var refs []*plumbing.Reference |
|
|
|
// Create resolved HEAD reference with remote prefix if it does not |
|
// exist. This is needed when using single branch and HEAD. |
|
for _, rs := range spec { |
|
name := resolvedHead.Name() |
|
if !rs.Match(name) { |
|
continue |
|
} |
|
|
|
name = rs.Dst(name) |
|
_, err := r.Storer.Reference(name) |
|
if err == plumbing.ErrReferenceNotFound { |
|
refs = append(refs, plumbing.NewHashReference(name, resolvedHead.Hash())) |
|
} |
|
} |
|
|
|
return refs |
|
} |
|
|
|
func checkAndUpdateReferenceStorerIfNeeded( |
|
s storer.ReferenceStorer, r, old *plumbing.Reference) ( |
|
updated bool, err error) { |
|
p, err := s.Reference(r.Name()) |
|
if err != nil && err != plumbing.ErrReferenceNotFound { |
|
return false, err |
|
} |
|
|
|
// we use the string method to compare references, is the easiest way |
|
if err == plumbing.ErrReferenceNotFound || r.String() != p.String() { |
|
if err := s.CheckAndSetReference(r, old); err != nil { |
|
return false, err |
|
} |
|
|
|
return true, nil |
|
} |
|
|
|
return false, nil |
|
} |
|
|
|
func updateReferenceStorerIfNeeded( |
|
s storer.ReferenceStorer, r *plumbing.Reference) (updated bool, err error) { |
|
return checkAndUpdateReferenceStorerIfNeeded(s, r, nil) |
|
} |
|
|
|
// Fetch fetches references along with the objects necessary to complete |
|
// their histories, from the remote named as FetchOptions.RemoteName. |
|
// |
|
// Returns nil if the operation is successful, NoErrAlreadyUpToDate if there are |
|
// no changes to be fetched, or an error. |
|
func (r *Repository) Fetch(o *FetchOptions) error { |
|
return r.FetchContext(context.Background(), o) |
|
} |
|
|
|
// FetchContext fetches references along with the objects necessary to complete |
|
// their histories, from the remote named as FetchOptions.RemoteName. |
|
// |
|
// Returns nil if the operation is successful, NoErrAlreadyUpToDate if there are |
|
// no changes to be fetched, or an error. |
|
// |
|
// The provided Context must be non-nil. If the context expires before the |
|
// operation is complete, an error is returned. The context only affects to the |
|
// transport operations. |
|
func (r *Repository) FetchContext(ctx context.Context, o *FetchOptions) error { |
|
if err := o.Validate(); err != nil { |
|
return err |
|
} |
|
|
|
remote, err := r.Remote(o.RemoteName) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
return remote.FetchContext(ctx, o) |
|
} |
|
|
|
// Push performs a push to the remote. Returns NoErrAlreadyUpToDate if |
|
// the remote was already up-to-date, from the remote named as |
|
// FetchOptions.RemoteName. |
|
func (r *Repository) Push(o *PushOptions) error { |
|
return r.PushContext(context.Background(), o) |
|
} |
|
|
|
// PushContext performs a push to the remote. Returns NoErrAlreadyUpToDate if |
|
// the remote was already up-to-date, from the remote named as |
|
// FetchOptions.RemoteName. |
|
// |
|
// The provided Context must be non-nil. If the context expires before the |
|
// operation is complete, an error is returned. The context only affects to the |
|
// transport operations. |
|
func (r *Repository) PushContext(ctx context.Context, o *PushOptions) error { |
|
if err := o.Validate(); err != nil { |
|
return err |
|
} |
|
|
|
remote, err := r.Remote(o.RemoteName) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
return remote.PushContext(ctx, o) |
|
} |
|
|
|
// Log returns the commit history from the given LogOptions. |
|
func (r *Repository) Log(o *LogOptions) (object.CommitIter, error) { |
|
fn := commitIterFunc(o.Order) |
|
if fn == nil { |
|
return nil, fmt.Errorf("invalid Order=%v", o.Order) |
|
} |
|
|
|
var ( |
|
it object.CommitIter |
|
err error |
|
) |
|
if o.All { |
|
it, err = r.logAll(fn) |
|
} else { |
|
it, err = r.log(o.From, fn) |
|
} |
|
|
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
if o.FileName != nil { |
|
// for `git log --all` also check parent (if the next commit comes from the real parent) |
|
it = r.logWithFile(*o.FileName, it, o.All) |
|
} |
|
|
|
return it, nil |
|
} |
|
|
|
func (r *Repository) log(from plumbing.Hash, commitIterFunc func(*object.Commit) object.CommitIter) (object.CommitIter, error) { |
|
h := from |
|
if from == plumbing.ZeroHash { |
|
head, err := r.Head() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
h = head.Hash() |
|
} |
|
|
|
commit, err := r.CommitObject(h) |
|
if err != nil { |
|
return nil, err |
|
} |
|
return commitIterFunc(commit), nil |
|
} |
|
|
|
func (r *Repository) logAll(commitIterFunc func(*object.Commit) object.CommitIter) (object.CommitIter, error) { |
|
return object.NewCommitAllIter(r.Storer, commitIterFunc) |
|
} |
|
|
|
func (*Repository) logWithFile(fileName string, commitIter object.CommitIter, checkParent bool) object.CommitIter { |
|
return object.NewCommitFileIterFromIter(fileName, commitIter, checkParent) |
|
} |
|
|
|
func commitIterFunc(order LogOrder) func(c *object.Commit) object.CommitIter { |
|
switch order { |
|
case LogOrderDefault: |
|
return func(c *object.Commit) object.CommitIter { |
|
return object.NewCommitPreorderIter(c, nil, nil) |
|
} |
|
case LogOrderDFS: |
|
return func(c *object.Commit) object.CommitIter { |
|
return object.NewCommitPreorderIter(c, nil, nil) |
|
} |
|
case LogOrderDFSPost: |
|
return func(c *object.Commit) object.CommitIter { |
|
return object.NewCommitPostorderIter(c, nil) |
|
} |
|
case LogOrderBSF: |
|
return func(c *object.Commit) object.CommitIter { |
|
return object.NewCommitIterBSF(c, nil, nil) |
|
} |
|
case LogOrderCommitterTime: |
|
return func(c *object.Commit) object.CommitIter { |
|
return object.NewCommitIterCTime(c, nil, nil) |
|
} |
|
} |
|
return nil |
|
} |
|
|
|
// Tags returns all the tag References in a repository. |
|
// |
|
// If you want to check to see if the tag is an annotated tag, you can call |
|
// TagObject on the hash Reference passed in through ForEach: |
|
// |
|
// iter, err := r.Tags() |
|
// if err != nil { |
|
// // Handle error |
|
// } |
|
// |
|
// if err := iter.ForEach(func (ref *plumbing.Reference) error { |
|
// obj, err := r.TagObject(ref.Hash()) |
|
// switch err { |
|
// case nil: |
|
// // Tag object present |
|
// case plumbing.ErrObjectNotFound: |
|
// // Not a tag object |
|
// default: |
|
// // Some other error |
|
// return err |
|
// } |
|
// }); err != nil { |
|
// // Handle outer iterator error |
|
// } |
|
// |
|
func (r *Repository) Tags() (storer.ReferenceIter, error) { |
|
refIter, err := r.Storer.IterReferences() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return storer.NewReferenceFilteredIter( |
|
func(r *plumbing.Reference) bool { |
|
return r.Name().IsTag() |
|
}, refIter), nil |
|
} |
|
|
|
// Branches returns all the References that are Branches. |
|
func (r *Repository) Branches() (storer.ReferenceIter, error) { |
|
refIter, err := r.Storer.IterReferences() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return storer.NewReferenceFilteredIter( |
|
func(r *plumbing.Reference) bool { |
|
return r.Name().IsBranch() |
|
}, refIter), nil |
|
} |
|
|
|
// Notes returns all the References that are notes. For more information: |
|
// https://git-scm.com/docs/git-notes |
|
func (r *Repository) Notes() (storer.ReferenceIter, error) { |
|
refIter, err := r.Storer.IterReferences() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return storer.NewReferenceFilteredIter( |
|
func(r *plumbing.Reference) bool { |
|
return r.Name().IsNote() |
|
}, refIter), nil |
|
} |
|
|
|
// TreeObject return a Tree with the given hash. If not found |
|
// plumbing.ErrObjectNotFound is returned |
|
func (r *Repository) TreeObject(h plumbing.Hash) (*object.Tree, error) { |
|
return object.GetTree(r.Storer, h) |
|
} |
|
|
|
// TreeObjects returns an unsorted TreeIter with all the trees in the repository |
|
func (r *Repository) TreeObjects() (*object.TreeIter, error) { |
|
iter, err := r.Storer.IterEncodedObjects(plumbing.TreeObject) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return object.NewTreeIter(r.Storer, iter), nil |
|
} |
|
|
|
// CommitObject return a Commit with the given hash. If not found |
|
// plumbing.ErrObjectNotFound is returned. |
|
func (r *Repository) CommitObject(h plumbing.Hash) (*object.Commit, error) { |
|
return object.GetCommit(r.Storer, h) |
|
} |
|
|
|
// CommitObjects returns an unsorted CommitIter with all the commits in the repository. |
|
func (r *Repository) CommitObjects() (object.CommitIter, error) { |
|
iter, err := r.Storer.IterEncodedObjects(plumbing.CommitObject) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return object.NewCommitIter(r.Storer, iter), nil |
|
} |
|
|
|
// BlobObject returns a Blob with the given hash. If not found |
|
// plumbing.ErrObjectNotFound is returned. |
|
func (r *Repository) BlobObject(h plumbing.Hash) (*object.Blob, error) { |
|
return object.GetBlob(r.Storer, h) |
|
} |
|
|
|
// BlobObjects returns an unsorted BlobIter with all the blobs in the repository. |
|
func (r *Repository) BlobObjects() (*object.BlobIter, error) { |
|
iter, err := r.Storer.IterEncodedObjects(plumbing.BlobObject) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return object.NewBlobIter(r.Storer, iter), nil |
|
} |
|
|
|
// TagObject returns a Tag with the given hash. If not found |
|
// plumbing.ErrObjectNotFound is returned. This method only returns |
|
// annotated Tags, no lightweight Tags. |
|
func (r *Repository) TagObject(h plumbing.Hash) (*object.Tag, error) { |
|
return object.GetTag(r.Storer, h) |
|
} |
|
|
|
// TagObjects returns a unsorted TagIter that can step through all of the annotated |
|
// tags in the repository. |
|
func (r *Repository) TagObjects() (*object.TagIter, error) { |
|
iter, err := r.Storer.IterEncodedObjects(plumbing.TagObject) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return object.NewTagIter(r.Storer, iter), nil |
|
} |
|
|
|
// Object returns an Object with the given hash. If not found |
|
// plumbing.ErrObjectNotFound is returned. |
|
func (r *Repository) Object(t plumbing.ObjectType, h plumbing.Hash) (object.Object, error) { |
|
obj, err := r.Storer.EncodedObject(t, h) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return object.DecodeObject(r.Storer, obj) |
|
} |
|
|
|
// Objects returns an unsorted ObjectIter with all the objects in the repository. |
|
func (r *Repository) Objects() (*object.ObjectIter, error) { |
|
iter, err := r.Storer.IterEncodedObjects(plumbing.AnyObject) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return object.NewObjectIter(r.Storer, iter), nil |
|
} |
|
|
|
// Head returns the reference where HEAD is pointing to. |
|
func (r *Repository) Head() (*plumbing.Reference, error) { |
|
return storer.ResolveReference(r.Storer, plumbing.HEAD) |
|
} |
|
|
|
// Reference returns the reference for a given reference name. If resolved is |
|
// true, any symbolic reference will be resolved. |
|
func (r *Repository) Reference(name plumbing.ReferenceName, resolved bool) ( |
|
*plumbing.Reference, error) { |
|
|
|
if resolved { |
|
return storer.ResolveReference(r.Storer, name) |
|
} |
|
|
|
return r.Storer.Reference(name) |
|
} |
|
|
|
// References returns an unsorted ReferenceIter for all references. |
|
func (r *Repository) References() (storer.ReferenceIter, error) { |
|
return r.Storer.IterReferences() |
|
} |
|
|
|
// Worktree returns a worktree based on the given fs, if nil the default |
|
// worktree will be used. |
|
func (r *Repository) Worktree() (*Worktree, error) { |
|
if r.wt == nil { |
|
return nil, ErrIsBareRepository |
|
} |
|
|
|
return &Worktree{r: r, Filesystem: r.wt}, nil |
|
} |
|
|
|
func countTrue(vals ...bool) int { |
|
sum := 0 |
|
for _, v := range vals { |
|
if v { |
|
sum++ |
|
} |
|
} |
|
return sum |
|
} |
|
|
|
// ResolveRevision resolves revision to corresponding hash. It will always |
|
// resolve to a commit hash, not a tree or annotated tag. |
|
// |
|
// Implemented resolvers : HEAD, branch, tag, heads/branch, refs/heads/branch, |
|
// refs/tags/tag, refs/remotes/origin/branch, refs/remotes/origin/HEAD, tilde and caret (HEAD~1, master~^, tag~2, ref/heads/master~1, ...), selection by text (HEAD^{/fix nasty bug}) |
|
func (r *Repository) ResolveRevision(rev plumbing.Revision) (*plumbing.Hash, error) { |
|
p := revision.NewParserFromString(string(rev)) |
|
|
|
items, err := p.Parse() |
|
|
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
var commit *object.Commit |
|
|
|
for _, item := range items { |
|
switch item.(type) { |
|
case revision.Ref: |
|
revisionRef := item.(revision.Ref) |
|
var ref *plumbing.Reference |
|
var hashCommit, refCommit, tagCommit *object.Commit |
|
var rErr, hErr, tErr error |
|
|
|
for _, rule := range append([]string{"%s"}, plumbing.RefRevParseRules...) { |
|
ref, err = storer.ResolveReference(r.Storer, plumbing.ReferenceName(fmt.Sprintf(rule, revisionRef))) |
|
|
|
if err == nil { |
|
break |
|
} |
|
} |
|
|
|
if ref != nil { |
|
tag, tObjErr := r.TagObject(ref.Hash()) |
|
if tObjErr != nil { |
|
tErr = tObjErr |
|
} else { |
|
tagCommit, tErr = tag.Commit() |
|
} |
|
refCommit, rErr = r.CommitObject(ref.Hash()) |
|
} else { |
|
rErr = plumbing.ErrReferenceNotFound |
|
tErr = plumbing.ErrReferenceNotFound |
|
} |
|
|
|
maybeHash := plumbing.NewHash(string(revisionRef)).String() == string(revisionRef) |
|
if maybeHash { |
|
hashCommit, hErr = r.CommitObject(plumbing.NewHash(string(revisionRef))) |
|
} else { |
|
hErr = plumbing.ErrReferenceNotFound |
|
} |
|
|
|
isTag := tErr == nil |
|
isCommit := rErr == nil |
|
isHash := hErr == nil |
|
|
|
switch { |
|
case countTrue(isTag, isCommit, isHash) > 1: |
|
return &plumbing.ZeroHash, fmt.Errorf(`refname "%s" is ambiguous`, revisionRef) |
|
case isTag: |
|
commit = tagCommit |
|
case isCommit: |
|
commit = refCommit |
|
case isHash: |
|
commit = hashCommit |
|
default: |
|
return &plumbing.ZeroHash, plumbing.ErrReferenceNotFound |
|
} |
|
case revision.CaretPath: |
|
depth := item.(revision.CaretPath).Depth |
|
|
|
if depth == 0 { |
|
break |
|
} |
|
|
|
iter := commit.Parents() |
|
|
|
c, err := iter.Next() |
|
|
|
if err != nil { |
|
return &plumbing.ZeroHash, err |
|
} |
|
|
|
if depth == 1 { |
|
commit = c |
|
|
|
break |
|
} |
|
|
|
c, err = iter.Next() |
|
|
|
if err != nil { |
|
return &plumbing.ZeroHash, err |
|
} |
|
|
|
commit = c |
|
case revision.TildePath: |
|
for i := 0; i < item.(revision.TildePath).Depth; i++ { |
|
c, err := commit.Parents().Next() |
|
|
|
if err != nil { |
|
return &plumbing.ZeroHash, err |
|
} |
|
|
|
commit = c |
|
} |
|
case revision.CaretReg: |
|
history := object.NewCommitPreorderIter(commit, nil, nil) |
|
|
|
re := item.(revision.CaretReg).Regexp |
|
negate := item.(revision.CaretReg).Negate |
|
|
|
var c *object.Commit |
|
|
|
err := history.ForEach(func(hc *object.Commit) error { |
|
if !negate && re.MatchString(hc.Message) { |
|
c = hc |
|
return storer.ErrStop |
|
} |
|
|
|
if negate && !re.MatchString(hc.Message) { |
|
c = hc |
|
return storer.ErrStop |
|
} |
|
|
|
return nil |
|
}) |
|
if err != nil { |
|
return &plumbing.ZeroHash, err |
|
} |
|
|
|
if c == nil { |
|
return &plumbing.ZeroHash, fmt.Errorf(`No commit message match regexp : "%s"`, re.String()) |
|
} |
|
|
|
commit = c |
|
} |
|
} |
|
|
|
return &commit.Hash, nil |
|
} |
|
|
|
type RepackConfig struct { |
|
// UseRefDeltas configures whether packfile encoder will use reference deltas. |
|
// By default OFSDeltaObject is used. |
|
UseRefDeltas bool |
|
// OnlyDeletePacksOlderThan if set to non-zero value |
|
// selects only objects older than the time provided. |
|
OnlyDeletePacksOlderThan time.Time |
|
} |
|
|
|
func (r *Repository) RepackObjects(cfg *RepackConfig) (err error) { |
|
pos, ok := r.Storer.(storer.PackedObjectStorer) |
|
if !ok { |
|
return ErrPackedObjectsNotSupported |
|
} |
|
|
|
// Get the existing object packs. |
|
hs, err := pos.ObjectPacks() |
|
if err != nil { |
|
return err |
|
} |
|
|
|
// Create a new pack. |
|
nh, err := r.createNewObjectPack(cfg) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
// Delete old packs. |
|
for _, h := range hs { |
|
// Skip if new hash is the same as an old one. |
|
if h == nh { |
|
continue |
|
} |
|
err = pos.DeleteOldObjectPackAndIndex(h, cfg.OnlyDeletePacksOlderThan) |
|
if err != nil { |
|
return err |
|
} |
|
} |
|
|
|
return nil |
|
} |
|
|
|
// createNewObjectPack is a helper for RepackObjects taking care |
|
// of creating a new pack. It is used so the the PackfileWriter |
|
// deferred close has the right scope. |
|
func (r *Repository) createNewObjectPack(cfg *RepackConfig) (h plumbing.Hash, err error) { |
|
ow := newObjectWalker(r.Storer) |
|
err = ow.walkAllRefs() |
|
if err != nil { |
|
return h, err |
|
} |
|
objs := make([]plumbing.Hash, 0, len(ow.seen)) |
|
for h := range ow.seen { |
|
objs = append(objs, h) |
|
} |
|
pfw, ok := r.Storer.(storer.PackfileWriter) |
|
if !ok { |
|
return h, fmt.Errorf("Repository storer is not a storer.PackfileWriter") |
|
} |
|
wc, err := pfw.PackfileWriter() |
|
if err != nil { |
|
return h, err |
|
} |
|
defer ioutil.CheckClose(wc, &err) |
|
scfg, err := r.Storer.Config() |
|
if err != nil { |
|
return h, err |
|
} |
|
enc := packfile.NewEncoder(wc, r.Storer, cfg.UseRefDeltas) |
|
h, err = enc.Encode(objs, scfg.Pack.Window) |
|
if err != nil { |
|
return h, err |
|
} |
|
|
|
// Delete the packed, loose objects. |
|
if los, ok := r.Storer.(storer.LooseObjectStorer); ok { |
|
err = los.ForEachObjectHash(func(hash plumbing.Hash) error { |
|
if ow.isSeen(hash) { |
|
err = los.DeleteLooseObject(hash) |
|
if err != nil { |
|
return err |
|
} |
|
} |
|
return nil |
|
}) |
|
if err != nil { |
|
return h, err |
|
} |
|
} |
|
|
|
return h, err |
|
}
|
|
|