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

256 lines
6.0 KiB
Go
Raw Normal View History

package semanticrelease
import (
"io/ioutil"
2019-05-15 22:09:52 +02:00
"strings"
"github.com/Masterminds/semver"
"github.com/Nightapes/go-semantic-release/internal/analyzer"
"github.com/Nightapes/go-semantic-release/internal/changelog"
"github.com/Nightapes/go-semantic-release/internal/gitutil"
"github.com/Nightapes/go-semantic-release/internal/releaser"
2019-06-15 23:03:27 +02:00
"github.com/Nightapes/go-semantic-release/internal/shared"
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 {
2019-06-15 23:03:27 +02:00
config *config.ReleaseConfig
gitutil *gitutil.GitUtil
analyzer *analyzer.Analyzer
releaser releaser.Releaser
repository string
2019-05-15 22:09:52 +02:00
}
// New SemanticRelease struct
2019-06-15 23:03:27 +02:00
func New(c *config.ReleaseConfig, repository string) (*SemanticRelease, error) {
util, err := gitutil.New(repository)
if err != nil {
return nil, err
2019-05-15 22:09:52 +02:00
}
2019-06-15 23:03:27 +02:00
analyzer, err := analyzer.New(c.CommitFormat, c.Changelog)
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
}
2019-06-15 23:03:27 +02:00
releaser, err := releaser.New(c).GetReleaser()
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
}
return &SemanticRelease{
config: c,
gitutil: util,
releaser: releaser,
analyzer: analyzer,
repository: repository,
}, nil
}
// GetNextVersion from .version or calculate new from commits
func (s *SemanticRelease) GetNextVersion(force bool) (*shared.ReleaseVersion, error) {
hash, err := s.gitutil.GetHash()
if err != nil {
return nil, err
}
2019-05-15 22:09:52 +02:00
log.Debugf("Ignore .version file if exits, %t", force)
if !force {
2019-06-15 23:03:27 +02:00
releaseVersion, err := s.readFromCache(hash)
if err != nil {
return nil, err
}
2019-06-15 23:03:27 +02:00
if releaseVersion != nil {
return releaseVersion, nil
2019-05-15 22:09:52 +02:00
}
}
2019-06-15 23:03:27 +02:00
currentBranch, err := s.gitutil.GetBranch()
if err != nil {
return nil, err
}
lastVersion, lastVersionHash, err := s.gitutil.GetLastVersion()
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
}
2019-06-15 23:03:27 +02:00
var newVersion semver.Version
if lastVersion == nil {
defaultVersion, _ := semver.NewVersion("1.0.0")
newVersion = *defaultVersion
lastVersion = defaultVersion
} else {
newVersion = *lastVersion
}
2019-06-15 23:03:27 +02:00
commits, err := s.gitutil.GetCommits(lastVersionHash)
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
}
log.Debugf("Found %d commits till last release", len(commits))
a, err := analyzer.New(s.config.CommitFormat, s.config.Changelog)
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
}
result := a.Analyze(commits)
isDraft := false
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 "beta", "alpha":
isDraft = true
case "rc":
newVersion = s.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
}
}
}
}
2019-06-15 23:03:27 +02:00
releaseVersion := shared.ReleaseVersion{
Next: shared.ReleaseVersionEntry{
Commit: hash,
Version: &newVersion,
},
Last: shared.ReleaseVersionEntry{
Commit: lastVersionHash,
Version: lastVersion,
},
Branch: currentBranch,
Draft: isDraft,
2019-06-15 23:03:27 +02:00
}
log.Infof("New version %s -> %s", lastVersion.String(), newVersion.String())
2019-06-15 23:03:27 +02:00
err = s.saveToCache(releaseVersion)
2019-05-14 20:19:36 +02:00
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
2019-05-14 20:19:36 +02:00
}
2019-06-15 23:03:27 +02:00
return &releaseVersion, err
}
2019-05-14 20:19:36 +02:00
//SetVersion for git repository
2019-06-15 23:03:27 +02:00
func (s *SemanticRelease) SetVersion(version string) error {
newVersion, err := semver.NewVersion(version)
if err != nil {
return err
}
2019-06-15 23:03:27 +02:00
lastVersion, lastVersionHash, err := s.gitutil.GetLastVersion()
if err != nil {
return err
}
2019-06-15 23:03:27 +02:00
if lastVersion == nil {
lastVersion, _ = semver.NewVersion("1.0.0")
}
2019-06-15 23:03:27 +02:00
hash, err := s.gitutil.GetHash()
if err != nil {
return err
}
2019-06-15 23:03:27 +02:00
currentBranch, err := s.gitutil.GetBranch()
if err != nil {
return err
}
2019-06-15 23:03:27 +02:00
return s.saveToCache(shared.ReleaseVersion{
Next: shared.ReleaseVersionEntry{
Commit: hash,
Version: newVersion,
},
Last: shared.ReleaseVersionEntry{
Commit: lastVersionHash,
Version: lastVersion,
},
Branch: currentBranch,
})
}
// GetChangelog from last version till now
2019-06-15 23:03:27 +02:00
func (s *SemanticRelease) GetChangelog(releaseVersion *shared.ReleaseVersion) (*shared.GeneratedChangelog, error) {
commits, err := s.gitutil.GetCommits(releaseVersion.Last.Commit)
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
}
2019-06-15 23:03:27 +02:00
result := s.analyzer.Analyze(commits)
log.Debugf("Found %d commits till last release", len(commits))
2019-06-15 23:03:27 +02:00
c := changelog.New(s.config, s.analyzer.GetRules())
return c.GenerateChanglog(shared.ChangelogTemplateConfig{
Version: releaseVersion.Next.Version.String(),
Hash: releaseVersion.Last.Commit,
CommitURL: s.releaser.GetCommitURL(),
CompareURL: s.releaser.GetCompareURL(releaseVersion.Last.Version.String(), releaseVersion.Next.Version.String()),
}, result)
}
// WriteChangeLog wirtes changelog content to the given file
func (s *SemanticRelease) WriteChangeLog(changelogContent, file string) error {
return ioutil.WriteFile(file, []byte(changelogContent), 0644)
}
// Release pusblish release to provider
2019-06-15 23:03:27 +02:00
func (s *SemanticRelease) Release(force bool) error {
currentBranch, err := s.gitutil.GetBranch()
2019-06-15 23:16:30 +02:00
if err != nil {
return err
}
if _, ok := s.config.Branch[currentBranch]; !ok {
log.Debugf("Will not perform a new release. Current %s branch is not configured in release config", currentBranch)
return nil
}
2019-06-15 23:03:27 +02:00
releaseVersion, err := s.GetNextVersion(force)
if err != nil {
log.Debugf("Could not get next version")
return err
}
2019-06-15 23:03:27 +02:00
generatedChanglog, err := s.GetChangelog(releaseVersion)
if err != nil {
log.Debugf("Could not get changelog")
return err
}
releaser, err := releaser.New(s.config).GetReleaser()
if err != nil {
return err
}
2019-06-15 23:03:27 +02:00
err = releaser.ValidateConfig()
if err != nil {
return err
}
if err = releaser.CreateRelease(releaseVersion, generatedChanglog); err != nil {
return err
}
if err = releaser.UploadAssets(s.config.Assets); err != nil {
return err
}
return nil
}
if file.Compress {