package common import ( "bufio" "bytes" "errors" "fmt" "io/ioutil" "os" "path" "path/filepath" "regexp" "strings" "sync" "github.com/go-ini/ini" log "github.com/sirupsen/logrus" git "gopkg.in/src-d/go-git.v4" "gopkg.in/src-d/go-git.v4/plumbing" yaml "gopkg.in/yaml.v2" ) var cloneLock sync.Mutex // 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 } ref, err := FindGitRef(file) if err != nil { return "", "", err } var refBuf []byte if strings.HasPrefix(ref, "refs/") { // load commitid ref refBuf, err = ioutil.ReadFile(fmt.Sprintf("%s/%s", gitDir, ref)) if err != nil { return "", "", err } } else { refBuf = []byte(ref) } log.Debugf("Found revision: %s", refBuf) return string(refBuf[:7]), string(refBuf), nil } // FindGitBranch get the current git branch func FindGitBranch(file string) (string, error) { ref, err := FindGitRef(file) if err != nil { return "", err } // get branch name branch := strings.TrimPrefix(ref, "refs/heads/") log.Debugf("Found branch: %s", branch) return branch, nil } // FindGitRef get the current git ref func FindGitRef(file string) (string, error) { gitDir, err := findGitDirectory(file) if err != nil { return "", err } log.Debugf("Loading revision from git directory '%s'", gitDir) // load HEAD ref headFile, err := os.Open(fmt.Sprintf("%s/HEAD", gitDir)) if err != nil { return "", err } defer func() { headFile.Close() }() headBuffer := new(bytes.Buffer) _, err = headBuffer.ReadFrom(bufio.NewReader(headFile)) if err != nil { log.Error(err) } headBytes := headBuffer.Bytes() var ref string head := make(map[string]string) err = yaml.Unmarshal(headBytes, head) if err != nil { ref = strings.TrimRight(string(headBytes), "\r\n") } else { ref = head["ref"] } log.Debugf("HEAD points to '%s'", ref) return ref, nil } // FindGithubRepo get the repo func FindGithubRepo(file string) (string, error) { url, err := findGitRemoteURL(file) if err != nil { return "", err } _, slug, err := findGitSlug(url) 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 } func findGitSlug(url string) (string, string, error) { codeCommitHTTPRegex := regexp.MustCompile(`^http(s?)://git-codecommit\.(.+)\.amazonaws.com/v1/repos/(.+)$`) codeCommitSSHRegex := regexp.MustCompile(`ssh://git-codecommit\.(.+)\.amazonaws.com/v1/repos/(.+)$`) httpRegex := regexp.MustCompile("^http(s?)://.*github.com.*/(.+)/(.+).git$") sshRegex := regexp.MustCompile("github.com:(.+)/(.+).git$") if matches := codeCommitHTTPRegex.FindStringSubmatch(url); matches != nil { return "CodeCommit", matches[3], nil } else if matches := codeCommitSSHRegex.FindStringSubmatch(url); matches != nil { return "CodeCommit", matches[2], nil } else if matches := httpRegex.FindStringSubmatch(url); matches != nil { return "GitHub", fmt.Sprintf("%s/%s", matches[2], matches[3]), nil } else if matches := sshRegex.FindStringSubmatch(url); matches != nil { return "GitHub", fmt.Sprintf("%s/%s", matches[1], matches[2]), nil } return "", url, nil } func findGitDirectory(fromFile string) (string, error) { absPath, err := filepath.Abs(fromFile) if err != nil { return "", err } log.Debugf("Searching for git directory in %s", absPath) fi, err := os.Stat(absPath) if err != nil { return "", err } var dir string if fi.Mode().IsDir() { dir = absPath } else { dir = path.Dir(absPath) } gitPath := path.Join(dir, ".git") 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 { URL string Ref string Dir string Logger *log.Entry Dryrun bool } // NewGitCloneExecutor creates an executor to clone git repos func NewGitCloneExecutor(input NewGitCloneExecutorInput) Executor { return func() error { input.Logger.Infof("git clone '%s' # ref=%s", input.URL, input.Ref) input.Logger.Debugf(" cloning %s to %s", input.URL, input.Dir) if input.Dryrun { return nil } cloneLock.Lock() defer cloneLock.Unlock() refName := plumbing.ReferenceName(fmt.Sprintf("refs/heads/%s", input.Ref)) r, err := git.PlainOpen(input.Dir) if err != nil { r, err = git.PlainClone(input.Dir, false, &git.CloneOptions{ URL: input.URL, Progress: input.Logger.WriterLevel(log.DebugLevel), //ReferenceName: refName, }) if err != nil { input.Logger.Errorf("Unable to clone %v %s: %v", input.URL, refName, err) return err } } w, err := r.Worktree() if err != nil { return err } err = w.Pull(&git.PullOptions{ //ReferenceName: refName, Force: true, }) if err != nil && err.Error() != "already up-to-date" { input.Logger.Errorf("Unable to pull %s: %v", refName, err) } input.Logger.Debugf("Cloned %s to %s", input.URL, input.Dir) hash, err := r.ResolveRevision(plumbing.Revision(input.Ref)) if err != nil { input.Logger.Errorf("Unable to resolve %s: %v", input.Ref, err) return err } err = w.Checkout(&git.CheckoutOptions{ //Branch: refName, Hash: *hash, Force: true, }) if err != nil { input.Logger.Errorf("Unable to checkout %s: %v", refName, err) return err } input.Logger.Debugf("Checked out %s", input.Ref) return nil } }