internal/lsp: improve completion support for untyped constants

Add some extra smarts when evaluating untyped constants as completion
candidates. Previously we called types.Default() on the expected type
and candidate type, but this loses the untypedness of an untyped
constant which prevents it from being assignable to any type or named
type other than the untyped constant's default type.

Note that the added logic does not take into account the untyped
constant's value, so you will still get some false positive
completions (e.g. suggesting an untyped negative integer constant when
only a uint would do). Unfortunately go/types doesn't provide a way of
answering the question "is this *types.Const assignable to this
types.Type" since types.AssignableTo only considers a constant's type,
not its value.

Change-Id: If7075642e928f712b127256ae7706a5190e2f42c
GitHub-Last-Rev: 124d2f05b0aec09c9d7004d9da0d900524185b92
GitHub-Pull-Request: golang/tools#128
Reviewed-on: https://go-review.googlesource.com/c/tools/+/184477
Reviewed-by: Suzy Mueller <suzmue@golang.org>
This commit is contained in:
Muir Manders 2019-07-01 21:28:46 +00:00 committed by Suzy Mueller
parent 44aeb8b7c3
commit 9d59f9e855
3 changed files with 45 additions and 4 deletions

View File

@ -1028,14 +1028,28 @@ func (c *completer) matchingType(cand *candidate) bool {
// are invoked by default.
cand.expandFuncCall = isFunc(cand.obj)
typeMatches := func(actual types.Type) bool {
typeMatches := func(candType types.Type) bool {
// Take into account any type modifiers on the expected type.
actual = c.expectedType.applyTypeModifiers(actual)
candType = c.expectedType.applyTypeModifiers(candType)
if c.expectedType.objType != nil {
wantType := types.Default(c.expectedType.objType)
// Handle untyped values specially since AssignableTo gives false negatives
// for them (see https://golang.org/issue/32146).
if candBasic, ok := candType.(*types.Basic); ok && candBasic.Info()&types.IsUntyped > 0 {
if wantBasic, ok := wantType.Underlying().(*types.Basic); ok {
// Check that their constant kind (bool|int|float|complex|string) matches.
// This doesn't take into account the constant value, so there will be some
// false positives due to integer sign and overflow.
return candBasic.Info()&types.IsConstType == wantBasic.Info()&types.IsConstType
}
return false
}
// AssignableTo covers the case where the types are equal, but also handles
// cases like assigning a concrete type to an interface type.
return types.AssignableTo(types.Default(actual), types.Default(c.expectedType.objType))
return types.AssignableTo(candType, wantType)
}
return false

View File

@ -10,3 +10,30 @@ func _() {
)
wantsStrList(strList(conv)) //@complete("))", convertB, convertA)
}
func _() {
const (
convC = "hi" //@item(convertC, "convC", "string", "const")
convD = 123 //@item(convertD, "convD", "int", "const")
convE int = 123 //@item(convertE, "convE", "int", "const")
convF string = "there" //@item(convertF, "convF", "string", "const")
)
var foo int
foo = conv //@complete(" //", convertD, convertE, convertC, convertF)
type myInt int
var mi myInt
mi = conv //@complete(" //", convertD, convertC, convertE, convertF)
mi + conv //@complete(" //", convertD, convertC, convertE, convertF)
1 + conv //@complete(" //", convertD, convertE, convertC, convertF)
type myString string
var ms myString
ms = conv //@complete(" //", convertC, convertD, convertE, convertF)
type myUint uint32
var mu myUint
mu = conv //@complete(" //", convertD, convertC, convertE, convertF)
}

View File

@ -25,7 +25,7 @@ import (
// We hardcode the expected number of test cases to ensure that all tests
// are being executed. If a test is added, this number must be changed.
const (
ExpectedCompletionsCount = 138
ExpectedCompletionsCount = 144
ExpectedCompletionSnippetCount = 15
ExpectedDiagnosticsCount = 17
ExpectedFormatCount = 5