databricks-cli/bundle/config/loader/process_include.go

168 lines
4.6 KiB
Go
Raw Normal View History

package loader
import (
"context"
"fmt"
2024-09-26 14:03:49 +00:00
"slices"
2024-09-24 14:39:02 +00:00
"sort"
"strings"
"github.com/databricks/cli/bundle"
"github.com/databricks/cli/bundle/config"
"github.com/databricks/cli/libs/diag"
"github.com/databricks/cli/libs/dyn"
)
2024-09-26 14:07:32 +00:00
func validateFileFormat(configRoot dyn.Value, filePath string) diag.Diagnostics {
2024-09-26 15:47:07 +00:00
for _, resourceDescription := range config.SupportedResources() {
singularName := resourceDescription.SingularName
for _, ext := range []string{fmt.Sprintf(".%s.yml", singularName), fmt.Sprintf(".%s.yaml", singularName)} {
2024-09-24 12:39:28 +00:00
if strings.HasSuffix(filePath, ext) {
2024-09-26 15:47:07 +00:00
return validateSingleResourceDefined(configRoot, ext, singularName)
2024-09-24 12:39:28 +00:00
}
}
}
return nil
}
2024-09-26 14:07:32 +00:00
func validateSingleResourceDefined(configRoot dyn.Value, ext, typ string) diag.Diagnostics {
2024-09-24 12:39:28 +00:00
type resource struct {
path dyn.Path
value dyn.Value
typ string
key string
}
resources := []resource{}
2024-09-26 15:47:07 +00:00
supportedResources := config.SupportedResources()
2024-09-24 12:39:28 +00:00
// Gather all resources defined in the resources block.
_, err := dyn.MapByPattern(
2024-09-26 14:07:32 +00:00
configRoot,
dyn.NewPattern(dyn.Key("resources"), dyn.AnyKey(), dyn.AnyKey()),
func(p dyn.Path, v dyn.Value) (dyn.Value, error) {
// The key for the resource. Eg: "my_job" for jobs.my_job.
k := p[2].Key()
// The type of the resource. Eg: "job" for jobs.my_job.
2024-09-26 15:47:07 +00:00
typ := supportedResources[p[1].Key()].SingularName
2024-09-24 12:39:28 +00:00
resources = append(resources, resource{path: p, value: v, typ: typ, key: k})
return v, nil
})
if err != nil {
2024-09-24 12:39:28 +00:00
return diag.FromErr(err)
}
// Gather all resources defined in a target block.
_, err = dyn.MapByPattern(
2024-09-26 14:07:32 +00:00
configRoot,
dyn.NewPattern(dyn.Key("targets"), dyn.AnyKey(), dyn.Key("resources"), dyn.AnyKey(), dyn.AnyKey()),
func(p dyn.Path, v dyn.Value) (dyn.Value, error) {
2024-09-24 12:39:28 +00:00
// The key for the resource. Eg: "my_job" for jobs.my_job.
k := p[4].Key()
2024-09-24 12:39:28 +00:00
// The type of the resource. Eg: "job" for jobs.my_job.
2024-09-26 15:47:07 +00:00
typ := supportedResources[p[3].Key()].SingularName
2024-09-24 12:39:28 +00:00
resources = append(resources, resource{path: p, value: v, typ: typ, key: k})
return v, nil
2024-09-24 12:39:28 +00:00
})
if err != nil {
return diag.FromErr(err)
}
2024-09-24 15:47:48 +00:00
typeMatch := true
2024-09-24 12:39:28 +00:00
seenKeys := map[string]struct{}{}
for _, rr := range resources {
2024-09-24 15:47:48 +00:00
// case: The resource is not of the correct type.
2024-09-24 12:39:28 +00:00
if rr.typ != typ {
2024-09-24 15:47:48 +00:00
typeMatch = false
2024-09-24 12:39:28 +00:00
break
}
2024-09-24 15:47:48 +00:00
seenKeys[rr.key] = struct{}{}
2024-09-24 12:39:28 +00:00
}
2024-09-26 13:39:41 +00:00
// Format matches. There's at most one resource defined in the file.
2024-09-24 15:47:48 +00:00
// The resource is also of the correct type.
if typeMatch && len(seenKeys) <= 1 {
2024-09-24 12:39:28 +00:00
return nil
}
2024-09-26 14:03:49 +00:00
detail := strings.Builder{}
detail.WriteString("The following resources are defined or configured in this file:\n")
lines := []string{}
2024-09-24 12:39:28 +00:00
for _, r := range resources {
2024-09-26 14:03:49 +00:00
lines = append(lines, fmt.Sprintf(" - %s (%s)\n", r.key, r.typ))
2024-09-24 14:39:02 +00:00
}
// Sort the line s to print to make the output deterministic.
2024-09-26 14:03:49 +00:00
sort.Strings(lines)
// Compact the lines before writing them to the message to remove any duplicate lines.
// This is needed because we do not dedup earlier when gathering the resources
// and it's valid to define the same resource in both the resources and targets block.
lines = slices.Compact(lines)
for _, l := range lines {
detail.WriteString(l)
2024-09-24 12:39:28 +00:00
}
locations := []dyn.Location{}
paths := []dyn.Path{}
for _, rr := range resources {
locations = append(locations, rr.value.Locations()...)
paths = append(paths, rr.path)
}
2024-09-24 14:39:02 +00:00
// Sort the locations and paths to make the output deterministic.
sort.Slice(locations, func(i, j int) bool {
return locations[i].String() < locations[j].String()
})
sort.Slice(paths, func(i, j int) bool {
return paths[i].String() < paths[j].String()
})
2024-09-24 12:39:28 +00:00
return diag.Diagnostics{
{
2024-09-26 15:14:43 +00:00
Severity: diag.Recommendation,
2024-09-26 14:03:49 +00:00
Summary: fmt.Sprintf("We recommend only defining a single %s in a file with the %s extension.", typ, ext),
Detail: detail.String(),
2024-09-24 12:39:28 +00:00
Locations: locations,
Paths: paths,
},
2024-09-24 12:39:28 +00:00
}
}
type processInclude struct {
fullPath string
relPath string
}
// ProcessInclude loads the configuration at [fullPath] and merges it into the configuration.
func ProcessInclude(fullPath, relPath string) bundle.Mutator {
return &processInclude{
fullPath: fullPath,
relPath: relPath,
}
}
func (m *processInclude) Name() string {
return fmt.Sprintf("ProcessInclude(%s)", m.relPath)
}
func (m *processInclude) Apply(_ context.Context, b *bundle.Bundle) diag.Diagnostics {
this, diags := config.Load(m.fullPath)
if diags.HasError() {
return diags
}
2024-09-24 12:39:28 +00:00
// Add any diagnostics associated with the file format.
2024-09-26 14:07:32 +00:00
diags = append(diags, validateFileFormat(this.Value(), m.relPath)...)
2024-09-24 15:42:03 +00:00
if diags.HasError() {
return diags
}
err := b.Config.Merge(this)
if err != nil {
diags = diags.Extend(diag.FromErr(err))
}
return diags
}