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

306 lines
8.3 KiB
Go
Raw Normal View History

package semanticrelease
import (
"fmt"
"os"
2019-07-21 15:07:13 +02:00
"time"
"github.com/Nightapes/go-semantic-release/internal/integrations"
"github.com/Masterminds/semver"
log "github.com/sirupsen/logrus"
"github.com/Nightapes/go-semantic-release/internal/analyzer"
"github.com/Nightapes/go-semantic-release/internal/assets"
"github.com/Nightapes/go-semantic-release/internal/cache"
"github.com/Nightapes/go-semantic-release/internal/calculator"
"github.com/Nightapes/go-semantic-release/internal/changelog"
"github.com/Nightapes/go-semantic-release/internal/ci"
"github.com/Nightapes/go-semantic-release/internal/gitutil"
"github.com/Nightapes/go-semantic-release/internal/hooks"
"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"
)
2019-05-15 22:09:52 +02:00
// SemanticRelease struct
type SemanticRelease struct {
config *config.ReleaseConfig
gitUtil *gitutil.GitUtil
analyzer *analyzer.Analyzer
calculator *calculator.Calculator
releaser releaser.Releaser
2020-03-24 19:56:57 +01:00
assets *assets.Set
repository string
checkConfig bool
2019-05-15 22:09:52 +02:00
}
// New SemanticRelease struct
func New(c *config.ReleaseConfig, repository string, checkConfig bool) (*SemanticRelease, error) {
2019-06-15 23:03:27 +02:00
util, err := gitutil.New(repository)
if err != nil {
return nil, err
2019-05-15 22:09:52 +02:00
}
analyzer, err := analyzer.New(c.CommitFormat, c.Analyzer, c.Changelog)
if err != nil {
2019-06-15 23:03:27 +02:00
return nil, err
}
if !checkConfig {
log.Infof("Ignore config checks!. No guarantee to run without issues")
}
assets := assets.New(repository, c.Checksum.Algorithm)
releaser, err := releaser.New(c, util).GetReleaser(checkConfig)
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,
assets: assets,
checkConfig: checkConfig,
calculator: calculator.New(),
2019-06-15 23:03:27 +02:00
}, nil
}
// GetCIProvider result with ci config
func (s *SemanticRelease) GetCIProvider() (*ci.ProviderConfig, error) {
return ci.GetCIProvider(s.gitUtil, s.checkConfig, ci.ReadAllEnvs())
}
// GetNextVersion from .version or calculate new from commits
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 {
releaseVersion, err := cache.Read(s.repository)
2019-06-15 23:03:27 +02:00
if err != nil {
return nil, err
2019-06-15 23:03:27 +02:00
}
if releaseVersion.Next.Commit == provider.Commit && releaseVersion != nil {
return releaseVersion, nil
2019-05-15 22:09:52 +02:00
}
}
lastVersion, lastVersionHash, err := s.gitUtil.GetLastVersion()
if err != nil {
return nil, err
}
2019-06-15 23:03:27 +02:00
firstRelease := false
if lastVersion == nil {
lastVersion, _ = semver.NewVersion("1.0.0")
log.Infof("This is the first release, will set version to %s", lastVersion.String())
firstRelease = true
}
commits, err := s.gitUtil.GetCommits(lastVersionHash)
if err != nil {
return nil, fmt.Errorf("could not get commits %w", err)
}
log.Debugf("Found %d commits till last release", len(commits))
analyzedCommits := s.analyzer.Analyze(commits)
var newVersion semver.Version
foundBranchConfig := false
2019-05-15 22:09:52 +02:00
for branch, releaseType := range s.config.Branch {
if provider.Branch == branch {
log.Debugf("Found branch config for branch %s with release type %s", provider.Branch, releaseType)
newVersion = s.calculator.CalculateNewVersion(analyzedCommits, lastVersion, releaseType, firstRelease)
foundBranchConfig = true
break
2019-05-15 22:09:52 +02:00
}
}
if !foundBranchConfig {
log.Warnf("No branch config found for branch %s, will return last known version", provider.Branch)
newVersion = *lastVersion
}
2019-06-15 23:03:27 +02:00
releaseVersion := shared.ReleaseVersion{
Next: shared.ReleaseVersionEntry{
Commit: provider.Commit,
2019-06-15 23:03:27 +02:00
Version: &newVersion,
},
Last: shared.ReleaseVersionEntry{
Commit: "",
2019-06-15 23:03:27 +02:00
Version: lastVersion,
},
Branch: provider.Branch,
Commits: analyzedCommits,
2019-06-15 23:03:27 +02:00
}
if lastVersionHash != nil {
releaseVersion.Last.Commit = lastVersionHash.Hash().String()
}
2019-06-15 23:03:27 +02:00
if firstRelease {
releaseVersion.Last.Version, _ = semver.NewVersion("0.0.0")
}
log.Infof("New version %s -> %s", lastVersion.String(), newVersion.String())
err = cache.Write(s.repository, releaseVersion)
2019-05-14 20:19:36 +02:00
if err != nil {
return nil, err
2019-05-14 20:19:36 +02:00
}
return &releaseVersion, err
}
// SetVersion for git repository
func (s *SemanticRelease) SetVersion(provider *ci.ProviderConfig, version string) error {
newVersion, err := semver.NewVersion(version)
if err != nil {
return err
}
lastVersion, lastVersionHash, err := s.gitUtil.GetLastVersion()
if err != nil {
return err
}
if lastVersion == nil {
lastVersion, _ = semver.NewVersion("1.0.0")
}
return cache.Write(s.repository, shared.ReleaseVersion{
2019-06-15 23:03:27 +02:00
Next: shared.ReleaseVersionEntry{
Commit: provider.Commit,
2019-06-15 23:03:27 +02:00
Version: newVersion,
},
Last: shared.ReleaseVersionEntry{
Commit: lastVersionHash.Hash().String(),
2019-06-15 23:03:27 +02:00
Version: lastVersion,
},
Branch: provider.Branch,
2019-06-15 23:03:27 +02:00
})
}
// GetChangelog from last version till now
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())
return c.GenerateChangelog(shared.ChangelogTemplateConfig{
2019-06-15 23:03:27 +02:00
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()),
}, releaseVersion.Commits)
}
// WriteChangeLog writes changelog content to the given file
func (s *SemanticRelease) WriteChangeLog(changelogContent, file string, overwrite bool) error {
newContent := make([]byte, 0)
newContent = append(newContent, []byte(changelogContent)...)
// only prepend the new changelog content if the the given file already exists
_, err := os.Stat(file)
if !overwrite && err == nil {
content, err := os.ReadFile(file)
if err != nil {
return err
}
newContent = append(newContent, []byte("\n---\n\n")...)
newContent = append(newContent, content...)
}
return os.WriteFile(file, newContent, 0644)
}
// Release publish release to provider
func (s *SemanticRelease) Release(provider *ci.ProviderConfig, force bool) error {
if provider.IsPR {
log.Infof("Will not perform a new release. This is a pull request")
return nil
2019-06-15 23:16:30 +02:00
}
if _, ok := s.config.Branch[provider.Branch]; !ok {
log.Infof("Will not perform a new release. Current %s branch is not configured in release config", provider.Branch)
return nil
}
if err := s.assets.Add(s.config.Assets...); err != nil {
return err
}
releaseVersion, err := s.GetNextVersion(provider, force)
if err != nil {
log.Debugf("Could not get next version")
return err
}
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())
return nil
}
generatedChangelog, err := s.GetChangelog(releaseVersion)
if err != nil {
log.Debugf("Could not get changelog")
return err
}
integrations := integrations.New(&s.config.Integrations, releaseVersion)
if err := integrations.Run(); err != nil {
log.Debugf("Error during integrations run")
return err
}
hook := hooks.New(s.config, releaseVersion)
if err := hook.PreRelease(); err != nil {
log.Debugf("Error during pre release hook")
2019-06-15 23:03:27 +02:00
return err
}
if s.config.Checksum.Algorithm != "" {
if err := s.assets.GenerateChecksum(); err != nil {
return err
}
}
for _, asset := range s.assets.All() {
if asset.IsCompressed() {
if _, err := asset.ZipFile(); err != nil {
return err
}
}
}
if err = s.releaser.CreateRelease(releaseVersion, generatedChangelog, s.assets); err != nil {
return err
}
if err := hook.PostRelease(); err != nil {
log.Debugf("Error during post release hook")
return err
}
return nil
}
// ZipFiles zip files configured in release config
func (s *SemanticRelease) ZipFiles() error {
assets := assets.New(s.repository, "")
if err := assets.Add(s.config.Assets...); err != nil {
return err
}
if err := assets.GenerateChecksum(); err != nil {
return err
}
for _, asset := range assets.All() {
path, err := asset.GetPath()
if err != nil {
return err
}
log.Infof("File %s under %s is zipped %t", asset.GetName(), path, asset.IsCompressed())
}
return nil
}