mirror of https://github.com/go-gitea/gitea
Backport #33262 by ExplodingDragon Only show the latest version of the package in the arch repo. closes #33534 Co-authored-by: Exploding Dragon <explodingfkl@gmail.com> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>pull/33592/head
parent
76bd60fc1d
commit
cc6ec56738
@ -0,0 +1,113 @@ |
||||
// Copyright 2025 The Gitea Authors. All rights reserved.
|
||||
// SPDX-License-Identifier: MIT
|
||||
|
||||
package arch |
||||
|
||||
import ( |
||||
"strings" |
||||
"unicode" |
||||
) |
||||
|
||||
// https://gitlab.archlinux.org/pacman/pacman/-/blob/d55b47e5512808b67bc944feb20c2bcc6c1a4c45/lib/libalpm/version.c
|
||||
|
||||
import ( |
||||
"strconv" |
||||
) |
||||
|
||||
func parseEVR(evr string) (epoch, version, release string) { |
||||
if before, after, f := strings.Cut(evr, ":"); f { |
||||
epoch = before |
||||
evr = after |
||||
} else { |
||||
epoch = "0" |
||||
} |
||||
|
||||
if before, after, f := strings.Cut(evr, "-"); f { |
||||
version = before |
||||
release = after |
||||
} else { |
||||
version = evr |
||||
release = "1" |
||||
} |
||||
return epoch, version, release |
||||
} |
||||
|
||||
func compareSegments(a, b []string) int { |
||||
lenA, lenB := len(a), len(b) |
||||
var l int |
||||
if lenA > lenB { |
||||
l = lenB |
||||
} else { |
||||
l = lenA |
||||
} |
||||
for i := 0; i < l; i++ { |
||||
if r := compare(a[i], b[i]); r != 0 { |
||||
return r |
||||
} |
||||
} |
||||
if lenA == lenB { |
||||
return 0 |
||||
} else if l == lenA { |
||||
return -1 |
||||
} |
||||
return 1 |
||||
} |
||||
|
||||
func compare(a, b string) int { |
||||
if a == b { |
||||
return 0 |
||||
} |
||||
|
||||
aNumeric := isNumeric(a) |
||||
bNumeric := isNumeric(b) |
||||
|
||||
if aNumeric && bNumeric { |
||||
aInt, _ := strconv.Atoi(a) |
||||
bInt, _ := strconv.Atoi(b) |
||||
switch { |
||||
case aInt < bInt: |
||||
return -1 |
||||
case aInt > bInt: |
||||
return 1 |
||||
default: |
||||
return 0 |
||||
} |
||||
} |
||||
|
||||
if aNumeric { |
||||
return 1 |
||||
} |
||||
if bNumeric { |
||||
return -1 |
||||
} |
||||
|
||||
return strings.Compare(a, b) |
||||
} |
||||
|
||||
func isNumeric(s string) bool { |
||||
for _, c := range s { |
||||
if !unicode.IsDigit(c) { |
||||
return false |
||||
} |
||||
} |
||||
return true |
||||
} |
||||
|
||||
func compareVersions(a, b string) int { |
||||
if a == b { |
||||
return 0 |
||||
} |
||||
|
||||
epochA, versionA, releaseA := parseEVR(a) |
||||
epochB, versionB, releaseB := parseEVR(b) |
||||
|
||||
if res := compareSegments([]string{epochA}, []string{epochB}); res != 0 { |
||||
return res |
||||
} |
||||
|
||||
if res := compareSegments(strings.Split(versionA, "."), strings.Split(versionB, ".")); res != 0 { |
||||
return res |
||||
} |
||||
|
||||
return compareSegments([]string{releaseA}, []string{releaseB}) |
||||
} |
@ -0,0 +1,27 @@ |
||||
// Copyright 2025 The Gitea Authors. All rights reserved.
|
||||
// SPDX-License-Identifier: MIT
|
||||
|
||||
package arch |
||||
|
||||
import ( |
||||
"testing" |
||||
|
||||
"github.com/stretchr/testify/require" |
||||
) |
||||
|
||||
func TestCompareVersions(t *testing.T) { |
||||
// https://man.archlinux.org/man/vercmp.8.en
|
||||
checks := [][]string{ |
||||
{"1.0a", "1.0b", "1.0beta", "1.0p", "1.0pre", "1.0rc", "1.0", "1.0.a", "1.0.1"}, |
||||
{"1", "1.0", "1.1", "1.1.1", "1.2", "2.0", "3.0.0"}, |
||||
} |
||||
for _, check := range checks { |
||||
for i := 0; i < len(check)-1; i++ { |
||||
require.Equal(t, -1, compareVersions(check[i], check[i+1])) |
||||
require.Equal(t, 1, compareVersions(check[i+1], check[i])) |
||||
} |
||||
} |
||||
require.Equal(t, 1, compareVersions("1.0-2", "1.0")) |
||||
require.Equal(t, 0, compareVersions("0:1.0-1", "1.0")) |
||||
require.Equal(t, 1, compareVersions("1:1.0-1", "2.0")) |
||||
} |
Loading…
Reference in new issue