mirror of https://github.com/databricks/cli.git
218 lines
4.9 KiB
Go
218 lines
4.9 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"reflect"
|
|
"strings"
|
|
|
|
"github.com/databricks/cli/bundle/internal/annotation"
|
|
"github.com/databricks/cli/libs/jsonschema"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
type Components struct {
|
|
Schemas map[string]jsonschema.Schema `json:"schemas,omitempty"`
|
|
}
|
|
|
|
type Specification struct {
|
|
Components Components `json:"components"`
|
|
}
|
|
|
|
type openapiParser struct {
|
|
ref map[string]jsonschema.Schema
|
|
}
|
|
|
|
const RootTypeKey = "_"
|
|
|
|
func newParser(path string) (*openapiParser, error) {
|
|
b, err := os.ReadFile(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
spec := Specification{}
|
|
err = json.Unmarshal(b, &spec)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
p := &openapiParser{}
|
|
p.ref = spec.Components.Schemas
|
|
return p, nil
|
|
}
|
|
|
|
// This function checks if the input type:
|
|
// 1. Is a Databricks Go SDK type.
|
|
// 2. Has a Databricks Go SDK type embedded in it.
|
|
//
|
|
// If the above conditions are met, the function returns the JSON schema
|
|
// corresponding to the Databricks Go SDK type from the OpenAPI spec.
|
|
func (p *openapiParser) findRef(typ reflect.Type) (jsonschema.Schema, bool) {
|
|
typs := []reflect.Type{typ}
|
|
|
|
// Check for embedded Databricks Go SDK types.
|
|
if typ.Kind() == reflect.Struct {
|
|
for i := range typ.NumField() {
|
|
if !typ.Field(i).Anonymous {
|
|
continue
|
|
}
|
|
|
|
// Deference current type if it's a pointer.
|
|
ctyp := typ.Field(i).Type
|
|
for ctyp.Kind() == reflect.Ptr {
|
|
ctyp = ctyp.Elem()
|
|
}
|
|
|
|
typs = append(typs, ctyp)
|
|
}
|
|
}
|
|
|
|
for _, ctyp := range typs {
|
|
// Skip if it's not a Go SDK type.
|
|
if !strings.HasPrefix(ctyp.PkgPath(), "github.com/databricks/databricks-sdk-go") {
|
|
continue
|
|
}
|
|
|
|
pkgName := path.Base(ctyp.PkgPath())
|
|
k := fmt.Sprintf("%s.%s", pkgName, ctyp.Name())
|
|
|
|
// Skip if the type is not in the openapi spec.
|
|
_, ok := p.ref[k]
|
|
if !ok {
|
|
k = mapIncorrectTypNames(k)
|
|
_, ok = p.ref[k]
|
|
if !ok {
|
|
continue
|
|
}
|
|
}
|
|
|
|
// Return the first Go SDK type found in the openapi spec.
|
|
return p.ref[k], true
|
|
}
|
|
|
|
return jsonschema.Schema{}, false
|
|
}
|
|
|
|
// Fix inconsistent type names between the Go SDK and the OpenAPI spec.
|
|
// E.g. "serving.PaLmConfig" in the Go SDK is "serving.PaLMConfig" in the OpenAPI spec.
|
|
func mapIncorrectTypNames(ref string) string {
|
|
switch ref {
|
|
case "serving.PaLmConfig":
|
|
return "serving.PaLMConfig"
|
|
case "serving.OpenAiConfig":
|
|
return "serving.OpenAIConfig"
|
|
case "serving.GoogleCloudVertexAiConfig":
|
|
return "serving.GoogleCloudVertexAIConfig"
|
|
case "serving.Ai21LabsConfig":
|
|
return "serving.AI21LabsConfig"
|
|
default:
|
|
return ref
|
|
}
|
|
}
|
|
|
|
// Use the OpenAPI spec to load descriptions for the given type.
|
|
func (p *openapiParser) extractAnnotations(typ reflect.Type, outputPath, overridesPath string) error {
|
|
annotations := annotation.File{}
|
|
overrides := annotation.File{}
|
|
|
|
b, err := os.ReadFile(overridesPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = yaml.Unmarshal(b, &overrides)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if overrides == nil {
|
|
overrides = annotation.File{}
|
|
}
|
|
|
|
_, err = jsonschema.FromType(typ, []func(reflect.Type, jsonschema.Schema) jsonschema.Schema{
|
|
func(typ reflect.Type, s jsonschema.Schema) jsonschema.Schema {
|
|
ref, ok := p.findRef(typ)
|
|
if !ok {
|
|
return s
|
|
}
|
|
|
|
basePath := getPath(typ)
|
|
pkg := map[string]annotation.Descriptor{}
|
|
annotations[basePath] = pkg
|
|
|
|
if ref.Description != "" || ref.Enum != nil {
|
|
pkg[RootTypeKey] = annotation.Descriptor{Description: ref.Description, Enum: ref.Enum}
|
|
}
|
|
|
|
for k := range s.Properties {
|
|
if refProp, ok := ref.Properties[k]; ok {
|
|
pkg[k] = annotation.Descriptor{Description: refProp.Description, Enum: refProp.Enum}
|
|
if refProp.Description == "" {
|
|
addEmptyOverride(k, basePath, overrides)
|
|
}
|
|
} else {
|
|
addEmptyOverride(k, basePath, overrides)
|
|
}
|
|
}
|
|
return s
|
|
},
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = saveYamlWithStyle(overridesPath, overrides)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = saveYamlWithStyle(outputPath, annotations)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = prependCommentToFile(outputPath, "# This file is auto-generated. DO NOT EDIT.\n")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func prependCommentToFile(outputPath, comment string) error {
|
|
b, err := os.ReadFile(outputPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
f, err := os.OpenFile(outputPath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0o644)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer f.Close()
|
|
|
|
_, err = f.WriteString(comment)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = f.Write(b)
|
|
return err
|
|
}
|
|
|
|
func addEmptyOverride(key, pkg string, overridesFile annotation.File) {
|
|
if overridesFile[pkg] == nil {
|
|
overridesFile[pkg] = map[string]annotation.Descriptor{}
|
|
}
|
|
|
|
overrides := overridesFile[pkg]
|
|
if overrides[key].Description == "" {
|
|
overrides[key] = annotation.Descriptor{Description: annotation.Placeholder}
|
|
}
|
|
|
|
a, ok := overrides[key]
|
|
if !ok {
|
|
a = annotation.Descriptor{}
|
|
}
|
|
if a.Description == "" {
|
|
a.Description = annotation.Placeholder
|
|
}
|
|
overrides[key] = a
|
|
}
|