liyujie
2025-08-28 d9927380ed7c8366f762049be9f3fee225860833
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
// Copyright 2009 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 doc
 
import "go/ast"
 
type Filter func(string) bool
 
func matchFields(fields *ast.FieldList, f Filter) bool {
   if fields != nil {
       for _, field := range fields.List {
           for _, name := range field.Names {
               if f(name.Name) {
                   return true
               }
           }
       }
   }
   return false
}
 
func matchDecl(d *ast.GenDecl, f Filter) bool {
   for _, d := range d.Specs {
       switch v := d.(type) {
       case *ast.ValueSpec:
           for _, name := range v.Names {
               if f(name.Name) {
                   return true
               }
           }
       case *ast.TypeSpec:
           if f(v.Name.Name) {
               return true
           }
           switch t := v.Type.(type) {
           case *ast.StructType:
               if matchFields(t.Fields, f) {
                   return true
               }
           case *ast.InterfaceType:
               if matchFields(t.Methods, f) {
                   return true
               }
           }
       }
   }
   return false
}
 
func filterValues(a []*Value, f Filter) []*Value {
   w := 0
   for _, vd := range a {
       if matchDecl(vd.Decl, f) {
           a[w] = vd
           w++
       }
   }
   return a[0:w]
}
 
func filterFuncs(a []*Func, f Filter) []*Func {
   w := 0
   for _, fd := range a {
       if f(fd.Name) {
           a[w] = fd
           w++
       }
   }
   return a[0:w]
}
 
func filterTypes(a []*Type, f Filter) []*Type {
   w := 0
   for _, td := range a {
       n := 0 // number of matches
       if matchDecl(td.Decl, f) {
           n = 1
       } else {
           // type name doesn't match, but we may have matching consts, vars, factories or methods
           td.Consts = filterValues(td.Consts, f)
           td.Vars = filterValues(td.Vars, f)
           td.Funcs = filterFuncs(td.Funcs, f)
           td.Methods = filterFuncs(td.Methods, f)
           n += len(td.Consts) + len(td.Vars) + len(td.Funcs) + len(td.Methods)
       }
       if n > 0 {
           a[w] = td
           w++
       }
   }
   return a[0:w]
}
 
// Filter eliminates documentation for names that don't pass through the filter f.
// TODO(gri): Recognize "Type.Method" as a name.
//
func (p *Package) Filter(f Filter) {
   p.Consts = filterValues(p.Consts, f)
   p.Vars = filterValues(p.Vars, f)
   p.Types = filterTypes(p.Types, f)
   p.Funcs = filterFuncs(p.Funcs, f)
   p.Doc = "" // don't show top-level package doc
}