gopls/internal/golang: Web, an abstraction of server.web

This change consolidates the two func types, PosURL and PkgURL,
into an interface, Web, that aligns with the server.web
implementation.

Also, strength-reduce PkgURL to require only a viewID,
not a view (as we did for freesymbolsURL in CL 591157).

Change-Id: Ic48e0d5808257934c56b31126fd4880ee88c7a33
Reviewed-on: https://go-review.googlesource.com/c/tools/+/591318
Commit-Queue: Alan Donovan <adonovan@google.com>
Reviewed-by: Robert Findley <rfindley@google.com>
LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
Auto-Submit: Alan Donovan <adonovan@google.com>
This commit is contained in:
Alan Donovan 2024-06-07 12:44:16 -04:00 коммит произвёл Gopher Robot
Родитель f5a26d251e
Коммит f41a407b04
5 изменённых файлов: 28 добавлений и 37 удалений

Просмотреть файл

@ -30,7 +30,7 @@ import (
// TODO(adonovan):
// - display a "Compiling..." message as a cold build can be slow.
// - cross-link jumps and block labels, like github.com/aclements/objbrowse.
func AssemblyHTML(ctx context.Context, snapshot *cache.Snapshot, pkg *cache.Package, symbol string, posURL PosURLFunc) ([]byte, error) {
func AssemblyHTML(ctx context.Context, snapshot *cache.Snapshot, pkg *cache.Package, symbol string, web Web) ([]byte, error) {
// Compile the package with -S, and capture its stderr stream.
inv, cleanupInvocation, err := snapshot.GoCommandInvocation(false, &gocommand.Invocation{
Verb: "build",
@ -145,7 +145,7 @@ function httpGET(url) {
if file, linenum, ok := cutLast(parts[2], ":"); ok && !strings.HasPrefix(file, "<") {
if linenum, err := strconv.Atoi(linenum); err == nil {
text := fmt.Sprintf("L%04d", linenum)
link = sourceLink(text, posURL(file, linenum, 1))
link = sourceLink(text, web.OpenURL(file, linenum, 1))
}
}
fmt.Fprintf(&buf, "%s\t%s\t%s", escape(parts[1]), link, escape(parts[3]))

Просмотреть файл

@ -27,7 +27,7 @@ import (
// FreeSymbolsHTML returns an HTML document containing the report of
// free symbols referenced by the selection.
func FreeSymbolsHTML(pkg *cache.Package, pgf *parsego.File, start, end token.Pos, posURL PosURLFunc, pkgURL PkgURLFunc) []byte {
func FreeSymbolsHTML(viewID string, pkg *cache.Package, pgf *parsego.File, start, end token.Pos, web Web) []byte {
// Compute free references.
refs := freeRefs(pkg.Types(), pkg.TypesInfo(), pgf.File, start, end)
@ -210,7 +210,7 @@ function httpGET(url) {
fmt.Fprintf(&buf, "<ul>\n")
for _, imp := range model.Imported {
fmt.Fprintf(&buf, "<li>import \"<a href='%s'>%s</a>\" // for %s</li>\n",
pkgURL(imp.Path, ""),
web.PkgURL(viewID, imp.Path, ""),
html.EscapeString(string(imp.Path)),
strings.Join(imp.Symbols, ", "))
}
@ -236,7 +236,7 @@ function httpGET(url) {
objHTML := func(obj types.Object) string {
text := obj.Name()
if posn := safetoken.StartPosition(pkg.FileSet(), obj.Pos()); posn.IsValid() {
return sourceLink(text, posURL(posn.Filename, posn.Line, posn.Column))
return sourceLink(text, web.OpenURL(posn.Filename, posn.Line, posn.Column))
}
return text
}

Просмотреть файл

@ -53,24 +53,21 @@ import (
"golang.org/x/tools/internal/typesinternal"
)
// TODO(adonovan): factor these two functions into an interface.
type (
// A PkgURLFunc forms URLs of package or symbol documentation.
PkgURLFunc = func(path PackagePath, fragment string) protocol.URI
// Web is an abstraction of gopls' web server.
type Web interface {
// PkgURL forms URLs of package or symbol documentation.
PkgURL(viewID string, path PackagePath, fragment string) protocol.URI
// A PosURLFunc forms URLs that cause the editor to navigate to a position.
PosURLFunc = func(filename string, line, col8 int) protocol.URI
)
// OpenURL forms URLs that cause the editor to open a file at a specific position.
OpenURL(filename string, line, col8 int) protocol.URI
}
// PackageDocHTML formats the package documentation page.
//
// The posURL function returns a URL that when visited, has the side
// effect of causing gopls to direct the client editor to navigate to
// the specified file/line/column position, in UTF-8 coordinates.
//
// The pkgURL function returns a URL for the documentation of the
// specified package and symbol.
func PackageDocHTML(pkg *cache.Package, posURL PosURLFunc, pkgURL PkgURLFunc) ([]byte, error) {
func PackageDocHTML(viewID string, pkg *cache.Package, web Web) ([]byte, error) {
// We can't use doc.NewFromFiles (even with doc.PreserveAST
// mode) as it calls ast.NewPackage which assumes that each
// ast.File has an ast.Scope and resolves identifiers to
@ -143,7 +140,7 @@ func PackageDocHTML(pkg *cache.Package, posURL PosURLFunc, pkgURL PkgURLFunc) ([
if link.Recv != "" {
fragment = link.Recv + "." + link.Name
}
return pkgURL(path, fragment)
return web.PkgURL(viewID, path, fragment)
}
parser := docpkg.Parser()
parser.LookupPackage = func(name string) (importPath string, ok bool) {
@ -334,7 +331,7 @@ window.onload = () => {
objHTML := func(obj types.Object) string {
text := obj.Name()
if posn := safetoken.StartPosition(pkg.FileSet(), obj.Pos()); posn.IsValid() {
return sourceLink(text, posURL(posn.Filename, posn.Line, posn.Column))
return sourceLink(text, web.OpenURL(posn.Filename, posn.Line, posn.Column))
}
return text
}
@ -350,7 +347,7 @@ window.onload = () => {
// imported package name?
if pkgname, ok := obj.(*types.PkgName); ok {
// TODO(adonovan): do this for Defs of PkgName too.
return pkgURL(PackagePath(pkgname.Imported().Path()), "")
return web.PkgURL(viewID, PackagePath(pkgname.Imported().Path()), "")
}
// package-level symbol?
@ -358,7 +355,7 @@ window.onload = () => {
if obj.Pkg() == pkg.Types() {
return "#" + obj.Name() // intra-package ref
} else {
return pkgURL(PackagePath(obj.Pkg().Path()), obj.Name())
return web.PkgURL(viewID, PackagePath(obj.Pkg().Path()), obj.Name())
}
}
@ -369,7 +366,7 @@ window.onload = () => {
_, named := typesinternal.ReceiverNamed(sig.Recv())
if named != nil {
fragment := named.Obj().Name() + "." + fn.Name()
return pkgURL(PackagePath(fn.Pkg().Path()), fragment)
return web.PkgURL(viewID, PackagePath(fn.Pkg().Path()), fragment)
}
}
return ""
@ -671,7 +668,7 @@ window.onload = () => {
fmt.Fprintf(&buf, "<h2 id='hdr-SourceFiles'>Source files</h2>\n")
for _, filename := range docpkg.Filenames {
fmt.Fprintf(&buf, "<div class='comment'>%s</div>\n",
sourceLink(filepath.Base(filename), posURL(filename, 1, 1)))
sourceLink(filepath.Base(filename), web.OpenURL(filename, 1, 1)))
}
fmt.Fprintf(&buf, "</main>\n")

Просмотреть файл

@ -571,7 +571,7 @@ func (c *commandHandler) Doc(ctx context.Context, loc protocol.Location) error {
}
// Direct the client to open the /pkg page.
url := web.pkgURL(deps.snapshot.View(), pkgpath, fragment)
url := web.PkgURL(deps.snapshot.View().ID(), pkgpath, fragment)
openClientBrowser(ctx, c.s.client, url)
return nil

Просмотреть файл

@ -347,10 +347,7 @@ func (s *server) initWeb() (*web, error) {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
pkgURL := func(path golang.PackagePath, fragment string) protocol.URI {
return web.pkgURL(view, path, fragment)
}
content, err := golang.PackageDocHTML(pkgs[0], web.openURL, pkgURL)
content, err := golang.PackageDocHTML(view.ID(), pkgs[0], web)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
@ -405,10 +402,7 @@ func (s *server) initWeb() (*web, error) {
}
// Produce report.
pkgURL := func(path golang.PackagePath, fragment string) protocol.URI {
return web.pkgURL(view, path, fragment)
}
html := golang.FreeSymbolsHTML(pkg, pgf, start, end, web.openURL, pkgURL)
html := golang.FreeSymbolsHTML(view.ID(), pkg, pgf, start, end, web)
w.Write(html)
})
@ -453,7 +447,7 @@ func (s *server) initWeb() (*web, error) {
pkg := pkgs[0]
// Produce report.
html, err := golang.AssemblyHTML(ctx, snapshot, pkg, symbol, web.openURL)
html, err := golang.AssemblyHTML(ctx, snapshot, pkg, symbol, web)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
@ -469,26 +463,26 @@ func (s *server) initWeb() (*web, error) {
//go:embed assets/*
var assets embed.FS
// openURL returns an /open URL that, when visited, causes the client
// OpenURL returns an /open URL that, when visited, causes the client
// editor to open the specified file/line/column (in 1-based UTF-8
// coordinates).
//
// (Rendering may generate hundreds of positions across files of many
// packages, so don't convert to LSP coordinates yet: wait until the
// URL is opened.)
func (w *web) openURL(filename string, line, col8 int) protocol.URI {
func (w *web) OpenURL(filename string, line, col8 int) protocol.URI {
return w.url(
"open",
fmt.Sprintf("file=%s&line=%d&col=%d", url.QueryEscape(filename), line, col8),
"")
}
// pkgURL returns a /pkg URL for the documentation of the specified package.
// PkgURL returns a /pkg URL for the documentation of the specified package.
// The optional fragment must be of the form "Println" or "Buffer.WriteString".
func (w *web) pkgURL(v *cache.View, path golang.PackagePath, fragment string) protocol.URI {
func (w *web) PkgURL(viewID string, path golang.PackagePath, fragment string) protocol.URI {
return w.url(
"pkg/"+string(path),
"view="+url.QueryEscape(v.ID()),
"view="+url.QueryEscape(viewID),
fragment)
}