mirror of
				https://github.com/siderolabs/talos.git
				synced 2025-10-31 00:11:36 +01:00 
			
		
		
		
	Set the additional description fields for vscode/monaco/jetbrains editors. Strip the markdown formatting from the plain description. Additionally, fix the description of the field `aescbcEncryptionSecret`. Related to siderolabs/talos#6705. Signed-off-by: Utku Ozdemir <utku.ozdemir@siderolabs.com>
		
			
				
	
	
		
			295 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			295 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // This Source Code Form is subject to the terms of the Mozilla Public
 | |
| // License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
| // file, You can obtain one at http://mozilla.org/MPL/2.0/.
 | |
| 
 | |
| package main
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"log"
 | |
| 	"os"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/gomarkdown/markdown"
 | |
| 	"github.com/gomarkdown/markdown/html"
 | |
| 	"github.com/iancoleman/orderedmap"
 | |
| 	"github.com/invopop/jsonschema"
 | |
| 	"github.com/microcosm-cc/bluemonday"
 | |
| 	validatejsonschema "github.com/santhosh-tekuri/jsonschema/v5"
 | |
| 	"github.com/siderolabs/gen/slices"
 | |
| )
 | |
| 
 | |
| const ConfigSchemaURLFormat = "https://talos.dev/%s/schemas/v1alpha1_config.schema.json"
 | |
| 
 | |
| // SchemaWrapper wraps jsonschema.Schema to provide correct YAML unmarshalling using its internal JSON marshaller.
 | |
| type SchemaWrapper struct {
 | |
| 	jsonschema.Schema
 | |
| }
 | |
| 
 | |
| // UnmarshalYAML unmarshals the schema from YAML.
 | |
| //
 | |
| // This converts the YAML that was read from the comments to JSON,
 | |
| // then uses the custom JSON unmarshaler of the wrapped Schema.
 | |
| func (t *SchemaWrapper) UnmarshalYAML(unmarshal func(any) error) error {
 | |
| 	var data map[string]any
 | |
| 
 | |
| 	err := unmarshal(&data)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	jsonBytes, err := json.Marshal(data)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	return t.UnmarshalJSON(jsonBytes)
 | |
| }
 | |
| 
 | |
| type SchemaTypeInfo struct {
 | |
| 	typeName string
 | |
| 	ref      string
 | |
| }
 | |
| 
 | |
| type SchemaDefinitionInfo struct {
 | |
| 	typeInfo           SchemaTypeInfo
 | |
| 	arrayItemsTypeInfo *SchemaTypeInfo
 | |
| 	mapValueTypeInfo   *SchemaTypeInfo
 | |
| 	enumValues         []any
 | |
| }
 | |
| 
 | |
| func goTypeToTypeInfo(goType string) *SchemaTypeInfo {
 | |
| 	switch goType {
 | |
| 	case "string":
 | |
| 		return &SchemaTypeInfo{typeName: "string"}
 | |
| 	case "int", "int8", "int16", "int32", "int64", "uint", "uint8", "uint16", "uint32", "uint64":
 | |
| 		return &SchemaTypeInfo{typeName: "integer"}
 | |
| 	case "bool":
 | |
| 		return &SchemaTypeInfo{typeName: "boolean"}
 | |
| 	default:
 | |
| 		return &SchemaTypeInfo{ref: "#/$defs/" + goType}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func fieldToDefinitionInfo(field *Field) SchemaDefinitionInfo {
 | |
| 	goType := field.Type
 | |
| 
 | |
| 	if field.Text != nil {
 | |
| 		// skip enumifying boolean fields so that we don't pick up the "on", "off", "yes", "no" from the values
 | |
| 		if goType != "bool" && field.Text.Values != nil {
 | |
| 			enumValues := make([]any, 0, len(field.Text.Values))
 | |
| 
 | |
| 			for _, val := range field.Text.Values {
 | |
| 				enumValues = append(enumValues, val)
 | |
| 			}
 | |
| 
 | |
| 			return SchemaDefinitionInfo{enumValues: enumValues}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if strings.HasPrefix(goType, "[]") {
 | |
| 		return SchemaDefinitionInfo{
 | |
| 			typeInfo:           SchemaTypeInfo{typeName: "array"},
 | |
| 			arrayItemsTypeInfo: goTypeToTypeInfo(strings.TrimPrefix(goType, "[]")),
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if strings.HasPrefix(goType, "map[string]") {
 | |
| 		return SchemaDefinitionInfo{
 | |
| 			typeInfo:         SchemaTypeInfo{typeName: "object"},
 | |
| 			mapValueTypeInfo: goTypeToTypeInfo(strings.TrimPrefix(goType, "map[string]")),
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return SchemaDefinitionInfo{
 | |
| 		typeInfo: *goTypeToTypeInfo(goType),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func typeInfoToSchema(typeInfo *SchemaTypeInfo) *jsonschema.Schema {
 | |
| 	schema := jsonschema.Schema{}
 | |
| 
 | |
| 	if typeInfo.typeName != "" {
 | |
| 		schema.Type = typeInfo.typeName
 | |
| 	}
 | |
| 
 | |
| 	if typeInfo.ref != "" {
 | |
| 		schema.Ref = typeInfo.ref
 | |
| 	}
 | |
| 
 | |
| 	return &schema
 | |
| }
 | |
| 
 | |
| func fieldToSchema(field *Field) *jsonschema.Schema {
 | |
| 	schema := jsonschema.Schema{}
 | |
| 
 | |
| 	if field.Text != nil {
 | |
| 		// if there is an explicit schema, use it
 | |
| 		if field.Text.Schema != nil {
 | |
| 			schema = field.Text.Schema.Schema
 | |
| 		}
 | |
| 
 | |
| 		// if no title is provided on the explicit schema, grab it from the comment
 | |
| 		if schema.Title == "" {
 | |
| 			schema.Title = strings.ReplaceAll(field.Tag, "\\n", "\n")
 | |
| 		}
 | |
| 
 | |
| 		populateDescriptionFields(field, &schema)
 | |
| 
 | |
| 		// if an explicit schema was provided, return it
 | |
| 		if field.Text.Schema != nil {
 | |
| 			return &schema
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// schema was not explicitly provided, generate it from the comment
 | |
| 
 | |
| 	info := fieldToDefinitionInfo(field)
 | |
| 
 | |
| 	if info.typeInfo.ref != "" {
 | |
| 		schema.Ref = info.typeInfo.ref
 | |
| 	}
 | |
| 
 | |
| 	if info.enumValues != nil {
 | |
| 		schema.Enum = info.enumValues
 | |
| 	}
 | |
| 
 | |
| 	if info.typeInfo.typeName != "" {
 | |
| 		schema.Type = info.typeInfo.typeName
 | |
| 	}
 | |
| 
 | |
| 	if info.arrayItemsTypeInfo != nil {
 | |
| 		schema.Items = typeInfoToSchema(info.arrayItemsTypeInfo)
 | |
| 	}
 | |
| 
 | |
| 	if info.mapValueTypeInfo != nil {
 | |
| 		schema.PatternProperties = map[string]*jsonschema.Schema{
 | |
| 			".*": typeInfoToSchema(info.mapValueTypeInfo),
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return &schema
 | |
| }
 | |
| 
 | |
| func populateDescriptionFields(field *Field, schema *jsonschema.Schema) {
 | |
| 	if schema.Extras == nil {
 | |
| 		schema.Extras = make(map[string]any)
 | |
| 	}
 | |
| 
 | |
| 	markdownDescription := normalizeDescription(field.Text.Description)
 | |
| 
 | |
| 	htmlFlags := html.CommonFlags | html.HrefTargetBlank
 | |
| 	opts := html.RendererOptions{Flags: htmlFlags}
 | |
| 	renderer := html.NewRenderer(opts)
 | |
| 
 | |
| 	htmlDescription := string(markdown.ToHTML([]byte(markdownDescription), nil, renderer))
 | |
| 
 | |
| 	policy := bluemonday.StrictPolicy()
 | |
| 
 | |
| 	plaintextDescription := policy.Sanitize(htmlDescription)
 | |
| 
 | |
| 	// set description
 | |
| 	if schema.Description == "" {
 | |
| 		schema.Description = plaintextDescription
 | |
| 	}
 | |
| 
 | |
| 	// set markdownDescription for vscode/monaco editor
 | |
| 	if schema.Extras["markdownDescription"] == nil {
 | |
| 		schema.Extras["markdownDescription"] = markdownDescription
 | |
| 	}
 | |
| 
 | |
| 	// set htmlDescription for Jetbrains IDEs
 | |
| 	if schema.Extras["x-intellij-html-description"] == nil {
 | |
| 		schema.Extras["x-intellij-html-description"] = htmlDescription
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func structToSchema(st *Struct) *jsonschema.Schema {
 | |
| 	schema := jsonschema.Schema{
 | |
| 		Type:                 "object",
 | |
| 		AdditionalProperties: jsonschema.FalseSchema,
 | |
| 	}
 | |
| 
 | |
| 	properties := orderedmap.New()
 | |
| 
 | |
| 	for _, field := range st.Fields {
 | |
| 		if field.Tag == "" {
 | |
| 			// skip unknown/untagged field
 | |
| 			continue
 | |
| 		}
 | |
| 		properties.Set(field.Tag, fieldToSchema(field))
 | |
| 	}
 | |
| 
 | |
| 	schema.Properties = properties
 | |
| 
 | |
| 	return &schema
 | |
| }
 | |
| 
 | |
| func docToSchema(doc *Doc, schemaURL string) *jsonschema.Schema {
 | |
| 	schema := jsonschema.Schema{
 | |
| 		Version: jsonschema.Version,
 | |
| 		ID:      jsonschema.ID(schemaURL),
 | |
| 		Ref:     "#/$defs/Config",
 | |
| 	}
 | |
| 
 | |
| 	schema.Definitions = slices.ToMap(doc.Structs, func(st *Struct) (string, *jsonschema.Schema) {
 | |
| 		return st.Name, structToSchema(st)
 | |
| 	})
 | |
| 
 | |
| 	return &schema
 | |
| }
 | |
| 
 | |
| func renderSchema(doc *Doc, destinationFile, versionTagFile string) {
 | |
| 	version := readMajorMinorVersion(versionTagFile)
 | |
| 
 | |
| 	schemaURL := fmt.Sprintf(ConfigSchemaURLFormat, version)
 | |
| 
 | |
| 	schema := docToSchema(doc, schemaURL)
 | |
| 
 | |
| 	marshaled, err := json.MarshalIndent(schema, "", "  ")
 | |
| 	if err != nil {
 | |
| 		log.Fatalf("failed to marshal schema: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	validateSchema(string(marshaled), schemaURL)
 | |
| 
 | |
| 	err = os.WriteFile(destinationFile, marshaled, 0o644)
 | |
| 	if err != nil {
 | |
| 		log.Fatalf("failed to write schema to %s: %v", destinationFile, err)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // validateSchema validates the schema itself by compiling it.
 | |
| func validateSchema(schema, schemaURL string) {
 | |
| 	_, err := validatejsonschema.CompileString(schemaURL, schema)
 | |
| 	if err != nil {
 | |
| 		log.Fatalf("failed to compile schema: %v", err)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func normalizeDescription(description string) string {
 | |
| 	description = strings.ReplaceAll(description, `\n`, "\n")
 | |
| 	description = strings.ReplaceAll(description, `\"`, `"`)
 | |
| 	description = strings.TrimSpace(description)
 | |
| 
 | |
| 	return description
 | |
| }
 | |
| 
 | |
| func readMajorMinorVersion(filePath string) string {
 | |
| 	fileBytes, err := os.ReadFile(filePath)
 | |
| 	if err != nil {
 | |
| 		log.Fatalf("failed to read version file: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	version := string(fileBytes)
 | |
| 
 | |
| 	versionParts := strings.Split(version, ".")
 | |
| 
 | |
| 	if len(versionParts) < 2 {
 | |
| 		log.Fatalf("unexpected version in version file: %s", version)
 | |
| 	}
 | |
| 
 | |
| 	return versionParts[0] + "." + versionParts[1]
 | |
| }
 |