Skip to content

Commit ebfd660

Browse files
committed
Bug 1339754 - Fix tag sorting according to semantic versioning rules
1 parent 3b2bbe5 commit ebfd660

File tree

3 files changed

+34
-22
lines changed

3 files changed

+34
-22
lines changed

pkg/image/api/helper.go

+13-16
Original file line numberDiff line numberDiff line change
@@ -858,17 +858,17 @@ func LatestObservedTagGeneration(stream *ImageStream, tag string) int64 {
858858
}
859859

860860
var (
861-
reMajorSemantic = regexp.MustCompile(`^[\d]+$`)
862-
reMinorSemantic = regexp.MustCompile(`^[\d]+\.[\d]+$`)
861+
reMinorSemantic = regexp.MustCompile(`^[\d]+\.[\d]+$`)
862+
reMinorReplacement = regexp.MustCompile(`[\d]+\.[\d]+`)
863+
reMinorWithPatch = regexp.MustCompile(`^[\d]+\.[\d]+-\w+$`)
863864
)
864865

865866
// PrioritizeTags orders a set of image tags with a few conventions:
866867
//
867868
// 1. the "latest" tag, if present, should be first
868-
// 2. any tags that represent a semantic major version ("5", "v5") should be next, in descending order
869-
// 3. any tags that represent a semantic minor version ("5.1", "v5.1") should be next, in descending order
870-
// 4. any tags that represent a full semantic version ("5.1.3-other", "v5.1.3-other") should be next, in descending order
871-
// 5. any remaining tags should be sorted in lexicographic order
869+
// 2. any tags that represent a semantic minor version ("5.1", "v5.1", "v5.1-rc1") should be next, in descending order
870+
// 3. any tags that represent a full semantic version ("5.1.3-other", "v5.1.3-other") should be next, in descending order
871+
// 4. any remaining tags should be sorted in lexicographic order
872872
//
873873
// The method updates the tags in place.
874874
func PrioritizeTags(tags []string) {
@@ -884,7 +884,7 @@ func PrioritizeTags(tags []string) {
884884
}
885885

886886
exact := make(map[string]string)
887-
var major, minor, micro semver.Versions
887+
var minor, micro semver.Versions
888888
other := make([]string, 0, len(remaining))
889889
for _, tag := range remaining {
890890
short := strings.TrimLeft(tag, "v")
@@ -894,28 +894,25 @@ func PrioritizeTags(tags []string) {
894894
exact[v.String()] = tag
895895
micro = append(micro, v)
896896
continue
897-
case reMajorSemantic.MatchString(short):
898-
if v, err = semver.Parse(short + ".0.0"); err == nil {
897+
case reMinorSemantic.MatchString(short):
898+
if v, err = semver.Parse(short + ".0"); err == nil {
899899
exact[v.String()] = tag
900-
major = append(major, v)
900+
minor = append(minor, v)
901901
continue
902902
}
903-
case reMinorSemantic.MatchString(short):
904-
if v, err = semver.Parse(short + ".0"); err == nil {
903+
case reMinorWithPatch.MatchString(short):
904+
repl := reMinorReplacement.FindString(short)
905+
if v, err = semver.Parse(strings.Replace(short, repl, repl+".0", 1)); err == nil {
905906
exact[v.String()] = tag
906907
minor = append(minor, v)
907908
continue
908909
}
909910
}
910911
other = append(other, tag)
911912
}
912-
sort.Sort(sort.Reverse(major))
913913
sort.Sort(sort.Reverse(minor))
914914
sort.Sort(sort.Reverse(micro))
915915
sort.Sort(sort.StringSlice(other))
916-
for _, v := range major {
917-
finalTags = append(finalTags, exact[v.String()])
918-
}
919916
for _, v := range minor {
920917
finalTags = append(finalTags, exact[v.String()])
921918
}

pkg/image/api/helper_test.go

+19-4
Original file line numberDiff line numberDiff line change
@@ -1412,10 +1412,25 @@ func TestDockerImageReferenceEquality(t *testing.T) {
14121412
}
14131413

14141414
func TestPrioritizeTags(t *testing.T) {
1415-
tags := []string{"5", "other", "latest", "v5.5", "v6", "5.2.3", "v5.3.6-bother", "5.3.6-abba", "5.6"}
1416-
PrioritizeTags(tags)
1417-
if !reflect.DeepEqual(tags, []string{"latest", "v6", "5", "5.6", "v5.5", "v5.3.6-bother", "5.3.6-abba", "5.2.3", "other"}) {
1418-
t.Errorf("unexpected order: %v", tags)
1415+
tests := []struct {
1416+
tags []string
1417+
expected []string
1418+
}{
1419+
{
1420+
tags: []string{"other", "latest", "v5.5", "5.2.3", "v5.3.6-bother", "5.3.6-abba", "5.6"},
1421+
expected: []string{"latest", "5.6", "v5.5", "v5.3.6-bother", "5.3.6-abba", "5.2.3", "other"},
1422+
},
1423+
{
1424+
tags: []string{"1.1-beta1", "1.2-rc1", "1.1-rc1", "1.1-beta2", "1.2-beta1", "1.2-alpha1", "1.2-beta4", "latest"},
1425+
expected: []string{"latest", "1.2-rc1", "1.2-beta4", "1.2-beta1", "1.2-alpha1", "1.1-rc1", "1.1-beta2", "1.1-beta1"},
1426+
},
1427+
}
1428+
1429+
for i, tc := range tests {
1430+
PrioritizeTags(tc.tags)
1431+
if !reflect.DeepEqual(tc.tags, tc.expected) {
1432+
t.Errorf("%d: unexpected order: %v", i, tc.tags)
1433+
}
14191434
}
14201435
}
14211436

pkg/image/importer/importer_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -239,13 +239,13 @@ func TestImport(t *testing.T) {
239239
},
240240
},
241241
expect: func(isi *api.ImageStreamImport, t *testing.T) {
242-
if !reflect.DeepEqual(isi.Status.Repository.AdditionalTags, []string{"other"}) {
242+
if !reflect.DeepEqual(isi.Status.Repository.AdditionalTags, []string{"v2"}) {
243243
t.Errorf("unexpected additional tags: %#v", isi.Status.Repository)
244244
}
245245
if len(isi.Status.Repository.Images) != 5 {
246246
t.Errorf("unexpected number of images: %#v", isi.Status.Repository.Images)
247247
}
248-
expectedTags := []string{"3", "v2", "v1", "3.1", "abc"}
248+
expectedTags := []string{"3.1", "3", "abc", "other", "v1"}
249249
for i, image := range isi.Status.Repository.Images {
250250
if image.Status.Status != unversioned.StatusFailure || image.Status.Message != "Internal error occurred: no such manifest tag" {
251251
t.Errorf("unexpected status %d: %#v", i, isi.Status.Repository.Images)

0 commit comments

Comments
 (0)