|
|
|
# We use a simple Go program to report many Go versions.
|
|
|
|
# The program also errors on any command other than "go version",
|
|
|
|
# which saves us having to rebuild main.go many times.
|
|
|
|
go build -o .bin/go$exe ./fakego
|
|
|
|
env PATH=${WORK}/.bin${:}${PATH}
|
|
|
|
|
|
|
|
# An empty go version.
|
|
|
|
env TOOLCHAIN_GOVERSION=''
|
|
|
|
! exec garble build
|
|
|
|
stderr 'Go version is too old; please upgrade to go1\.23\.5 or newer'
|
|
|
|
|
|
|
|
# We should error on a devel version that's too old.
|
|
|
|
# Note that they lacked the "goN.M-" prefix.
|
|
|
|
env TOOLCHAIN_GOVERSION='devel +afb5fca Sun Aug 07 00:00:00 2020 +0000'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'Go version is too old; please upgrade to go1\.23\.5 or newer'
|
|
|
|
|
|
|
|
# Another form of old version; with an old "goN.M-" prefix.
|
|
|
|
env TOOLCHAIN_GOVERSION='devel go1.15-afb5fca Sun Aug 07 00:00:00 2020 +0000'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'Go version "devel go1\.15-.*2020.*" is too old; please upgrade to go1\.23'
|
|
|
|
|
|
|
|
# A current devel version should be fine.
|
|
|
|
# Note that we don't look at devel version timestamps.
|
|
|
|
env GARBLE_TEST_GOVERSION='go1.23.5'
|
|
|
|
# TODO: temporarily disabled while we do not support tip.
|
|
|
|
# env TOOLCHAIN_GOVERSION='devel go1.23-ad97d204f0 Sun Sep 12 16:46:58 2023 +0000'
|
|
|
|
# ! exec garble build
|
|
|
|
# stderr 'mocking the real build'
|
|
|
|
|
|
|
|
# We should error on a stable version that's too old.
|
|
|
|
env TOOLCHAIN_GOVERSION='go1.14'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'Go version "go1\.14" is too old; please upgrade to go1\.23\.5 or newer'
|
|
|
|
|
|
|
|
# We should reject a future stable version, as we don't have linker patches yet.
|
|
|
|
# Note that we need to bump the version of Go that supposedly built it, too.
|
|
|
|
env GARBLE_TEST_GOVERSION='go1.28.2'
|
|
|
|
env TOOLCHAIN_GOVERSION='go1.28.2'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'Go version "go1\.28\.2" is too new; Go linker patches aren''t available for go1\.25 or later yet'
|
|
|
|
|
CI: pin a commit when testing against Go tip
Since it changes rapidly, especially during merge cycles, and we don't
want CI to surprisingly blow up in our faces from one day to another.
Pin this to a commit from yesterday which works, since some changes
merged today moved where the Go build version is recorded and broke
garble.
While at it, replace "git clone" with a wget of a source archive. This
is much, much faster, mainly because a tarball is significantly smaller.
We now download about 20MiB instead of over 350MiB.
One downside is that, without git, make.bash can't construct a devel
version on its own. For that reason, add a pretty basic manual version
via the VERSION file.
This means that we must not reject custom devel version strings. This is
a good thing anyway, because custom devel strings are already common
when building Go in custom ways. Those people tend to be advanced users,
such as CI, so fall back to assuming they know what they are doing and
don't error.
Plus, starting last week, devel versions in Go master now contain the
major Go version like in build tags, such as "go1.17-commit...", so we
will soon start relying on that instead of parsing dates:
$ go version
go version devel go1.17-a7e16abb22 Thu Apr 8 07:33:58 2021 +0000 linux/amd64
4 years ago
|
|
|
# We should accept custom devel strings.
|
|
|
|
env TOOLCHAIN_GOVERSION='devel go1.23.5-somecustomversion'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'mocking the real build'
|
|
|
|
|
|
|
|
# The current toolchain may be older than the one that built garble.
|
|
|
|
env GARBLE_TEST_GOVERSION='go1.23.22'
|
|
|
|
env TOOLCHAIN_GOVERSION='go1.23.21'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'mocking the real build'
|
|
|
|
|
|
|
|
# The current toolchain may be equal to the one that built garble.
|
|
|
|
env GARBLE_TEST_GOVERSION='go1.23.25'
|
|
|
|
env TOOLCHAIN_GOVERSION='go1.23.25'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'mocking the real build'
|
|
|
|
|
|
|
|
# The current toolchain must not be newer than the one that built garble.
|
|
|
|
env GARBLE_TEST_GOVERSION='go1.18'
|
|
|
|
env TOOLCHAIN_GOVERSION='go1.23.25'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'garble was built with "go1\.18" and can''t be used with the newer "go1\.23\.25"; rebuild '
|
|
|
|
|
|
|
|
# We'll error even if the difference is a minor (bugfix) level.
|
|
|
|
# In practice it probably wouldn't matter, but in theory it could still lead to tricky bugs.
|
|
|
|
env GARBLE_TEST_GOVERSION='go1.23.11'
|
|
|
|
env TOOLCHAIN_GOVERSION='go1.23.14'
|
|
|
|
! exec garble build
|
|
|
|
stderr 'garble was built with "go1\.23\.11" and can''t be used with the newer "go1\.23\.14"; rebuild '
|
|
|
|
|
|
|
|
# If garble builds itself and is then used, it won't know what version built it.
|
|
|
|
# As a fallback, we drop the comparison against the toolchain's version.
|
|
|
|
env GARBLE_TEST_GOVERSION='bogus version'
|
|
|
|
env TOOLCHAIN_GOVERSION='go1.23.25'
|
|
|
|
! exec garble build
|
CI: pin a commit when testing against Go tip
Since it changes rapidly, especially during merge cycles, and we don't
want CI to surprisingly blow up in our faces from one day to another.
Pin this to a commit from yesterday which works, since some changes
merged today moved where the Go build version is recorded and broke
garble.
While at it, replace "git clone" with a wget of a source archive. This
is much, much faster, mainly because a tarball is significantly smaller.
We now download about 20MiB instead of over 350MiB.
One downside is that, without git, make.bash can't construct a devel
version on its own. For that reason, add a pretty basic manual version
via the VERSION file.
This means that we must not reject custom devel version strings. This is
a good thing anyway, because custom devel strings are already common
when building Go in custom ways. Those people tend to be advanced users,
such as CI, so fall back to assuming they know what they are doing and
don't error.
Plus, starting last week, devel versions in Go master now contain the
major Go version like in build tags, such as "go1.17-commit...", so we
will soon start relying on that instead of parsing dates:
$ go version
go version devel go1.17-a7e16abb22 Thu Apr 8 07:33:58 2021 +0000 linux/amd64
4 years ago
|
|
|
stderr 'mocking the real build'
|
|
|
|
-- go.mod --
|
|
|
|
module test/main
|
|
|
|
|
|
|
|
go 1.23
|
|
|
|
-- main.go --
|
|
|
|
package main
|
|
|
|
|
|
|
|
func main() {}
|
|
|
|
|
|
|
|
-- fakego/main.go --
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
use "go env -json" to collect env info all at once
In the worst case scenario, when GOPRIVATE isn't set at all, we would
run these three commands:
* "go env GOPRIVATE", to fetch GOPRIVATE itself
* "go list -m", for GOPRIVATE's fallback
* "go version", to check the version of Go being used
Now that we support Go 1.16 and later, all these three can be obtained
via "go env -json":
$ go env -json GOPRIVATE GOMOD GOVERSION
{
"GOMOD": "/home/mvdan/src/garble/go.mod",
"GOPRIVATE": "",
"GOVERSION": "go1.16.3"
}
Note that we don't get the module path directly, but we can use the
x/mod/modfile Go API to parse it from the GOMOD file cheaply.
Notably, this also simplifies our Go version checking logic, as now we
get just the version string without the "go version" prefix and
"GOOS/GOARCH" suffix we don't care about.
This makes our code a bit more maintainable and robust. When running a
short incremental build, we can also see a small speed-up, as saving two
"go" invocations can save a few milliseconds:
name old time/op new time/op delta
Build/Cache-8 168ms ± 0% 166ms ± 1% -1.26% (p=0.009 n=6+6)
name old bin-B new bin-B delta
Build/Cache-8 6.36M ± 0% 6.36M ± 0% +0.12% (p=0.002 n=6+6)
name old sys-time/op new sys-time/op delta
Build/Cache-8 222ms ± 2% 219ms ± 3% ~ (p=0.589 n=6+6)
name old user-time/op new user-time/op delta
Build/Cache-8 857ms ± 1% 846ms ± 1% -1.31% (p=0.041 n=6+6)
4 years ago
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
use "go env -json" to collect env info all at once
In the worst case scenario, when GOPRIVATE isn't set at all, we would
run these three commands:
* "go env GOPRIVATE", to fetch GOPRIVATE itself
* "go list -m", for GOPRIVATE's fallback
* "go version", to check the version of Go being used
Now that we support Go 1.16 and later, all these three can be obtained
via "go env -json":
$ go env -json GOPRIVATE GOMOD GOVERSION
{
"GOMOD": "/home/mvdan/src/garble/go.mod",
"GOPRIVATE": "",
"GOVERSION": "go1.16.3"
}
Note that we don't get the module path directly, but we can use the
x/mod/modfile Go API to parse it from the GOMOD file cheaply.
Notably, this also simplifies our Go version checking logic, as now we
get just the version string without the "go version" prefix and
"GOOS/GOARCH" suffix we don't care about.
This makes our code a bit more maintainable and robust. When running a
short incremental build, we can also see a small speed-up, as saving two
"go" invocations can save a few milliseconds:
name old time/op new time/op delta
Build/Cache-8 168ms ± 0% 166ms ± 1% -1.26% (p=0.009 n=6+6)
name old bin-B new bin-B delta
Build/Cache-8 6.36M ± 0% 6.36M ± 0% +0.12% (p=0.002 n=6+6)
name old sys-time/op new sys-time/op delta
Build/Cache-8 222ms ± 2% 219ms ± 3% ~ (p=0.589 n=6+6)
name old user-time/op new user-time/op delta
Build/Cache-8 857ms ± 1% 846ms ± 1% -1.31% (p=0.041 n=6+6)
4 years ago
|
|
|
if len(os.Args) > 0 && os.Args[1] == "env" {
|
|
|
|
enc, _ := json.Marshal(struct{
|
|
|
|
GOVERSION string
|
|
|
|
} {
|
|
|
|
GOVERSION: os.Getenv("TOOLCHAIN_GOVERSION"),
|
use "go env -json" to collect env info all at once
In the worst case scenario, when GOPRIVATE isn't set at all, we would
run these three commands:
* "go env GOPRIVATE", to fetch GOPRIVATE itself
* "go list -m", for GOPRIVATE's fallback
* "go version", to check the version of Go being used
Now that we support Go 1.16 and later, all these three can be obtained
via "go env -json":
$ go env -json GOPRIVATE GOMOD GOVERSION
{
"GOMOD": "/home/mvdan/src/garble/go.mod",
"GOPRIVATE": "",
"GOVERSION": "go1.16.3"
}
Note that we don't get the module path directly, but we can use the
x/mod/modfile Go API to parse it from the GOMOD file cheaply.
Notably, this also simplifies our Go version checking logic, as now we
get just the version string without the "go version" prefix and
"GOOS/GOARCH" suffix we don't care about.
This makes our code a bit more maintainable and robust. When running a
short incremental build, we can also see a small speed-up, as saving two
"go" invocations can save a few milliseconds:
name old time/op new time/op delta
Build/Cache-8 168ms ± 0% 166ms ± 1% -1.26% (p=0.009 n=6+6)
name old bin-B new bin-B delta
Build/Cache-8 6.36M ± 0% 6.36M ± 0% +0.12% (p=0.002 n=6+6)
name old sys-time/op new sys-time/op delta
Build/Cache-8 222ms ± 2% 219ms ± 3% ~ (p=0.589 n=6+6)
name old user-time/op new user-time/op delta
Build/Cache-8 857ms ± 1% 846ms ± 1% -1.31% (p=0.041 n=6+6)
4 years ago
|
|
|
})
|
|
|
|
fmt.Printf("%s\n", enc)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Fprintln(os.Stderr, "mocking the real build")
|
|
|
|
os.Exit(1)
|
|
|
|
}
|