Files
go-semantic-release/pkg/semanticrelease/semantic-release.go

223 lines
5.6 KiB
Go
Raw Normal View History

2019-05-14 20:19:36 +02:00
// Package semanticrelease provides public methods to include in own code
package semanticrelease
import (
"io/ioutil"
2019-05-15 22:09:52 +02:00
"strconv"
"strings"
"github.com/Masterminds/semver"
"github.com/Nightapes/go-semantic-release/internal/analyzer"
2019-05-15 22:09:52 +02:00
"github.com/Nightapes/go-semantic-release/internal/cache"
"github.com/Nightapes/go-semantic-release/internal/changelog"
"github.com/Nightapes/go-semantic-release/internal/gitutil"
2019-05-15 22:09:52 +02:00
"github.com/Nightapes/go-semantic-release/pkg/config"
log "github.com/sirupsen/logrus"
)
2019-05-15 22:09:52 +02:00
// SemanticRelease struct
type SemanticRelease struct {
config *config.ReleaseConfig
}
// New SemanticRelease struct
func New(c *config.ReleaseConfig) *SemanticRelease {
return &SemanticRelease{
config: c,
}
}
2019-05-14 20:19:36 +02:00
// GetNextVersion from .version or calculate new from commits
func (s *SemanticRelease) GetNextVersion(repro string, force bool) (string, error) {
util, err := gitutil.New(repro)
if err != nil {
return "", err
}
hash, err := util.GetHash()
if err != nil {
return "", err
}
2019-05-15 22:09:52 +02:00
log.Debugf("Ignore .version file if exits, %t", force)
if !force {
content, err := cache.Read()
2019-05-15 22:09:52 +02:00
if err == nil && content.Commit == hash {
log.Infof("Found cache, will return cached version %s", content.NextVersion)
return content.NextVersion, err
2019-05-15 22:09:52 +02:00
}
log.Debugf("Mismatch git and version file %s - %s", content.Commit, hash)
}
lastVersion, lastVersionHash, err := util.GetLastVersion()
if err != nil {
return "", err
}
var newVersion semver.Version
if lastVersion == nil {
defaultVersion, _ := semver.NewVersion("1.0.0")
newVersion = *defaultVersion
lastVersion = defaultVersion
} else {
newVersion = *lastVersion
}
commits, err := util.GetCommits(lastVersionHash)
if err != nil {
return "", err
}
log.Debugf("Found %d commits till last release", len(commits))
a, err := analyzer.New(s.config.CommitFormat, s.config.Changelog)
if err != nil {
return "", err
}
result := a.Analyze(commits)
2019-05-15 22:09:52 +02:00
currentBranch, err := util.GetBranch()
if err != nil {
return "", err
2019-05-15 22:09:52 +02:00
}
for branch, releaseType := range s.config.Branch {
if currentBranch == branch || strings.HasPrefix(currentBranch, branch) {
log.Debugf("Found branch config for branch %s with release type %s", currentBranch, releaseType)
switch releaseType {
case "rc", "beta", "alpha":
newVersion = incPrerelease(releaseType, newVersion)
case "release":
if len(result["major"]) > 0 {
newVersion = newVersion.IncMajor()
} else if len(result["minor"]) > 0 {
newVersion = newVersion.IncMinor()
} else if len(result["patch"]) > 0 {
newVersion = newVersion.IncPatch()
2019-05-15 22:09:52 +02:00
}
}
}
}
log.Infof("New version %s -> %s", lastVersion.String(), newVersion.String())
err = saveToCache(util, lastVersion, &newVersion)
2019-05-14 20:19:36 +02:00
if err != nil {
return "", err
2019-05-14 20:19:36 +02:00
}
c := changelog.New(s.config, a.GetRules())
c.GenerateChanglog(newVersion.String(), "https://github.com/Nightapes/go-semantic-release/commit/{{hash}}", result)
return newVersion.String(), err
}
2019-05-14 20:19:36 +02:00
//SetVersion for git repository
2019-05-15 22:09:52 +02:00
func (s *SemanticRelease) SetVersion(version string, repro string) error {
util, err := gitutil.New(repro)
if err != nil {
return err
}
newVersion, err := semver.NewVersion(version)
if err != nil {
return err
}
lastVersion, _, err := util.GetLastVersion()
if err != nil {
return err
}
return saveToCache(util, lastVersion, newVersion)
}
func saveToCache(util *gitutil.GitUtil, lastVersion *semver.Version, nextVersion *semver.Version) error {
hash, err := util.GetHash()
if err != nil {
return err
}
branch, err := util.GetBranch()
if err != nil {
return err
}
2019-05-15 22:09:52 +02:00
newVersionContent := cache.VersionFileContent{
Commit: hash,
NextVersion: nextVersion.String(),
Branch: branch,
}
if lastVersion != nil {
newVersionContent.Version = lastVersion.String()
}
log.Debugf("Save %s with hash %s to cache", nextVersion.String(), hash)
2019-05-15 22:09:52 +02:00
return cache.Write(newVersionContent)
}
func incPrerelease(preReleaseType string, version semver.Version) semver.Version {
defaultPrerelease := preReleaseType + ".0"
if version.Prerelease() == "" || !strings.HasPrefix(version.Prerelease(), preReleaseType) {
version, _ = version.SetPrerelease(defaultPrerelease)
} else {
parts := strings.Split(version.Prerelease(), ".")
if len(parts) == 2 {
i, err := strconv.Atoi(parts[1])
if err != nil {
version, _ = version.SetPrerelease(defaultPrerelease)
log.Warnf("Could not parse release tag %s, use version %s", version.Prerelease(), version.String())
} else {
version, _ = version.SetPrerelease(preReleaseType + "." + strconv.Itoa((i + 1)))
}
} else {
version, _ = version.SetPrerelease(defaultPrerelease)
log.Warnf("Could not parse release tag %s, use version %s", version.Prerelease(), version.String())
}
}
return version
}
// GetChangelog from last version till now
func (s *SemanticRelease) GetChangelog(repro, file string) error {
nextVersion, err := s.GetNextVersion(repro, false)
if err != nil {
log.Debugf("Could not get next version")
return err
}
util, err := gitutil.New(repro)
if err != nil {
return err
}
_, lastVersionHash, err := util.GetLastVersion()
if err != nil {
return err
}
commits, err := util.GetCommits(lastVersionHash)
if err != nil {
return err
}
log.Debugf("Found %d commits till last release", len(commits))
a, err := analyzer.New(s.config.CommitFormat, s.config.Changelog)
if err != nil {
return err
}
result := a.Analyze(commits)
c := changelog.New(s.config, a.GetRules())
_, content, err := c.GenerateChanglog(nextVersion, s.config.GitProvider.URL+"{{hash}}", result)
if err != nil {
return err
}
return ioutil.WriteFile(file, []byte(content), 0644)
}