2019-01-13 04:45:25 +00:00
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
2020-02-07 06:17:58 +00:00
|
|
|
"context"
|
2019-01-13 04:45:25 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2020-02-10 23:27:05 +00:00
|
|
|
"io"
|
2019-01-13 04:45:25 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2021-01-12 06:47:33 +00:00
|
|
|
"path"
|
2019-01-13 04:45:25 +00:00
|
|
|
"path/filepath"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
|
2020-04-16 23:24:30 +00:00
|
|
|
git "github.com/go-git/go-git/v5"
|
|
|
|
"github.com/go-git/go-git/v5/plumbing"
|
2021-05-05 16:42:34 +00:00
|
|
|
"github.com/go-git/go-git/v5/plumbing/transport/http"
|
2019-01-13 04:45:25 +00:00
|
|
|
"github.com/go-ini/ini"
|
2021-05-06 13:55:23 +00:00
|
|
|
"github.com/mattn/go-isatty"
|
2019-01-13 04:45:25 +00:00
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
2019-02-14 20:24:18 +00:00
|
|
|
var (
|
|
|
|
codeCommitHTTPRegex = regexp.MustCompile(`^https?://git-codecommit\.(.+)\.amazonaws.com/v1/repos/(.+)$`)
|
|
|
|
codeCommitSSHRegex = regexp.MustCompile(`ssh://git-codecommit\.(.+)\.amazonaws.com/v1/repos/(.+)$`)
|
|
|
|
githubHTTPRegex = regexp.MustCompile(`^https?://.*github.com.*/(.+)/(.+?)(?:.git)?$`)
|
|
|
|
githubSSHRegex = regexp.MustCompile(`github.com[:/](.+)/(.+).git$`)
|
|
|
|
|
|
|
|
cloneLock sync.Mutex
|
|
|
|
)
|
2019-01-13 04:45:25 +00:00
|
|
|
|
|
|
|
// FindGitRevision get the current git revision
|
|
|
|
func FindGitRevision(file string) (shortSha string, sha string, err error) {
|
|
|
|
gitDir, err := findGitDirectory(file)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
|
2019-05-23 04:39:57 +00:00
|
|
|
bts, err := ioutil.ReadFile(filepath.Join(gitDir, "HEAD"))
|
2019-01-13 04:45:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
2019-01-17 23:53:03 +00:00
|
|
|
|
2019-05-23 04:39:57 +00:00
|
|
|
var ref = strings.TrimSpace(strings.TrimPrefix(string(bts), "ref:"))
|
2019-01-17 23:53:03 +00:00
|
|
|
var refBuf []byte
|
2019-01-18 21:28:53 +00:00
|
|
|
if strings.HasPrefix(ref, "refs/") {
|
2019-01-17 23:53:03 +00:00
|
|
|
// load commitid ref
|
2019-05-23 04:39:57 +00:00
|
|
|
refBuf, err = ioutil.ReadFile(filepath.Join(gitDir, ref))
|
2019-01-17 23:53:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
} else {
|
2019-01-18 21:28:53 +00:00
|
|
|
refBuf = []byte(ref)
|
2019-01-13 04:45:25 +00:00
|
|
|
}
|
2019-01-17 23:53:03 +00:00
|
|
|
|
|
|
|
log.Debugf("Found revision: %s", refBuf)
|
2019-02-13 04:02:24 +00:00
|
|
|
return string(refBuf[:7]), strings.TrimSpace(string(refBuf)), nil
|
2019-01-13 04:45:25 +00:00
|
|
|
}
|
|
|
|
|
2019-01-18 21:28:53 +00:00
|
|
|
// FindGitRef get the current git ref
|
|
|
|
func FindGitRef(file string) (string, error) {
|
2019-01-13 04:45:25 +00:00
|
|
|
gitDir, err := findGitDirectory(file)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2019-05-23 04:39:57 +00:00
|
|
|
log.Debugf("Loading revision from git directory '%s'", gitDir)
|
2019-01-13 04:45:25 +00:00
|
|
|
|
2019-05-23 04:39:57 +00:00
|
|
|
_, ref, err := FindGitRevision(file)
|
2019-01-13 04:45:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2019-01-17 22:13:15 +00:00
|
|
|
log.Debugf("HEAD points to '%s'", ref)
|
2019-01-13 04:45:25 +00:00
|
|
|
|
2021-05-03 14:32:00 +00:00
|
|
|
// Prefer the git library to iterate over the references and find a matching tag or branch.
|
|
|
|
var refTag = ""
|
|
|
|
var refBranch = ""
|
|
|
|
r, err := git.PlainOpen(filepath.Join(gitDir, ".."))
|
|
|
|
if err == nil {
|
|
|
|
iter, err := r.References()
|
|
|
|
if err == nil {
|
|
|
|
for {
|
|
|
|
r, err := iter.Next()
|
|
|
|
if r == nil || err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
log.Debugf("Reference: name=%s sha=%s", r.Name().String(), r.Hash().String())
|
|
|
|
if r.Hash().String() == ref {
|
|
|
|
if r.Name().IsTag() {
|
|
|
|
refTag = r.Name().String()
|
|
|
|
}
|
|
|
|
if r.Name().IsBranch() {
|
|
|
|
refBranch = r.Name().String()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
iter.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if refTag != "" {
|
|
|
|
return refTag, nil
|
|
|
|
}
|
|
|
|
if refBranch != "" {
|
|
|
|
return refBranch, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the above doesn't work, fall back to the old way
|
|
|
|
|
2019-05-23 03:05:14 +00:00
|
|
|
// try tags first
|
2019-05-23 04:45:50 +00:00
|
|
|
tag, err := findGitPrettyRef(ref, gitDir, "refs/tags")
|
2019-05-23 03:05:14 +00:00
|
|
|
if err != nil || tag != "" {
|
|
|
|
return tag, err
|
|
|
|
}
|
|
|
|
// and then branches
|
2019-05-23 04:45:50 +00:00
|
|
|
return findGitPrettyRef(ref, gitDir, "refs/heads")
|
2019-05-23 03:05:14 +00:00
|
|
|
}
|
|
|
|
|
2019-05-23 04:45:50 +00:00
|
|
|
func findGitPrettyRef(head, root, sub string) (string, error) {
|
2019-05-23 03:05:14 +00:00
|
|
|
var name string
|
|
|
|
var err = filepath.Walk(filepath.Join(root, sub), func(path string, info os.FileInfo, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if name != "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if info.IsDir() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
bts, err := ioutil.ReadFile(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var pointsTo = strings.TrimSpace(string(bts))
|
|
|
|
if head == pointsTo {
|
2021-03-13 00:23:03 +00:00
|
|
|
// On Windows paths are separated with backslash character so they should be replaced to provide proper git refs format
|
|
|
|
name = strings.TrimPrefix(strings.ReplaceAll(strings.Replace(path, root, "", 1), `\`, `/`), "/")
|
2019-05-23 03:05:14 +00:00
|
|
|
log.Debugf("HEAD matches %s", name)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return name, err
|
2019-01-13 04:45:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// FindGithubRepo get the repo
|
2021-05-05 16:42:34 +00:00
|
|
|
func FindGithubRepo(file string, githubInstance string) (string, error) {
|
2019-01-13 04:45:25 +00:00
|
|
|
url, err := findGitRemoteURL(file)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2021-05-05 16:42:34 +00:00
|
|
|
_, slug, err := findGitSlug(url, githubInstance)
|
2019-01-13 04:45:25 +00:00
|
|
|
return slug, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func findGitRemoteURL(file string) (string, error) {
|
|
|
|
gitDir, err := findGitDirectory(file)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
log.Debugf("Loading slug from git directory '%s'", gitDir)
|
|
|
|
|
|
|
|
gitconfig, err := ini.InsensitiveLoad(fmt.Sprintf("%s/config", gitDir))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
remote, err := gitconfig.GetSection("remote \"origin\"")
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
urlKey, err := remote.GetKey("url")
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
url := urlKey.String()
|
|
|
|
return url, nil
|
|
|
|
}
|
|
|
|
|
2021-05-05 16:42:34 +00:00
|
|
|
func findGitSlug(url string, githubInstance string) (string, string, error) {
|
2019-01-13 04:45:25 +00:00
|
|
|
if matches := codeCommitHTTPRegex.FindStringSubmatch(url); matches != nil {
|
2019-02-14 20:24:18 +00:00
|
|
|
return "CodeCommit", matches[2], nil
|
2019-01-13 04:45:25 +00:00
|
|
|
} else if matches := codeCommitSSHRegex.FindStringSubmatch(url); matches != nil {
|
|
|
|
return "CodeCommit", matches[2], nil
|
2019-02-14 20:24:18 +00:00
|
|
|
} else if matches := githubHTTPRegex.FindStringSubmatch(url); matches != nil {
|
|
|
|
return "GitHub", fmt.Sprintf("%s/%s", matches[1], matches[2]), nil
|
|
|
|
} else if matches := githubSSHRegex.FindStringSubmatch(url); matches != nil {
|
2019-01-13 04:45:25 +00:00
|
|
|
return "GitHub", fmt.Sprintf("%s/%s", matches[1], matches[2]), nil
|
2021-05-05 16:42:34 +00:00
|
|
|
} else if githubInstance != "github.com" {
|
|
|
|
gheHTTPRegex := regexp.MustCompile(fmt.Sprintf(`^https?://%s/(.+)/(.+?)(?:.git)?$`, githubInstance))
|
|
|
|
gheSSHRegex := regexp.MustCompile(fmt.Sprintf(`%s[:/](.+)/(.+).git$`, githubInstance))
|
|
|
|
if matches := gheHTTPRegex.FindStringSubmatch(url); matches != nil {
|
|
|
|
return "GitHubEnterprise", fmt.Sprintf("%s/%s", matches[1], matches[2]), nil
|
|
|
|
} else if matches := gheSSHRegex.FindStringSubmatch(url); matches != nil {
|
|
|
|
return "GitHubEnterprise", fmt.Sprintf("%s/%s", matches[1], matches[2]), nil
|
|
|
|
}
|
2019-01-13 04:45:25 +00:00
|
|
|
}
|
|
|
|
return "", url, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func findGitDirectory(fromFile string) (string, error) {
|
|
|
|
absPath, err := filepath.Abs(fromFile)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
fi, err := os.Stat(absPath)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
var dir string
|
|
|
|
if fi.Mode().IsDir() {
|
|
|
|
dir = absPath
|
|
|
|
} else {
|
2019-05-23 04:41:56 +00:00
|
|
|
dir = filepath.Dir(absPath)
|
2019-01-13 04:45:25 +00:00
|
|
|
}
|
|
|
|
|
2019-05-23 04:39:57 +00:00
|
|
|
gitPath := filepath.Join(dir, ".git")
|
2019-01-13 04:45:25 +00:00
|
|
|
fi, err = os.Stat(gitPath)
|
|
|
|
if err == nil && fi.Mode().IsDir() {
|
|
|
|
return gitPath, nil
|
|
|
|
} else if dir == "/" || dir == "C:\\" || dir == "c:\\" {
|
|
|
|
return "", errors.New("unable to find git repo")
|
|
|
|
}
|
|
|
|
|
|
|
|
return findGitDirectory(filepath.Dir(dir))
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewGitCloneExecutorInput the input for the NewGitCloneExecutor
|
|
|
|
type NewGitCloneExecutorInput struct {
|
2021-05-05 16:42:34 +00:00
|
|
|
URL string
|
|
|
|
Ref string
|
|
|
|
Dir string
|
|
|
|
Token string
|
2019-01-13 04:45:25 +00:00
|
|
|
}
|
|
|
|
|
2021-03-30 19:26:25 +00:00
|
|
|
// CloneIfRequired ...
|
2021-04-06 13:43:02 +00:00
|
|
|
func CloneIfRequired(ctx context.Context, refName plumbing.ReferenceName, input NewGitCloneExecutorInput, logger log.FieldLogger) (*git.Repository, error) {
|
2020-06-09 14:43:26 +00:00
|
|
|
r, err := git.PlainOpen(input.Dir)
|
|
|
|
if err != nil {
|
|
|
|
var progressWriter io.Writer
|
2021-05-06 13:55:23 +00:00
|
|
|
if isatty.IsTerminal(os.Stdout.Fd()) || isatty.IsCygwinTerminal(os.Stdout.Fd()) {
|
|
|
|
if entry, ok := logger.(*log.Entry); ok {
|
|
|
|
progressWriter = entry.WriterLevel(log.DebugLevel)
|
|
|
|
} else if lgr, ok := logger.(*log.Logger); ok {
|
|
|
|
progressWriter = lgr.WriterLevel(log.DebugLevel)
|
|
|
|
} else {
|
|
|
|
log.Errorf("Unable to get writer from logger (type=%T)", logger)
|
|
|
|
progressWriter = os.Stdout
|
|
|
|
}
|
2020-06-09 14:43:26 +00:00
|
|
|
}
|
|
|
|
|
2021-05-06 13:55:23 +00:00
|
|
|
cloneOptions := git.CloneOptions{
|
|
|
|
URL: input.URL,
|
|
|
|
Progress: progressWriter,
|
|
|
|
}
|
2021-05-05 16:42:34 +00:00
|
|
|
if input.Token != "" {
|
2021-05-06 13:55:23 +00:00
|
|
|
cloneOptions.Auth = &http.BasicAuth{
|
|
|
|
Username: "token",
|
|
|
|
Password: input.Token,
|
2021-05-05 16:42:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err = git.PlainCloneContext(ctx, input.Dir, false, &cloneOptions)
|
2020-06-09 14:43:26 +00:00
|
|
|
if err != nil {
|
|
|
|
logger.Errorf("Unable to clone %v %s: %v", input.URL, refName, err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
_ = os.Chmod(input.Dir, 0755)
|
|
|
|
}
|
|
|
|
|
|
|
|
return r, nil
|
|
|
|
}
|
|
|
|
|
2019-01-13 04:45:25 +00:00
|
|
|
// NewGitCloneExecutor creates an executor to clone git repos
|
|
|
|
func NewGitCloneExecutor(input NewGitCloneExecutorInput) Executor {
|
2020-02-07 06:17:58 +00:00
|
|
|
return func(ctx context.Context) error {
|
2020-02-10 23:27:05 +00:00
|
|
|
logger := Logger(ctx)
|
2020-02-11 17:10:35 +00:00
|
|
|
logger.Infof(" \u2601 git clone '%s' # ref=%s", input.URL, input.Ref)
|
2020-02-10 23:27:05 +00:00
|
|
|
logger.Debugf(" cloning %s to %s", input.URL, input.Dir)
|
2019-01-13 04:45:25 +00:00
|
|
|
|
|
|
|
cloneLock.Lock()
|
|
|
|
defer cloneLock.Unlock()
|
|
|
|
|
2019-01-15 17:05:27 +00:00
|
|
|
refName := plumbing.ReferenceName(fmt.Sprintf("refs/heads/%s", input.Ref))
|
2021-04-06 13:43:02 +00:00
|
|
|
r, err := CloneIfRequired(ctx, refName, input, logger)
|
2019-01-13 04:45:25 +00:00
|
|
|
if err != nil {
|
2020-06-09 14:43:26 +00:00
|
|
|
return err
|
2019-01-13 04:45:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
w, err := r.Worktree()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-01-12 06:47:33 +00:00
|
|
|
// At this point we need to know if it's a tag or a branch
|
|
|
|
// And the easiest way to do it is duck typing
|
2021-01-15 05:24:17 +00:00
|
|
|
//
|
|
|
|
// If err is nil, it's a tag so let's proceed with that hash like we would if
|
|
|
|
// it was a sha
|
2021-01-12 06:47:33 +00:00
|
|
|
refType := "tag"
|
|
|
|
rev := plumbing.Revision(path.Join("refs", "tags", input.Ref))
|
|
|
|
if _, err := r.Tag(input.Ref); errors.Is(err, git.ErrTagNotFound) {
|
2021-02-23 17:50:28 +00:00
|
|
|
rName := plumbing.ReferenceName(path.Join("refs", "remotes", "origin", input.Ref))
|
|
|
|
if _, err := r.Reference(rName, false); errors.Is(err, plumbing.ErrReferenceNotFound) {
|
|
|
|
refType = "sha"
|
|
|
|
rev = plumbing.Revision(input.Ref)
|
|
|
|
} else {
|
|
|
|
refType = "branch"
|
|
|
|
rev = plumbing.Revision(rName)
|
|
|
|
}
|
2021-01-12 06:47:33 +00:00
|
|
|
}
|
|
|
|
hash, err := r.ResolveRevision(rev)
|
2020-06-09 14:43:26 +00:00
|
|
|
if err != nil {
|
|
|
|
logger.Errorf("Unable to resolve %s: %v", input.Ref, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the hash resolved doesn't match the ref provided in a workflow then we're
|
|
|
|
// using a branch or tag ref, not a sha
|
|
|
|
//
|
|
|
|
// Repos on disk point to commit hashes, and need to checkout input.Ref before
|
|
|
|
// we try and pull down any changes
|
|
|
|
if hash.String() != input.Ref {
|
|
|
|
// Run git fetch to make sure we have the latest sha
|
2021-01-15 05:24:17 +00:00
|
|
|
err := r.Fetch(&git.FetchOptions{})
|
|
|
|
if err != nil && !errors.Is(err, git.NoErrAlreadyUpToDate) {
|
2020-06-09 14:43:26 +00:00
|
|
|
logger.Debugf("Unable to fetch: %v", err)
|
|
|
|
}
|
|
|
|
|
2021-01-12 06:47:33 +00:00
|
|
|
if refType == "branch" {
|
2020-06-09 14:43:26 +00:00
|
|
|
logger.Debugf("Provided ref is not a sha. Checking out branch before pulling changes")
|
2021-01-15 05:24:17 +00:00
|
|
|
sourceRef := plumbing.ReferenceName(path.Join("refs", "remotes", "origin", input.Ref))
|
2021-01-12 06:47:33 +00:00
|
|
|
err := w.Checkout(&git.CheckoutOptions{
|
2021-01-15 05:24:17 +00:00
|
|
|
Branch: sourceRef,
|
2020-06-09 14:43:26 +00:00
|
|
|
Force: true,
|
|
|
|
})
|
|
|
|
if err != nil {
|
2021-01-15 05:24:17 +00:00
|
|
|
logger.Errorf("Unable to checkout %s: %v", sourceRef, err)
|
2020-06-09 14:43:26 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-16 01:41:02 +00:00
|
|
|
err = w.Pull(&git.PullOptions{
|
2019-01-31 08:18:52 +00:00
|
|
|
Force: true,
|
2019-01-15 17:57:36 +00:00
|
|
|
})
|
2019-01-16 18:10:24 +00:00
|
|
|
if err != nil && err.Error() != "already up-to-date" {
|
2020-03-06 18:17:20 +00:00
|
|
|
logger.Debugf("Unable to pull %s: %v", refName, err)
|
2019-01-16 01:41:02 +00:00
|
|
|
}
|
2020-02-10 23:27:05 +00:00
|
|
|
logger.Debugf("Cloned %s to %s", input.URL, input.Dir)
|
2019-01-13 04:45:25 +00:00
|
|
|
|
|
|
|
err = w.Checkout(&git.CheckoutOptions{
|
2019-01-31 08:18:52 +00:00
|
|
|
Hash: *hash,
|
|
|
|
Force: true,
|
2019-01-13 04:45:25 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
2020-02-24 20:48:12 +00:00
|
|
|
logger.Errorf("Unable to checkout %s: %v", *hash, err)
|
2019-01-13 04:45:25 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-02-21 01:47:21 +00:00
|
|
|
err = w.Reset(&git.ResetOptions{
|
|
|
|
Mode: git.HardReset,
|
|
|
|
Commit: *hash,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
logger.Errorf("Unable to reset to %s: %v", hash.String(), err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-02-10 23:27:05 +00:00
|
|
|
logger.Debugf("Checked out %s", input.Ref)
|
2019-01-13 04:45:25 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|