-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
godoc: show version information for stdlib
This change reads $GOROOT/api/go1.*.txt when godoc starts and caches information about which versions of Go introduce functions, types, and methods. This information is displayed currently only in HTML output. Functions, types, and methods introduced as part of Go 1 are not annotated, as their presence at that version is implied. This change does not address constants or variables, and completely ignores the syscall package. The former are future work, the latter is likely an exercise in futility. In all cases, this is because the story around displaying the version information is not well developed. Fixes golang/go#5778 Change-Id: Ieb3cc0da7b18e195bc9c443f14fd8a82e8b2bbf8 Reviewed-on: https://go-review.googlesource.com/85396 Run-TryBot: Brad Fitzpatrick <[email protected]> TryBot-Result: Gobot Gobot <[email protected]> Reviewed-by: Devon O'Dell <[email protected]> Reviewed-by: Brad Fitzpatrick <[email protected]>
- Loading branch information
Showing
7 changed files
with
340 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,211 @@ | ||
// Copyright 2018 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. | ||
|
||
// This file caches information about which standard library types, methods, | ||
// and functions appeared in what version of Go | ||
|
||
package godoc | ||
|
||
import ( | ||
"bufio" | ||
"go/build" | ||
"log" | ||
"os" | ||
"path/filepath" | ||
"strings" | ||
"unicode" | ||
) | ||
|
||
// apiVersions is a map of packages to information about those packages' | ||
// symbols and when they were added to Go. | ||
// | ||
// Only things added after Go1 are tracked. Version strings are of the | ||
// form "1.1", "1.2", etc. | ||
type apiVersions map[string]pkgAPIVersions // keyed by Go package ("net/http") | ||
|
||
// pkgAPIVersions contains information about which version of Go added | ||
// certain package symbols. | ||
// | ||
// Only things added after Go1 are tracked. Version strings are of the | ||
// form "1.1", "1.2", etc. | ||
type pkgAPIVersions struct { | ||
typeSince map[string]string // "Server" -> "1.7" | ||
methodSince map[string]map[string]string // "*Server"->"Shutdown"->1.8 | ||
funcSince map[string]string // "NewServer" -> "1.7" | ||
} | ||
|
||
// sinceVersionFunc returns a string (such as "1.7") specifying which Go | ||
// version introduced a symbol, unless it was introduced in Go1, in | ||
// which case it returns the empty string. | ||
// | ||
// The kind is one of "type", "method", or "func". | ||
// | ||
// The receiver is only used for "methods" and specifies the receiver type, | ||
// such as "*Server". | ||
// | ||
// The name is the symbol name ("Server") and the pkg is the package | ||
// ("net/http"). | ||
func (v apiVersions) sinceVersionFunc(kind, receiver, name, pkg string) string { | ||
pv := v[pkg] | ||
switch kind { | ||
case "func": | ||
return pv.funcSince[name] | ||
case "type": | ||
return pv.typeSince[name] | ||
case "method": | ||
return pv.methodSince[receiver][name] | ||
} | ||
return "" | ||
} | ||
|
||
// versionedRow represents an API feature, a parsed line of a | ||
// $GOROOT/api/go.*txt file. | ||
type versionedRow struct { | ||
pkg string // "net/http" | ||
kind string // "type", "func", "method", TODO: "const", "var" | ||
recv string // for methods, the receiver type ("Server", "*Server") | ||
name string // name of type, func, or method | ||
} | ||
|
||
// versionParser parses $GOROOT/api/go*.txt files and stores them in in its rows field. | ||
type versionParser struct { | ||
res apiVersions // initialized lazily | ||
} | ||
|
||
func (vp *versionParser) parseFile(name string) error { | ||
base := filepath.Base(name) | ||
ver := strings.TrimPrefix(strings.TrimSuffix(base, ".txt"), "go") | ||
if ver == "1" { | ||
return nil | ||
} | ||
f, err := os.Open(name) | ||
if err != nil { | ||
return err | ||
} | ||
defer f.Close() | ||
|
||
sc := bufio.NewScanner(f) | ||
for sc.Scan() { | ||
row, ok := parseRow(sc.Text()) | ||
if !ok { | ||
continue | ||
} | ||
if vp.res == nil { | ||
vp.res = make(apiVersions) | ||
} | ||
pkgi, ok := vp.res[row.pkg] | ||
if !ok { | ||
pkgi = pkgAPIVersions{ | ||
typeSince: make(map[string]string), | ||
methodSince: make(map[string]map[string]string), | ||
funcSince: make(map[string]string), | ||
} | ||
vp.res[row.pkg] = pkgi | ||
} | ||
switch row.kind { | ||
case "func": | ||
pkgi.funcSince[row.name] = ver | ||
case "type": | ||
pkgi.typeSince[row.name] = ver | ||
case "method": | ||
if _, ok := pkgi.methodSince[row.recv]; !ok { | ||
pkgi.methodSince[row.recv] = make(map[string]string) | ||
} | ||
pkgi.methodSince[row.recv][row.name] = ver | ||
} | ||
} | ||
return sc.Err() | ||
} | ||
|
||
func parseRow(s string) (vr versionedRow, ok bool) { | ||
if !strings.HasPrefix(s, "pkg ") { | ||
// Skip comments, blank lines, etc. | ||
return | ||
} | ||
rest := s[len("pkg "):] | ||
endPkg := strings.IndexFunc(rest, func(r rune) bool { return !(unicode.IsLetter(r) || r == '/') }) | ||
if endPkg == -1 { | ||
return | ||
} | ||
vr.pkg, rest = rest[:endPkg], rest[endPkg:] | ||
if !strings.HasPrefix(rest, ", ") { | ||
// If the part after the pkg name isn't ", ", then it's a OS/ARCH-dependent line of the form: | ||
// pkg syscall (darwin-amd64), const ImplementsGetwd = false | ||
// We skip those for now. | ||
return | ||
} | ||
rest = rest[len(", "):] | ||
|
||
switch { | ||
case strings.HasPrefix(rest, "type "): | ||
vr.kind = "type" | ||
rest = rest[len("type "):] | ||
sp := strings.IndexByte(rest, ' ') | ||
if sp == -1 { | ||
return | ||
} | ||
vr.name, rest = rest[:sp], rest[sp+1:] | ||
if strings.HasPrefix(rest, "struct, ") { | ||
// TODO: handle struct fields | ||
return | ||
} | ||
return vr, true | ||
case strings.HasPrefix(rest, "func "): | ||
vr.kind = "func" | ||
rest = rest[len("func "):] | ||
if i := strings.IndexByte(rest, '('); i != -1 { | ||
vr.name = rest[:i] | ||
return vr, true | ||
} | ||
case strings.HasPrefix(rest, "method "): // "method (*File) SetModTime(time.Time)" | ||
vr.kind = "method" | ||
rest = rest[len("method "):] // "(*File) SetModTime(time.Time)" | ||
sp := strings.IndexByte(rest, ' ') | ||
if sp == -1 { | ||
return | ||
} | ||
vr.recv = strings.Trim(rest[:sp], "()") // "*File" | ||
rest = rest[sp+1:] // SetMode(os.FileMode) | ||
paren := strings.IndexByte(rest, '(') | ||
if paren == -1 { | ||
return | ||
} | ||
vr.name = rest[:paren] | ||
return vr, true | ||
} | ||
return // TODO: handle more cases | ||
} | ||
|
||
// InitVersionInfo parses the $GOROOT/api/go*.txt API definition files to discover | ||
// which API features were added in which Go releases. | ||
func (c *Corpus) InitVersionInfo() { | ||
var err error | ||
c.pkgAPIInfo, err = parsePackageAPIInfo() | ||
if err != nil { | ||
// TODO: consider making this fatal, after the Go 1.11 cycle. | ||
log.Printf("godoc: error parsing API version files: %v", err) | ||
} | ||
} | ||
|
||
func parsePackageAPIInfo() (apiVersions, error) { | ||
var apiGlob string | ||
if os.Getenv("GOROOT") == "" { | ||
apiGlob = filepath.Join(build.Default.GOROOT, "api", "go*.txt") | ||
} else { | ||
apiGlob = filepath.Join(os.Getenv("GOROOT"), "api", "go*.txt") | ||
} | ||
|
||
files, err := filepath.Glob(apiGlob) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
vp := new(versionParser) | ||
for _, f := range files { | ||
if err := vp.parseFile(f); err != nil { | ||
return nil, err | ||
} | ||
} | ||
return vp.res, nil | ||
} |
Oops, something went wrong.