2020-11-07 16:52:09 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2020-11-10 16:28:58 +00:00
|
|
|
"errors"
|
2020-11-10 10:56:26 +00:00
|
|
|
"fmt"
|
2020-11-27 17:36:34 +00:00
|
|
|
"io/ioutil"
|
2020-11-07 16:52:09 +00:00
|
|
|
"log"
|
|
|
|
"os"
|
2020-11-10 10:56:26 +00:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"time"
|
2020-11-07 16:52:09 +00:00
|
|
|
|
2020-11-10 10:56:26 +00:00
|
|
|
"github.com/haunt98/changeloguru/pkg/changelog"
|
2020-11-09 08:46:42 +00:00
|
|
|
"github.com/haunt98/changeloguru/pkg/convention"
|
2020-11-07 16:52:09 +00:00
|
|
|
"github.com/haunt98/changeloguru/pkg/git"
|
|
|
|
"github.com/urfave/cli/v2"
|
2020-11-10 10:56:26 +00:00
|
|
|
"golang.org/x/mod/semver"
|
2020-11-07 16:52:09 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2020-11-10 16:16:51 +00:00
|
|
|
name = "changeloguru"
|
|
|
|
description = "generate changelog from conventional commits"
|
|
|
|
|
2020-12-17 09:54:32 +00:00
|
|
|
currentPath = "."
|
|
|
|
markdownFiletype = "md"
|
2020-12-17 10:52:04 +00:00
|
|
|
|
|
|
|
defaultPath = currentPath
|
|
|
|
defaultFilename = "CHANGELOG"
|
|
|
|
defaultFiletype = markdownFiletype
|
|
|
|
defaultVersion = "0.1.0"
|
2020-11-10 10:56:26 +00:00
|
|
|
|
2020-11-10 16:28:58 +00:00
|
|
|
fromFlag = "from"
|
|
|
|
excludeToFlag = "exclude-to"
|
|
|
|
includeToFlag = "include-to"
|
|
|
|
versionFlag = "version"
|
2020-12-17 09:54:32 +00:00
|
|
|
filenameFlag = "filename"
|
|
|
|
filetypeFlag = "filetype"
|
2020-11-10 16:28:58 +00:00
|
|
|
verboseFlag = "verbose"
|
2020-11-30 11:19:05 +00:00
|
|
|
|
|
|
|
pathArgs = "path"
|
2020-11-07 16:52:09 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
2020-11-10 08:33:19 +00:00
|
|
|
a := &action{
|
|
|
|
verbose: false,
|
2020-11-30 11:19:05 +00:00
|
|
|
flags: make(map[string]string),
|
|
|
|
args: make(map[string]string),
|
2020-11-10 08:33:19 +00:00
|
|
|
}
|
|
|
|
|
2020-11-07 16:52:09 +00:00
|
|
|
app := &cli.App{
|
2020-11-10 16:16:51 +00:00
|
|
|
Name: name,
|
|
|
|
Usage: description,
|
2020-11-07 16:52:09 +00:00
|
|
|
Flags: []cli.Flag{
|
|
|
|
&cli.StringFlag{
|
|
|
|
Name: fromFlag,
|
2020-12-17 10:34:50 +00:00
|
|
|
Usage: "generate from `COMMIT`",
|
2020-11-07 16:52:09 +00:00
|
|
|
},
|
|
|
|
&cli.StringFlag{
|
2020-11-10 16:28:58 +00:00
|
|
|
Name: excludeToFlag,
|
2020-12-17 10:34:50 +00:00
|
|
|
Usage: "generate to `COMMIT` (exclude that commit)",
|
2020-11-10 16:28:58 +00:00
|
|
|
},
|
|
|
|
&cli.StringFlag{
|
|
|
|
Name: includeToFlag,
|
2020-12-17 10:34:50 +00:00
|
|
|
Usage: "generate to `COMMIT` (include that commit)",
|
2020-11-07 16:52:09 +00:00
|
|
|
},
|
2020-11-10 10:56:26 +00:00
|
|
|
&cli.StringFlag{
|
|
|
|
Name: versionFlag,
|
2020-12-17 10:34:50 +00:00
|
|
|
Usage: "generate new `VERSION`",
|
2020-11-10 10:56:26 +00:00
|
|
|
},
|
2020-12-17 09:54:32 +00:00
|
|
|
&cli.StringFlag{
|
|
|
|
Name: filenameFlag,
|
2020-12-17 10:34:50 +00:00
|
|
|
Usage: fmt.Sprintf("output `FILENAME`, default is %s", defaultFilename),
|
2020-12-17 09:54:32 +00:00
|
|
|
},
|
|
|
|
&cli.StringFlag{
|
|
|
|
Name: filetypeFlag,
|
2020-12-17 10:34:50 +00:00
|
|
|
Usage: fmt.Sprintf("output `FILETYPE`, default is %s", defaultFiletype),
|
2020-12-17 09:54:32 +00:00
|
|
|
},
|
2020-11-07 16:52:09 +00:00
|
|
|
&cli.BoolFlag{
|
|
|
|
Name: "verbose",
|
|
|
|
Aliases: []string{"v"},
|
|
|
|
Usage: "show what is going on",
|
|
|
|
},
|
|
|
|
},
|
2020-11-10 08:33:19 +00:00
|
|
|
Action: a.Run,
|
2020-11-07 16:52:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-10 08:33:19 +00:00
|
|
|
type action struct {
|
|
|
|
verbose bool
|
2020-11-30 11:19:05 +00:00
|
|
|
flags map[string]string
|
|
|
|
args map[string]string
|
2020-11-10 08:33:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *action) Run(c *cli.Context) error {
|
2020-11-30 11:19:05 +00:00
|
|
|
// set up
|
|
|
|
a.getFlags(c)
|
|
|
|
a.getArgs(c)
|
2020-11-07 16:52:09 +00:00
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
commits, err := a.getCommits()
|
2020-11-10 08:33:19 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-11-07 16:52:09 +00:00
|
|
|
}
|
2020-11-10 08:33:19 +00:00
|
|
|
a.log("commits %+v", commits)
|
2020-11-07 16:52:09 +00:00
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
conventionalCommits := a.getConventionalCommits(commits)
|
2020-11-10 08:33:19 +00:00
|
|
|
a.log("conventional commits %+v", conventionalCommits)
|
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
if err := a.generateChangelog(conventionalCommits); err != nil {
|
2020-11-10 10:56:26 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-11-10 08:33:19 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
func (a *action) getArgs(c *cli.Context) {
|
2020-12-17 09:54:32 +00:00
|
|
|
a.args[pathArgs] = defaultPath
|
2020-11-30 11:19:05 +00:00
|
|
|
if c.NArg() > 0 {
|
|
|
|
a.args[pathArgs] = c.Args().Get(0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *action) getFlags(c *cli.Context) {
|
|
|
|
a.verbose = c.Bool(verboseFlag)
|
|
|
|
a.flags[fromFlag] = c.String(fromFlag)
|
|
|
|
a.flags[excludeToFlag] = c.String(excludeToFlag)
|
|
|
|
a.flags[includeToFlag] = c.String(includeToFlag)
|
|
|
|
a.flags[versionFlag] = c.String(versionFlag)
|
2020-12-17 09:54:32 +00:00
|
|
|
a.flags[filenameFlag] = c.String(filenameFlag)
|
|
|
|
a.flags[filetypeFlag] = c.String(filetypeFlag)
|
2020-11-30 11:19:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *action) getCommits() ([]git.Commit, error) {
|
|
|
|
path := a.args[pathArgs]
|
|
|
|
a.log("path %s", path)
|
|
|
|
|
2020-11-07 16:52:09 +00:00
|
|
|
r, err := git.NewRepository(path)
|
|
|
|
if err != nil {
|
2020-11-10 08:33:19 +00:00
|
|
|
return nil, err
|
2020-11-07 16:52:09 +00:00
|
|
|
}
|
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
fromRev := a.flags[fromFlag]
|
2020-11-10 08:33:19 +00:00
|
|
|
a.log("from revision %s", fromRev)
|
2020-11-07 16:52:09 +00:00
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
excludeToRev := a.flags[excludeToFlag]
|
2020-11-10 16:48:41 +00:00
|
|
|
a.log("exclude to revision %s", excludeToRev)
|
2020-11-07 16:52:09 +00:00
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
includeToRev := a.flags[includeToFlag]
|
2020-11-10 16:48:41 +00:00
|
|
|
a.log("include to revision %s", includeToRev)
|
2020-11-10 16:28:58 +00:00
|
|
|
|
|
|
|
if excludeToRev != "" && includeToRev != "" {
|
|
|
|
return nil, errors.New("excludeToFlag and includeToFlag can not appear same time")
|
2020-11-07 16:52:09 +00:00
|
|
|
}
|
2020-11-10 16:28:58 +00:00
|
|
|
|
2020-11-10 16:33:35 +00:00
|
|
|
if excludeToRev != "" {
|
2020-11-10 16:28:58 +00:00
|
|
|
return r.LogExcludeTo(fromRev, excludeToRev)
|
|
|
|
}
|
|
|
|
|
2020-11-10 16:33:35 +00:00
|
|
|
if includeToRev != "" {
|
|
|
|
return r.LogIncludeTo(fromRev, includeToRev)
|
2020-11-10 16:28:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return r.Log(fromRev)
|
2020-11-10 08:33:19 +00:00
|
|
|
}
|
2020-11-07 16:52:09 +00:00
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
func (a *action) getConventionalCommits(commits []git.Commit) []convention.Commit {
|
2020-11-09 08:46:42 +00:00
|
|
|
conventionalCommits := make([]convention.Commit, 0, len(commits))
|
2020-11-10 08:33:19 +00:00
|
|
|
|
2020-11-09 08:46:42 +00:00
|
|
|
for _, commit := range commits {
|
|
|
|
conventionalCommit, err := convention.NewCommit(commit)
|
|
|
|
if err != nil {
|
2020-11-10 08:33:19 +00:00
|
|
|
a.log("failed to new conventional commits %+v: %s", commit, err)
|
2020-11-09 08:46:42 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
conventionalCommits = append(conventionalCommits, conventionalCommit)
|
|
|
|
}
|
|
|
|
|
2020-11-10 08:33:19 +00:00
|
|
|
return conventionalCommits
|
|
|
|
}
|
|
|
|
|
2020-11-30 11:19:05 +00:00
|
|
|
func (a *action) generateChangelog(commits []convention.Commit) error {
|
2020-12-17 10:52:04 +00:00
|
|
|
changelogPath, _, filetype := a.getChangelogPath()
|
|
|
|
|
|
|
|
version, err := a.getVersion()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch filetype {
|
|
|
|
case markdownFiletype:
|
|
|
|
return a.generateMarkdownChangelog(changelogPath, version, commits)
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("unknown filetype %s", filetype)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *action) getChangelogPath() (string, string, string) {
|
2020-11-30 11:19:05 +00:00
|
|
|
path := a.args[pathArgs]
|
2020-12-17 09:54:32 +00:00
|
|
|
|
|
|
|
filename := a.flags[filenameFlag]
|
|
|
|
if filename == "" {
|
|
|
|
filename = defaultFilename
|
|
|
|
}
|
|
|
|
|
|
|
|
filetype := a.flags[filetypeFlag]
|
2020-12-17 10:52:04 +00:00
|
|
|
if filetype == "" {
|
2020-12-17 09:54:32 +00:00
|
|
|
filetype = defaultFiletype
|
|
|
|
}
|
|
|
|
|
2020-12-17 10:52:04 +00:00
|
|
|
changelogName := filename + "." + filetype
|
|
|
|
changelogPath := filepath.Join(path, changelogName)
|
2020-12-17 10:53:38 +00:00
|
|
|
|
2020-12-17 10:52:04 +00:00
|
|
|
a.log("changelog path %s", changelogPath)
|
|
|
|
|
|
|
|
return changelogPath, filename, filetype
|
|
|
|
}
|
2020-11-10 10:56:26 +00:00
|
|
|
|
2020-12-17 10:52:04 +00:00
|
|
|
func (a *action) getVersion() (string, error) {
|
2020-11-30 11:19:05 +00:00
|
|
|
version := a.flags[versionFlag]
|
2020-12-17 10:52:04 +00:00
|
|
|
if version == "" {
|
|
|
|
version = defaultVersion
|
|
|
|
}
|
|
|
|
|
2020-11-10 10:56:26 +00:00
|
|
|
if !strings.HasPrefix(version, "v") {
|
|
|
|
version = "v" + version
|
|
|
|
}
|
2020-12-17 10:52:04 +00:00
|
|
|
|
2020-11-10 10:56:26 +00:00
|
|
|
if !semver.IsValid(version) {
|
2020-12-17 10:52:04 +00:00
|
|
|
return "", fmt.Errorf("invalid semver %s", version)
|
2020-11-10 10:56:26 +00:00
|
|
|
}
|
|
|
|
|
2020-12-17 10:53:38 +00:00
|
|
|
a.log("version %s", version)
|
|
|
|
|
2020-12-17 10:52:04 +00:00
|
|
|
return version, nil
|
2020-12-17 09:54:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *action) generateMarkdownChangelog(path, version string, commits []convention.Commit) error {
|
2020-11-27 17:36:34 +00:00
|
|
|
var oldData string
|
2020-12-17 09:54:32 +00:00
|
|
|
bytes, err := ioutil.ReadFile(path)
|
2020-11-27 17:36:34 +00:00
|
|
|
if err == nil {
|
|
|
|
oldData = string(bytes)
|
|
|
|
}
|
2020-11-10 10:56:26 +00:00
|
|
|
|
2020-11-27 17:36:34 +00:00
|
|
|
markdownGenerator := changelog.NewMarkdownGenerator(oldData, version, time.Now())
|
|
|
|
newData := markdownGenerator.Generate(commits)
|
|
|
|
|
2020-12-17 09:54:32 +00:00
|
|
|
if err := ioutil.WriteFile(path, []byte(newData), 0644); err != nil {
|
|
|
|
return fmt.Errorf("failed to write file %s: %w", path, err)
|
2020-11-10 10:56:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-11-10 08:33:19 +00:00
|
|
|
func (a *action) log(format string, v ...interface{}) {
|
|
|
|
if a.verbose {
|
|
|
|
log.Printf(format, v...)
|
|
|
|
}
|
2020-11-07 16:52:09 +00:00
|
|
|
}
|