2024-12-18 16:07:09 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"os"
|
2025-01-02 17:24:18 +00:00
|
|
|
"path"
|
2024-12-18 16:07:09 +00:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/databricks/cli/bundle/config"
|
2025-01-02 15:40:06 +00:00
|
|
|
"github.com/databricks/cli/bundle/internal/annotation"
|
2024-12-18 16:07:09 +00:00
|
|
|
"github.com/databricks/cli/libs/jsonschema"
|
|
|
|
)
|
|
|
|
|
2025-01-15 15:48:07 +00:00
|
|
|
const (
|
|
|
|
rootFileName = "reference.md"
|
|
|
|
resourcesFileName = "resources.md"
|
|
|
|
)
|
|
|
|
|
2024-12-18 16:07:09 +00:00
|
|
|
func main() {
|
|
|
|
if len(os.Args) != 3 {
|
|
|
|
fmt.Println("Usage: go run main.go <annotation-file> <output-file>")
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
2025-01-02 17:24:18 +00:00
|
|
|
annotationDir := os.Args[1]
|
|
|
|
docsDir := os.Args[2]
|
|
|
|
outputDir := path.Join(docsDir, "output")
|
2025-01-15 15:48:07 +00:00
|
|
|
templatesDir := path.Join(docsDir, "templates")
|
2025-01-02 17:24:18 +00:00
|
|
|
|
|
|
|
if _, err := os.Stat(outputDir); os.IsNotExist(err) {
|
|
|
|
if err := os.MkdirAll(outputDir, 0o755); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
2024-12-18 16:07:09 +00:00
|
|
|
|
2025-01-15 15:48:07 +00:00
|
|
|
rootHeader, err := os.ReadFile(path.Join(templatesDir, rootFileName))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
err = generateDocs(
|
2025-01-02 17:24:18 +00:00
|
|
|
[]string{path.Join(annotationDir, "annotations.yml")},
|
|
|
|
path.Join(outputDir, rootFileName),
|
|
|
|
reflect.TypeOf(config.Root{}),
|
2025-01-15 15:48:07 +00:00
|
|
|
string(rootHeader),
|
2025-01-02 17:24:18 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2025-01-15 15:48:07 +00:00
|
|
|
resourcesHeader, err := os.ReadFile(path.Join(templatesDir, resourcesFileName))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2025-01-02 17:24:18 +00:00
|
|
|
err = generateDocs(
|
2025-01-10 11:52:20 +00:00
|
|
|
[]string{path.Join(annotationDir, "annotations_openapi.yml"), path.Join(annotationDir, "annotations_openapi_overrides.yml"), path.Join(annotationDir, "annotations.yml")},
|
2025-01-02 17:24:18 +00:00
|
|
|
path.Join(outputDir, resourcesFileName),
|
|
|
|
reflect.TypeOf(config.Resources{}),
|
2025-01-15 15:48:07 +00:00
|
|
|
string(resourcesHeader),
|
2025-01-02 17:24:18 +00:00
|
|
|
)
|
2024-12-18 16:07:09 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-02 17:24:18 +00:00
|
|
|
func generateDocs(inputPaths []string, outputPath string, rootType reflect.Type, header string) error {
|
|
|
|
annotations, err := annotation.LoadAndMerge(inputPaths)
|
2024-12-18 16:07:09 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2025-01-15 18:47:39 +00:00
|
|
|
schemas := map[string]*jsonschema.Schema{}
|
2024-12-18 19:53:38 +00:00
|
|
|
customFields := map[string]bool{}
|
2024-12-18 16:07:09 +00:00
|
|
|
|
2025-01-02 17:24:18 +00:00
|
|
|
s, err := jsonschema.FromType(rootType, []func(reflect.Type, jsonschema.Schema) jsonschema.Schema{
|
2024-12-18 16:07:09 +00:00
|
|
|
func(typ reflect.Type, s jsonschema.Schema) jsonschema.Schema {
|
2024-12-18 19:53:38 +00:00
|
|
|
_, isCustomField := annotations[jsonschema.TypePath(typ)]
|
|
|
|
if isCustomField {
|
|
|
|
customFields[jsonschema.TypePath(typ)] = true
|
|
|
|
}
|
2024-12-18 16:07:09 +00:00
|
|
|
|
|
|
|
refPath := getPath(typ)
|
|
|
|
shouldHandle := strings.HasPrefix(refPath, "github.com")
|
|
|
|
if !shouldHandle {
|
2025-01-15 18:47:39 +00:00
|
|
|
schemas[jsonschema.TypePath(typ)] = &s
|
2024-12-18 16:07:09 +00:00
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
a := annotations[refPath]
|
|
|
|
if a == nil {
|
2025-01-02 15:40:06 +00:00
|
|
|
a = map[string]annotation.Descriptor{}
|
2024-12-18 16:07:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
rootTypeAnnotation, ok := a["_"]
|
|
|
|
if ok {
|
|
|
|
assignAnnotation(&s, rootTypeAnnotation)
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range s.Properties {
|
|
|
|
assignAnnotation(v, a[k])
|
|
|
|
}
|
|
|
|
|
2025-01-15 18:47:39 +00:00
|
|
|
schemas[jsonschema.TypePath(typ)] = &s
|
2024-12-18 16:07:09 +00:00
|
|
|
return s
|
|
|
|
},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2024-12-18 19:53:38 +00:00
|
|
|
nodes := getNodes(s, schemas, customFields)
|
2025-01-02 17:24:18 +00:00
|
|
|
err = buildMarkdown(nodes, outputPath, header)
|
2024-12-18 16:07:09 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getPath(typ reflect.Type) string {
|
|
|
|
return typ.PkgPath() + "." + typ.Name()
|
|
|
|
}
|
|
|
|
|
2025-01-02 15:40:06 +00:00
|
|
|
func assignAnnotation(s *jsonschema.Schema, a annotation.Descriptor) {
|
|
|
|
if a.Description != "" && a.Description != annotation.Placeholder {
|
2024-12-18 16:07:09 +00:00
|
|
|
s.Description = a.Description
|
|
|
|
}
|
|
|
|
if a.MarkdownDescription != "" {
|
|
|
|
s.MarkdownDescription = a.MarkdownDescription
|
|
|
|
}
|
2025-01-02 15:13:24 +00:00
|
|
|
if a.MarkdownExamples != "" {
|
|
|
|
s.Examples = []any{a.MarkdownExamples}
|
|
|
|
}
|
2024-12-18 16:07:09 +00:00
|
|
|
}
|