blob: 83a50fbec100d11afcde9252e3ad3d52158fac13 [file] [log] [blame]
// Copyright 2021 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 safetoken_test
import (
"fmt"
"go/parser"
"go/token"
"go/types"
"os"
"testing"
"golang.org/x/tools/go/packages"
"golang.org/x/tools/gopls/internal/lsp/safetoken"
"golang.org/x/tools/internal/testenv"
)
func TestWorkaroundIssue57490(t *testing.T) {
// During error recovery the parser synthesizes various close
// tokens at EOF, causing the End position of incomplete
// syntax nodes, computed as Rbrace+len("}"), to be beyond EOF.
src := `package p; func f() { var x struct`
fset := token.NewFileSet()
file, _ := parser.ParseFile(fset, "a.go", src, 0)
tf := fset.File(file.Pos())
// Add another file to the FileSet.
file2, _ := parser.ParseFile(fset, "b.go", "package q", 0)
// This is the ambiguity of #57490...
if file.End() != file2.Pos() {
t.Errorf("file.End() %d != %d file2.Pos()", file.End(), file2.Pos())
}
// ...which causes these statements to panic.
if false {
tf.Offset(file.End()) // panic: invalid Pos value 36 (should be in [1, 35])
tf.Position(file.End()) // panic: invalid Pos value 36 (should be in [1, 35])
}
// The offset of the EOF position is the file size.
offset, err := safetoken.Offset(tf, file.End()-1)
if err != nil || offset != tf.Size() {
t.Errorf("Offset(EOF) = (%d, %v), want token.File.Size %d", offset, err, tf.Size())
}
// The offset of the file.End() position, 1 byte beyond EOF,
// is also the size of the file.
offset, err = safetoken.Offset(tf, file.End())
if err != nil || offset != tf.Size() {
t.Errorf("Offset(ast.File.End()) = (%d, %v), want token.File.Size %d", offset, err, tf.Size())
}
if got, want := safetoken.Position(tf, file.End()).String(), "a.go:1:35"; got != want {
t.Errorf("Position(ast.File.End()) = %s, want %s", got, want)
}
if got, want := safetoken.EndPosition(fset, file.End()).String(), "a.go:1:35"; got != want {
t.Errorf("EndPosition(ast.File.End()) = %s, want %s", got, want)
}
// Note that calling StartPosition on an end may yield the wrong file:
if got, want := safetoken.StartPosition(fset, file.End()).String(), "b.go:1:1"; got != want {
t.Errorf("StartPosition(ast.File.End()) = %s, want %s", got, want)
}
}
// To reduce the risk of panic, or bugs for which this package
// provides a workaround, this test statically reports references to
// forbidden methods of token.File or FileSet throughout gopls and
// suggests alternatives.
func TestGoplsSourceDoesNotCallTokenFileMethods(t *testing.T) {
testenv.NeedsGoPackages(t)
testenv.NeedsGo1Point(t, 18)
testenv.NeedsLocalXTools(t)
cfg := &packages.Config{
Mode: packages.NeedName | packages.NeedModule | packages.NeedCompiledGoFiles | packages.NeedTypes | packages.NeedTypesInfo | packages.NeedSyntax | packages.NeedImports | packages.NeedDeps,
}
cfg.Env = os.Environ()
cfg.Env = append(cfg.Env,
"GOPACKAGESDRIVER=off",
"GOWORK=off", // necessary for -mod=mod below
"GOFLAGS=-mod=mod",
)
pkgs, err := packages.Load(cfg, "go/token", "golang.org/x/tools/gopls/...")
if err != nil {
t.Fatal(err)
}
var tokenPkg *packages.Package
for _, pkg := range pkgs {
if pkg.PkgPath == "go/token" {
tokenPkg = pkg
break
}
}
if tokenPkg == nil {
t.Fatal("missing package go/token")
}
File := tokenPkg.Types.Scope().Lookup("File")
FileSet := tokenPkg.Types.Scope().Lookup("FileSet")
alternative := make(map[types.Object]string)
setAlternative := func(recv types.Object, old, new string) {
oldMethod, _, _ := types.LookupFieldOrMethod(recv.Type(), true, recv.Pkg(), old)
alternative[oldMethod] = new
}
setAlternative(File, "Line", "safetoken.Line")
setAlternative(File, "Offset", "safetoken.Offset")
setAlternative(File, "Position", "safetoken.Position")
setAlternative(File, "PositionFor", "safetoken.Position")
setAlternative(FileSet, "Position", "safetoken.StartPosition or EndPosition")
setAlternative(FileSet, "PositionFor", "safetoken.StartPosition or EndPosition")
for _, pkg := range pkgs {
switch pkg.PkgPath {
case "go/token", "golang.org/x/tools/gopls/internal/lsp/safetoken":
continue // allow calls within these packages
}
for ident, obj := range pkg.TypesInfo.Uses {
if alt, ok := alternative[obj]; ok {
posn := safetoken.StartPosition(pkg.Fset, ident.Pos())
fmt.Fprintf(os.Stderr, "%s: forbidden use of %v; use %s instead.\n", posn, obj, alt)
t.Fail()
}
}
}
}