mirror of
https://github.com/golang/go
synced 2024-09-28 22:14:28 -06:00
go/types, types2: remove local version processing in favor of go/version
In the Checker, maintain a map of versions for each file, even if the file doensn't specify a version. In that case, the version is the module version. If Info.FileVersions is set, use that map directly; otherwise allocate a Checker-local map. Introduce a new type, goVersion, which represents a Go language version. This type effectively takes the role of the earlier version struct. Replace all versions-related logic accordingly and use the go/version package for version parsing/validation/comparison. Added more tests. Fixes #63974. Change-Id: Ia05ff47a9eae0f0bb03c6b4cb65a7ce0a5857402 Reviewed-on: https://go-review.googlesource.com/c/go/+/541395 Run-TryBot: Robert Griesemer <gri@google.com> TryBot-Result: Gopher Robot <gobot@golang.org> Reviewed-by: Robert Findley <rfindley@google.com> Reviewed-by: Robert Griesemer <gri@google.com>
This commit is contained in:
parent
3073f3f941
commit
56c91c0502
@ -8,6 +8,7 @@ import (
|
|||||||
"cmd/compile/internal/syntax"
|
"cmd/compile/internal/syntax"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"internal/goversion"
|
||||||
"internal/testenv"
|
"internal/testenv"
|
||||||
"reflect"
|
"reflect"
|
||||||
"regexp"
|
"regexp"
|
||||||
@ -2839,11 +2840,28 @@ var _ = f(1, 2)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestModuleVersion(t *testing.T) {
|
||||||
|
// version go1.dd must be able to typecheck go1.dd.0, go1.dd.1, etc.
|
||||||
|
goversion := fmt.Sprintf("go1.%d", goversion.Version)
|
||||||
|
for _, v := range []string{
|
||||||
|
goversion,
|
||||||
|
goversion + ".0",
|
||||||
|
goversion + ".1",
|
||||||
|
goversion + ".rc",
|
||||||
|
} {
|
||||||
|
conf := Config{GoVersion: v}
|
||||||
|
pkg := mustTypecheck("package p", &conf, nil)
|
||||||
|
if pkg.GoVersion() != conf.GoVersion {
|
||||||
|
t.Errorf("got %s; want %s", pkg.GoVersion(), conf.GoVersion)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestFileVersions(t *testing.T) {
|
func TestFileVersions(t *testing.T) {
|
||||||
for _, test := range []struct {
|
for _, test := range []struct {
|
||||||
moduleVersion string
|
goVersion string
|
||||||
fileVersion string
|
fileVersion string
|
||||||
wantVersion string
|
wantVersion string
|
||||||
}{
|
}{
|
||||||
{"", "", ""}, // no versions specified
|
{"", "", ""}, // no versions specified
|
||||||
{"go1.19", "", "go1.19"}, // module version specified
|
{"go1.19", "", "go1.19"}, // module version specified
|
||||||
@ -2851,6 +2869,16 @@ func TestFileVersions(t *testing.T) {
|
|||||||
{"go1.19", "go1.20", "go1.20"}, // file upgrade permitted
|
{"go1.19", "go1.20", "go1.20"}, // file upgrade permitted
|
||||||
{"go1.20", "go1.19", "go1.20"}, // file downgrade not permitted
|
{"go1.20", "go1.19", "go1.20"}, // file downgrade not permitted
|
||||||
{"go1.21", "go1.19", "go1.19"}, // file downgrade permitted (module version is >= go1.21)
|
{"go1.21", "go1.19", "go1.19"}, // file downgrade permitted (module version is >= go1.21)
|
||||||
|
|
||||||
|
// versions containing release numbers
|
||||||
|
// (file versions containing release numbers are considered invalid)
|
||||||
|
{"go1.19.0", "", "go1.19.0"}, // no file version specified
|
||||||
|
{"go1.20", "go1.20.1", "go1.20"}, // file upgrade ignored
|
||||||
|
{"go1.20.1", "go1.20", "go1.20.1"}, // file upgrade ignored
|
||||||
|
{"go1.20.1", "go1.21", "go1.21"}, // file upgrade permitted
|
||||||
|
{"go1.20.1", "go1.19", "go1.20.1"}, // file downgrade not permitted
|
||||||
|
{"go1.21.1", "go1.19.1", "go1.21.1"}, // file downgrade not permitted (invalid file version)
|
||||||
|
{"go1.21.1", "go1.19", "go1.19"}, // file downgrade permitted (module version is >= go1.21)
|
||||||
} {
|
} {
|
||||||
var src string
|
var src string
|
||||||
if test.fileVersion != "" {
|
if test.fileVersion != "" {
|
||||||
@ -2858,7 +2886,7 @@ func TestFileVersions(t *testing.T) {
|
|||||||
}
|
}
|
||||||
src += "package p"
|
src += "package p"
|
||||||
|
|
||||||
conf := Config{GoVersion: test.moduleVersion}
|
conf := Config{GoVersion: test.goVersion}
|
||||||
versions := make(map[*syntax.PosBase]string)
|
versions := make(map[*syntax.PosBase]string)
|
||||||
var info Info
|
var info Info
|
||||||
info.FileVersions = versions
|
info.FileVersions = versions
|
||||||
|
@ -12,7 +12,6 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"go/constant"
|
"go/constant"
|
||||||
"internal/godebug"
|
"internal/godebug"
|
||||||
"internal/goversion"
|
|
||||||
. "internal/types/errors"
|
. "internal/types/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -107,12 +106,11 @@ type Checker struct {
|
|||||||
ctxt *Context // context for de-duplicating instances
|
ctxt *Context // context for de-duplicating instances
|
||||||
pkg *Package
|
pkg *Package
|
||||||
*Info
|
*Info
|
||||||
version version // accepted language version
|
version goVersion // accepted language version
|
||||||
posVers map[*syntax.PosBase]version // maps file PosBases to versions (may be nil)
|
nextID uint64 // unique Id for type parameters (first valid Id is 1)
|
||||||
nextID uint64 // unique Id for type parameters (first valid Id is 1)
|
objMap map[Object]*declInfo // maps package-level objects and (non-interface) methods to declaration info
|
||||||
objMap map[Object]*declInfo // maps package-level objects and (non-interface) methods to declaration info
|
impMap map[importKey]*Package // maps (import path, source directory) to (complete or fake) package
|
||||||
impMap map[importKey]*Package // maps (import path, source directory) to (complete or fake) package
|
valids instanceLookup // valid *Named (incl. instantiated) types per the validType check
|
||||||
valids instanceLookup // valid *Named (incl. instantiated) types per the validType check
|
|
||||||
|
|
||||||
// pkgPathMap maps package names to the set of distinct import paths we've
|
// pkgPathMap maps package names to the set of distinct import paths we've
|
||||||
// seen for that name, anywhere in the import graph. It is used for
|
// seen for that name, anywhere in the import graph. It is used for
|
||||||
@ -128,6 +126,7 @@ type Checker struct {
|
|||||||
// (initialized by Files, valid only for the duration of check.Files;
|
// (initialized by Files, valid only for the duration of check.Files;
|
||||||
// maps and lists are allocated on demand)
|
// maps and lists are allocated on demand)
|
||||||
files []*syntax.File // list of package files
|
files []*syntax.File // list of package files
|
||||||
|
versions map[*syntax.PosBase]string // maps file bases to version strings (each file has an entry)
|
||||||
imports []*PkgName // list of imported packages
|
imports []*PkgName // list of imported packages
|
||||||
dotImportMap map[dotImportKey]*PkgName // maps dot-imported objects to the package they were dot-imported through
|
dotImportMap map[dotImportKey]*PkgName // maps dot-imported objects to the package they were dot-imported through
|
||||||
recvTParamMap map[*syntax.Name]*TypeParam // maps blank receiver type parameters to their type
|
recvTParamMap map[*syntax.Name]*TypeParam // maps blank receiver type parameters to their type
|
||||||
@ -261,6 +260,7 @@ func NewChecker(conf *Config, pkg *Package, info *Info) *Checker {
|
|||||||
ctxt: conf.Context,
|
ctxt: conf.Context,
|
||||||
pkg: pkg,
|
pkg: pkg,
|
||||||
Info: info,
|
Info: info,
|
||||||
|
version: asGoVersion(conf.GoVersion),
|
||||||
objMap: make(map[Object]*declInfo),
|
objMap: make(map[Object]*declInfo),
|
||||||
impMap: make(map[importKey]*Package),
|
impMap: make(map[importKey]*Package),
|
||||||
}
|
}
|
||||||
@ -302,36 +302,51 @@ func (check *Checker) initFiles(files []*syntax.File) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// reuse Info.FileVersions if provided
|
||||||
|
versions := check.Info.FileVersions
|
||||||
|
if versions == nil {
|
||||||
|
versions = make(map[*syntax.PosBase]string)
|
||||||
|
}
|
||||||
|
check.versions = versions
|
||||||
|
|
||||||
|
pkgVersionOk := check.version.isValid()
|
||||||
|
downgradeOk := check.version.cmp(go1_21) >= 0
|
||||||
|
|
||||||
|
// determine Go version for each file
|
||||||
for _, file := range check.files {
|
for _, file := range check.files {
|
||||||
fbase := base(file.Pos()) // fbase may be nil for tests
|
// use unaltered Config.GoVersion by default
|
||||||
check.recordFileVersion(fbase, check.conf.GoVersion) // record package version (possibly zero version)
|
// (This version string may contain dot-release numbers as in go1.20.1,
|
||||||
v, _ := parseGoVersion(file.GoVersion)
|
// unlike file versions which are Go language versions only, if valid.)
|
||||||
if v.major > 0 {
|
v := check.conf.GoVersion
|
||||||
if v.equal(check.version) {
|
// use the file version, if applicable
|
||||||
continue
|
// (file versions are either the empty string or of the form go1.dd)
|
||||||
|
if pkgVersionOk {
|
||||||
|
fileVersion := asGoVersion(file.GoVersion)
|
||||||
|
if fileVersion.isValid() {
|
||||||
|
cmp := fileVersion.cmp(check.version)
|
||||||
|
// Go 1.21 introduced the feature of setting the go.mod
|
||||||
|
// go line to an early version of Go and allowing //go:build lines
|
||||||
|
// to “upgrade” (cmp > 0) the Go version in a given file.
|
||||||
|
// We can do that backwards compatibly.
|
||||||
|
//
|
||||||
|
// Go 1.21 also introduced the feature of allowing //go:build lines
|
||||||
|
// to “downgrade” (cmp < 0) the Go version in a given file.
|
||||||
|
// That can't be done compatibly in general, since before the
|
||||||
|
// build lines were ignored and code got the module's Go version.
|
||||||
|
// To work around this, downgrades are only allowed when the
|
||||||
|
// module's Go version is Go 1.21 or later.
|
||||||
|
//
|
||||||
|
// If there is no valid check.version, then we don't really know what
|
||||||
|
// Go version to apply.
|
||||||
|
// Legacy tools may do this, and they historically have accepted everything.
|
||||||
|
// Preserve that behavior by ignoring //go:build constraints entirely in that
|
||||||
|
// case (!pkgVersionOk).
|
||||||
|
if cmp > 0 || cmp < 0 && downgradeOk {
|
||||||
|
v = file.GoVersion
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Go 1.21 introduced the feature of setting the go.mod
|
|
||||||
// go line to an early version of Go and allowing //go:build lines
|
|
||||||
// to “upgrade” the Go version in a given file.
|
|
||||||
// We can do that backwards compatibly.
|
|
||||||
// Go 1.21 also introduced the feature of allowing //go:build lines
|
|
||||||
// to “downgrade” the Go version in a given file.
|
|
||||||
// That can't be done compatibly in general, since before the
|
|
||||||
// build lines were ignored and code got the module's Go version.
|
|
||||||
// To work around this, downgrades are only allowed when the
|
|
||||||
// module's Go version is Go 1.21 or later.
|
|
||||||
// If there is no check.version, then we don't really know what Go version to apply.
|
|
||||||
// Legacy tools may do this, and they historically have accepted everything.
|
|
||||||
// Preserve that behavior by ignoring //go:build constraints entirely in that case.
|
|
||||||
if (v.before(check.version) && check.version.before(go1_21)) || check.version.equal(go0_0) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if check.posVers == nil {
|
|
||||||
check.posVers = make(map[*syntax.PosBase]version)
|
|
||||||
}
|
|
||||||
check.posVers[fbase] = v
|
|
||||||
check.recordFileVersion(fbase, file.GoVersion) // overwrite package version
|
|
||||||
}
|
}
|
||||||
|
versions[base(file.Pos())] = v // base(file.Pos()) may be nil for tests
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -362,15 +377,8 @@ func (check *Checker) checkFiles(files []*syntax.File) (err error) {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: parseGoVersion and the subsequent checks should happen once,
|
// Note: NewChecker doesn't return an error, so we need to check the version here.
|
||||||
// when we create a new Checker, not for each batch of files.
|
if check.version.cmp(go_current) > 0 {
|
||||||
// We can't change it at this point because NewChecker doesn't
|
|
||||||
// return an error.
|
|
||||||
check.version, err = parseGoVersion(check.conf.GoVersion)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if check.version.after(version{1, goversion.Version}) {
|
|
||||||
return fmt.Errorf("package requires newer Go version %v", check.version)
|
return fmt.Errorf("package requires newer Go version %v", check.version)
|
||||||
}
|
}
|
||||||
if check.conf.FakeImportC && check.conf.go115UsesCgo {
|
if check.conf.FakeImportC && check.conf.go115UsesCgo {
|
||||||
@ -694,9 +702,3 @@ func (check *Checker) recordScope(node syntax.Node, scope *Scope) {
|
|||||||
m[node] = scope
|
m[node] = scope
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (check *Checker) recordFileVersion(fbase *syntax.PosBase, version string) {
|
|
||||||
if m := check.FileVersions; m != nil {
|
|
||||||
m[fbase] = version
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -297,7 +297,7 @@ func (check *Checker) softErrorf(at poser, code Code, format string, args ...int
|
|||||||
check.err(at, code, check.sprintf(format, args...), true)
|
check.err(at, code, check.sprintf(format, args...), true)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (check *Checker) versionErrorf(at poser, v version, format string, args ...interface{}) {
|
func (check *Checker) versionErrorf(at poser, v goVersion, format string, args ...interface{}) {
|
||||||
msg := check.sprintf(format, args...)
|
msg := check.sprintf(format, args...)
|
||||||
msg = fmt.Sprintf("%s requires %s or later", msg, v)
|
msg = fmt.Sprintf("%s requires %s or later", msg, v)
|
||||||
check.err(at, UnsupportedFeature, msg, true)
|
check.err(at, UnsupportedFeature, msg, true)
|
||||||
|
@ -7,90 +7,46 @@ package types2
|
|||||||
import (
|
import (
|
||||||
"cmd/compile/internal/syntax"
|
"cmd/compile/internal/syntax"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"go/version"
|
||||||
|
"internal/goversion"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
// A version represents a released Go version.
|
// A goVersion is a Go language version string of the form "go1.%d"
|
||||||
type version struct {
|
// where d is the minor version number. goVersion strings don't
|
||||||
major, minor int
|
// contain release numbers ("go1.20.1" is not a valid goVersion).
|
||||||
|
type goVersion string
|
||||||
|
|
||||||
|
// asGoVersion returns v as a goVersion (e.g., "go1.20.1" becomes "go1.20").
|
||||||
|
// If v is not a valid Go version, the result is the empty string.
|
||||||
|
func asGoVersion(v string) goVersion {
|
||||||
|
return goVersion(version.Lang(v))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v version) String() string {
|
// isValid reports whether v is a valid Go version.
|
||||||
return fmt.Sprintf("go%d.%d", v.major, v.minor)
|
func (v goVersion) isValid() bool {
|
||||||
|
return v != ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v version) equal(u version) bool {
|
// cmp returns -1, 0, or +1 depending on whether x < y, x == y, or x > y,
|
||||||
return v.major == u.major && v.minor == u.minor
|
// interpreted as Go versions.
|
||||||
|
func (x goVersion) cmp(y goVersion) int {
|
||||||
|
return version.Compare(string(x), string(y))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v version) before(u version) bool {
|
|
||||||
return v.major < u.major || v.major == u.major && v.minor < u.minor
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v version) after(u version) bool {
|
|
||||||
return v.major > u.major || v.major == u.major && v.minor > u.minor
|
|
||||||
}
|
|
||||||
|
|
||||||
// Go versions that introduced language changes.
|
|
||||||
var (
|
var (
|
||||||
go0_0 = version{0, 0} // no version specified
|
// Go versions that introduced language changes
|
||||||
go1_9 = version{1, 9}
|
go1_9 = asGoVersion("go1.9")
|
||||||
go1_13 = version{1, 13}
|
go1_13 = asGoVersion("go1.13")
|
||||||
go1_14 = version{1, 14}
|
go1_14 = asGoVersion("go1.14")
|
||||||
go1_17 = version{1, 17}
|
go1_17 = asGoVersion("go1.17")
|
||||||
go1_18 = version{1, 18}
|
go1_18 = asGoVersion("go1.18")
|
||||||
go1_20 = version{1, 20}
|
go1_20 = asGoVersion("go1.20")
|
||||||
go1_21 = version{1, 21}
|
go1_21 = asGoVersion("go1.21")
|
||||||
)
|
|
||||||
|
|
||||||
// parseGoVersion parses a Go version string (such as "go1.12")
|
// current (deployed) Go version
|
||||||
// and returns the version, or an error. If s is the empty
|
go_current = asGoVersion(fmt.Sprintf("go1.%d", goversion.Version))
|
||||||
// string, the version is 0.0.
|
)
|
||||||
func parseGoVersion(s string) (v version, err error) {
|
|
||||||
bad := func() (version, error) {
|
|
||||||
return version{}, fmt.Errorf("invalid Go version syntax %q", s)
|
|
||||||
}
|
|
||||||
if s == "" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if !strings.HasPrefix(s, "go") {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
s = s[len("go"):]
|
|
||||||
i := 0
|
|
||||||
for ; i < len(s) && '0' <= s[i] && s[i] <= '9'; i++ {
|
|
||||||
if i >= 10 || i == 0 && s[i] == '0' {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
v.major = 10*v.major + int(s[i]) - '0'
|
|
||||||
}
|
|
||||||
if i > 0 && i == len(s) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if i == 0 || s[i] != '.' {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
s = s[i+1:]
|
|
||||||
if s == "0" {
|
|
||||||
// We really should not accept "go1.0",
|
|
||||||
// but we didn't reject it from the start
|
|
||||||
// and there are now programs that use it.
|
|
||||||
// So accept it.
|
|
||||||
return
|
|
||||||
}
|
|
||||||
i = 0
|
|
||||||
for ; i < len(s) && '0' <= s[i] && s[i] <= '9'; i++ {
|
|
||||||
if i >= 10 || i == 0 && s[i] == '0' {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
v.minor = 10*v.minor + int(s[i]) - '0'
|
|
||||||
}
|
|
||||||
// Accept any suffix after the minor number.
|
|
||||||
// We are only looking for the language version (major.minor)
|
|
||||||
// but want to accept any valid Go version, like go1.21.0
|
|
||||||
// and go1.21rc2.
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// langCompat reports an error if the representation of a numeric
|
// langCompat reports an error if the representation of a numeric
|
||||||
// literal is not compatible with the current language version.
|
// literal is not compatible with the current language version.
|
||||||
@ -121,30 +77,30 @@ func (check *Checker) langCompat(lit *syntax.BasicLit) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// allowVersion reports whether the given package
|
// allowVersion reports whether the given package is allowed to use version v.
|
||||||
// is allowed to use version major.minor.
|
func (check *Checker) allowVersion(pkg *Package, at poser, v goVersion) bool {
|
||||||
func (check *Checker) allowVersion(pkg *Package, at poser, v version) bool {
|
|
||||||
// We assume that imported packages have all been checked,
|
// We assume that imported packages have all been checked,
|
||||||
// so we only have to check for the local package.
|
// so we only have to check for the local package.
|
||||||
if pkg != check.pkg {
|
if pkg != check.pkg {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the source file declares its Go version, use that to decide.
|
// If no explicit file version is specified,
|
||||||
if check.posVers != nil {
|
// fileVersion corresponds to the module version.
|
||||||
if src, ok := check.posVers[base(at.Pos())]; ok && src.major >= 1 {
|
var fileVersion goVersion
|
||||||
return !src.before(v)
|
if pos := at.Pos(); pos.IsKnown() {
|
||||||
}
|
// We need version.Lang below because file versions
|
||||||
|
// can be (unaltered) Config.GoVersion strings that
|
||||||
|
// may contain dot-release information.
|
||||||
|
fileVersion = asGoVersion(check.versions[base(pos)])
|
||||||
}
|
}
|
||||||
|
return !fileVersion.isValid() || fileVersion.cmp(v) >= 0
|
||||||
// Otherwise fall back to the version in the checker.
|
|
||||||
return check.version.equal(go0_0) || !check.version.before(v)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// verifyVersionf is like allowVersion but also accepts a format string and arguments
|
// verifyVersionf is like allowVersion but also accepts a format string and arguments
|
||||||
// which are used to report a version error if allowVersion returns false. It uses the
|
// which are used to report a version error if allowVersion returns false. It uses the
|
||||||
// current package.
|
// current package.
|
||||||
func (check *Checker) verifyVersionf(at poser, v version, format string, args ...interface{}) bool {
|
func (check *Checker) verifyVersionf(at poser, v goVersion, format string, args ...interface{}) bool {
|
||||||
if !check.allowVersion(check.pkg, at, v) {
|
if !check.allowVersion(check.pkg, at, v) {
|
||||||
check.versionErrorf(at, v, format, args...)
|
check.versionErrorf(at, v, format, args...)
|
||||||
return false
|
return false
|
||||||
@ -154,7 +110,9 @@ func (check *Checker) verifyVersionf(at poser, v version, format string, args ..
|
|||||||
|
|
||||||
// base finds the underlying PosBase of the source file containing pos,
|
// base finds the underlying PosBase of the source file containing pos,
|
||||||
// skipping over intermediate PosBase layers created by //line directives.
|
// skipping over intermediate PosBase layers created by //line directives.
|
||||||
|
// The positions must be known.
|
||||||
func base(pos syntax.Pos) *syntax.PosBase {
|
func base(pos syntax.Pos) *syntax.PosBase {
|
||||||
|
assert(pos.IsKnown())
|
||||||
b := pos.Base()
|
b := pos.Base()
|
||||||
for {
|
for {
|
||||||
bb := b.Pos().Base()
|
bb := b.Pos().Base()
|
||||||
|
@ -1,24 +0,0 @@
|
|||||||
// Copyright 2023 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package types2
|
|
||||||
|
|
||||||
import "testing"
|
|
||||||
|
|
||||||
var parseGoVersionTests = []struct {
|
|
||||||
in string
|
|
||||||
out version
|
|
||||||
}{
|
|
||||||
{"go1.21", version{1, 21}},
|
|
||||||
{"go1.21.0", version{1, 21}},
|
|
||||||
{"go1.21rc2", version{1, 21}},
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestParseGoVersion(t *testing.T) {
|
|
||||||
for _, tt := range parseGoVersionTests {
|
|
||||||
if out, err := parseGoVersion(tt.in); out != tt.out || err != nil {
|
|
||||||
t.Errorf("parseGoVersion(%q) = %v, %v, want %v, nil", tt.in, out, err, tt.out)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -11,6 +11,7 @@ import (
|
|||||||
"go/importer"
|
"go/importer"
|
||||||
"go/parser"
|
"go/parser"
|
||||||
"go/token"
|
"go/token"
|
||||||
|
"internal/goversion"
|
||||||
"internal/testenv"
|
"internal/testenv"
|
||||||
"reflect"
|
"reflect"
|
||||||
"regexp"
|
"regexp"
|
||||||
@ -2849,11 +2850,28 @@ var _ = f(1, 2)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestModuleVersion(t *testing.T) {
|
||||||
|
// version go1.dd must be able to typecheck go1.dd.0, go1.dd.1, etc.
|
||||||
|
goversion := fmt.Sprintf("go1.%d", goversion.Version)
|
||||||
|
for _, v := range []string{
|
||||||
|
goversion,
|
||||||
|
goversion + ".0",
|
||||||
|
goversion + ".1",
|
||||||
|
goversion + ".rc",
|
||||||
|
} {
|
||||||
|
conf := Config{GoVersion: v}
|
||||||
|
pkg := mustTypecheck("package p", &conf, nil)
|
||||||
|
if pkg.GoVersion() != conf.GoVersion {
|
||||||
|
t.Errorf("got %s; want %s", pkg.GoVersion(), conf.GoVersion)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestFileVersions(t *testing.T) {
|
func TestFileVersions(t *testing.T) {
|
||||||
for _, test := range []struct {
|
for _, test := range []struct {
|
||||||
moduleVersion string
|
goVersion string
|
||||||
fileVersion string
|
fileVersion string
|
||||||
wantVersion string
|
wantVersion string
|
||||||
}{
|
}{
|
||||||
{"", "", ""}, // no versions specified
|
{"", "", ""}, // no versions specified
|
||||||
{"go1.19", "", "go1.19"}, // module version specified
|
{"go1.19", "", "go1.19"}, // module version specified
|
||||||
@ -2861,6 +2879,16 @@ func TestFileVersions(t *testing.T) {
|
|||||||
{"go1.19", "go1.20", "go1.20"}, // file upgrade permitted
|
{"go1.19", "go1.20", "go1.20"}, // file upgrade permitted
|
||||||
{"go1.20", "go1.19", "go1.20"}, // file downgrade not permitted
|
{"go1.20", "go1.19", "go1.20"}, // file downgrade not permitted
|
||||||
{"go1.21", "go1.19", "go1.19"}, // file downgrade permitted (module version is >= go1.21)
|
{"go1.21", "go1.19", "go1.19"}, // file downgrade permitted (module version is >= go1.21)
|
||||||
|
|
||||||
|
// versions containing release numbers
|
||||||
|
// (file versions containing release numbers are considered invalid)
|
||||||
|
{"go1.19.0", "", "go1.19.0"}, // no file version specified
|
||||||
|
{"go1.20", "go1.20.1", "go1.20"}, // file upgrade ignored
|
||||||
|
{"go1.20.1", "go1.20", "go1.20.1"}, // file upgrade ignored
|
||||||
|
{"go1.20.1", "go1.21", "go1.21"}, // file upgrade permitted
|
||||||
|
{"go1.20.1", "go1.19", "go1.20.1"}, // file downgrade not permitted
|
||||||
|
{"go1.21.1", "go1.19.1", "go1.21.1"}, // file downgrade not permitted (invalid file version)
|
||||||
|
{"go1.21.1", "go1.19", "go1.19"}, // file downgrade permitted (module version is >= go1.21)
|
||||||
} {
|
} {
|
||||||
var src string
|
var src string
|
||||||
if test.fileVersion != "" {
|
if test.fileVersion != "" {
|
||||||
@ -2868,7 +2896,7 @@ func TestFileVersions(t *testing.T) {
|
|||||||
}
|
}
|
||||||
src += "package p"
|
src += "package p"
|
||||||
|
|
||||||
conf := Config{GoVersion: test.moduleVersion}
|
conf := Config{GoVersion: test.goVersion}
|
||||||
versions := make(map[*ast.File]string)
|
versions := make(map[*ast.File]string)
|
||||||
var info Info
|
var info Info
|
||||||
info.FileVersions = versions
|
info.FileVersions = versions
|
||||||
|
@ -13,7 +13,6 @@ import (
|
|||||||
"go/constant"
|
"go/constant"
|
||||||
"go/token"
|
"go/token"
|
||||||
"internal/godebug"
|
"internal/godebug"
|
||||||
"internal/goversion"
|
|
||||||
. "internal/types/errors"
|
. "internal/types/errors"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -109,8 +108,7 @@ type Checker struct {
|
|||||||
fset *token.FileSet
|
fset *token.FileSet
|
||||||
pkg *Package
|
pkg *Package
|
||||||
*Info
|
*Info
|
||||||
version version // accepted language version
|
version goVersion // accepted language version
|
||||||
posVers map[token.Pos]version // maps file start positions to versions (may be nil)
|
|
||||||
nextID uint64 // unique Id for type parameters (first valid Id is 1)
|
nextID uint64 // unique Id for type parameters (first valid Id is 1)
|
||||||
objMap map[Object]*declInfo // maps package-level objects and (non-interface) methods to declaration info
|
objMap map[Object]*declInfo // maps package-level objects and (non-interface) methods to declaration info
|
||||||
impMap map[importKey]*Package // maps (import path, source directory) to (complete or fake) package
|
impMap map[importKey]*Package // maps (import path, source directory) to (complete or fake) package
|
||||||
@ -130,6 +128,7 @@ type Checker struct {
|
|||||||
// (initialized by Files, valid only for the duration of check.Files;
|
// (initialized by Files, valid only for the duration of check.Files;
|
||||||
// maps and lists are allocated on demand)
|
// maps and lists are allocated on demand)
|
||||||
files []*ast.File // package files
|
files []*ast.File // package files
|
||||||
|
versions map[*ast.File]string // maps files to version strings (each file has an entry)
|
||||||
imports []*PkgName // list of imported packages
|
imports []*PkgName // list of imported packages
|
||||||
dotImportMap map[dotImportKey]*PkgName // maps dot-imported objects to the package they were dot-imported through
|
dotImportMap map[dotImportKey]*PkgName // maps dot-imported objects to the package they were dot-imported through
|
||||||
recvTParamMap map[*ast.Ident]*TypeParam // maps blank receiver type parameters to their type
|
recvTParamMap map[*ast.Ident]*TypeParam // maps blank receiver type parameters to their type
|
||||||
@ -264,6 +263,7 @@ func NewChecker(conf *Config, fset *token.FileSet, pkg *Package, info *Info) *Ch
|
|||||||
fset: fset,
|
fset: fset,
|
||||||
pkg: pkg,
|
pkg: pkg,
|
||||||
Info: info,
|
Info: info,
|
||||||
|
version: asGoVersion(conf.GoVersion),
|
||||||
objMap: make(map[Object]*declInfo),
|
objMap: make(map[Object]*declInfo),
|
||||||
impMap: make(map[importKey]*Package),
|
impMap: make(map[importKey]*Package),
|
||||||
}
|
}
|
||||||
@ -305,35 +305,51 @@ func (check *Checker) initFiles(files []*ast.File) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// collect file versions
|
// reuse Info.FileVersions if provided
|
||||||
|
versions := check.Info.FileVersions
|
||||||
|
if versions == nil {
|
||||||
|
versions = make(map[*ast.File]string)
|
||||||
|
}
|
||||||
|
check.versions = versions
|
||||||
|
|
||||||
|
pkgVersionOk := check.version.isValid()
|
||||||
|
downgradeOk := check.version.cmp(go1_21) >= 0
|
||||||
|
|
||||||
|
// determine Go version for each file
|
||||||
for _, file := range check.files {
|
for _, file := range check.files {
|
||||||
check.recordFileVersion(file, check.conf.GoVersion) // record package version (possibly zero version)
|
// use unaltered Config.GoVersion by default
|
||||||
if v, _ := parseGoVersion(file.GoVersion); v.major > 0 {
|
// (This version string may contain dot-release numbers as in go1.20.1,
|
||||||
if v.equal(check.version) {
|
// unlike file versions which are Go language versions only, if valid.)
|
||||||
continue
|
v := check.conf.GoVersion
|
||||||
|
// use the file version, if applicable
|
||||||
|
// (file versions are either the empty string or of the form go1.dd)
|
||||||
|
if pkgVersionOk {
|
||||||
|
fileVersion := asGoVersion(file.GoVersion)
|
||||||
|
if fileVersion.isValid() {
|
||||||
|
cmp := fileVersion.cmp(check.version)
|
||||||
|
// Go 1.21 introduced the feature of setting the go.mod
|
||||||
|
// go line to an early version of Go and allowing //go:build lines
|
||||||
|
// to “upgrade” (cmp > 0) the Go version in a given file.
|
||||||
|
// We can do that backwards compatibly.
|
||||||
|
//
|
||||||
|
// Go 1.21 also introduced the feature of allowing //go:build lines
|
||||||
|
// to “downgrade” (cmp < 0) the Go version in a given file.
|
||||||
|
// That can't be done compatibly in general, since before the
|
||||||
|
// build lines were ignored and code got the module's Go version.
|
||||||
|
// To work around this, downgrades are only allowed when the
|
||||||
|
// module's Go version is Go 1.21 or later.
|
||||||
|
//
|
||||||
|
// If there is no valid check.version, then we don't really know what
|
||||||
|
// Go version to apply.
|
||||||
|
// Legacy tools may do this, and they historically have accepted everything.
|
||||||
|
// Preserve that behavior by ignoring //go:build constraints entirely in that
|
||||||
|
// case (!pkgVersionOk).
|
||||||
|
if cmp > 0 || cmp < 0 && downgradeOk {
|
||||||
|
v = file.GoVersion
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Go 1.21 introduced the feature of setting the go.mod
|
|
||||||
// go line to an early version of Go and allowing //go:build lines
|
|
||||||
// to “upgrade” the Go version in a given file.
|
|
||||||
// We can do that backwards compatibly.
|
|
||||||
// Go 1.21 also introduced the feature of allowing //go:build lines
|
|
||||||
// to “downgrade” the Go version in a given file.
|
|
||||||
// That can't be done compatibly in general, since before the
|
|
||||||
// build lines were ignored and code got the module's Go version.
|
|
||||||
// To work around this, downgrades are only allowed when the
|
|
||||||
// module's Go version is Go 1.21 or later.
|
|
||||||
// If there is no check.version, then we don't really know what Go version to apply.
|
|
||||||
// Legacy tools may do this, and they historically have accepted everything.
|
|
||||||
// Preserve that behavior by ignoring //go:build constraints entirely in that case.
|
|
||||||
if (v.before(check.version) && check.version.before(go1_21)) || check.version.equal(go0_0) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if check.posVers == nil {
|
|
||||||
check.posVers = make(map[token.Pos]version)
|
|
||||||
}
|
|
||||||
check.posVers[file.FileStart] = v
|
|
||||||
check.recordFileVersion(file, file.GoVersion) // overwrite package version
|
|
||||||
}
|
}
|
||||||
|
versions[file] = v
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -364,15 +380,8 @@ func (check *Checker) checkFiles(files []*ast.File) (err error) {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: parseGoVersion and the subsequent checks should happen once,
|
// Note: NewChecker doesn't return an error, so we need to check the version here.
|
||||||
// when we create a new Checker, not for each batch of files.
|
if check.version.cmp(go_current) > 0 {
|
||||||
// We can't change it at this point because NewChecker doesn't
|
|
||||||
// return an error.
|
|
||||||
check.version, err = parseGoVersion(check.conf.GoVersion)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if check.version.after(version{1, goversion.Version}) {
|
|
||||||
return fmt.Errorf("package requires newer Go version %v", check.version)
|
return fmt.Errorf("package requires newer Go version %v", check.version)
|
||||||
}
|
}
|
||||||
if check.conf.FakeImportC && check.conf.go115UsesCgo {
|
if check.conf.FakeImportC && check.conf.go115UsesCgo {
|
||||||
@ -650,9 +659,3 @@ func (check *Checker) recordScope(node ast.Node, scope *Scope) {
|
|||||||
m[node] = scope
|
m[node] = scope
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (check *Checker) recordFileVersion(file *ast.File, version string) {
|
|
||||||
if m := check.FileVersions; m != nil {
|
|
||||||
m[file] = version
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -316,7 +316,7 @@ func (check *Checker) softErrorf(at positioner, code Code, format string, args .
|
|||||||
check.report(err)
|
check.report(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (check *Checker) versionErrorf(at positioner, v version, format string, args ...interface{}) {
|
func (check *Checker) versionErrorf(at positioner, v goVersion, format string, args ...interface{}) {
|
||||||
msg := check.sprintf(format, args...)
|
msg := check.sprintf(format, args...)
|
||||||
var err *error_
|
var err *error_
|
||||||
err = newErrorf(at, UnsupportedFeature, "%s requires %s or later", msg, v)
|
err = newErrorf(at, UnsupportedFeature, "%s requires %s or later", msg, v)
|
||||||
|
@ -143,7 +143,6 @@ var filemap = map[string]action{
|
|||||||
"universe.go": fixGlobalTypVarDecl,
|
"universe.go": fixGlobalTypVarDecl,
|
||||||
"util_test.go": fixTokenPos,
|
"util_test.go": fixTokenPos,
|
||||||
"validtype.go": nil,
|
"validtype.go": nil,
|
||||||
"version_test.go": nil,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO(gri) We should be able to make these rewriters more configurable/composable.
|
// TODO(gri) We should be able to make these rewriters more configurable/composable.
|
||||||
|
@ -8,90 +8,46 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"go/ast"
|
"go/ast"
|
||||||
"go/token"
|
"go/token"
|
||||||
|
"go/version"
|
||||||
|
"internal/goversion"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
// A version represents a released Go version.
|
// A goVersion is a Go language version string of the form "go1.%d"
|
||||||
type version struct {
|
// where d is the minor version number. goVersion strings don't
|
||||||
major, minor int
|
// contain release numbers ("go1.20.1" is not a valid goVersion).
|
||||||
|
type goVersion string
|
||||||
|
|
||||||
|
// asGoVersion returns v as a goVersion (e.g., "go1.20.1" becomes "go1.20").
|
||||||
|
// If v is not a valid Go version, the result is the empty string.
|
||||||
|
func asGoVersion(v string) goVersion {
|
||||||
|
return goVersion(version.Lang(v))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v version) String() string {
|
// isValid reports whether v is a valid Go version.
|
||||||
return fmt.Sprintf("go%d.%d", v.major, v.minor)
|
func (v goVersion) isValid() bool {
|
||||||
|
return v != ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v version) equal(u version) bool {
|
// cmp returns -1, 0, or +1 depending on whether x < y, x == y, or x > y,
|
||||||
return v.major == u.major && v.minor == u.minor
|
// interpreted as Go versions.
|
||||||
|
func (x goVersion) cmp(y goVersion) int {
|
||||||
|
return version.Compare(string(x), string(y))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (v version) before(u version) bool {
|
|
||||||
return v.major < u.major || v.major == u.major && v.minor < u.minor
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v version) after(u version) bool {
|
|
||||||
return v.major > u.major || v.major == u.major && v.minor > u.minor
|
|
||||||
}
|
|
||||||
|
|
||||||
// Go versions that introduced language changes.
|
|
||||||
var (
|
var (
|
||||||
go0_0 = version{0, 0} // no version specified
|
// Go versions that introduced language changes
|
||||||
go1_9 = version{1, 9}
|
go1_9 = asGoVersion("go1.9")
|
||||||
go1_13 = version{1, 13}
|
go1_13 = asGoVersion("go1.13")
|
||||||
go1_14 = version{1, 14}
|
go1_14 = asGoVersion("go1.14")
|
||||||
go1_17 = version{1, 17}
|
go1_17 = asGoVersion("go1.17")
|
||||||
go1_18 = version{1, 18}
|
go1_18 = asGoVersion("go1.18")
|
||||||
go1_20 = version{1, 20}
|
go1_20 = asGoVersion("go1.20")
|
||||||
go1_21 = version{1, 21}
|
go1_21 = asGoVersion("go1.21")
|
||||||
)
|
|
||||||
|
|
||||||
// parseGoVersion parses a Go version string (such as "go1.12")
|
// current (deployed) Go version
|
||||||
// and returns the version, or an error. If s is the empty
|
go_current = asGoVersion(fmt.Sprintf("go1.%d", goversion.Version))
|
||||||
// string, the version is 0.0.
|
)
|
||||||
func parseGoVersion(s string) (v version, err error) {
|
|
||||||
bad := func() (version, error) {
|
|
||||||
return version{}, fmt.Errorf("invalid Go version syntax %q", s)
|
|
||||||
}
|
|
||||||
if s == "" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if !strings.HasPrefix(s, "go") {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
s = s[len("go"):]
|
|
||||||
i := 0
|
|
||||||
for ; i < len(s) && '0' <= s[i] && s[i] <= '9'; i++ {
|
|
||||||
if i >= 10 || i == 0 && s[i] == '0' {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
v.major = 10*v.major + int(s[i]) - '0'
|
|
||||||
}
|
|
||||||
if i > 0 && i == len(s) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if i == 0 || s[i] != '.' {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
s = s[i+1:]
|
|
||||||
if s == "0" {
|
|
||||||
// We really should not accept "go1.0",
|
|
||||||
// but we didn't reject it from the start
|
|
||||||
// and there are now programs that use it.
|
|
||||||
// So accept it.
|
|
||||||
return
|
|
||||||
}
|
|
||||||
i = 0
|
|
||||||
for ; i < len(s) && '0' <= s[i] && s[i] <= '9'; i++ {
|
|
||||||
if i >= 10 || i == 0 && s[i] == '0' {
|
|
||||||
return bad()
|
|
||||||
}
|
|
||||||
v.minor = 10*v.minor + int(s[i]) - '0'
|
|
||||||
}
|
|
||||||
// Accept any suffix after the minor number.
|
|
||||||
// We are only looking for the language version (major.minor)
|
|
||||||
// but want to accept any valid Go version, like go1.21.0
|
|
||||||
// and go1.21rc2.
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// langCompat reports an error if the representation of a numeric
|
// langCompat reports an error if the representation of a numeric
|
||||||
// literal is not compatible with the current language version.
|
// literal is not compatible with the current language version.
|
||||||
@ -122,35 +78,54 @@ func (check *Checker) langCompat(lit *ast.BasicLit) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// allowVersion reports whether the given package
|
// allowVersion reports whether the given package is allowed to use version v.
|
||||||
// is allowed to use version major.minor.
|
func (check *Checker) allowVersion(pkg *Package, at positioner, v goVersion) bool {
|
||||||
func (check *Checker) allowVersion(pkg *Package, at positioner, v version) bool {
|
|
||||||
// We assume that imported packages have all been checked,
|
// We assume that imported packages have all been checked,
|
||||||
// so we only have to check for the local package.
|
// so we only have to check for the local package.
|
||||||
if pkg != check.pkg {
|
if pkg != check.pkg {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the source file declares its Go version and at references a valid
|
// If no explicit file version is specified,
|
||||||
// position, use that to decide.
|
// fileVersion corresponds to the module version.
|
||||||
if pos := at.Pos(); pos.IsValid() && check.posVers != nil {
|
var fileVersion goVersion
|
||||||
fileStart := check.fset.File(pos).Pos(0)
|
if pos := at.Pos(); pos.IsValid() {
|
||||||
if src, ok := check.posVers[fileStart]; ok && src.major >= 1 {
|
// We need version.Lang below because file versions
|
||||||
return !src.before(v)
|
// can be (unaltered) Config.GoVersion strings that
|
||||||
}
|
// may contain dot-release information.
|
||||||
|
fileVersion = asGoVersion(check.versions[check.fileFor(pos)])
|
||||||
}
|
}
|
||||||
|
return !fileVersion.isValid() || fileVersion.cmp(v) >= 0
|
||||||
// Otherwise fall back to the version in the checker.
|
|
||||||
return check.version.equal(go0_0) || !check.version.before(v)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// verifyVersionf is like allowVersion but also accepts a format string and arguments
|
// verifyVersionf is like allowVersion but also accepts a format string and arguments
|
||||||
// which are used to report a version error if allowVersion returns false. It uses the
|
// which are used to report a version error if allowVersion returns false. It uses the
|
||||||
// current package.
|
// current package.
|
||||||
func (check *Checker) verifyVersionf(at positioner, v version, format string, args ...interface{}) bool {
|
func (check *Checker) verifyVersionf(at positioner, v goVersion, format string, args ...interface{}) bool {
|
||||||
if !check.allowVersion(check.pkg, at, v) {
|
if !check.allowVersion(check.pkg, at, v) {
|
||||||
check.versionErrorf(at, v, format, args...)
|
check.versionErrorf(at, v, format, args...)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO(gri) Consider a more direct (position-independent) mechanism
|
||||||
|
// to identify which file we're in so that version checks
|
||||||
|
// work correctly in the absence of correct position info.
|
||||||
|
|
||||||
|
// fileFor returns the *ast.File which contains the position pos.
|
||||||
|
// If there are no files, the result is nil.
|
||||||
|
// The position must be valid.
|
||||||
|
func (check *Checker) fileFor(pos token.Pos) *ast.File {
|
||||||
|
assert(pos.IsValid())
|
||||||
|
// Eval and CheckExpr tests may not have any source files.
|
||||||
|
if len(check.files) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
for _, file := range check.files {
|
||||||
|
if file.FileStart <= pos && pos < file.FileEnd {
|
||||||
|
return file
|
||||||
|
}
|
||||||
|
}
|
||||||
|
panic(check.sprintf("file not found for pos = %d (%s)", int(pos), check.fset.Position(pos)))
|
||||||
|
}
|
||||||
|
@ -1,26 +0,0 @@
|
|||||||
// Code generated by "go test -run=Generate -write=all"; DO NOT EDIT.
|
|
||||||
|
|
||||||
// Copyright 2023 The Go Authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style
|
|
||||||
// license that can be found in the LICENSE file.
|
|
||||||
|
|
||||||
package types
|
|
||||||
|
|
||||||
import "testing"
|
|
||||||
|
|
||||||
var parseGoVersionTests = []struct {
|
|
||||||
in string
|
|
||||||
out version
|
|
||||||
}{
|
|
||||||
{"go1.21", version{1, 21}},
|
|
||||||
{"go1.21.0", version{1, 21}},
|
|
||||||
{"go1.21rc2", version{1, 21}},
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestParseGoVersion(t *testing.T) {
|
|
||||||
for _, tt := range parseGoVersionTests {
|
|
||||||
if out, err := parseGoVersion(tt.in); out != tt.out || err != nil {
|
|
||||||
t.Errorf("parseGoVersion(%q) = %v, %v, want %v, nil", tt.in, out, err, tt.out)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user