mirror of
https://github.com/thegeeklab/git-sv.git
synced 2024-11-24 01:00:39 +00:00
320 lines
7.1 KiB
Go
320 lines
7.1 KiB
Go
package sv
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/Masterminds/semver/v3"
|
|
)
|
|
|
|
const (
|
|
logSeparator = "###"
|
|
endLine = "~~~"
|
|
)
|
|
|
|
// Git commands.
|
|
type Git interface {
|
|
LastTag() string
|
|
Log(lr LogRange) ([]GitCommitLog, error)
|
|
Commit(header, body, footer string) error
|
|
Tag(version semver.Version) (string, error)
|
|
Tags() ([]GitTag, error)
|
|
Branch() string
|
|
IsDetached() (bool, error)
|
|
}
|
|
|
|
// GitCommitLog description of a single commit log.
|
|
type GitCommitLog struct {
|
|
Date string `json:"date,omitempty"`
|
|
Timestamp int `json:"timestamp,omitempty"`
|
|
AuthorName string `json:"authorName,omitempty"`
|
|
Hash string `json:"hash,omitempty"`
|
|
Message CommitMessage `json:"message,omitempty"`
|
|
}
|
|
|
|
// GitTag git tag info.
|
|
type GitTag struct {
|
|
Name string
|
|
Date time.Time
|
|
}
|
|
|
|
// LogRangeType type of log range.
|
|
type LogRangeType string
|
|
|
|
// constants for log range type.
|
|
const (
|
|
TagRange LogRangeType = "tag"
|
|
DateRange LogRangeType = "date"
|
|
HashRange LogRangeType = "hash"
|
|
)
|
|
|
|
// LogRange git log range.
|
|
type LogRange struct {
|
|
rangeType LogRangeType
|
|
start string
|
|
end string
|
|
}
|
|
|
|
// NewLogRange LogRange constructor.
|
|
func NewLogRange(t LogRangeType, start, end string) LogRange {
|
|
return LogRange{rangeType: t, start: start, end: end}
|
|
}
|
|
|
|
// GitImpl git command implementation.
|
|
type GitImpl struct {
|
|
messageProcessor MessageProcessor
|
|
tagCfg TagConfig
|
|
}
|
|
|
|
// NewGit constructor.
|
|
func NewGit(messageProcessor MessageProcessor, cfg TagConfig) *GitImpl {
|
|
return &GitImpl{
|
|
messageProcessor: messageProcessor,
|
|
tagCfg: cfg,
|
|
}
|
|
}
|
|
|
|
// LastTag get last tag, if no tag found, return empty.
|
|
func (g GitImpl) LastTag() string {
|
|
//nolint:gosec
|
|
cmd := exec.Command(
|
|
"git",
|
|
"for-each-ref",
|
|
fmt.Sprintf("refs/tags/%s", *g.tagCfg.Filter),
|
|
"--sort",
|
|
"-creatordate",
|
|
"--format",
|
|
"%(refname:short)",
|
|
"--count",
|
|
"1",
|
|
)
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
|
|
return strings.TrimSpace(strings.Trim(string(out), "\n"))
|
|
}
|
|
|
|
// Log return git log.
|
|
func (g GitImpl) Log(lr LogRange) ([]GitCommitLog, error) {
|
|
format := "--pretty=format:\"%ad" + logSeparator +
|
|
"%at" + logSeparator +
|
|
"%cN" + logSeparator +
|
|
"%h" + logSeparator +
|
|
"%s" + logSeparator +
|
|
"%b" + endLine + "\""
|
|
params := []string{"log", "--date=short", format}
|
|
|
|
if lr.start != "" || lr.end != "" {
|
|
switch lr.rangeType {
|
|
case DateRange:
|
|
params = append(params, "--since", lr.start, "--until", addDay(lr.end))
|
|
default:
|
|
if lr.start == "" {
|
|
params = append(params, lr.end)
|
|
} else {
|
|
params = append(params, lr.start+".."+str(lr.end, "HEAD"))
|
|
}
|
|
}
|
|
}
|
|
|
|
cmd := exec.Command("git", params...)
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return nil, combinedOutputErr(err, out)
|
|
}
|
|
|
|
logs, parseErr := parseLogOutput(g.messageProcessor, string(out))
|
|
if parseErr != nil {
|
|
return nil, parseErr
|
|
}
|
|
|
|
return logs, 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) (string, error) {
|
|
tag := fmt.Sprintf(*g.tagCfg.Pattern, 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 out, err := tagCommand.CombinedOutput(); err != nil {
|
|
return tag, combinedOutputErr(err, out)
|
|
}
|
|
|
|
pushCommand := exec.Command("git", "push", "origin", tag)
|
|
if out, err := pushCommand.CombinedOutput(); err != nil {
|
|
return tag, combinedOutputErr(err, out)
|
|
}
|
|
|
|
return tag, nil
|
|
}
|
|
|
|
// Tags list repository tags.
|
|
func (g GitImpl) Tags() ([]GitTag, error) {
|
|
//nolint:gosec
|
|
cmd := exec.Command(
|
|
"git",
|
|
"for-each-ref",
|
|
"--sort",
|
|
"creatordate",
|
|
"--format",
|
|
"%(creatordate:iso8601)#%(refname:short)",
|
|
fmt.Sprintf("refs/tags/%s", *g.tagCfg.Filter),
|
|
)
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return nil, combinedOutputErr(err, out)
|
|
}
|
|
|
|
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"))
|
|
}
|
|
|
|
// IsDetached check if is detached.
|
|
func (GitImpl) IsDetached() (bool, error) {
|
|
cmd := exec.Command("git", "symbolic-ref", "-q", "HEAD")
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
// -q: do not issue an error message if the <name> is not a symbolic ref, but a detached HEAD;
|
|
// instead exit with non-zero status silently.
|
|
if output := string(out); err != nil {
|
|
if output == "" {
|
|
return true, nil
|
|
}
|
|
|
|
return false, fmt.Errorf("%w: %s", errUnknownGitError, output)
|
|
}
|
|
|
|
return false, nil
|
|
}
|
|
|
|
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(messageProcessor MessageProcessor, log string) ([]GitCommitLog, error) {
|
|
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 != "" {
|
|
log, err := parseCommitLog(messageProcessor, text)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
logs = append(logs, log)
|
|
}
|
|
}
|
|
|
|
return logs, nil
|
|
}
|
|
|
|
func parseCommitLog(messageProcessor MessageProcessor, commit string) (GitCommitLog, error) {
|
|
content := strings.Split(strings.Trim(commit, "\""), logSeparator)
|
|
timestamp, _ := strconv.Atoi(content[1])
|
|
|
|
message, err := messageProcessor.Parse(content[4], content[5])
|
|
if err != nil {
|
|
return GitCommitLog{}, err
|
|
}
|
|
|
|
return GitCommitLog{
|
|
Date: content[0],
|
|
Timestamp: timestamp,
|
|
AuthorName: content[2],
|
|
Hash: content[3],
|
|
Message: message,
|
|
}, nil
|
|
}
|
|
|
|
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) { //nolint:nonamedreturns
|
|
if atEOF && len(data) == 0 {
|
|
return 0, nil, nil
|
|
}
|
|
|
|
if i := bytes.Index(data, b); i >= 0 {
|
|
return i + len(b), data[0:i], nil
|
|
}
|
|
|
|
if atEOF {
|
|
return len(data), data, nil
|
|
}
|
|
|
|
return 0, nil, nil
|
|
}
|
|
}
|
|
|
|
func addDay(value string) string {
|
|
if value == "" {
|
|
return value
|
|
}
|
|
|
|
t, err := time.Parse("2006-01-02", value)
|
|
if err != nil { // keep original value if is not date format
|
|
return value
|
|
}
|
|
|
|
return t.AddDate(0, 0, 1).Format("2006-01-02")
|
|
}
|
|
|
|
func str(value, defaultValue string) string {
|
|
if value != "" {
|
|
return value
|
|
}
|
|
|
|
return defaultValue
|
|
}
|
|
|
|
func combinedOutputErr(err error, out []byte) error {
|
|
msg := strings.Split(string(out), "\n")
|
|
|
|
return fmt.Errorf("%w - %s", err, msg[0])
|
|
}
|