2019-05-13 22:36:24 +02:00
|
|
|
package semanticrelease
|
|
|
|
|
|
|
|
|
|
import (
|
2019-05-25 18:10:24 +02:00
|
|
|
"io/ioutil"
|
2019-05-15 22:09:52 +02:00
|
|
|
"strings"
|
2019-07-21 15:07:13 +02:00
|
|
|
"time"
|
2019-05-13 22:36:24 +02:00
|
|
|
|
|
|
|
|
"github.com/Masterminds/semver"
|
|
|
|
|
"github.com/Nightapes/go-semantic-release/internal/analyzer"
|
2019-07-22 20:46:48 +02:00
|
|
|
"github.com/Nightapes/go-semantic-release/internal/cache"
|
|
|
|
|
"github.com/Nightapes/go-semantic-release/internal/calculator"
|
2019-05-25 18:10:24 +02:00
|
|
|
"github.com/Nightapes/go-semantic-release/internal/changelog"
|
2019-07-15 21:20:44 +02:00
|
|
|
"github.com/Nightapes/go-semantic-release/internal/ci"
|
2019-05-13 22:36:24 +02:00
|
|
|
"github.com/Nightapes/go-semantic-release/internal/gitutil"
|
2019-06-13 23:33:42 +02:00
|
|
|
"github.com/Nightapes/go-semantic-release/internal/releaser"
|
2019-07-09 01:46:31 +02:00
|
|
|
"github.com/Nightapes/go-semantic-release/internal/releaser/util"
|
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"
|
2019-05-13 22:36:24 +02:00
|
|
|
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
|
2019-07-22 20:46:48 +02:00
|
|
|
calculator *calculator.Calculator
|
2019-06-15 23:03:27 +02:00
|
|
|
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)
|
2019-05-13 22:36:24 +02:00
|
|
|
if err != nil {
|
2019-06-15 23:03:27 +02:00
|
|
|
return nil, err
|
2019-05-13 22:36:24 +02:00
|
|
|
}
|
|
|
|
|
|
2019-06-15 23:03:27 +02:00
|
|
|
releaser, err := releaser.New(c).GetReleaser()
|
2019-05-13 22:36:24 +02:00
|
|
|
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,
|
2019-07-22 20:46:48 +02:00
|
|
|
calculator: calculator.New(),
|
2019-06-15 23:03:27 +02:00
|
|
|
}, nil
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-24 22:14:03 +02:00
|
|
|
//GetCIProvider result with ci config
|
|
|
|
|
func (s *SemanticRelease) GetCIProvider() (*ci.ProviderConfig, error) {
|
|
|
|
|
return ci.GetCIProvider(s.gitutil, ci.ReadAllEnvs())
|
|
|
|
|
}
|
2019-05-13 22:36:24 +02:00
|
|
|
|
2019-07-24 22:14:03 +02:00
|
|
|
// GetNextVersion from .version or calculate new from commits
|
2019-08-11 18:27:52 +02:00
|
|
|
func (s *SemanticRelease) GetNextVersion(provider *ci.ProviderConfig, force bool) (*shared.ReleaseVersion, error) {
|
2019-05-15 22:09:52 +02:00
|
|
|
log.Debugf("Ignore .version file if exits, %t", force)
|
|
|
|
|
if !force {
|
2019-07-22 20:46:48 +02:00
|
|
|
releaseVersion, err := cache.Read(s.repository)
|
2019-06-15 23:03:27 +02:00
|
|
|
if err != nil {
|
2019-08-11 18:27:52 +02:00
|
|
|
return nil, err
|
2019-06-15 23:03:27 +02:00
|
|
|
}
|
2019-05-13 22:36:24 +02:00
|
|
|
|
2019-07-22 20:46:48 +02:00
|
|
|
if releaseVersion.Next.Commit == provider.Commit && releaseVersion != nil {
|
2019-08-11 18:27:52 +02:00
|
|
|
return releaseVersion, nil
|
2019-05-15 22:09:52 +02:00
|
|
|
}
|
2019-05-13 22:36:24 +02:00
|
|
|
}
|
|
|
|
|
|
2019-06-15 23:03:27 +02:00
|
|
|
lastVersion, lastVersionHash, err := s.gitutil.GetLastVersion()
|
2019-05-13 22:36:24 +02:00
|
|
|
if err != nil {
|
2019-08-11 18:27:52 +02:00
|
|
|
return nil, err
|
2019-05-13 22:36:24 +02:00
|
|
|
}
|
2019-06-15 23:03:27 +02:00
|
|
|
|
2019-07-15 21:20:44 +02:00
|
|
|
firstRelease := false
|
2019-05-13 22:36:24 +02:00
|
|
|
|
|
|
|
|
if lastVersion == nil {
|
2019-08-23 08:58:44 +02:00
|
|
|
lastVersion, _ = semver.NewVersion("1.0.0")
|
|
|
|
|
log.Infof("This is the first release, will set version to %s", lastVersion.String())
|
2019-07-15 21:20:44 +02:00
|
|
|
firstRelease = true
|
2019-05-13 22:36:24 +02:00
|
|
|
}
|
|
|
|
|
|
2019-06-15 23:03:27 +02:00
|
|
|
commits, err := s.gitutil.GetCommits(lastVersionHash)
|
2019-05-13 22:36:24 +02:00
|
|
|
if err != nil {
|
2019-08-11 18:27:52 +02:00
|
|
|
return nil, err
|
2019-05-13 22:36:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
log.Debugf("Found %d commits till last release", len(commits))
|
|
|
|
|
|
2019-07-24 22:14:03 +02:00
|
|
|
analyzedCommits := s.analyzer.Analyze(commits)
|
2019-07-22 20:46:48 +02:00
|
|
|
|
|
|
|
|
var newVersion semver.Version
|
2019-05-15 22:09:52 +02:00
|
|
|
for branch, releaseType := range s.config.Branch {
|
2019-07-15 21:20:44 +02:00
|
|
|
if provider.Branch == branch || strings.HasPrefix(provider.Branch, branch) {
|
|
|
|
|
log.Debugf("Found branch config for branch %s with release type %s", provider.Branch, releaseType)
|
2019-08-13 23:01:38 +02:00
|
|
|
newVersion = s.calculator.CalculateNewVersion(analyzedCommits, lastVersion, releaseType, firstRelease)
|
2019-07-22 20:46:48 +02:00
|
|
|
break
|
2019-05-15 22:09:52 +02:00
|
|
|
}
|
2019-05-13 22:36:24 +02:00
|
|
|
}
|
|
|
|
|
|
2019-06-15 23:03:27 +02:00
|
|
|
releaseVersion := shared.ReleaseVersion{
|
|
|
|
|
Next: shared.ReleaseVersionEntry{
|
2019-07-15 21:20:44 +02:00
|
|
|
Commit: provider.Commit,
|
2019-06-15 23:03:27 +02:00
|
|
|
Version: &newVersion,
|
|
|
|
|
},
|
|
|
|
|
Last: shared.ReleaseVersionEntry{
|
|
|
|
|
Commit: lastVersionHash,
|
|
|
|
|
Version: lastVersion,
|
|
|
|
|
},
|
2019-08-11 18:27:52 +02:00
|
|
|
Branch: provider.Branch,
|
|
|
|
|
Commits: analyzedCommits,
|
2019-06-15 23:03:27 +02:00
|
|
|
}
|
|
|
|
|
|
2019-08-23 08:58:44 +02:00
|
|
|
if firstRelease {
|
|
|
|
|
releaseVersion.Last.Version, _ = semver.NewVersion("0.0.0")
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-16 21:30:35 +02:00
|
|
|
log.Infof("New version %s -> %s", lastVersion.String(), newVersion.String())
|
2019-07-22 20:46:48 +02:00
|
|
|
err = cache.Write(s.repository, releaseVersion)
|
2019-05-14 20:19:36 +02:00
|
|
|
if err != nil {
|
2019-08-11 18:27:52 +02:00
|
|
|
return nil, err
|
2019-05-14 20:19:36 +02:00
|
|
|
}
|
2019-08-11 18:27:52 +02:00
|
|
|
return &releaseVersion, err
|
2019-05-13 22:36:24 +02:00
|
|
|
}
|
|
|
|
|
|
2019-05-14 20:19:36 +02:00
|
|
|
//SetVersion for git repository
|
2019-07-24 22:14:03 +02:00
|
|
|
func (s *SemanticRelease) SetVersion(provider *ci.ProviderConfig, version string) error {
|
2019-05-13 22:36:24 +02:00
|
|
|
|
|
|
|
|
newVersion, err := semver.NewVersion(version)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-15 21:20:44 +02:00
|
|
|
lastVersion, lastVersionHash, err := s.gitutil.GetLastVersion()
|
2019-05-13 22:36:24 +02:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2019-07-15 21:20:44 +02:00
|
|
|
if lastVersion == nil {
|
|
|
|
|
lastVersion, _ = semver.NewVersion("1.0.0")
|
|
|
|
|
}
|
2019-05-13 22:36:24 +02:00
|
|
|
|
2019-07-22 20:46:48 +02:00
|
|
|
return cache.Write(s.repository, shared.ReleaseVersion{
|
2019-06-15 23:03:27 +02:00
|
|
|
Next: shared.ReleaseVersionEntry{
|
2019-07-15 21:20:44 +02:00
|
|
|
Commit: provider.Commit,
|
2019-06-15 23:03:27 +02:00
|
|
|
Version: newVersion,
|
|
|
|
|
},
|
|
|
|
|
Last: shared.ReleaseVersionEntry{
|
|
|
|
|
Commit: lastVersionHash,
|
|
|
|
|
Version: lastVersion,
|
|
|
|
|
},
|
2019-07-15 21:20:44 +02:00
|
|
|
Branch: provider.Branch,
|
2019-06-15 23:03:27 +02:00
|
|
|
})
|
2019-05-16 21:30:35 +02:00
|
|
|
}
|
2019-05-25 18:10:24 +02:00
|
|
|
|
|
|
|
|
// GetChangelog from last version till now
|
2019-08-11 18:27:52 +02:00
|
|
|
func (s *SemanticRelease) GetChangelog(releaseVersion *shared.ReleaseVersion) (*shared.GeneratedChangelog, error) {
|
2019-07-21 15:07:13 +02:00
|
|
|
c := changelog.New(s.config, s.analyzer.GetRules(), time.Now())
|
2019-06-15 23:03:27 +02:00
|
|
|
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()),
|
2019-08-11 18:27:52 +02:00
|
|
|
}, releaseVersion.Commits)
|
2019-06-13 23:33:42 +02:00
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 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-07-24 22:14:03 +02:00
|
|
|
func (s *SemanticRelease) Release(provider *ci.ProviderConfig, force bool) error {
|
2019-07-15 21:20:44 +02:00
|
|
|
|
|
|
|
|
if provider.IsPR {
|
2019-08-22 00:12:24 +02:00
|
|
|
log.Infof("Will not perform a new release. This is a pull request")
|
2019-07-15 21:20:44 +02:00
|
|
|
return nil
|
2019-06-15 23:16:30 +02:00
|
|
|
}
|
2019-06-13 23:45:21 +02:00
|
|
|
|
2019-07-15 21:20:44 +02:00
|
|
|
if _, ok := s.config.Branch[provider.Branch]; !ok {
|
2019-08-22 00:12:24 +02:00
|
|
|
log.Infof("Will not perform a new release. Current %s branch is not configured in release config", provider.Branch)
|
2019-06-13 23:45:21 +02:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-11 18:27:52 +02:00
|
|
|
releaseVersion, err := s.GetNextVersion(provider, force)
|
2019-06-13 23:33:42 +02:00
|
|
|
if err != nil {
|
|
|
|
|
log.Debugf("Could not get next version")
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-07 23:24:13 +02:00
|
|
|
if releaseVersion.Next.Version.Equal(releaseVersion.Last.Version) {
|
|
|
|
|
log.Infof("No new version, no release needed %s <> %s", releaseVersion.Next.Version.String(), releaseVersion.Last.Version.String())
|
2019-07-24 22:14:03 +02:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-11 18:27:52 +02:00
|
|
|
generatedChanglog, err := s.GetChangelog(releaseVersion)
|
2019-06-13 23:33:42 +02:00
|
|
|
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 {
|
2019-06-10 16:24:44 +02:00
|
|
|
return err
|
|
|
|
|
}
|
2019-05-25 18:10:24 +02:00
|
|
|
|
2019-07-09 01:46:31 +02:00
|
|
|
if err = releaser.UploadAssets(s.repository, s.config.Assets); err != nil {
|
2019-07-07 23:42:52 +02:00
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-13 23:33:42 +02:00
|
|
|
return nil
|
2019-05-25 18:10:24 +02:00
|
|
|
}
|
2019-07-09 01:45:16 +02:00
|
|
|
|
2019-07-09 01:46:31 +02:00
|
|
|
// ZipFiles zip files configured in release config
|
|
|
|
|
func (s *SemanticRelease) ZipFiles() error {
|
|
|
|
|
for _, file := range s.config.Assets {
|
2019-07-09 01:45:16 +02:00
|
|
|
if file.Compress {
|
2019-07-09 01:46:31 +02:00
|
|
|
if _, err := util.PrepareAssets(s.repository, s.config.Assets); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|