mirror of https://github.com/golang/go.git
271 lines
6.7 KiB
Go
271 lines
6.7 KiB
Go
// Copyright 2019 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 debug exports debug information for gopls.
|
|
package debug
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"reflect"
|
|
"runtime/debug"
|
|
"sort"
|
|
"strings"
|
|
|
|
"golang.org/x/tools/internal/lsp/source"
|
|
)
|
|
|
|
type PrintMode int
|
|
|
|
const (
|
|
PlainText = PrintMode(iota)
|
|
Markdown
|
|
HTML
|
|
)
|
|
|
|
// Version is a manually-updated mechanism for tracking versions.
|
|
const Version = "master"
|
|
|
|
// ServerVersion is the format used by gopls to report its version to the
|
|
// client. This format is structured so that the client can parse it easily.
|
|
type ServerVersion struct {
|
|
Module
|
|
Deps []*Module `json:"deps,omitempty"`
|
|
}
|
|
|
|
type Module struct {
|
|
ModuleVersion
|
|
Replace *ModuleVersion `json:"replace,omitempty"`
|
|
}
|
|
|
|
type ModuleVersion struct {
|
|
Path string `json:"path,omitempty"`
|
|
Version string `json:"version,omitempty"`
|
|
Sum string `json:"sum,omitempty"`
|
|
}
|
|
|
|
// VersionInfo returns the build info for the gopls process. If it was not
|
|
// built in module mode, we return a GOPATH-specific message with the
|
|
// hardcoded version.
|
|
func VersionInfo() *ServerVersion {
|
|
if info, ok := debug.ReadBuildInfo(); ok {
|
|
return getVersion(info)
|
|
}
|
|
path := "gopls, built in GOPATH mode"
|
|
return &ServerVersion{
|
|
Module: Module{
|
|
ModuleVersion: ModuleVersion{
|
|
Path: path,
|
|
Version: Version,
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
func getVersion(info *debug.BuildInfo) *ServerVersion {
|
|
serverVersion := ServerVersion{
|
|
Module: Module{
|
|
ModuleVersion: ModuleVersion{
|
|
Path: info.Main.Path,
|
|
Version: info.Main.Version,
|
|
Sum: info.Main.Sum,
|
|
},
|
|
},
|
|
}
|
|
for _, d := range info.Deps {
|
|
m := &Module{
|
|
ModuleVersion: ModuleVersion{
|
|
Path: d.Path,
|
|
Version: d.Version,
|
|
Sum: d.Sum,
|
|
},
|
|
}
|
|
if d.Replace != nil {
|
|
m.Replace = &ModuleVersion{
|
|
Path: d.Replace.Path,
|
|
Version: d.Replace.Version,
|
|
}
|
|
}
|
|
serverVersion.Deps = append(serverVersion.Deps, m)
|
|
}
|
|
return &serverVersion
|
|
}
|
|
|
|
// PrintServerInfo writes HTML debug info to w for the Instance.
|
|
func (i *Instance) PrintServerInfo(ctx context.Context, w io.Writer) {
|
|
section(w, HTML, "Server Instance", func() {
|
|
fmt.Fprintf(w, "Start time: %v\n", i.StartTime)
|
|
fmt.Fprintf(w, "LogFile: %s\n", i.Logfile)
|
|
fmt.Fprintf(w, "Working directory: %s\n", i.Workdir)
|
|
fmt.Fprintf(w, "Address: %s\n", i.ServerAddress)
|
|
fmt.Fprintf(w, "Debug address: %s\n", i.DebugAddress())
|
|
})
|
|
PrintVersionInfo(ctx, w, true, HTML)
|
|
section(w, HTML, "Command Line", func() {
|
|
fmt.Fprintf(w, "<a href=/debug/pprof/cmdline>cmdline</a>")
|
|
})
|
|
}
|
|
|
|
// PrintVersionInfo writes version information to w, using the output format
|
|
// specified by mode. verbose controls whether additional information is
|
|
// written, including section headers.
|
|
func PrintVersionInfo(ctx context.Context, w io.Writer, verbose bool, mode PrintMode) {
|
|
info := VersionInfo()
|
|
if !verbose {
|
|
printBuildInfo(w, info, false, mode)
|
|
return
|
|
}
|
|
section(w, mode, "Build info", func() {
|
|
printBuildInfo(w, info, true, mode)
|
|
})
|
|
}
|
|
|
|
func section(w io.Writer, mode PrintMode, title string, body func()) {
|
|
switch mode {
|
|
case PlainText:
|
|
fmt.Fprintln(w, title)
|
|
fmt.Fprintln(w, strings.Repeat("-", len(title)))
|
|
body()
|
|
case Markdown:
|
|
fmt.Fprintf(w, "#### %s\n\n```\n", title)
|
|
body()
|
|
fmt.Fprintf(w, "```\n")
|
|
case HTML:
|
|
fmt.Fprintf(w, "<h3>%s</h3>\n<pre>\n", title)
|
|
body()
|
|
fmt.Fprint(w, "</pre>\n")
|
|
}
|
|
}
|
|
|
|
func printBuildInfo(w io.Writer, info *ServerVersion, verbose bool, mode PrintMode) {
|
|
fmt.Fprintf(w, "%v %v\n", info.Path, Version)
|
|
printModuleInfo(w, &info.Module, mode)
|
|
if !verbose {
|
|
return
|
|
}
|
|
for _, dep := range info.Deps {
|
|
printModuleInfo(w, dep, mode)
|
|
}
|
|
}
|
|
|
|
func printModuleInfo(w io.Writer, m *Module, mode PrintMode) {
|
|
fmt.Fprintf(w, " %s@%s", m.Path, m.Version)
|
|
if m.Sum != "" {
|
|
fmt.Fprintf(w, " %s", m.Sum)
|
|
}
|
|
if m.Replace != nil {
|
|
fmt.Fprintf(w, " => %v", m.Replace.Path)
|
|
}
|
|
fmt.Fprintf(w, "\n")
|
|
}
|
|
|
|
type field struct {
|
|
index []int
|
|
}
|
|
|
|
var fields []field
|
|
|
|
// find all the options. The presumption is that the Options are nested structs
|
|
// and that pointers don't need to be dereferenced
|
|
func swalk(t reflect.Type, ix []int, indent string) {
|
|
switch t.Kind() {
|
|
case reflect.Struct:
|
|
for i := 0; i < t.NumField(); i++ {
|
|
fld := t.Field(i)
|
|
ixx := append(append([]int{}, ix...), i)
|
|
swalk(fld.Type, ixx, indent+". ")
|
|
}
|
|
default:
|
|
// everything is either a struct or a field (that's an assumption about Options)
|
|
fields = append(fields, field{ix})
|
|
}
|
|
}
|
|
|
|
type sessionOption struct {
|
|
Name string
|
|
Type string
|
|
Current string
|
|
Default string
|
|
}
|
|
|
|
func showOptions(o *source.Options) []sessionOption {
|
|
var out []sessionOption
|
|
t := reflect.TypeOf(*o)
|
|
swalk(t, []int{}, "")
|
|
v := reflect.ValueOf(*o)
|
|
do := reflect.ValueOf(*source.DefaultOptions())
|
|
for _, f := range fields {
|
|
val := v.FieldByIndex(f.index)
|
|
def := do.FieldByIndex(f.index)
|
|
tx := t.FieldByIndex(f.index)
|
|
is := strVal(val)
|
|
was := strVal(def)
|
|
out = append(out, sessionOption{
|
|
Name: tx.Name,
|
|
Type: tx.Type.String(),
|
|
Current: is,
|
|
Default: was,
|
|
})
|
|
}
|
|
sort.Slice(out, func(i, j int) bool {
|
|
rd := out[i].Current == out[i].Default
|
|
ld := out[j].Current == out[j].Default
|
|
if rd != ld {
|
|
return ld
|
|
}
|
|
return out[i].Name < out[j].Name
|
|
})
|
|
return out
|
|
}
|
|
|
|
func strVal(val reflect.Value) string {
|
|
switch val.Kind() {
|
|
case reflect.Bool:
|
|
return fmt.Sprintf("%v", val.Interface())
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
return fmt.Sprintf("%v", val.Interface())
|
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
|
return fmt.Sprintf("%v", val.Interface())
|
|
case reflect.Uintptr, reflect.UnsafePointer:
|
|
return fmt.Sprintf("0x%x", val.Pointer())
|
|
case reflect.Complex64, reflect.Complex128:
|
|
return fmt.Sprintf("%v", val.Complex())
|
|
case reflect.Array, reflect.Slice:
|
|
ans := []string{}
|
|
for i := 0; i < val.Len(); i++ {
|
|
ans = append(ans, strVal(val.Index(i)))
|
|
}
|
|
sort.Strings(ans)
|
|
return fmt.Sprintf("%v", ans)
|
|
case reflect.Chan, reflect.Func, reflect.Ptr:
|
|
return val.Kind().String()
|
|
case reflect.Struct:
|
|
var x source.Analyzer
|
|
if val.Type() != reflect.TypeOf(x) {
|
|
return val.Kind().String()
|
|
}
|
|
// this is sort of ugly, but usable
|
|
str := val.FieldByName("Analyzer").Elem().FieldByName("Doc").String()
|
|
ix := strings.Index(str, "\n")
|
|
if ix == -1 {
|
|
ix = len(str)
|
|
}
|
|
return str[:ix]
|
|
case reflect.String:
|
|
return fmt.Sprintf("%q", val.Interface())
|
|
case reflect.Map:
|
|
ans := []string{}
|
|
iter := val.MapRange()
|
|
for iter.Next() {
|
|
k := iter.Key()
|
|
v := iter.Value()
|
|
ans = append(ans, fmt.Sprintf("%s:%s, ", strVal(k), strVal(v)))
|
|
}
|
|
sort.Strings(ans)
|
|
return fmt.Sprintf("%v", ans)
|
|
}
|
|
return fmt.Sprintf("??%s??", val.Type())
|
|
}
|