aboutsummaryrefslogtreecommitdiffstats
path: root/services/packages/arch/vercmp.go
blob: 0d33dda0f1232ef44721592e4618c57de3005289 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
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})
}