From 24f7e3aaaf4a6e94d2fc05a7ef0fac4bb80b7e01 Mon Sep 17 00:00:00 2001 From: Luc Talatinian <102624213+lucix-aws@users.noreply.github.com> Date: Tue, 16 Apr 2024 11:51:07 -0400 Subject: [PATCH] cmd/docextractor: generate enum and union variants as type members (#2603) --- .../repotools/cmd/docextractor/extract.go | 178 ++++++++++++------ internal/repotools/cmd/docextractor/index.go | 16 +- .../repotools/cmd/docextractor/jewelry.go | 3 + 3 files changed, 140 insertions(+), 57 deletions(-) diff --git a/internal/repotools/cmd/docextractor/extract.go b/internal/repotools/cmd/docextractor/extract.go index ea0d3ff6436..d058fc90a9e 100644 --- a/internal/repotools/cmd/docextractor/extract.go +++ b/internal/repotools/cmd/docextractor/extract.go @@ -40,10 +40,11 @@ func Extract(servicePath string, serviceDir fs.DirEntry, items map[string]jewelr } index := astIndex{ - Types: map[string]*ast.TypeSpec{}, - Functions: map[string]*ast.FuncDecl{}, - Fields: map[string]*ast.Field{}, - Other: []*ast.GenDecl{}, + Types: map[string]*ast.TypeSpec{}, + Functions: map[string]*ast.FuncDecl{}, + Fields: map[string]*ast.Field{}, + StringEnumConsts: map[string]string{}, + Other: []*ast.GenDecl{}, } for _, p := range directory { @@ -56,7 +57,7 @@ func Extract(servicePath string, serviceDir fs.DirEntry, items map[string]jewelr indexFromAst(p, &index) - err = extractTypes(packageName, index.Types, items) + err = extractTypes(d.Name(), packageName, index, items) if err != nil { log.Fatal(err) } @@ -74,11 +75,13 @@ func Extract(servicePath string, serviceDir fs.DirEntry, items map[string]jewelr } // extractType iterates through -func extractTypes(packageName string, types map[string]*ast.TypeSpec, items map[string]jewelryItem) error { +func extractTypes(pkg, module string, index astIndex, items map[string]jewelryItem) error { + types := index.Types for kt, vt := range types { typeName := vt.Name.Name item := jewelryItem{ + Package: pkg, Name: typeName, Summary: formatComment(vt.Doc), Members: []jewelryItem{}, @@ -87,29 +90,15 @@ func extractTypes(packageName string, types map[string]*ast.TypeSpec, items map[ Params: []jewelryParam{}, BreadCrumbs: []breadCrumb{ { - Name: packageName, + Name: module, 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 { + switch st := vt.Type.(type) { + case *ast.StructType: item.Type = jewelryItemKindStruct bc := item.BreadCrumbs bc = append(bc, breadCrumb{ @@ -120,9 +109,6 @@ func extractTypes(packageName string, types map[string]*ast.TypeSpec, items map[ 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++ { @@ -141,7 +127,7 @@ func extractTypes(packageName string, types map[string]*ast.TypeSpec, items map[ fieldItem.Type = jewelryItemKindField fieldItem.BreadCrumbs = []breadCrumb{ { - Name: packageName, + Name: module, Kind: jewelryItemKindPackage, }, { @@ -154,20 +140,76 @@ func extractTypes(packageName string, types map[string]*ast.TypeSpec, items map[ }, } fieldItem.Signature = typeSignature{ - Signature: toSignature(vf.Type, packageName), + Signature: toSignature(vf.Type, module), // Location is unused - links have to be embedded in signature } members = append(members, fieldItem) } } + case *ast.Ident: + if st.Name != "string" { + continue + } + + // probably an enum, map its variants to members + item.Type = jewelryItemKindEnum + item.BreadCrumbs = append(item.BreadCrumbs, breadCrumb{ + Name: typeName, + Kind: jewelryItemKindEnum, + }) + for name, value := range index.StringEnumConsts { + if strings.HasPrefix(name, typeName) { // good enough + members = append(members, jewelryItem{ + Name: name, + Signature: typeSignature{Signature: typeName}, + Summary: value, + }) + } + } + case *ast.InterfaceType: + if !isProbablyUnion(typeName, st) { + continue + } + + item.Type = jewelryItemKindUnion + item.BreadCrumbs = append(item.BreadCrumbs, breadCrumb{ + Name: typeName, + Kind: jewelryItemKindUnion, + }) + for name, typ := range index.Types { + if strings.HasPrefix(name, typeName+"Member") { + members = append(members, jewelryItem{ + Name: name, + Signature: typeSignature{ + Signature: fmt.Sprintf("[%s](-aws-sdk-client-%s!%s:Struct)", name, module, name), + }, + Summary: formatComment(typ.Doc), + }) + } + } + default: + continue } + item.Members = members items[kt] = item } return nil } +func isProbablyUnion(name string, i *ast.InterfaceType) bool { + for _, field := range i.Methods.List { + if len(field.Names) == 0 { + continue + } + if field.Names[0].Name == "is"+name { + return true + } + } + return false +} + // We've already converted the model's HTML to Go docs, now for ref docs we // must convert back. We can't use the model's original docs directly because // that doesn't include extra content we may inject at codegen. @@ -260,7 +302,7 @@ func toSignature(v ast.Expr, pkg string) string { case *ast.FuncType: return toFuncSignature(vv, pkg) default: - return "" + return fmt.Sprintf("[unhandled %T]", v) } } @@ -359,35 +401,65 @@ func extractFunctions(packageName string, types map[string]*ast.TypeSpec, functi } members := i.Members - members = append(members, - jewelryItem{ - Type: jewelryItemKindMethod, - Name: methodName, - Members: []jewelryItem{}, - Tags: []string{}, - OtherBlocks: map[string]string{}, - Params: params, - Returns: returns, - Summary: formatComment(vf.Doc), - BreadCrumbs: []breadCrumb{ - { - Name: packageName, - Kind: jewelryItemKindPackage, - }, - { - Name: receiverName, - Kind: jewelryItemKindStruct, - }, - { - Name: methodName, - Kind: jewelryItemKindMethod, + + // without proper runtime documentation, we have to bridge the gap to + // event payloads for now + if vf.Name.Name == "GetStream" && isInputOutput(receiverName) { + stream := strings.TrimSuffix(receiverName, whichSuffix(receiverName)) + "EventStream" + members = append(members, jewelryItem{ + Name: "(event stream payload)", + Summary: "The event streaming payload union for this structure.", + Signature: typeSignature{ + Signature: fmt.Sprintf("[%s](-aws-sdk-client-%s!%s:Union)", stream, packageName, stream), + }, + }) + } else { + members = append(members, + jewelryItem{ + Type: jewelryItemKindMethod, + Name: methodName, + Members: []jewelryItem{}, + Tags: []string{}, + OtherBlocks: map[string]string{}, + Params: params, + Returns: returns, + Summary: formatComment(vf.Doc), + BreadCrumbs: []breadCrumb{ + { + Name: packageName, + Kind: jewelryItemKindPackage, + }, + { + Name: receiverName, + Kind: jewelryItemKindStruct, + }, + { + Name: methodName, + Kind: jewelryItemKindMethod, + }, }, }, - }, - ) + ) + } + i.Members = members items[receiverName] = i } return nil } + +// whether there's "Input" or "Output" on a structure +func isInputOutput(name string) bool { + return strings.HasSuffix(name, "Input") || strings.HasSuffix(name, "Output") +} + +// "Input" or "Output" on a structure +func whichSuffix(name string) string { + if strings.HasSuffix(name, "Input") { + return "Input" + } else if strings.HasSuffix(name, "Output") { + return "Output" + } + panic("expected -Input or -Output suffix") +} diff --git a/internal/repotools/cmd/docextractor/index.go b/internal/repotools/cmd/docextractor/index.go index 78f8aac65b6..69f9d60f12e 100644 --- a/internal/repotools/cmd/docextractor/index.go +++ b/internal/repotools/cmd/docextractor/index.go @@ -6,10 +6,11 @@ import ( ) type astIndex struct { - Types map[string]*ast.TypeSpec - Functions map[string]*ast.FuncDecl - Fields map[string]*ast.Field - Other []*ast.GenDecl + Types map[string]*ast.TypeSpec + Functions map[string]*ast.FuncDecl + Fields map[string]*ast.Field + StringEnumConsts map[string]string + Other []*ast.GenDecl } func indexFromAst(p *ast.Package, index *astIndex) { @@ -81,6 +82,13 @@ func indexFromAst(p *ast.Package, index *astIndex) { } index.Types[name] = xt + } else if x.Tok == token.CONST { + for _, spec := range x.Specs { + vs := spec.(*ast.ValueSpec) + if vl, ok := vs.Values[0].(*ast.BasicLit); ok { + index.StringEnumConsts[vs.Names[0].Name] = vl.Value + } + } } else { index.Other = append(index.Other, x) } diff --git a/internal/repotools/cmd/docextractor/jewelry.go b/internal/repotools/cmd/docextractor/jewelry.go index 6c1c6567bc2..37ebc9b82e1 100644 --- a/internal/repotools/cmd/docextractor/jewelry.go +++ b/internal/repotools/cmd/docextractor/jewelry.go @@ -9,6 +9,8 @@ const ( jewelryItemKindFunc = "Function" jewelryItemKindMethod = "Method" jewelryItemKindField = "Field" + jewelryItemKindEnum = "Enum" + jewelryItemKindUnion = "Union" jewelryItemKindOther = "Other" ) @@ -30,6 +32,7 @@ type jewelryParam struct { } type jewelryItem struct { + Package string `json:"package"` Name string `json:"name"` Summary string `json:"summary"` Type jewelryItemKind `json:"type"`