Skip to content

Commit

Permalink
internal: api ref doc extractor (#2577)
Browse files Browse the repository at this point in the history
  • Loading branch information
lucix-aws authored Mar 25, 2024
1 parent 64cb287 commit 1b28ac4
Show file tree
Hide file tree
Showing 6 changed files with 643 additions and 0 deletions.
295 changes: 295 additions & 0 deletions internal/repotools/cmd/docextractor/extract.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,295 @@
package main

import (
"fmt"
"go/ast"
"go/parser"
"go/token"
"io/fs"
"log"
"path/filepath"
"strings"
)

// Extract will extract documentation from serviceDir and all sub-directories,
// populate items with newly-created JewelryItem(s).
// The overall strategy is to do a
func Extract(servicePath string, serviceDir fs.DirEntry, items map[string]jewelryItem) {

if serviceDir.Name() == "service" {
return
}

packageName := serviceDir.Name()

filepath.WalkDir(servicePath,
func(path string, d fs.DirEntry, e error) error {
if !d.IsDir() {
return nil
}

isInternal := strings.Count(path, "/internal") > 0
if isInternal {
return nil
}

fset := token.NewFileSet()
directory, err := parser.ParseDir(fset, path, nil, parser.ParseComments)
if err != nil {
panic(err)
}

index := astIndex{
Types: map[string]*ast.TypeSpec{},
Functions: map[string]*ast.FuncDecl{},
Fields: map[string]*ast.Field{},
Other: []*ast.GenDecl{},
}

for _, p := range directory {
removeTestFiles(p.Files)

packageItem, err := getPackageItem(packageName, p.Files)
if err == nil {
items["packageDocumentation"] = packageItem
}

indexFromAst(p, &index)

err = extractTypes(packageName, index.Types, items)
if err != nil {
log.Fatal(err)
}

err = extractFunctions(packageName, index.Types, index.Functions, items)
if err != nil {
log.Fatal(err)
}
}

return nil
})

serialize(packageName, items)
}

// extractType iterates through
func extractTypes(packageName string, types map[string]*ast.TypeSpec, items map[string]jewelryItem) error {
for kt, vt := range types {
summary := ""
if vt.Doc != nil {
summary = vt.Doc.Text()
}
typeName := vt.Name.Name

item := jewelryItem{
Name: typeName,
Summary: summary,
Members: []jewelryItem{},
Tags: []string{},
OtherBlocks: map[string]string{},
Params: []jewelryParam{},
BreadCrumbs: []breadCrumb{
{
Name: packageName,
Kind: jewelryItemKindPackage,
},
},
}
members := []jewelryItem{}

st, ok := vt.Type.(*ast.StructType)

if !ok {
item.Type = jewelryItemKindInterface

bc := item.BreadCrumbs
bc = append(bc, breadCrumb{
Name: typeName,
Kind: jewelryItemKindInterface,
})
item.BreadCrumbs = bc
item.Signature = typeSignature{
Signature: fmt.Sprintf("type %v interface", typeName),
}

} else {
item.Type = jewelryItemKindStruct
bc := item.BreadCrumbs
bc = append(bc, breadCrumb{
Name: typeName,
Kind: jewelryItemKindStruct,
})
item.BreadCrumbs = bc
item.Signature = typeSignature{
Signature: fmt.Sprintf("type %v struct", typeName),
}
}

if ok && st.Fields != nil && st.Fields.List != nil {
for _, vf := range st.Fields.List {
namesNum := len(vf.Names)
for i := 0; i < namesNum; i++ {
if !isExported(vf.Names[i].Name) {
break
}
fieldName := vf.Names[i].Name
var fieldItem jewelryItem
if vf.Doc == nil || vf.Doc.List == nil || vf.Doc.List[i] == nil {
fieldItem = jewelryItem{
Name: fieldName,
Tags: []string{},
OtherBlocks: map[string]string{},
Params: []jewelryParam{},
Members: []jewelryItem{},
Summary: "",
}

} else {
fieldItem = jewelryItem{
Name: fieldName,
Tags: []string{},
OtherBlocks: map[string]string{},
Params: []jewelryParam{},
Members: []jewelryItem{},
Summary: vf.Doc.List[i].Text,
}
}
fieldItem.Type = jewelryItemKindField
fieldItem.BreadCrumbs = []breadCrumb{
{
Name: packageName,
Kind: jewelryItemKindPackage,
},
{
Name: typeName,
Kind: jewelryItemKindStruct,
},
{
Name: fieldName,
Kind: jewelryItemKindField,
},
}
se, ok := vf.Type.(*ast.StarExpr)
if ok {
ident, ok := se.X.(*ast.Ident)
if ok {
fieldItem.Signature = typeSignature{
Signature: ident.Name,
}
}
}
members = append(members, fieldItem)
}
}
}
item.Members = members
items[kt] = item
}
return nil
}

func extractFunctions(packageName string, types map[string]*ast.TypeSpec, functions map[string]*ast.FuncDecl, items map[string]jewelryItem) error {
for _, vf := range functions {

// extract top-level functions
if vf.Recv == nil {
functionName := vf.Name.Name
items[functionName] = jewelryItem{
Type: jewelryItemKindFunc,
Name: functionName,
Tags: []string{},
OtherBlocks: map[string]string{},
Params: []jewelryParam{},
Members: []jewelryItem{},
Summary: vf.Doc.Text(),
BreadCrumbs: []breadCrumb{
{
Name: packageName,
Kind: jewelryItemKindPackage,
},
{
Name: functionName,
Kind: jewelryItemKindFunc,
},
},
}
continue
}
var receiverName string
switch r := vf.Recv.List[0].Type.(type) {
case *ast.StarExpr:
rName, _ := r.X.(*ast.Ident)
receiverName = rName.Name
case *ast.Ident:
receiverName = r.Name
}

// grab existing type
_, ok := types[receiverName]
if !ok {
// type doesnt exist
continue
}

methodName := vf.Name.Name

i := items[receiverName]

params := []jewelryParam{}
returns := ""

// extract operations
// assumes that all receiver methods on Client are
// service API operations except for the Options method.
if receiverName == "Client" && methodName != "Options" {
inputItem := items[fmt.Sprintf("%vInput", methodName)]
input := jewelryParam{
jewelryItem: jewelryItem{
Name: inputItem.Name,
Summary: inputItem.Summary,
Type: inputItem.Type,
Members: inputItem.Members,
BreadCrumbs: inputItem.BreadCrumbs,
Signature: inputItem.Signature,
},
IsOptional: false,
IsReadonly: false,
}
params = append(params, input)
returns = fmt.Sprintf("%vOutput", methodName)
}

members := i.Members
members = append(members,
jewelryItem{
Type: jewelryItemKindMethod,
Name: methodName,
Members: []jewelryItem{},
Tags: []string{},
OtherBlocks: map[string]string{},
Params: params,
Returns: returns,
Summary: vf.Doc.Text(),
BreadCrumbs: []breadCrumb{
{
Name: packageName,
Kind: jewelryItemKindPackage,
},
{
Name: receiverName,
Kind: jewelryItemKindStruct,
},
{
Name: methodName,
Kind: jewelryItemKindMethod,
},
},
},
)
i.Members = members
items[receiverName] = i
}

return nil
}
91 changes: 91 additions & 0 deletions internal/repotools/cmd/docextractor/index.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
package main

import (
"go/ast"
"go/token"
)

type astIndex struct {
Types map[string]*ast.TypeSpec
Functions map[string]*ast.FuncDecl
Fields map[string]*ast.Field
Other []*ast.GenDecl
}


func indexFromAst(p *ast.Package, index *astIndex) {
ast.Inspect(p, func(n ast.Node) bool {
switch x := n.(type) {
case *ast.FuncDecl:

// remove unexported items
if !isExported(x.Name.Name) {
break
}
name := x.Name.Name
index.Functions[name] = x

// use TypeSpec (over like StructType) because
// StructType doesnt have the name of the thing for some reason
// and TypeSpec contains the StructType obj as a field.
case *ast.TypeSpec:

if !isExported(x.Name.Name) {
break
}

// if a type exists AND it has a doc comment, then
// dont add anything -- were good.
// if not, then just add whatever.
name := x.Name.Name
if _, ok := index.Types[name]; ok && index.Types[name].Doc.Text() != "" {
break
}

index.Types[name] = x
case *ast.Field:
namesNum := len(x.Names)
for i := 0; i < namesNum; i++ {
if !isExported(x.Names[i].Name) {
break
}
name := x.Names[i].Name
index.Fields[name] = x
}
case *ast.GenDecl:

// for some reason, the same type will show up in the AST node list
// one with documentation and one without documentation
if x.Tok == token.TYPE {
xt, _ := x.Specs[0].(*ast.TypeSpec)

name := xt.Name.Name
if !isExported(name) {
break
}

// if a type exists AND it has a doc comment, then
// dont add anything -- were good.
// if not, then just add whatever.
if _, ok := index.Types[name]; ok && index.Types[name].Doc.Text() != "" {
break
}

// its a comment group, and each item in the list
// is a line
// summary := ""
if x.Doc != nil && x.Doc.List != nil {
xt.Doc = x.Doc
// for _, line := range x.Doc.List {
// summary += line.Text
// }
}

index.Types[name] = xt
} else {
index.Other = append(index.Other, x)
}
}
return true
})
}
Loading

0 comments on commit 1b28ac4

Please sign in to comment.