blob: 57576e10282e771923e516d088dcc53121851a2a [file] [log] [blame]
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +10001// Copyright 2013 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5package godoc
6
7import (
8 "bytes"
Alan Donovan80c4f062014-03-14 18:58:22 -04009 "encoding/json"
Agniva De Sarker006ac432018-02-10 09:59:54 +053010 "errors"
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100011 "fmt"
12 "go/ast"
13 "go/build"
14 "go/doc"
15 "go/token"
16 htmlpkg "html"
Alan Donovan80c4f062014-03-14 18:58:22 -040017 htmltemplate "html/template"
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100018 "io"
19 "io/ioutil"
20 "log"
21 "net/http"
22 "os"
23 pathpkg "path"
24 "path/filepath"
Andrew Gerrand2e6fbd82013-08-28 09:39:02 +100025 "sort"
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100026 "strings"
27 "text/template"
28 "time"
29
Andrew Gerrand5ebbcd12014-11-10 08:50:40 +110030 "golang.org/x/tools/godoc/analysis"
31 "golang.org/x/tools/godoc/util"
32 "golang.org/x/tools/godoc/vfs"
aarzilli47547482022-01-04 13:16:06 +010033 "golang.org/x/tools/internal/typeparams"
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100034)
35
Brad Fitzpatrick705bb7f2013-07-19 10:27:53 +100036// handlerServer is a migration from an old godoc http Handler type.
37// This should probably merge into something else.
38type handlerServer struct {
Andrew Gerrand31971b72015-08-28 11:00:04 +100039 p *Presentation
40 c *Corpus // copy of p.Corpus
41 pattern string // url pattern; e.g. "/pkg/"
42 stripPrefix string // prefix to strip from import path; e.g. "pkg/"
43 fsRoot string // file system root to which the pattern is mapped; e.g. "/src"
44 exclude []string // file system paths to exclude; e.g. "/src/cmd"
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100045}
46
Brad Fitzpatrick705bb7f2013-07-19 10:27:53 +100047func (s *handlerServer) registerWithMux(mux *http.ServeMux) {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100048 mux.Handle(s.pattern, s)
49}
50
Dmitri Shuralyove7abfed2019-09-23 10:59:13 -040051// GetPageInfo returns the PageInfo for a package directory abspath. If the
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100052// parameter genAST is set, an AST containing only the package exports is
53// computed (PageInfo.PAst), otherwise package documentation (PageInfo.Doc)
54// is extracted from the AST. If there is no corresponding package in the
55// directory, PageInfo.PAst and PageInfo.PDoc are nil. If there are no sub-
56// directories, PageInfo.Dirs is nil. If an error occurred, PageInfo.Err is
57// set to the respective error but the error is not logged.
Larz Conwell9a6ae372014-12-11 10:10:04 -050058func (h *handlerServer) GetPageInfo(abspath, relpath string, mode PageInfoMode, goos, goarch string) *PageInfo {
Ted Kornish396c1762017-01-18 10:52:36 -080059 info := &PageInfo{Dirname: abspath, Mode: mode}
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100060
61 // Restrict to the package files that would be used when building
62 // the package on this system. This makes sure that if there are
63 // separate implementations for, say, Windows vs Unix, we don't
64 // jumble them all together.
Larz Conwell9a6ae372014-12-11 10:10:04 -050065 // Note: If goos/goarch aren't set, the current binary's GOOS/GOARCH
66 // are used.
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +100067 ctxt := build.Default
68 ctxt.IsAbsPath = pathpkg.IsAbs
Dmitri Shuralyovf595fb52017-03-04 19:06:07 -050069 ctxt.IsDir = func(path string) bool {
70 fi, err := h.c.fs.Stat(filepath.ToSlash(path))
71 return err == nil && fi.IsDir()
72 }
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +100073 ctxt.ReadDir = func(dir string) ([]os.FileInfo, error) {
Hana Kim13837d22014-10-13 18:47:02 +020074 f, err := h.c.fs.ReadDir(filepath.ToSlash(dir))
75 filtered := make([]os.FileInfo, 0, len(f))
76 for _, i := range f {
77 if mode&NoFiltering != 0 || i.Name() != "internal" {
78 filtered = append(filtered, i)
79 }
80 }
81 return filtered, err
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +100082 }
83 ctxt.OpenFile = func(name string) (r io.ReadCloser, err error) {
84 data, err := vfs.ReadFile(h.c.fs, filepath.ToSlash(name))
85 if err != nil {
86 return nil, err
87 }
88 return ioutil.NopCloser(bytes.NewReader(data)), nil
89 }
90
Brad Fitzpatrick1c99e122018-06-27 20:58:43 +000091 // Make the syscall/js package always visible by default.
92 // It defaults to the host's GOOS/GOARCH, and golang.org's
93 // linux/amd64 means the wasm syscall/js package was blank.
94 // And you can't run godoc on js/wasm anyway, so host defaults
95 // don't make sense here.
96 if goos == "" && goarch == "" && relpath == "syscall/js" {
97 goos, goarch = "js", "wasm"
98 }
Larz Conwell9a6ae372014-12-11 10:10:04 -050099 if goos != "" {
100 ctxt.GOOS = goos
101 }
102 if goarch != "" {
103 ctxt.GOARCH = goarch
104 }
105
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000106 pkginfo, err := ctxt.ImportDir(abspath, 0)
107 // continue if there are no Go source files; we still want the directory info
108 if _, nogo := err.(*build.NoGoError); err != nil && !nogo {
109 info.Err = err
110 return info
111 }
112
113 // collect package files
114 pkgname := pkginfo.Name
115 pkgfiles := append(pkginfo.GoFiles, pkginfo.CgoFiles...)
116 if len(pkgfiles) == 0 {
117 // Commands written in C have no .go files in the build.
118 // Instead, documentation may be found in an ignored file.
119 // The file may be ignored via an explicit +build ignore
120 // constraint (recommended), or by defining the package
121 // documentation (historic).
122 pkgname = "main" // assume package main since pkginfo.Name == ""
123 pkgfiles = pkginfo.IgnoredGoFiles
Agniva De Sarker1a83a0b2018-04-15 03:03:16 +0530124 }
125
126 // get package information, if any
127 if len(pkgfiles) > 0 {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000128 // build package AST
129 fset := token.NewFileSet()
Brad Garcia2ca2bfc2014-01-30 06:28:19 -0500130 files, err := h.c.parseFiles(fset, relpath, abspath, pkgfiles)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000131 if err != nil {
132 info.Err = err
133 return info
134 }
135
136 // ignore any errors - they are due to unresolved identifiers
137 pkg, _ := ast.NewPackage(fset, files, poorMansImporter, nil)
138
139 // extract package documentation
140 info.FSet = fset
141 if mode&ShowSource == 0 {
142 // show extracted documentation
143 var m doc.Mode
144 if mode&NoFiltering != 0 {
Andrew Gerrand2e6fbd82013-08-28 09:39:02 +1000145 m |= doc.AllDecls
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000146 }
147 if mode&AllMethods != 0 {
148 m |= doc.AllMethods
149 }
150 info.PDoc = doc.New(pkg, pathpkg.Clean(relpath), m) // no trailing '/' in importpath
Robert Griesemer5351a1c2014-02-04 10:26:38 -0800151 if mode&NoTypeAssoc != 0 {
Andrew Gerrand2e6fbd82013-08-28 09:39:02 +1000152 for _, t := range info.PDoc.Types {
Robert Griesemer5351a1c2014-02-04 10:26:38 -0800153 info.PDoc.Consts = append(info.PDoc.Consts, t.Consts...)
154 info.PDoc.Vars = append(info.PDoc.Vars, t.Vars...)
Andrew Gerrand2e6fbd82013-08-28 09:39:02 +1000155 info.PDoc.Funcs = append(info.PDoc.Funcs, t.Funcs...)
Robert Griesemer5351a1c2014-02-04 10:26:38 -0800156 t.Consts = nil
157 t.Vars = nil
Andrew Gerrand2e6fbd82013-08-28 09:39:02 +1000158 t.Funcs = nil
159 }
Robert Griesemer5351a1c2014-02-04 10:26:38 -0800160 // for now we cannot easily sort consts and vars since
161 // go/doc.Value doesn't export the order information
Andrew Gerrand2e6fbd82013-08-28 09:39:02 +1000162 sort.Sort(funcsByName(info.PDoc.Funcs))
163 }
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000164
165 // collect examples
166 testfiles := append(pkginfo.TestGoFiles, pkginfo.XTestGoFiles...)
Brad Garcia2ca2bfc2014-01-30 06:28:19 -0500167 files, err = h.c.parseFiles(fset, relpath, abspath, testfiles)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000168 if err != nil {
169 log.Println("parsing examples:", err)
170 }
Brad Fitzpatrick88f792c2013-11-14 09:01:08 -0800171 info.Examples = collectExamples(h.c, pkg, files)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000172
173 // collect any notes that we want to show
174 if info.PDoc.Notes != nil {
175 // could regexp.Compile only once per godoc, but probably not worth it
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000176 if rx := h.p.NotesRx; rx != nil {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000177 for m, n := range info.PDoc.Notes {
178 if rx.MatchString(m) {
179 if info.Notes == nil {
180 info.Notes = make(map[string][]*doc.Note)
181 }
182 info.Notes[m] = n
183 }
184 }
185 }
186 }
187
188 } else {
189 // show source code
190 // TODO(gri) Consider eliminating export filtering in this mode,
191 // or perhaps eliminating the mode altogether.
192 if mode&NoFiltering == 0 {
193 packageExports(fset, pkg)
194 }
Brad Garcia2ca2bfc2014-01-30 06:28:19 -0500195 info.PAst = files
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000196 }
197 info.IsMain = pkgname == "main"
198 }
199
200 // get directory information, if any
201 var dir *Directory
202 var timestamp time.Time
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000203 if tree, ts := h.c.fsTree.Get(); tree != nil && tree.(*Directory) != nil {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000204 // directory tree is present; lookup respective directory
205 // (may still fail if the file system was updated and the
206 // new directory tree has not yet been computed)
207 dir = tree.(*Directory).lookup(abspath)
208 timestamp = ts
209 }
210 if dir == nil {
Agniva De Sarkerdef26772018-10-18 12:51:06 +0530211 // TODO(agnivade): handle this case better, now since there is no CLI mode.
Agniva De Sarker2ae76fd2018-02-01 22:39:07 +0530212 // no directory tree present (happens in command-line mode);
213 // compute 2 levels for this page. The second level is to
214 // get the synopses of sub-directories.
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000215 // note: cannot use path filter here because in general
Agniva De Sarker2ae76fd2018-02-01 22:39:07 +0530216 // it doesn't contain the FSTree path
217 dir = h.c.newDirectory(abspath, 2)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000218 timestamp = time.Now()
219 }
Hana Kim13837d22014-10-13 18:47:02 +0200220 info.Dirs = dir.listing(true, func(path string) bool { return h.includePath(path, mode) })
Ted Kornish396c1762017-01-18 10:52:36 -0800221
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000222 info.DirTime = timestamp
223 info.DirFlat = mode&FlatDir != 0
224
225 return info
226}
227
Hana Kim13837d22014-10-13 18:47:02 +0200228func (h *handlerServer) includePath(path string, mode PageInfoMode) (r bool) {
229 // if the path is under one of the exclusion paths, don't list.
230 for _, e := range h.exclude {
231 if strings.HasPrefix(path, e) {
232 return false
233 }
234 }
235
236 // if the path includes 'internal', don't list unless we are in the NoFiltering mode.
237 if mode&NoFiltering != 0 {
238 return true
239 }
Andrew Gerrandb81ea3a2016-01-13 13:06:35 +1100240 if strings.Contains(path, "internal") || strings.Contains(path, "vendor") {
Alex Brainman30f7e632014-10-28 12:26:29 +1100241 for _, c := range strings.Split(filepath.Clean(path), string(os.PathSeparator)) {
Andrew Gerrandb81ea3a2016-01-13 13:06:35 +1100242 if c == "internal" || c == "vendor" {
Hana Kim13837d22014-10-13 18:47:02 +0200243 return false
244 }
245 }
246 }
247 return true
248}
249
Andrew Gerrand2e6fbd82013-08-28 09:39:02 +1000250type funcsByName []*doc.Func
251
252func (s funcsByName) Len() int { return len(s) }
253func (s funcsByName) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
254func (s funcsByName) Less(i, j int) bool { return s[i].Name < s[j].Name }
255
Brad Fitzpatrick705bb7f2013-07-19 10:27:53 +1000256func (h *handlerServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000257 if redirect(w, r) {
258 return
259 }
260
Andrew Gerrand31971b72015-08-28 11:00:04 +1000261 relpath := pathpkg.Clean(r.URL.Path[len(h.stripPrefix)+1:])
Agniva De Sarker006ac432018-02-10 09:59:54 +0530262
263 if !h.corpusInitialized() {
264 h.p.ServeError(w, r, relpath, errors.New("Scan is not yet complete. Please retry after a few moments"))
265 return
266 }
267
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000268 abspath := pathpkg.Join(h.fsRoot, relpath)
Brad Fitzpatrick705bb7f2013-07-19 10:27:53 +1000269 mode := h.p.GetPageInfoMode(r)
270 if relpath == builtinPkgPath {
Dmitri Shuralyov93949562019-02-18 13:49:49 -0500271 // The fake built-in package contains unexported identifiers,
272 // but we want to show them. Also, disable type association,
273 // since it's not helpful for this fake package (see issue 6645).
274 mode |= NoFiltering | NoTypeAssoc
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000275 }
Larz Conwell9a6ae372014-12-11 10:10:04 -0500276 info := h.GetPageInfo(abspath, relpath, mode, r.FormValue("GOOS"), r.FormValue("GOARCH"))
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000277 if info.Err != nil {
278 log.Print(info.Err)
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000279 h.p.ServeError(w, r, relpath, info.Err)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000280 return
281 }
282
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000283 var tabtitle, title, subtitle string
284 switch {
285 case info.PAst != nil:
Brad Garcia2ca2bfc2014-01-30 06:28:19 -0500286 for _, ast := range info.PAst {
287 tabtitle = ast.Name.Name
288 break
289 }
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000290 case info.PDoc != nil:
291 tabtitle = info.PDoc.Name
292 default:
293 tabtitle = info.Dirname
294 title = "Directory "
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000295 if h.p.ShowTimestamps {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000296 subtitle = "Last update: " + info.DirTime.String()
297 }
298 }
299 if title == "" {
300 if info.IsMain {
301 // assume that the directory name is the command name
302 _, tabtitle = pathpkg.Split(relpath)
303 title = "Command "
304 } else {
305 title = "Package "
306 }
307 }
308 title += tabtitle
309
310 // special cases for top-level package/command directories
311 switch tabtitle {
Alan Donovan6c93dbf2014-09-10 09:02:54 -0400312 case "/src":
Andrew Gerrand277e5a12014-08-21 09:53:57 +1000313 title = "Packages"
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000314 tabtitle = "Packages"
315 case "/src/cmd":
Andrew Gerrand277e5a12014-08-21 09:53:57 +1000316 title = "Commands"
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000317 tabtitle = "Commands"
318 }
319
Alan Donovan80c4f062014-03-14 18:58:22 -0400320 // Emit JSON array for type information.
Alan Donovan99d45c02014-07-09 07:59:55 -0400321 pi := h.c.Analysis.PackageInfo(relpath)
Agniva De Sarkerc75e7e62018-05-12 10:43:07 +0530322 hasTreeView := len(pi.CallGraph) != 0
Alan Donovan99d45c02014-07-09 07:59:55 -0400323 info.CallGraphIndex = pi.CallGraphIndex
324 info.CallGraph = htmltemplate.JS(marshalJSON(pi.CallGraph))
325 info.AnalysisData = htmltemplate.JS(marshalJSON(pi.Types))
Alan Donovan80c4f062014-03-14 18:58:22 -0400326 info.TypeInfoIndex = make(map[string]int)
Alan Donovan99d45c02014-07-09 07:59:55 -0400327 for i, ti := range pi.Types {
Alan Donovan80c4f062014-03-14 18:58:22 -0400328 info.TypeInfoIndex[ti.Name] = i
329 }
330
Agniva De Sarkerfaed9972018-03-18 00:46:28 +0530331 var body []byte
332 if info.Dirname == "/src" {
333 body = applyTemplate(h.p.PackageRootHTML, "packageRootHTML", info)
334 } else {
335 body = applyTemplate(h.p.PackageHTML, "packageHTML", info)
336 }
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000337 h.p.ServePage(w, Page{
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000338 Title: title,
339 Tabtitle: tabtitle,
340 Subtitle: subtitle,
Agniva De Sarkerfaed9972018-03-18 00:46:28 +0530341 Body: body,
Agniva De Sarkerc75e7e62018-05-12 10:43:07 +0530342 TreeView: hasTreeView,
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000343 })
344}
345
Agniva De Sarker006ac432018-02-10 09:59:54 +0530346func (h *handlerServer) corpusInitialized() bool {
347 h.c.initMu.RLock()
348 defer h.c.initMu.RUnlock()
349 return h.c.initDone
350}
351
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000352type PageInfoMode uint
353
354const (
Ted Kornish396c1762017-01-18 10:52:36 -0800355 PageInfoModeQueryString = "m" // query string where PageInfoMode is stored
356
Robert Griesemer5351a1c2014-02-04 10:26:38 -0800357 NoFiltering PageInfoMode = 1 << iota // do not filter exports
358 AllMethods // show all embedded methods
359 ShowSource // show source code, do not extract documentation
Robert Griesemer5351a1c2014-02-04 10:26:38 -0800360 FlatDir // show directory in a flat (non-indented) manner
Dmitri Shuralyov93949562019-02-18 13:49:49 -0500361 NoTypeAssoc // don't associate consts, vars, and factory functions with types (not exposed via ?m= query parameter, used for package builtin, see issue 6645)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000362)
363
364// modeNames defines names for each PageInfoMode flag.
365var modeNames = map[string]PageInfoMode{
366 "all": NoFiltering,
367 "methods": AllMethods,
368 "src": ShowSource,
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000369 "flat": FlatDir,
370}
371
Ted Kornish396c1762017-01-18 10:52:36 -0800372// generate a query string for persisting PageInfoMode between pages.
373func modeQueryString(mode PageInfoMode) string {
374 if modeNames := mode.names(); len(modeNames) > 0 {
375 return "?m=" + strings.Join(modeNames, ",")
376 }
377 return ""
378}
379
380// alphabetically sorted names of active flags for a PageInfoMode.
381func (m PageInfoMode) names() []string {
382 var names []string
383 for name, mode := range modeNames {
384 if m&mode != 0 {
385 names = append(names, name)
386 }
387 }
388 sort.Strings(names)
389 return names
390}
391
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000392// GetPageInfoMode computes the PageInfoMode flags by analyzing the request
393// URL form value "m". It is value is a comma-separated list of mode names
394// as defined by modeNames (e.g.: m=src,text).
Brad Fitzpatrick705bb7f2013-07-19 10:27:53 +1000395func (p *Presentation) GetPageInfoMode(r *http.Request) PageInfoMode {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000396 var mode PageInfoMode
Ted Kornish396c1762017-01-18 10:52:36 -0800397 for _, k := range strings.Split(r.FormValue(PageInfoModeQueryString), ",") {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000398 if m, found := modeNames[strings.TrimSpace(k)]; found {
399 mode |= m
400 }
401 }
Brad Fitzpatrick705bb7f2013-07-19 10:27:53 +1000402 if p.AdjustPageInfoMode != nil {
403 mode = p.AdjustPageInfoMode(r, mode)
404 }
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000405 return mode
406}
407
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000408// poorMansImporter returns a (dummy) package object named
409// by the last path component of the provided package path
410// (as is the convention for packages). This is sufficient
411// to resolve package identifiers without doing an actual
412// import. It never returns an error.
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000413func poorMansImporter(imports map[string]*ast.Object, path string) (*ast.Object, error) {
414 pkg := imports[path]
415 if pkg == nil {
416 // note that strings.LastIndex returns -1 if there is no "/"
417 pkg = ast.NewObj(ast.Pkg, path[strings.LastIndex(path, "/")+1:])
418 pkg.Data = ast.NewScope(nil) // required by ast.NewPackage for dot-import
419 imports[path] = pkg
420 }
421 return pkg, nil
422}
423
424// globalNames returns a set of the names declared by all package-level
425// declarations. Method names are returned in the form Receiver_Method.
426func globalNames(pkg *ast.Package) map[string]bool {
427 names := make(map[string]bool)
428 for _, file := range pkg.Files {
429 for _, decl := range file.Decls {
430 addNames(names, decl)
431 }
432 }
433 return names
434}
435
436// collectExamples collects examples for pkg from testfiles.
Brad Fitzpatrick88f792c2013-11-14 09:01:08 -0800437func collectExamples(c *Corpus, pkg *ast.Package, testfiles map[string]*ast.File) []*doc.Example {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000438 var files []*ast.File
439 for _, f := range testfiles {
440 files = append(files, f)
441 }
442
443 var examples []*doc.Example
444 globals := globalNames(pkg)
445 for _, e := range doc.Examples(files...) {
446 name := stripExampleSuffix(e.Name)
447 if name == "" || globals[name] {
448 examples = append(examples, e)
Brad Fitzpatrick88f792c2013-11-14 09:01:08 -0800449 } else if c.Verbose {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000450 log.Printf("skipping example 'Example%s' because '%s' is not a known function or type", e.Name, e.Name)
451 }
452 }
453
454 return examples
455}
456
457// addNames adds the names declared by decl to the names set.
458// Method names are added in the form ReceiverTypeName_Method.
459func addNames(names map[string]bool, decl ast.Decl) {
460 switch d := decl.(type) {
461 case *ast.FuncDecl:
462 name := d.Name.Name
463 if d.Recv != nil {
aarzilli47547482022-01-04 13:16:06 +0100464 r := d.Recv.List[0].Type
465 if rr, isstar := r.(*ast.StarExpr); isstar {
466 r = rr.X
467 }
468
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000469 var typeName string
aarzilli47547482022-01-04 13:16:06 +0100470 switch x := r.(type) {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000471 case *ast.Ident:
aarzilli47547482022-01-04 13:16:06 +0100472 typeName = x.Name
473 case *ast.IndexExpr:
474 typeName = x.X.(*ast.Ident).Name
475 case *typeparams.IndexListExpr:
476 typeName = x.X.(*ast.Ident).Name
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000477 }
478 name = typeName + "_" + name
479 }
480 names[name] = true
481 case *ast.GenDecl:
482 for _, spec := range d.Specs {
483 switch s := spec.(type) {
484 case *ast.TypeSpec:
485 names[s.Name.Name] = true
486 case *ast.ValueSpec:
487 for _, id := range s.Names {
488 names[id.Name] = true
489 }
490 }
491 }
492 }
493}
494
495// packageExports is a local implementation of ast.PackageExports
496// which correctly updates each package file's comment list.
497// (The ast.PackageExports signature is frozen, hence the local
498// implementation).
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000499func packageExports(fset *token.FileSet, pkg *ast.Package) {
500 for _, src := range pkg.Files {
501 cmap := ast.NewCommentMap(fset, src, src.Comments)
502 ast.FileExports(src)
503 src.Comments = cmap.Filter(src).Comments()
504 }
505}
506
507func applyTemplate(t *template.Template, name string, data interface{}) []byte {
508 var buf bytes.Buffer
509 if err := t.Execute(&buf, data); err != nil {
510 log.Printf("%s.Execute: %s", name, err)
511 }
512 return buf.Bytes()
513}
514
Brad Garciaefd232e2014-01-29 10:53:45 -0500515type writerCapturesErr struct {
516 w io.Writer
517 err error
518}
519
520func (w *writerCapturesErr) Write(p []byte) (int, error) {
521 n, err := w.w.Write(p)
522 if err != nil {
523 w.err = err
524 }
525 return n, err
526}
527
Brad Garciaefd232e2014-01-29 10:53:45 -0500528// applyTemplateToResponseWriter uses an http.ResponseWriter as the io.Writer
529// for the call to template.Execute. It uses an io.Writer wrapper to capture
Sameer Ajmani87156cb2015-07-15 14:49:51 -0400530// errors from the underlying http.ResponseWriter. Errors are logged only when
531// they come from the template processing and not the Writer; this avoid
532// polluting log files with error messages due to networking issues, such as
533// client disconnects and http HEAD protocol violations.
Brad Garciaefd232e2014-01-29 10:53:45 -0500534func applyTemplateToResponseWriter(rw http.ResponseWriter, t *template.Template, data interface{}) {
535 w := &writerCapturesErr{w: rw}
536 err := t.Execute(w, data)
537 // There are some cases where template.Execute does not return an error when
538 // rw returns an error, and some where it does. So check w.err first.
Sameer Ajmani87156cb2015-07-15 14:49:51 -0400539 if w.err == nil && err != nil {
Brad Garciaefd232e2014-01-29 10:53:45 -0500540 // Log template errors.
541 log.Printf("%s.Execute: %s", t.Name(), err)
542 }
543}
544
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000545func redirect(w http.ResponseWriter, r *http.Request) (redirected bool) {
546 canonical := pathpkg.Clean(r.URL.Path)
547 if !strings.HasSuffix(canonical, "/") {
548 canonical += "/"
549 }
550 if r.URL.Path != canonical {
551 url := *r.URL
552 url.Path = canonical
553 http.Redirect(w, r, url.String(), http.StatusMovedPermanently)
554 redirected = true
555 }
556 return
557}
558
559func redirectFile(w http.ResponseWriter, r *http.Request) (redirected bool) {
560 c := pathpkg.Clean(r.URL.Path)
561 c = strings.TrimRight(c, "/")
562 if r.URL.Path != c {
563 url := *r.URL
564 url.Path = c
565 http.Redirect(w, r, url.String(), http.StatusMovedPermanently)
566 redirected = true
567 }
568 return
569}
570
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000571func (p *Presentation) serveTextFile(w http.ResponseWriter, r *http.Request, abspath, relpath, title string) {
572 src, err := vfs.ReadFile(p.Corpus.fs, abspath)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000573 if err != nil {
574 log.Printf("ReadFile: %s", err)
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000575 p.ServeError(w, r, relpath, err)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000576 return
577 }
578
Ted Kornish396c1762017-01-18 10:52:36 -0800579 if r.FormValue(PageInfoModeQueryString) == "text" {
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000580 p.ServeText(w, src)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000581 return
582 }
583
Alan Donovan80c4f062014-03-14 18:58:22 -0400584 h := r.FormValue("h")
585 s := RangeSelection(r.FormValue("s"))
586
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000587 var buf bytes.Buffer
Alan Donovan80c4f062014-03-14 18:58:22 -0400588 if pathpkg.Ext(abspath) == ".go" {
Alan Donovan6c93dbf2014-09-10 09:02:54 -0400589 // Find markup links for this file (e.g. "/src/fmt/print.go").
Alan Donovan99d45c02014-07-09 07:59:55 -0400590 fi := p.Corpus.Analysis.FileInfo(abspath)
Alan Donovan80c4f062014-03-14 18:58:22 -0400591 buf.WriteString("<script type='text/javascript'>document.ANALYSIS_DATA = ")
Alan Donovan99d45c02014-07-09 07:59:55 -0400592 buf.Write(marshalJSON(fi.Data))
Alan Donovan80c4f062014-03-14 18:58:22 -0400593 buf.WriteString(";</script>\n")
594
Alan Donovan99d45c02014-07-09 07:59:55 -0400595 if status := p.Corpus.Analysis.Status(); status != "" {
596 buf.WriteString("<a href='/lib/godoc/analysis/help.html'>Static analysis features</a> ")
597 // TODO(adonovan): show analysis status at per-file granularity.
598 fmt.Fprintf(&buf, "<span style='color: grey'>[%s]</span><br/>", htmlpkg.EscapeString(status))
Alan Donovan503140c2014-04-16 16:35:08 -0400599 }
600
Alan Donovan80c4f062014-03-14 18:58:22 -0400601 buf.WriteString("<pre>")
Alan Donovan99d45c02014-07-09 07:59:55 -0400602 formatGoSource(&buf, src, fi.Links, h, s)
Alan Donovan80c4f062014-03-14 18:58:22 -0400603 buf.WriteString("</pre>")
604 } else {
605 buf.WriteString("<pre>")
606 FormatText(&buf, src, 1, false, h, s)
607 buf.WriteString("</pre>")
608 }
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000609 fmt.Fprintf(&buf, `<p><a href="/%s?m=text">View as plain text</a></p>`, htmlpkg.EscapeString(relpath))
610
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000611 p.ServePage(w, Page{
Sina Siadat5128de72016-09-16 17:12:50 +0430612 Title: title,
613 SrcPath: relpath,
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000614 Tabtitle: relpath,
615 Body: buf.Bytes(),
616 })
617}
618
Alan Donovan80c4f062014-03-14 18:58:22 -0400619// formatGoSource HTML-escapes Go source text and writes it to w,
620// decorating it with the specified analysis links.
Alan Donovan80c4f062014-03-14 18:58:22 -0400621func formatGoSource(buf *bytes.Buffer, text []byte, links []analysis.Link, pattern string, selection Selection) {
Alan Donovan7f24a832014-04-09 15:41:18 -0400622 // Emit to a temp buffer so that we can add line anchors at the end.
623 saved, buf := buf, new(bytes.Buffer)
624
Alan Donovan80c4f062014-03-14 18:58:22 -0400625 var i int
626 var link analysis.Link // shared state of the two funcs below
627 segmentIter := func() (seg Segment) {
628 if i < len(links) {
629 link = links[i]
630 i++
631 seg = Segment{link.Start(), link.End()}
632 }
633 return
634 }
635 linkWriter := func(w io.Writer, offs int, start bool) {
636 link.Write(w, offs, start)
637 }
638
639 comments := tokenSelection(text, token.COMMENT)
640 var highlights Selection
641 if pattern != "" {
642 highlights = regexpSelection(text, pattern)
643 }
644
645 FormatSelections(buf, text, linkWriter, segmentIter, selectionTag, comments, highlights, selection)
Alan Donovan7f24a832014-04-09 15:41:18 -0400646
647 // Now copy buf to saved, adding line anchors.
648
649 // The lineSelection mechanism can't be composed with our
650 // linkWriter, so we have to add line spans as another pass.
651 n := 1
652 for _, line := range bytes.Split(buf.Bytes(), []byte("\n")) {
Kevin Burke75e5ff32017-04-21 19:13:36 -0700653 // The line numbers are inserted into the document via a CSS ::before
654 // pseudo-element. This prevents them from being copied when users
655 // highlight and copy text.
656 // ::before is supported in 98% of browsers: https://caniuse.com/#feat=css-gencontent
657 // This is also the trick Github uses to hide line numbers.
658 //
659 // The first tab for the code snippet needs to start in column 9, so
660 // it indents a full 8 spaces, hence the two nbsp's. Otherwise the tab
Kevin Burke4bc20fc2018-08-27 16:02:08 -0600661 // character only indents a short amount.
662 //
663 // Due to rounding and font width Firefox might not treat 8 rendered
664 // characters as 8 characters wide, and subsequently may treat the tab
665 // character in the 9th position as moving the width from (7.5 or so) up
666 // to 8. See
667 // https://github.com/webcompat/web-bugs/issues/17530#issuecomment-402675091
668 // for a fuller explanation. The solution is to add a CSS class to
669 // explicitly declare the width to be 8 characters.
Agniva De Sarkerce871d12018-02-14 18:36:10 +0530670 fmt.Fprintf(saved, `<span id="L%d" class="ln">%6d&nbsp;&nbsp;</span>`, n, n)
Alan Donovan7f24a832014-04-09 15:41:18 -0400671 n++
672 saved.Write(line)
673 saved.WriteByte('\n')
674 }
Alan Donovan80c4f062014-03-14 18:58:22 -0400675}
676
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000677func (p *Presentation) serveDirectory(w http.ResponseWriter, r *http.Request, abspath, relpath string) {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000678 if redirect(w, r) {
679 return
680 }
681
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000682 list, err := p.Corpus.fs.ReadDir(abspath)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000683 if err != nil {
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000684 p.ServeError(w, r, relpath, err)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000685 return
686 }
687
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000688 p.ServePage(w, Page{
Sina Siadat5128de72016-09-16 17:12:50 +0430689 Title: "Directory",
690 SrcPath: relpath,
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000691 Tabtitle: relpath,
Brad Fitzpatrick66f0d6e2013-07-18 13:51:17 +1000692 Body: applyTemplate(p.DirlistHTML, "dirlistHTML", list),
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000693 })
694}
695
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000696func (p *Presentation) ServeHTMLDoc(w http.ResponseWriter, r *http.Request, abspath, relpath string) {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000697 // get HTML body contents
Russ Cox5bd3da92020-08-28 12:33:28 -0400698 isMarkdown := false
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000699 src, err := vfs.ReadFile(p.Corpus.fs, abspath)
Russ Cox5bd3da92020-08-28 12:33:28 -0400700 if err != nil && strings.HasSuffix(abspath, ".html") {
701 if md, errMD := vfs.ReadFile(p.Corpus.fs, strings.TrimSuffix(abspath, ".html")+".md"); errMD == nil {
702 src = md
703 isMarkdown = true
704 err = nil
705 }
706 }
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000707 if err != nil {
708 log.Printf("ReadFile: %s", err)
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000709 p.ServeError(w, r, relpath, err)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000710 return
711 }
712
713 // if it begins with "<!DOCTYPE " assume it is standalone
714 // html that doesn't need the template wrapping.
715 if bytes.HasPrefix(src, doctype) {
716 w.Write(src)
717 return
718 }
719
720 // if it begins with a JSON blob, read in the metadata.
721 meta, src, err := extractMetadata(src)
722 if err != nil {
723 log.Printf("decoding metadata %s: %v", relpath, err)
724 }
725
Andrew Gerrand1330b282015-09-02 09:49:30 +1000726 page := Page{
727 Title: meta.Title,
728 Subtitle: meta.Subtitle,
Andrew Gerrand1330b282015-09-02 09:49:30 +1000729 }
730
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000731 // evaluate as template if indicated
732 if meta.Template {
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000733 tmpl, err := template.New("main").Funcs(p.TemplateFuncs()).Parse(string(src))
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000734 if err != nil {
735 log.Printf("parsing template %s: %v", relpath, err)
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000736 p.ServeError(w, r, relpath, err)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000737 return
738 }
739 var buf bytes.Buffer
Andrew Gerrand1330b282015-09-02 09:49:30 +1000740 if err := tmpl.Execute(&buf, page); err != nil {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000741 log.Printf("executing template %s: %v", relpath, err)
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000742 p.ServeError(w, r, relpath, err)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000743 return
744 }
745 src = buf.Bytes()
746 }
747
Russ Cox5bd3da92020-08-28 12:33:28 -0400748 // Apply markdown as indicated.
749 // (Note template applies before Markdown.)
750 if isMarkdown {
751 html, err := renderMarkdown(src)
752 if err != nil {
753 log.Printf("executing markdown %s: %v", relpath, err)
754 p.ServeError(w, r, relpath, err)
755 return
756 }
757 src = html
758 }
759
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000760 // if it's the language spec, add tags to EBNF productions
761 if strings.HasSuffix(abspath, "go_spec.html") {
762 var buf bytes.Buffer
763 Linkify(&buf, src)
764 src = buf.Bytes()
765 }
766
Andrew Gerrand1330b282015-09-02 09:49:30 +1000767 page.Body = src
768 p.ServePage(w, page)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000769}
770
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000771func (p *Presentation) ServeFile(w http.ResponseWriter, r *http.Request) {
772 p.serveFile(w, r)
773}
774
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000775func (p *Presentation) serveFile(w http.ResponseWriter, r *http.Request) {
Russ Cox97606e32020-08-27 10:42:22 -0400776 if strings.HasSuffix(r.URL.Path, "/index.html") {
777 // We'll show index.html for the directory.
778 // Use the dir/ version as canonical instead of dir/index.html.
779 http.Redirect(w, r, r.URL.Path[0:len(r.URL.Path)-len("index.html")], http.StatusMovedPermanently)
780 return
781 }
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000782
783 // Check to see if we need to redirect or serve another file.
Russ Cox97606e32020-08-27 10:42:22 -0400784 relpath := r.URL.Path
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000785 if m := p.Corpus.MetadataFor(relpath); m != nil {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000786 if m.Path != relpath {
787 // Redirect to canonical path.
788 http.Redirect(w, r, m.Path, http.StatusMovedPermanently)
789 return
790 }
791 // Serve from the actual filesystem path.
792 relpath = m.filePath
793 }
794
795 abspath := relpath
796 relpath = relpath[1:] // strip leading slash
797
798 switch pathpkg.Ext(relpath) {
799 case ".html":
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000800 p.ServeHTMLDoc(w, r, abspath, relpath)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000801 return
802
803 case ".go":
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000804 p.serveTextFile(w, r, abspath, relpath, "Source file")
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000805 return
806 }
807
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000808 dir, err := p.Corpus.fs.Lstat(abspath)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000809 if err != nil {
810 log.Print(err)
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000811 p.ServeError(w, r, relpath, err)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000812 return
813 }
814
815 if dir != nil && dir.IsDir() {
816 if redirect(w, r) {
817 return
818 }
Russ Cox5bd3da92020-08-28 12:33:28 -0400819 index := pathpkg.Join(abspath, "index.html")
820 if util.IsTextFile(p.Corpus.fs, index) || util.IsTextFile(p.Corpus.fs, pathpkg.Join(abspath, "index.md")) {
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000821 p.ServeHTMLDoc(w, r, index, index)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000822 return
823 }
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000824 p.serveDirectory(w, r, abspath, relpath)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000825 return
826 }
827
Brad Fitzpatrick5395cfe2013-07-18 13:14:09 +1000828 if util.IsTextFile(p.Corpus.fs, abspath) {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000829 if redirectFile(w, r) {
830 return
831 }
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000832 p.serveTextFile(w, r, abspath, relpath, "Text file")
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000833 return
834 }
835
Brad Fitzpatrick705bb7f2013-07-19 10:27:53 +1000836 p.fileServer.ServeHTTP(w, r)
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000837}
838
Brad Fitzpatrick4fc63232013-07-18 09:52:45 +1000839func (p *Presentation) ServeText(w http.ResponseWriter, text []byte) {
Brad Fitzpatrickca3319f2013-07-17 17:17:12 +1000840 w.Header().Set("Content-Type", "text/plain; charset=utf-8")
841 w.Write(text)
842}
Alan Donovan80c4f062014-03-14 18:58:22 -0400843
844func marshalJSON(x interface{}) []byte {
845 var data []byte
846 var err error
847 const indentJSON = false // for easier debugging
848 if indentJSON {
849 data, err = json.MarshalIndent(x, "", " ")
850 } else {
851 data, err = json.Marshal(x)
852 }
853 if err != nil {
854 panic(fmt.Sprintf("json.Marshal failed: %s", err))
855 }
856 return data
857}