224 lines
5.9 KiB
Go
224 lines
5.9 KiB
Go
package sv
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/Masterminds/semver"
|
|
)
|
|
|
|
const (
|
|
logSeparator = "##"
|
|
endLine = "~~"
|
|
|
|
// BreakingChangesKey key to breaking change metadata
|
|
BreakingChangesKey = "breakingchange"
|
|
// IssueIDKey key to issue id metadata
|
|
IssueIDKey = "issueid"
|
|
)
|
|
|
|
// Git commands
|
|
type Git interface {
|
|
Describe() string
|
|
Log(initialTag, endTag string) ([]GitCommitLog, error)
|
|
Commit(header, body, footer string) error
|
|
Tag(version semver.Version) error
|
|
Tags() ([]GitTag, error)
|
|
Branch() string
|
|
}
|
|
|
|
// GitCommitLog description of a single commit log
|
|
type GitCommitLog struct {
|
|
Hash string `json:"hash,omitempty"`
|
|
Type string `json:"type,omitempty"`
|
|
Scope string `json:"scope,omitempty"`
|
|
Subject string `json:"subject,omitempty"`
|
|
Body string `json:"body,omitempty"`
|
|
Metadata map[string]string `json:"metadata,omitempty"`
|
|
}
|
|
|
|
// GitTag git tag info
|
|
type GitTag struct {
|
|
Name string
|
|
Date time.Time
|
|
}
|
|
|
|
// GitImpl git command implementation
|
|
type GitImpl struct {
|
|
messageMetadata map[string][]string
|
|
tagPattern string
|
|
}
|
|
|
|
// NewGit constructor
|
|
func NewGit(breakinChangePrefixes, issueIDPrefixes []string, tagPattern string) *GitImpl {
|
|
return &GitImpl{
|
|
messageMetadata: map[string][]string{BreakingChangesKey: breakinChangePrefixes, IssueIDKey: issueIDPrefixes},
|
|
tagPattern: tagPattern,
|
|
}
|
|
}
|
|
|
|
// Describe runs git describe, it no tag found, return empty
|
|
func (GitImpl) Describe() string {
|
|
cmd := exec.Command("git", "describe", "--abbrev=0")
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
return strings.TrimSpace(strings.Trim(string(out), "\n"))
|
|
}
|
|
|
|
// Log return git log
|
|
func (g GitImpl) Log(initialTag, endTag string) ([]GitCommitLog, error) {
|
|
format := "--pretty=format:\"%h" + logSeparator + "%s" + logSeparator + "%b" + endLine + "\""
|
|
var cmd *exec.Cmd
|
|
if initialTag == "" && endTag == "" {
|
|
cmd = exec.Command("git", "log", format)
|
|
} else if endTag == "" {
|
|
cmd = exec.Command("git", "log", initialTag+"..HEAD", format)
|
|
} else if initialTag == "" {
|
|
cmd = exec.Command("git", "log", endTag, format)
|
|
} else {
|
|
cmd = exec.Command("git", "log", initialTag+".."+endTag, format)
|
|
}
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return parseLogOutput(g.messageMetadata, string(out)), nil
|
|
}
|
|
|
|
// Commit runs git commit
|
|
func (g GitImpl) Commit(header, body, footer string) error {
|
|
cmd := exec.Command("git", "commit", "-m", header, "-m", "", "-m", body, "-m", "", "-m", footer)
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
return cmd.Run()
|
|
}
|
|
|
|
// Tag create a git tag
|
|
func (g GitImpl) Tag(version semver.Version) error {
|
|
tag := fmt.Sprintf(g.tagPattern, version.Major(), version.Minor(), version.Patch())
|
|
tagMsg := fmt.Sprintf("Version %d.%d.%d", version.Major(), version.Minor(), version.Patch())
|
|
|
|
tagCommand := exec.Command("git", "tag", "-a", tag, "-m", tagMsg)
|
|
if err := tagCommand.Run(); err != nil {
|
|
return err
|
|
}
|
|
|
|
pushCommand := exec.Command("git", "push", "origin", tag)
|
|
return pushCommand.Run()
|
|
}
|
|
|
|
// Tags list repository tags
|
|
func (g GitImpl) Tags() ([]GitTag, error) {
|
|
cmd := exec.Command("git", "tag", "-l", "--format", "%(taggerdate:iso8601)#%(refname:short)")
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return parseTagsOutput(string(out))
|
|
}
|
|
|
|
// Branch get git branch
|
|
func (GitImpl) Branch() string {
|
|
cmd := exec.Command("git", "symbolic-ref", "--short", "HEAD")
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
return strings.TrimSpace(strings.Trim(string(out), "\n"))
|
|
}
|
|
|
|
func parseTagsOutput(input string) ([]GitTag, error) {
|
|
scanner := bufio.NewScanner(strings.NewReader(input))
|
|
var result []GitTag
|
|
for scanner.Scan() {
|
|
if line := strings.TrimSpace(scanner.Text()); line != "" {
|
|
values := strings.Split(line, "#")
|
|
date, _ := time.Parse("2006-01-02 15:04:05 -0700", values[0]) // ignore invalid dates
|
|
result = append(result, GitTag{Name: values[1], Date: date})
|
|
}
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
func parseLogOutput(messageMetadata map[string][]string, log string) []GitCommitLog {
|
|
scanner := bufio.NewScanner(strings.NewReader(log))
|
|
scanner.Split(splitAt([]byte(endLine)))
|
|
var logs []GitCommitLog
|
|
for scanner.Scan() {
|
|
if text := strings.TrimSpace(strings.Trim(scanner.Text(), "\"")); text != "" {
|
|
logs = append(logs, parseCommitLog(messageMetadata, text))
|
|
}
|
|
}
|
|
return logs
|
|
}
|
|
|
|
func parseCommitLog(messageMetadata map[string][]string, commit string) GitCommitLog {
|
|
content := strings.Split(strings.Trim(commit, "\""), logSeparator)
|
|
commitType, scope, subject := parseCommitLogMessage(content[1])
|
|
|
|
metadata := make(map[string]string)
|
|
for key, prefixes := range messageMetadata {
|
|
for _, prefix := range prefixes {
|
|
if tagValue := extractTag(prefix, content[2]); tagValue != "" {
|
|
metadata[key] = tagValue
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
return GitCommitLog{
|
|
Hash: content[0],
|
|
Type: commitType,
|
|
Scope: scope,
|
|
Subject: subject,
|
|
Body: content[2],
|
|
Metadata: metadata,
|
|
}
|
|
}
|
|
|
|
func parseCommitLogMessage(message string) (string, string, string) {
|
|
regex := regexp.MustCompile("([a-z]+)(\\((.*)\\))?: (.*)")
|
|
result := regex.FindStringSubmatch(message)
|
|
if len(result) != 5 {
|
|
return "", "", message
|
|
}
|
|
return result[1], result[3], strings.TrimSpace(result[4])
|
|
}
|
|
|
|
func extractTag(tag, text string) string {
|
|
regex := regexp.MustCompile(tag + " (.*)")
|
|
result := regex.FindStringSubmatch(text)
|
|
if len(result) < 2 {
|
|
return ""
|
|
}
|
|
return result[1]
|
|
}
|
|
|
|
func splitAt(b []byte) func(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
|
return func(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
|
dataLen := len(data)
|
|
|
|
if atEOF && dataLen == 0 {
|
|
return 0, nil, nil
|
|
}
|
|
|
|
if i := bytes.Index(data, b); i >= 0 {
|
|
return i + len(b), data[0:i], nil
|
|
}
|
|
|
|
if atEOF {
|
|
return dataLen, data, nil
|
|
}
|
|
|
|
return 0, nil, nil
|
|
}
|
|
}
|