summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorAvery Pennarun <apenwarr@tailscale.com>2026-04-13 03:52:19 +0200
committerBrad Fitzpatrick <brad@danga.com>2026-04-14 07:11:44 -0700
commit621dc9cf1b5afb7d2ba00fc1c525d503175a4052 (patch)
tree8a0a129c3a83df7b1d3fb632721df3dababcc860
parent6aa10576c902b0930c72f0c0113bb6e8636cfd1e (diff)
downloadtailscale-621dc9cf1b5afb7d2ba00fc1c525d503175a4052.tar.xz
tailscale-621dc9cf1b5afb7d2ba00fc1c525d503175a4052.zip
tstest: fix kernel version parsing for Debian-style version strings
The kernel version parser used strings.Cut with "-" to handle versions like "5.4.0-76-generic", but Debian uses "+" in versions like "6.12.41+deb13-amd64". Use strings.IndexAny to find the first "-" or "+" and truncate there. Fixes TestKernelVersion on Debian systems. Fixes #19395 Change-Id: I70e5f95682d54baf908e51f9f4b51c130b00aaaa Co-Authored-By: Brad Fitzpatrick <bradfitz@tailscale.com> Signed-off-by: Avery Pennarun <apenwarr@tailscale.com>
-rw-r--r--tstest/kernel_linux.go16
-rw-r--r--tstest/kernel_linux_test.go34
2 files changed, 46 insertions, 4 deletions
diff --git a/tstest/kernel_linux.go b/tstest/kernel_linux.go
index ab7c0d529..ed48fd071 100644
--- a/tstest/kernel_linux.go
+++ b/tstest/kernel_linux.go
@@ -20,8 +20,13 @@ func KernelVersion() (major, minor, patch int) {
return 0, 0, 0
}
release := unix.ByteSliceToString(uname.Release[:])
+ return parseKernelVersion(release)
+}
- // Parse version string (e.g., "5.15.0-...")
+// parseKernelVersion parses a Linux kernel version string like "6.12.73+deb13-amd64"
+// or "5.15.0-76-generic" and returns the major, minor, and patch components.
+// It returns (0, 0, 0) if the version cannot be parsed.
+func parseKernelVersion(release string) (major, minor, patch int) {
parts := strings.Split(release, ".")
if len(parts) < 3 {
return 0, 0, 0
@@ -37,9 +42,12 @@ func KernelVersion() (major, minor, patch int) {
return 0, 0, 0
}
- // Patch version may have additional info after a hyphen (e.g., "0-76-generic")
- // Extract just the numeric part before any hyphen
- patchStr, _, _ := strings.Cut(parts[2], "-")
+ // Patch version may have additional info after a hyphen or plus (e.g., "0-76-generic" or "41+deb13-amd64")
+ // Extract just the numeric part before any hyphen or plus
+ patchStr := parts[2]
+ if idx := strings.IndexAny(patchStr, "-+"); idx != -1 {
+ patchStr = patchStr[:idx]
+ }
patch, err = strconv.Atoi(patchStr)
if err != nil {
diff --git a/tstest/kernel_linux_test.go b/tstest/kernel_linux_test.go
new file mode 100644
index 000000000..9445ebe2c
--- /dev/null
+++ b/tstest/kernel_linux_test.go
@@ -0,0 +1,34 @@
+// Copyright (c) Tailscale Inc & contributors
+// SPDX-License-Identifier: BSD-3-Clause
+
+//go:build linux
+
+package tstest
+
+import "testing"
+
+func TestParseKernelVersion(t *testing.T) {
+ tests := []struct {
+ release string
+ major, minor, patch int
+ }{
+ {"5.15.0-76-generic", 5, 15, 0},
+ {"6.12.73+deb13-amd64", 6, 12, 73},
+ {"6.1.0-18-amd64", 6, 1, 0},
+ {"5.4.0", 5, 4, 0},
+ {"6.8.12", 6, 8, 12},
+ {"4.19.0+1", 4, 19, 0},
+ {"6.12.41+deb13-amd64", 6, 12, 41},
+ {"", 0, 0, 0},
+ {"not-a-version", 0, 0, 0},
+ {"1.2", 0, 0, 0},
+ {"a.b.c", 0, 0, 0},
+ }
+ for _, tt := range tests {
+ major, minor, patch := parseKernelVersion(tt.release)
+ if major != tt.major || minor != tt.minor || patch != tt.patch {
+ t.Errorf("parseKernelVersion(%q) = (%d, %d, %d), want (%d, %d, %d)",
+ tt.release, major, minor, patch, tt.major, tt.minor, tt.patch)
+ }
+ }
+}