101 lines
2.5 KiB
Go
101 lines
2.5 KiB
Go
package githistory
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/git-lfs/git-lfs/errors"
|
|
"github.com/git-lfs/git-lfs/git"
|
|
"github.com/git-lfs/git-lfs/tasklog"
|
|
"github.com/git-lfs/git-lfs/tools"
|
|
"github.com/git-lfs/gitobj"
|
|
)
|
|
|
|
// refUpdater is a type responsible for moving references from one point in the
|
|
// Git object graph to another.
|
|
type refUpdater struct {
|
|
// CacheFn is a function that returns the SHA1 transformation from an
|
|
// original hash to a new one. It specifies a "bool" return value
|
|
// signaling whether or not that given "old" SHA1 was migrated.
|
|
CacheFn func(old []byte) ([]byte, bool)
|
|
// Logger logs the progress of reference updating.
|
|
Logger *tasklog.Logger
|
|
// Refs is a set of *git.Ref's to migrate.
|
|
Refs []*git.Ref
|
|
// Root is the given directory on disk in which the repository is
|
|
// located.
|
|
Root string
|
|
|
|
db *gitobj.ObjectDatabase
|
|
}
|
|
|
|
// UpdateRefs performs the reference update(s) from existing locations (see:
|
|
// Refs) to their respective new locations in the graph (see CacheFn).
|
|
//
|
|
// It creates reflog entries as well as stderr log entries as it progresses
|
|
// through the reference updates.
|
|
//
|
|
// It returns any error encountered, or nil if the reference update(s) was/were
|
|
// successful.
|
|
func (r *refUpdater) UpdateRefs() error {
|
|
list := r.Logger.List("migrate: Updating refs")
|
|
defer list.Complete()
|
|
|
|
var maxNameLen int
|
|
for _, ref := range r.Refs {
|
|
maxNameLen = tools.MaxInt(maxNameLen, len(ref.Name))
|
|
}
|
|
|
|
for _, ref := range r.Refs {
|
|
sha1, err := hex.DecodeString(ref.Sha)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "could not decode: %q", ref.Sha)
|
|
}
|
|
|
|
to, ok := r.CacheFn(sha1)
|
|
|
|
if ref.Type == git.RefTypeLocalTag {
|
|
tag, _ := r.db.Tag(sha1)
|
|
if tag != nil && tag.ObjectType == gitobj.CommitObjectType {
|
|
// Assume that a non-nil error is an indication
|
|
// that the tag is bare (without annotation).
|
|
|
|
toObj, okObj := r.CacheFn(tag.Object)
|
|
if !okObj {
|
|
continue
|
|
}
|
|
|
|
newTag, err := r.db.WriteTag(&gitobj.Tag{
|
|
Object: toObj,
|
|
ObjectType: tag.ObjectType,
|
|
Name: tag.Name,
|
|
Tagger: tag.Tagger,
|
|
|
|
Message: tag.Message,
|
|
})
|
|
|
|
if err != nil {
|
|
return errors.Wrapf(err, "could not rewrite tag: %s", tag.Name)
|
|
}
|
|
|
|
to = newTag
|
|
ok = true
|
|
}
|
|
}
|
|
|
|
if !ok {
|
|
continue
|
|
}
|
|
|
|
if err := git.UpdateRefIn(r.Root, ref, to, ""); err != nil {
|
|
return err
|
|
}
|
|
|
|
namePadding := tools.MaxInt(maxNameLen-len(ref.Name), 0)
|
|
list.Entry(fmt.Sprintf(" %s%s\t%s -> %x", ref.Name, strings.Repeat(" ", namePadding), ref.Sha, to))
|
|
}
|
|
|
|
return nil
|
|
}
|