mirror of
https://github.com/outbackdingo/cozystack.git
synced 2026-01-28 10:18:42 +00:00
177 lines
5.3 KiB
Go
177 lines
5.3 KiB
Go
package dashboard
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"strings"
|
|
|
|
dashv1alpha1 "github.com/cozystack/cozystack/api/dashboard/v1alpha1"
|
|
cozyv1alpha1 "github.com/cozystack/cozystack/api/v1alpha1"
|
|
|
|
apiextv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
|
|
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
|
|
"sigs.k8s.io/controller-runtime/pkg/log"
|
|
)
|
|
|
|
// ensureCustomFormsOverride creates or updates a CustomFormsOverride resource for the given CRD
|
|
func (m *Manager) ensureCustomFormsOverride(ctx context.Context, crd *cozyv1alpha1.CozystackResourceDefinition) error {
|
|
g, v, kind := pickGVK(crd)
|
|
plural := pickPlural(kind, crd)
|
|
|
|
name := fmt.Sprintf("%s.%s.%s", g, v, plural)
|
|
customizationID := fmt.Sprintf("default-/%s/%s/%s", g, v, plural)
|
|
|
|
obj := &dashv1alpha1.CustomFormsOverride{}
|
|
obj.SetName(name)
|
|
|
|
// Replicates your Helm includes (system metadata + api + status).
|
|
hidden := []any{}
|
|
hidden = append(hidden, hiddenMetadataSystem()...)
|
|
hidden = append(hidden, hiddenMetadataAPI()...)
|
|
hidden = append(hidden, hiddenStatus()...)
|
|
|
|
// If Name is set, hide metadata
|
|
if crd.Spec.Dashboard != nil && strings.TrimSpace(crd.Spec.Dashboard.Name) != "" {
|
|
hidden = append([]interface{}{
|
|
[]any{"metadata"},
|
|
}, hidden...)
|
|
}
|
|
|
|
var sort []any
|
|
if crd.Spec.Dashboard != nil && len(crd.Spec.Dashboard.KeysOrder) > 0 {
|
|
sort = make([]any, len(crd.Spec.Dashboard.KeysOrder))
|
|
for i, v := range crd.Spec.Dashboard.KeysOrder {
|
|
sort[i] = v
|
|
}
|
|
}
|
|
|
|
// Build schema with multilineString for string fields without enum
|
|
l := log.FromContext(ctx)
|
|
schema, err := buildMultilineStringSchema(crd.Spec.Application.OpenAPISchema)
|
|
if err != nil {
|
|
// If schema parsing fails, log the error and use an empty schema
|
|
l.Error(err, "failed to build multiline string schema, using empty schema", "crd", crd.Name)
|
|
schema = map[string]any{}
|
|
}
|
|
|
|
spec := map[string]any{
|
|
"customizationId": customizationID,
|
|
"hidden": hidden,
|
|
"sort": sort,
|
|
"schema": schema,
|
|
"strategy": "merge",
|
|
}
|
|
|
|
_, err = controllerutil.CreateOrUpdate(ctx, m.Client, obj, func() error {
|
|
if err := controllerutil.SetOwnerReference(crd, obj, m.Scheme); err != nil {
|
|
return err
|
|
}
|
|
// Add dashboard labels to dynamic resources
|
|
m.addDashboardLabels(obj, crd, ResourceTypeDynamic)
|
|
b, err := json.Marshal(spec)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Only update spec if it's different to avoid unnecessary updates
|
|
newSpec := dashv1alpha1.ArbitrarySpec{JSON: apiextv1.JSON{Raw: b}}
|
|
if !compareArbitrarySpecs(obj.Spec, newSpec) {
|
|
obj.Spec = newSpec
|
|
}
|
|
return nil
|
|
})
|
|
return err
|
|
}
|
|
|
|
// buildMultilineStringSchema parses OpenAPI schema and creates schema with multilineString
|
|
// for all string fields inside spec that don't have enum
|
|
func buildMultilineStringSchema(openAPISchema string) (map[string]any, error) {
|
|
if openAPISchema == "" {
|
|
return map[string]any{}, nil
|
|
}
|
|
|
|
var root map[string]any
|
|
if err := json.Unmarshal([]byte(openAPISchema), &root); err != nil {
|
|
return nil, fmt.Errorf("cannot parse openAPISchema: %w", err)
|
|
}
|
|
|
|
props, _ := root["properties"].(map[string]any)
|
|
if props == nil {
|
|
return map[string]any{}, nil
|
|
}
|
|
|
|
schema := map[string]any{
|
|
"properties": map[string]any{},
|
|
}
|
|
|
|
// Process spec properties recursively
|
|
processSpecProperties(props, schema["properties"].(map[string]any))
|
|
|
|
return schema, nil
|
|
}
|
|
|
|
// processSpecProperties recursively processes spec properties and adds multilineString type
|
|
// for string fields without enum
|
|
func processSpecProperties(props map[string]any, schemaProps map[string]any) {
|
|
for pname, raw := range props {
|
|
sub, ok := raw.(map[string]any)
|
|
if !ok {
|
|
continue
|
|
}
|
|
|
|
typ, _ := sub["type"].(string)
|
|
|
|
switch typ {
|
|
case "string":
|
|
// Check if this string field has enum
|
|
if !hasEnum(sub) {
|
|
// Add multilineString type for this field
|
|
if schemaProps[pname] == nil {
|
|
schemaProps[pname] = map[string]any{}
|
|
}
|
|
fieldSchema := schemaProps[pname].(map[string]any)
|
|
fieldSchema["type"] = "multilineString"
|
|
}
|
|
case "object":
|
|
// Recursively process nested objects
|
|
if childProps, ok := sub["properties"].(map[string]any); ok {
|
|
fieldSchema, ok := schemaProps[pname].(map[string]any)
|
|
if !ok {
|
|
fieldSchema = map[string]any{}
|
|
schemaProps[pname] = fieldSchema
|
|
}
|
|
nestedSchemaProps, ok := fieldSchema["properties"].(map[string]any)
|
|
if !ok {
|
|
nestedSchemaProps = map[string]any{}
|
|
fieldSchema["properties"] = nestedSchemaProps
|
|
}
|
|
processSpecProperties(childProps, nestedSchemaProps)
|
|
}
|
|
case "array":
|
|
// Check if array items are objects with properties
|
|
if items, ok := sub["items"].(map[string]any); ok {
|
|
if itemProps, ok := items["properties"].(map[string]any); ok {
|
|
// Create array item schema
|
|
fieldSchema, ok := schemaProps[pname].(map[string]any)
|
|
if !ok {
|
|
fieldSchema = map[string]any{}
|
|
schemaProps[pname] = fieldSchema
|
|
}
|
|
itemSchema, ok := fieldSchema["items"].(map[string]any)
|
|
if !ok {
|
|
itemSchema = map[string]any{}
|
|
fieldSchema["items"] = itemSchema
|
|
}
|
|
itemSchemaProps, ok := itemSchema["properties"].(map[string]any)
|
|
if !ok {
|
|
itemSchemaProps = map[string]any{}
|
|
itemSchema["properties"] = itemSchemaProps
|
|
}
|
|
processSpecProperties(itemProps, itemSchemaProps)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|