mirror of
https://github.com/vbatts/git-validation.git
synced 2025-04-18 12:54:55 +00:00
*: run tests in a runner
for cleanliness and ease of testing Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
This commit is contained in:
parent
b9413c60c8
commit
d614ccf997
4 changed files with 119 additions and 52 deletions
|
@ -17,7 +17,7 @@ import (
|
||||||
// If commitrange is a single commit, all ancestor commits up through the hash provided.
|
// If commitrange is a single commit, all ancestor commits up through the hash provided.
|
||||||
func Commits(commitrange string) ([]CommitEntry, error) {
|
func Commits(commitrange string) ([]CommitEntry, error) {
|
||||||
cmdArgs := []string{"git", "log", prettyFormat + formatCommit, commitrange}
|
cmdArgs := []string{"git", "log", prettyFormat + formatCommit, commitrange}
|
||||||
if Verbose {
|
if debug() {
|
||||||
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
||||||
}
|
}
|
||||||
output, err := exec.Command(cmdArgs[0], cmdArgs[1:]...).Output()
|
output, err := exec.Command(cmdArgs[0], cmdArgs[1:]...).Output()
|
||||||
|
@ -40,9 +40,6 @@ func Commits(commitrange string) ([]CommitEntry, error) {
|
||||||
type CommitEntry map[string]string
|
type CommitEntry map[string]string
|
||||||
|
|
||||||
var (
|
var (
|
||||||
// Verbose output of commands run
|
|
||||||
Verbose = false
|
|
||||||
|
|
||||||
prettyFormat = `--pretty=format:`
|
prettyFormat = `--pretty=format:`
|
||||||
formatSubject = `%s`
|
formatSubject = `%s`
|
||||||
formatBody = `%b`
|
formatBody = `%b`
|
||||||
|
@ -60,7 +57,7 @@ var (
|
||||||
func LogCommit(commit string) (*CommitEntry, error) {
|
func LogCommit(commit string) (*CommitEntry, error) {
|
||||||
buf := bytes.NewBuffer([]byte{})
|
buf := bytes.NewBuffer([]byte{})
|
||||||
cmdArgs := []string{"git", "log", "-1", prettyFormat + formatMap, commit}
|
cmdArgs := []string{"git", "log", "-1", prettyFormat + formatMap, commit}
|
||||||
if Verbose {
|
if debug() {
|
||||||
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
||||||
}
|
}
|
||||||
cmd := exec.Command(cmdArgs[0], cmdArgs[1:]...)
|
cmd := exec.Command(cmdArgs[0], cmdArgs[1:]...)
|
||||||
|
@ -99,10 +96,14 @@ func LogCommit(commit string) (*CommitEntry, error) {
|
||||||
return &c, nil
|
return &c, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func debug() bool {
|
||||||
|
return len(os.Getenv("DEBUG")) > 0
|
||||||
|
}
|
||||||
|
|
||||||
// FetchHeadCommit returns the hash of FETCH_HEAD
|
// FetchHeadCommit returns the hash of FETCH_HEAD
|
||||||
func FetchHeadCommit() (string, error) {
|
func FetchHeadCommit() (string, error) {
|
||||||
cmdArgs := []string{"git", "rev-parse", "--verify", "FETCH_HEAD"}
|
cmdArgs := []string{"git", "rev-parse", "--verify", "FETCH_HEAD"}
|
||||||
if Verbose {
|
if debug() {
|
||||||
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
||||||
}
|
}
|
||||||
output, err := exec.Command(cmdArgs[0], cmdArgs[1:]...).Output()
|
output, err := exec.Command(cmdArgs[0], cmdArgs[1:]...).Output()
|
||||||
|
@ -115,7 +116,7 @@ func FetchHeadCommit() (string, error) {
|
||||||
// HeadCommit returns the hash of HEAD
|
// HeadCommit returns the hash of HEAD
|
||||||
func HeadCommit() (string, error) {
|
func HeadCommit() (string, error) {
|
||||||
cmdArgs := []string{"git", "rev-parse", "--verify", "HEAD"}
|
cmdArgs := []string{"git", "rev-parse", "--verify", "HEAD"}
|
||||||
if Verbose {
|
if debug() {
|
||||||
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
logrus.Infof("[git] cmd: %q", strings.Join(cmdArgs, " "))
|
||||||
}
|
}
|
||||||
output, err := exec.Command(cmdArgs[0], cmdArgs[1:]...).Output()
|
output, err := exec.Command(cmdArgs[0], cmdArgs[1:]...).Output()
|
||||||
|
|
52
main.go
52
main.go
|
@ -6,7 +6,6 @@ import (
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
"github.com/vbatts/git-validation/git"
|
|
||||||
_ "github.com/vbatts/git-validation/rules/dco"
|
_ "github.com/vbatts/git-validation/rules/dco"
|
||||||
_ "github.com/vbatts/git-validation/rules/shortsubject"
|
_ "github.com/vbatts/git-validation/rules/shortsubject"
|
||||||
"github.com/vbatts/git-validation/validate"
|
"github.com/vbatts/git-validation/validate"
|
||||||
|
@ -17,13 +16,15 @@ var (
|
||||||
flListRules = flag.Bool("list-rules", false, "list the rules registered")
|
flListRules = flag.Bool("list-rules", false, "list the rules registered")
|
||||||
flRun = flag.String("run", "", "comma delimited list of rules to run. Defaults to all.")
|
flRun = flag.String("run", "", "comma delimited list of rules to run. Defaults to all.")
|
||||||
flVerbose = flag.Bool("v", false, "verbose")
|
flVerbose = flag.Bool("v", false, "verbose")
|
||||||
|
flDebug = flag.Bool("D", false, "debug output")
|
||||||
|
flDir = flag.String("d", ".", "git directory to validate from")
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
if *flVerbose {
|
if *flDebug {
|
||||||
git.Verbose = true
|
os.Setenv("DEBUG", "1")
|
||||||
}
|
}
|
||||||
|
|
||||||
if *flListRules {
|
if *flListRules {
|
||||||
|
@ -39,53 +40,18 @@ func main() {
|
||||||
rules = validate.FilterRules(rules, validate.SanitizeFilters(*flRun))
|
rules = validate.FilterRules(rules, validate.SanitizeFilters(*flRun))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Guess the commits we're working with
|
runner, err := validate.NewRunner(*flDir, rules, *flCommitRange, *flVerbose)
|
||||||
var commitrange string
|
|
||||||
if *flCommitRange != "" {
|
|
||||||
commitrange = *flCommitRange
|
|
||||||
} else {
|
|
||||||
var err error
|
|
||||||
commitrange, err = git.FetchHeadCommit()
|
|
||||||
if err != nil {
|
|
||||||
commitrange, err = git.HeadCommit()
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// collect the entries
|
|
||||||
c, err := git.Commits(commitrange)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// run them and show results
|
if err := runner.Run(); err != nil {
|
||||||
results := validate.Results{}
|
log.Fatal(err)
|
||||||
for _, commit := range c {
|
|
||||||
fmt.Printf(" * %s %s ... ", commit["abbreviated_commit"], commit["subject"])
|
|
||||||
vr := validate.Commit(commit, rules)
|
|
||||||
results = append(results, vr...)
|
|
||||||
if _, fail := vr.PassFail(); fail == 0 {
|
|
||||||
fmt.Println("PASS")
|
|
||||||
} else {
|
|
||||||
fmt.Println("FAIL")
|
|
||||||
}
|
|
||||||
for _, r := range vr {
|
|
||||||
if *flVerbose {
|
|
||||||
if r.Pass {
|
|
||||||
fmt.Printf("ok %s\n", r.Msg)
|
|
||||||
} else {
|
|
||||||
fmt.Printf("not ok %s\n", r.Msg)
|
|
||||||
}
|
|
||||||
} else if !r.Pass {
|
|
||||||
fmt.Printf("not ok %s\n", r.Msg)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
_, fail := results.PassFail()
|
_, fail := runner.Results.PassFail()
|
||||||
if fail > 0 {
|
if fail > 0 {
|
||||||
fmt.Printf("%d issues to fix\n", fail)
|
fmt.Printf("%d issues to fix\n", fail)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
// DcoRule is the rule being registered
|
// ShortSubjectRule is the rule being registered
|
||||||
ShortSubjectRule = validate.Rule{
|
ShortSubjectRule = validate.Rule{
|
||||||
Name: "short-subject",
|
Name: "short-subject",
|
||||||
Description: "commit subjects are strictly less than 90 (github ellipsis length)",
|
Description: "commit subjects are strictly less than 90 (github ellipsis length)",
|
||||||
|
@ -18,6 +18,8 @@ func init() {
|
||||||
validate.RegisterRule(ShortSubjectRule)
|
validate.RegisterRule(ShortSubjectRule)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ValidateShortSubject checks that the commit's subject is strictly less than
|
||||||
|
// 90 characters (preferrably not more than 72 chars).
|
||||||
func ValidateShortSubject(c git.CommitEntry) (vr validate.Result) {
|
func ValidateShortSubject(c git.CommitEntry) (vr validate.Result) {
|
||||||
if len(c["subject"]) >= 90 {
|
if len(c["subject"]) >= 90 {
|
||||||
vr.Pass = false
|
vr.Pass = false
|
||||||
|
@ -25,6 +27,10 @@ func ValidateShortSubject(c git.CommitEntry) (vr validate.Result) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
vr.Pass = true
|
vr.Pass = true
|
||||||
vr.Msg = "commit subject is not more than 90 characters"
|
if len(c["subject"]) > 72 {
|
||||||
|
vr.Msg = "commit subject is not more than 90 characters, but is still more than 72 chars"
|
||||||
|
} else {
|
||||||
|
vr.Msg = "commit subject is 72 characters or less! *yay*"
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
94
validate/runner.go
Normal file
94
validate/runner.go
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
package validate
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/vbatts/git-validation/git"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Runner is the for processing a set of rules against a range of commits
|
||||||
|
type Runner struct {
|
||||||
|
Root string
|
||||||
|
Rules []Rule
|
||||||
|
Results Results
|
||||||
|
Verbose bool
|
||||||
|
CommitRange string // if this is empty, then it will default to FETCH_HEAD, then HEAD
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewRunner returns an initiallized Runner.
|
||||||
|
func NewRunner(root string, rules []Rule, commitrange string, verbose bool) (*Runner, error) {
|
||||||
|
newroot, err := filepath.Abs(root)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to get absolute path of %q: %s", root, err)
|
||||||
|
}
|
||||||
|
if commitrange == "" {
|
||||||
|
var err error
|
||||||
|
cwd, err := os.Getwd()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer os.Chdir(cwd)
|
||||||
|
|
||||||
|
if err := os.Chdir(newroot); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
commitrange, err = git.FetchHeadCommit()
|
||||||
|
if err != nil {
|
||||||
|
commitrange, err = git.HeadCommit()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return &Runner{
|
||||||
|
Root: newroot,
|
||||||
|
Rules: rules,
|
||||||
|
CommitRange: commitrange,
|
||||||
|
Verbose: verbose,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run processes the rules for each commit in the range provided
|
||||||
|
func (r *Runner) Run() error {
|
||||||
|
cwd, err := os.Getwd()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer os.Chdir(cwd)
|
||||||
|
|
||||||
|
if err := os.Chdir(r.Root); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// collect the entries
|
||||||
|
c, err := git.Commits(r.CommitRange)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// run them and show results
|
||||||
|
for _, commit := range c {
|
||||||
|
fmt.Printf(" * %s %q ... ", commit["abbreviated_commit"], commit["subject"])
|
||||||
|
vr := Commit(commit, r.Rules)
|
||||||
|
r.Results = append(r.Results, vr...)
|
||||||
|
if _, fail := vr.PassFail(); fail == 0 {
|
||||||
|
fmt.Println("PASS")
|
||||||
|
} else {
|
||||||
|
fmt.Println("FAIL")
|
||||||
|
}
|
||||||
|
for _, res := range vr {
|
||||||
|
if r.Verbose {
|
||||||
|
if res.Pass {
|
||||||
|
fmt.Printf(" - PASS - %s\n", res.Msg)
|
||||||
|
} else {
|
||||||
|
fmt.Printf(" - FAIL - %s\n", res.Msg)
|
||||||
|
}
|
||||||
|
} else if !res.Pass {
|
||||||
|
fmt.Printf(" - FAIL - %s\n", res.Msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue