2020-07-29 17:11:41 +08:00
|
|
|
package docgen
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"html/template"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
2020-08-08 16:40:10 +08:00
|
|
|
"github.com/tal-tech/go-zero/core/stringx"
|
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/api/gogen"
|
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/api/spec"
|
|
|
|
"github.com/tal-tech/go-zero/tools/goctl/api/util"
|
2020-07-29 17:11:41 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
markdownTemplate = `
|
|
|
|
### {{.index}}. {{.routeComment}}
|
|
|
|
|
|
|
|
1. 路由定义
|
|
|
|
|
|
|
|
- Url: {{.uri}}
|
|
|
|
- Method: {{.method}}
|
|
|
|
- Request: {{.requestType}}
|
|
|
|
- Response: {{.responseType}}
|
|
|
|
|
|
|
|
|
|
|
|
2. 类型定义
|
|
|
|
|
|
|
|
{{.responseContent}}
|
|
|
|
|
|
|
|
`
|
|
|
|
)
|
|
|
|
|
|
|
|
func genDoc(api *spec.ApiSpec, dir string, filename string) error {
|
|
|
|
fp, _, err := util.MaybeCreateFile(dir, "", filename)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer fp.Close()
|
|
|
|
|
|
|
|
var builder strings.Builder
|
2020-11-13 23:01:19 +08:00
|
|
|
for index, route := range api.Service.Routes() {
|
2021-01-11 15:10:51 +08:00
|
|
|
routeComment := route.JoinedDoc()
|
2020-07-29 17:11:41 +08:00
|
|
|
if len(routeComment) == 0 {
|
|
|
|
routeComment = "N/A"
|
|
|
|
}
|
|
|
|
|
|
|
|
responseContent, err := responseBody(api, route)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
t := template.Must(template.New("markdownTemplate").Parse(markdownTemplate))
|
|
|
|
var tmplBytes bytes.Buffer
|
|
|
|
err = t.Execute(&tmplBytes, map[string]string{
|
|
|
|
"index": strconv.Itoa(index + 1),
|
|
|
|
"routeComment": routeComment,
|
|
|
|
"method": strings.ToUpper(route.Method),
|
|
|
|
"uri": route.Path,
|
2021-01-11 15:10:51 +08:00
|
|
|
"requestType": "`" + stringx.TakeOne(route.RequestTypeName(), "-") + "`",
|
|
|
|
"responseType": "`" + stringx.TakeOne(route.ResponseTypeName(), "-") + "`",
|
2020-07-29 17:11:41 +08:00
|
|
|
"responseContent": responseContent,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
builder.Write(tmplBytes.Bytes())
|
|
|
|
}
|
|
|
|
_, err = fp.WriteString(strings.Replace(builder.String(), """, `"`, -1))
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func responseBody(api *spec.ApiSpec, route spec.Route) (string, error) {
|
2021-01-11 15:10:51 +08:00
|
|
|
if len(route.ResponseTypeName()) == 0 {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var tps = make([]spec.Type, 0)
|
|
|
|
tps = append(tps, route.ResponseType)
|
|
|
|
if definedType, ok := route.ResponseType.(spec.DefineStruct); ok {
|
|
|
|
associatedTypes(definedType, &tps)
|
|
|
|
}
|
2020-07-29 17:11:41 +08:00
|
|
|
value, err := gogen.BuildTypes(tps)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
2020-07-29 17:11:41 +08:00
|
|
|
return fmt.Sprintf("\n\n```golang\n%s\n```\n", value), nil
|
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
|
|
|
func associatedTypes(tp spec.DefineStruct, tps *[]spec.Type) {
|
|
|
|
*tps = append(*tps, tp)
|
|
|
|
for _, item := range tp.Members {
|
|
|
|
if definedType, ok := item.Type.(spec.DefineStruct); ok {
|
|
|
|
associatedTypes(definedType, tps)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|