2019-11-17 13:17:24 -03:00
|
|
|
package sv
|
|
|
|
|
2021-02-13 15:40:09 -03:00
|
|
|
import "github.com/Masterminds/semver/v3"
|
2019-11-17 13:17:24 -03:00
|
|
|
|
|
|
|
type versionType int
|
|
|
|
|
|
|
|
const (
|
|
|
|
none versionType = iota
|
|
|
|
patch
|
|
|
|
minor
|
|
|
|
major
|
|
|
|
)
|
|
|
|
|
2023-10-15 21:29:29 +02:00
|
|
|
// CommitLog description of a single commit log.
|
|
|
|
type CommitLog struct {
|
|
|
|
Date string `json:"date,omitempty"`
|
|
|
|
Timestamp int `json:"timestamp,omitempty"`
|
|
|
|
AuthorName string `json:"authorName,omitempty"`
|
|
|
|
Hash string `json:"hash,omitempty"`
|
|
|
|
Message CommitMessage `json:"message,omitempty"`
|
|
|
|
}
|
|
|
|
|
2022-01-29 19:25:52 -03:00
|
|
|
// IsValidVersion return true when a version is valid.
|
|
|
|
func IsValidVersion(value string) bool {
|
|
|
|
_, err := semver.NewVersion(value)
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2022-01-29 19:25:52 -03:00
|
|
|
return err == nil
|
|
|
|
}
|
|
|
|
|
2021-07-31 16:03:58 -03:00
|
|
|
// ToVersion parse string to semver.Version.
|
2021-12-29 18:02:50 -03:00
|
|
|
func ToVersion(value string) (*semver.Version, error) {
|
2019-11-17 13:17:24 -03:00
|
|
|
version := value
|
|
|
|
if version == "" {
|
|
|
|
version = "0.0.0"
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2021-12-29 18:02:50 -03:00
|
|
|
return semver.NewVersion(version)
|
2019-11-17 13:17:24 -03:00
|
|
|
}
|
|
|
|
|
2023-10-15 21:29:29 +02:00
|
|
|
// CommitProcessor interface.
|
|
|
|
type CommitProcessor interface {
|
|
|
|
NextVersion(version *semver.Version, commits []CommitLog) (*semver.Version, bool)
|
2019-11-17 13:17:24 -03:00
|
|
|
}
|
|
|
|
|
2023-10-15 21:29:29 +02:00
|
|
|
// SemVerCommitProcessor process versions using commit log.
|
|
|
|
type SemVerCommitProcessor struct {
|
2019-11-17 13:17:24 -03:00
|
|
|
MajorVersionTypes map[string]struct{}
|
|
|
|
MinorVersionTypes map[string]struct{}
|
|
|
|
PatchVersionTypes map[string]struct{}
|
2021-02-15 02:41:43 -03:00
|
|
|
KnownTypes []string
|
2019-11-17 13:17:24 -03:00
|
|
|
IncludeUnknownTypeAsPatch bool
|
|
|
|
}
|
|
|
|
|
2023-10-15 21:29:29 +02:00
|
|
|
// VersioningConfig versioning preferences.
|
|
|
|
type VersioningConfig struct {
|
|
|
|
UpdateMajor []string `yaml:"update-major,flow"`
|
|
|
|
UpdateMinor []string `yaml:"update-minor,flow"`
|
|
|
|
UpdatePatch []string `yaml:"update-patch,flow"`
|
|
|
|
IgnoreUnknown bool `yaml:"ignore-unknown"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewSemVerCommitProcessor SemanticVersionCommitProcessorImpl constructor.
|
|
|
|
func NewSemVerCommitProcessor(vcfg VersioningConfig, mcfg CommitMessageConfig) *SemVerCommitProcessor {
|
|
|
|
return &SemVerCommitProcessor{
|
2021-02-15 02:41:43 -03:00
|
|
|
IncludeUnknownTypeAsPatch: !vcfg.IgnoreUnknown,
|
|
|
|
MajorVersionTypes: toMap(vcfg.UpdateMajor),
|
|
|
|
MinorVersionTypes: toMap(vcfg.UpdateMinor),
|
|
|
|
PatchVersionTypes: toMap(vcfg.UpdatePatch),
|
|
|
|
KnownTypes: mcfg.Types,
|
2019-11-17 13:17:24 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-31 16:03:58 -03:00
|
|
|
// NextVersion calculates next version based on commit log.
|
2023-10-15 21:29:29 +02:00
|
|
|
func (p SemVerCommitProcessor) NextVersion(
|
|
|
|
version *semver.Version, commits []CommitLog,
|
2023-10-12 16:18:25 +02:00
|
|
|
) (*semver.Version, bool) {
|
2021-07-31 16:03:58 -03:00
|
|
|
versionToUpdate := none
|
2019-11-17 13:17:24 -03:00
|
|
|
for _, commit := range commits {
|
|
|
|
if v := p.versionTypeToUpdate(commit); v > versionToUpdate {
|
|
|
|
versionToUpdate = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-29 18:02:50 -03:00
|
|
|
updated := versionToUpdate != none
|
|
|
|
if version == nil {
|
|
|
|
return nil, updated
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2021-12-29 18:02:50 -03:00
|
|
|
newVersion := updateVersion(*version, versionToUpdate)
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2021-12-29 18:02:50 -03:00
|
|
|
return &newVersion, updated
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateVersion(version semver.Version, versionToUpdate versionType) semver.Version {
|
2019-11-17 13:17:24 -03:00
|
|
|
switch versionToUpdate {
|
|
|
|
case major:
|
2021-12-29 18:02:50 -03:00
|
|
|
return version.IncMajor()
|
2019-11-17 13:17:24 -03:00
|
|
|
case minor:
|
2021-12-29 18:02:50 -03:00
|
|
|
return version.IncMinor()
|
2019-11-17 13:17:24 -03:00
|
|
|
case patch:
|
2021-12-29 18:02:50 -03:00
|
|
|
return version.IncPatch()
|
2019-11-17 13:17:24 -03:00
|
|
|
default:
|
2021-12-29 18:02:50 -03:00
|
|
|
return version
|
2019-11-17 13:17:24 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-15 21:29:29 +02:00
|
|
|
func (p SemVerCommitProcessor) versionTypeToUpdate(commit CommitLog) versionType {
|
2021-02-13 23:35:31 -03:00
|
|
|
if commit.Message.IsBreakingChange {
|
2019-11-17 13:17:24 -03:00
|
|
|
return major
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2021-02-13 23:35:31 -03:00
|
|
|
if _, exists := p.MajorVersionTypes[commit.Message.Type]; exists {
|
2019-11-17 13:17:24 -03:00
|
|
|
return major
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2021-02-13 23:35:31 -03:00
|
|
|
if _, exists := p.MinorVersionTypes[commit.Message.Type]; exists {
|
2019-11-17 13:17:24 -03:00
|
|
|
return minor
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2021-02-13 23:35:31 -03:00
|
|
|
if _, exists := p.PatchVersionTypes[commit.Message.Type]; exists {
|
2019-11-17 13:17:24 -03:00
|
|
|
return patch
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2021-02-15 02:41:43 -03:00
|
|
|
if !contains(commit.Message.Type, p.KnownTypes) && p.IncludeUnknownTypeAsPatch {
|
2019-11-17 13:17:24 -03:00
|
|
|
return patch
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2019-11-17 13:17:24 -03:00
|
|
|
return none
|
|
|
|
}
|
|
|
|
|
|
|
|
func toMap(values []string) map[string]struct{} {
|
|
|
|
result := make(map[string]struct{})
|
|
|
|
for _, v := range values {
|
|
|
|
result[v] = struct{}{}
|
|
|
|
}
|
2023-10-12 16:18:25 +02:00
|
|
|
|
2019-11-17 13:17:24 -03:00
|
|
|
return result
|
|
|
|
}
|