Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

x/tools/gopls: better struct field docs messages on hover #73016

Open
Cianidos opened this issue Mar 24, 2025 · 8 comments
Open

x/tools/gopls: better struct field docs messages on hover #73016

Cianidos opened this issue Mar 24, 2025 · 8 comments
Assignees
Labels
Documentation Issues describing a change to documentation. FeatureRequest Issues asking for a new feature that does not need a proposal. gopls Issues related to the Go language server, gopls. Tools This label describes issues relating to any tools in the x/tools repository.
Milestone

Comments

@Cianidos
Copy link

Cianidos commented Mar 24, 2025

gopls version

Build info

golang.org/x/tools/gopls v0.18.1
golang.org/x/tools/gopls@v0.18.1 h1:2xJBNzdImS5u/kV/ZzqDLSvlBSeZX+pWY9uKVP7Pask=
github.com/BurntSushi/toml@v1.4.1-0.20240526193622-a339e1f7089c h1:pxW6RcqyfI9/kWtOwnv/G+AzdKuy2ZrqINhenH4HyNs=
github.com/google/go-cmp@v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
golang.org/x/exp/typeparams@v0.0.0-20241210194714-1829a127f884 h1:1xaZTydL5Gsg78QharTwKfA9FY9CZ1VQj6D/AZEvHR0=
golang.org/x/mod@v0.23.0 h1:Zb7khfcRGKk+kqfxFaP5tZqCnDZMjC5VtUBs87Hr6QM=
golang.org/x/sync@v0.11.0 h1:GGz8+XQP4FvTTrjZPzNKTMFtSXH80RAzG+5ghFPgK9w=
golang.org/x/telemetry@v0.0.0-20241220003058-cc96b6e0d3d9 h1:L2k9GUV2TpQKVRGMjN94qfUMgUwOFimSQ6gipyJIjKw=
golang.org/x/text@v0.22.0 h1:bofq7m3/HAFvbF51jz3Q9wLg3jkvSPuiZu/pD1XwgtM=
golang.org/x/tools@v0.30.1-0.20250221230316-5055f70f240c h1:Ja/5gV5a9Vvho3p2NC/T2TtxhHjrWS/2DvCKMvA0a+Y=
golang.org/x/vuln@v1.1.3 h1:NPGnvPOTgnjBc9HTaUx+nj+EaUYxl5SJOWqaDYGaFYw=
honnef.co/go/tools@v0.5.1 h1:4bH5o3b5ZULQ4UrBmP+63W9r7qIkqJClEA9ko5YKx+I=
mvdan.cc/gofumpt@v0.7.0 h1:bg91ttqXmi9y2xawvkuMXyvAA/1ZGJqYAEGjXuP0JXU=
mvdan.cc/xurls/v2@v2.5.0 h1:lyBNOm8Wo71UknhUs4QTFUNNMyxy2JEIaKKo0RWOh+8=
go: go1.24.1

go env

AR='ar'
CC='cc'
CGO_CFLAGS='-O2 -g'
CGO_CPPFLAGS=''
CGO_CXXFLAGS='-O2 -g'
CGO_ENABLED='1'
CGO_FFLAGS='-O2 -g'
CGO_LDFLAGS='-O2 -g'
CXX='c++'
GCCGO='gccgo'
GO111MODULE='on'
GOARCH='arm64'
GOARM64='v8.0'
GOAUTH='netrc'
GOBIN='/Users/ariel/go/bin'
GOCACHE='/Users/ariel/Library/Caches/go-build'
GOCACHEPROG=''
GODEBUG=''
GOENV='/Users/ariel/Library/Application Support/go/env'
GOEXE=''
GOEXPERIMENT=''
GOFIPS140='off'
GOFLAGS=''
GOGCCFLAGS='-fPIC -arch arm64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -ffile-prefix-map=/var/folders/_c/mpmdn7k9503ff_hrbg5sj7nh0000gn/T/go-build4247256077=/tmp/go-build -gno-record-gcc-switches -fno-common'
GOHOSTARCH='arm64'
GOHOSTOS='darwin'
GOINSECURE=''
GOMOD='private/go.mod'
GOMODCACHE='/Users/ariel/go/pkg/mod'
GONOPROXY='pirvate'
GONOSUMDB='private'
GOOS='darwin'
GOPATH='/Users/ariel/go'
GOPRIVATE='pirvate'
GOPROXY='https://proxy.golang.org,direct'
GOROOT='/opt/homebrew/Cellar/go/1.24.1/libexec'
GOSUMDB='off'
GOTELEMETRY='off'
GOTELEMETRYDIR='/Users/ariel/Library/Application Support/go/telemetry'
GOTMPDIR=''
GOTOOLCHAIN='auto'
GOTOOLDIR='/opt/homebrew/Cellar/go/1.24.1/libexec/pkg/tool/darwin_arm64'
GOVCS=''
GOVERSION='go1.24.1'
GOWORK=''
PKG_CONFIG='pkg-config'

What did you do?

Suggest I have such code with struct that embeds other structure as pointer.

type OtherStruct struct {
	Some int
}

type MyStruct struct {
	*OtherStruct
}

func init() {
	var v MyStruct
	_ = v.Some // <- cursor here
}

I put my point (cursor) on v.Some, exactly on Some word to know its type.

What did you see happen?

In minibuffer, and with doc buffer "shift + k" I have following:

field Some int
field Some int

---

(mypkg.OtherStruct).Some on pkg.go.dev

I found this message a little bit confusing, when I try to understand if field may cause nil pointer dereference panic.

What did you expect to see?

I think it will be better to show this declaration fully as:

field *OtherStruct.Some int

It shows that it is from an embedded structure which is a pointer.

Otherwise it may show an embedded structure as a provider like this:

field Some int // Provided by *OtherStruct.Some

Editor and settings

Default, stock Doom Emacs with Eglot and gopls.

:language
(go +lsp)

Logs

No response

@Cianidos Cianidos added gopls Issues related to the Go language server, gopls. Tools This label describes issues relating to any tools in the x/tools repository. labels Mar 24, 2025
@gopherbot gopherbot added this to the Unreleased milestone Mar 24, 2025
@gopherbot gopherbot added the Documentation Issues describing a change to documentation. label Mar 24, 2025
@madelinekalil madelinekalil added the FeatureRequest Issues asking for a new feature that does not need a proposal. label Mar 24, 2025
@madelinekalil
Copy link

This information is also available via a "find all references" query on v.Some. The cursor (hover) docs message functionality focuses more on local type information.

@Cianidos
Copy link
Author

The primary reason I opened this issue is that the information provided about the safety of using this field is now misleading.
If I use it as it is, v.Some, as field of MyStruct, I can encounter an unexpected nil pointer dereference panic because it is actually provided by an embedded pointer to OtherStruct.
Therefore, if I am aware that a struct may contain embedded structs, I cannot rely on the LSP information.
I encountered this issue after a lengthy search for a nil pointer dereference panic in a code base that I am not familiar with.
I believe my proposal can enhance the usefulness of this LSP information and prevent panics for some developers or assist in diagnosing issues.

@Cianidos Cianidos changed the title x/tools/gopls: better struct field docs messages x/tools/gopls: better struct field docs messages on hover Mar 25, 2025
@acehinnnqru
Copy link

What about

field Some int

---

(mypkg.MyStruct.*OtherStruct).Some on pkg.go.dev

@Cianidos
Copy link
Author

Thank you for your suggestion.
I believe it is an improvement over the current state.
At the very least, it would be beneficial to modify the quick message feature in Emacs to display two lines of hover quick documentation, the first and last.

Do you think extending the first-line message would make it excessively long?
In that case, if the chain of embedded structures is lengthy, the message could become very long. However, I am unsure why LSP cares about this.

Perhaps a way to indicate that this structure is not ‘by value’ could be implemented.
field Some *int would be more confusing than the current format, lol.
Therefore, a better suggestion would be to use

embedded field Some int

to emphasize the importance of caution when using this field.

There is another related issue.
Assume we have the following code:

_ = v.OtherStruct

and the cursor hovers over OtherStruct.
The message would then be:

field OtherStruct *OtherStruct

—

(main.MyStruct).OtherStruct on pkg.go.dev

This message does not make sense if the embedded structure or field is named the same as the type name (until the full message is opened).

It is possible that this is an issue specific to Emacs’ view, and other clients display the entire message by default.

So it is also may be solved with additional mention that it is embedded, for example:

embedded *OtherStruct

@acehinnnqru
Copy link

I agreed with your proposal but there could be hard to determine how many depth should we search the ref backward.

Like the following sample, what should we show as doc?

type A struct {
    V int
}

type B struct {A}

type C struct {*B}

// Then we do something
_ = C{}

_ = C.V <- cursor

On the other hand, I think there could be some perf issue to determine whether a ref's parent would be nil when there got deep-embed structs.

But if we only focus the first depth of embed, this should be a good enhancement.

@Cianidos
Copy link
Author

I am uncertain about the current state of doc rendering. If “gopls” is already aware that c.V is a field of C, in other words, a “forward search” for the field has been performed, then it should be possible to unwind this backward to collect the embedding chain. Therefore, I believe this is not a significant challenge.

The final line of the document could potentially display the complete chain of embeddings, as suggested. I believe this is a necessary feature.

The question remains about the first line of the document.

In my opinion, there are two possible edge solutions, with everything in between being reasonable.

One option is to display only the information that the field is an embedded field, rather than simply indicating that it is a field (owned or not). This would provide a concise representation.

The other option is to display the complete chain of embeddings.

Of course, there may be limitations on the depth level of the embeddings, such as a maximum depth of 1, 2, or other predefined values.

However, I believe limiting the depth to the server side is not a good idea. Clients can shorten the embedding chain if necessary.

I am currently leaning towards the second option, which would display the complete chain of embeddings as follows:

field *B.A.V int

---

(mypkg.*B.A).V on pkg.go.dev

However, it may be acceptable to simplify the representation to the following:

embedded field V int

---

(mypkg.*B.A).V on pkg.go.dev

@madelinekalil
Copy link

Apologies, I think I misunderstood earlier. Augmenting the hover information with the embedded representation would certainly be helpful. Please feel free to pick this up if you have any interest, I've added a help wanted label

@gopherbot
Copy link
Contributor

Change https://go.dev/cl/662275 mentions this issue: gopls/internal/golang: add embedded struct info to hover

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Documentation Issues describing a change to documentation. FeatureRequest Issues asking for a new feature that does not need a proposal. gopls Issues related to the Go language server, gopls. Tools This label describes issues relating to any tools in the x/tools repository.
Projects
None yet
Development

No branches or pull requests

5 participants