2024-09-19 09:41:40 +00:00
|
|
|
package loader
|
2022-11-18 09:57:31 +00:00
|
|
|
|
|
|
|
import (
|
2022-11-28 09:59:43 +00:00
|
|
|
"context"
|
2022-11-18 09:57:31 +00:00
|
|
|
"path/filepath"
|
2024-09-19 09:41:40 +00:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
2022-11-18 09:57:31 +00:00
|
|
|
"testing"
|
|
|
|
|
2023-05-16 16:35:39 +00:00
|
|
|
"github.com/databricks/cli/bundle"
|
|
|
|
"github.com/databricks/cli/bundle/config"
|
2024-09-19 09:41:40 +00:00
|
|
|
"github.com/databricks/cli/bundle/config/resources"
|
2024-09-24 14:39:02 +00:00
|
|
|
"github.com/databricks/cli/bundle/internal/bundletest"
|
|
|
|
"github.com/databricks/cli/libs/diag"
|
|
|
|
"github.com/databricks/cli/libs/dyn"
|
2022-11-18 09:57:31 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestProcessInclude(t *testing.T) {
|
2023-11-15 14:03:36 +00:00
|
|
|
b := &bundle.Bundle{
|
2024-09-24 14:39:02 +00:00
|
|
|
RootPath: "testdata/basic",
|
2022-11-28 09:59:43 +00:00
|
|
|
Config: config.Root{
|
|
|
|
Workspace: config.Workspace{
|
|
|
|
Host: "foo",
|
|
|
|
},
|
2022-11-18 09:57:31 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2024-09-19 09:41:40 +00:00
|
|
|
m := ProcessInclude(filepath.Join(b.RootPath, "host.yml"), "host.yml")
|
2024-03-27 10:49:05 +00:00
|
|
|
assert.Equal(t, "ProcessInclude(host.yml)", m.Name())
|
2022-11-18 09:57:31 +00:00
|
|
|
|
2024-03-27 10:49:05 +00:00
|
|
|
// Assert the host value prior to applying the mutator
|
2023-11-15 14:03:36 +00:00
|
|
|
assert.Equal(t, "foo", b.Config.Workspace.Host)
|
2024-03-27 10:49:05 +00:00
|
|
|
|
|
|
|
// Apply the mutator and assert that the host value has been updated
|
|
|
|
diags := bundle.Apply(context.Background(), b, m)
|
2024-03-25 14:18:47 +00:00
|
|
|
require.NoError(t, diags.Error())
|
2023-11-15 14:03:36 +00:00
|
|
|
assert.Equal(t, "bar", b.Config.Workspace.Host)
|
2022-11-18 09:57:31 +00:00
|
|
|
}
|
2024-09-19 09:41:40 +00:00
|
|
|
|
2024-09-24 14:39:02 +00:00
|
|
|
func TestProcessIncludeValidatesFileFormat(t *testing.T) {
|
|
|
|
b := &bundle.Bundle{
|
|
|
|
RootPath: "testdata/format",
|
|
|
|
Config: config.Root{
|
|
|
|
Bundle: config.Bundle{
|
|
|
|
Name: "format_test",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
m := ProcessInclude(filepath.Join(b.RootPath, "foo.job.yml"), "foo.job.yml")
|
|
|
|
diags := bundle.Apply(context.Background(), b, m)
|
|
|
|
require.NoError(t, diags.Error())
|
|
|
|
|
|
|
|
// Assert that the diagnostics contain the expected information
|
|
|
|
assert.Len(t, diags, 1)
|
|
|
|
assert.Equal(t, diag.Diagnostics{
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
2024-09-26 14:03:49 +00:00
|
|
|
Summary: "We recommend only defining a single job in a file with the .job.yml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - bar (job)\n - foo (job)\n",
|
2024-09-24 14:39:02 +00:00
|
|
|
Locations: []dyn.Location{
|
2024-09-24 15:54:13 +00:00
|
|
|
{File: filepath.FromSlash("testdata/format/foo.job.yml"), Line: 4, Column: 7},
|
|
|
|
{File: filepath.FromSlash("testdata/format/foo.job.yml"), Line: 7, Column: 7},
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.bar"),
|
|
|
|
dyn.MustPathFromString("resources.jobs.foo"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, diags)
|
|
|
|
}
|
|
|
|
|
2024-09-19 09:41:40 +00:00
|
|
|
func TestResourceNames(t *testing.T) {
|
|
|
|
names := []string{}
|
|
|
|
typ := reflect.TypeOf(config.Resources{})
|
|
|
|
for i := 0; i < typ.NumField(); i++ {
|
|
|
|
field := typ.Field(i)
|
|
|
|
jsonTags := strings.Split(field.Tag.Get("json"), ",")
|
|
|
|
singularName := strings.TrimSuffix(jsonTags[0], "s")
|
|
|
|
names = append(names, singularName)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert the contents of the two lists are equal. Please add the singular
|
|
|
|
// name of your resource to resourceNames global if you are adding a new
|
|
|
|
// resource.
|
|
|
|
assert.Equal(t, len(resourceTypes), len(names))
|
|
|
|
for _, name := range names {
|
|
|
|
assert.Contains(t, resourceTypes, name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-24 12:39:28 +00:00
|
|
|
func TestValidateFileFormat(t *testing.T) {
|
|
|
|
onlyJob := config.Root{
|
|
|
|
Resources: config.Resources{
|
|
|
|
Jobs: map[string]*resources.Job{
|
|
|
|
"job1": {},
|
2024-09-19 09:41:40 +00:00
|
|
|
},
|
2024-09-24 12:39:28 +00:00
|
|
|
},
|
|
|
|
Targets: map[string]*config.Target{
|
|
|
|
"target1": {
|
|
|
|
Resources: &config.Resources{
|
|
|
|
Jobs: map[string]*resources.Job{
|
|
|
|
"job1": {},
|
|
|
|
},
|
2024-09-19 09:41:40 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2024-09-24 14:39:02 +00:00
|
|
|
onlyJobBundle := bundle.Bundle{Config: onlyJob}
|
|
|
|
|
|
|
|
onlyPipeline := config.Root{
|
|
|
|
Resources: config.Resources{
|
|
|
|
Pipelines: map[string]*resources.Pipeline{
|
|
|
|
"pipeline1": {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
onlyPipelineBundle := bundle.Bundle{Config: onlyPipeline}
|
|
|
|
|
|
|
|
bothJobAndPipeline := config.Root{
|
|
|
|
Resources: config.Resources{
|
|
|
|
Jobs: map[string]*resources.Job{
|
|
|
|
"job1": {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Targets: map[string]*config.Target{
|
|
|
|
"target1": {
|
|
|
|
Resources: &config.Resources{
|
|
|
|
Pipelines: map[string]*resources.Pipeline{
|
|
|
|
"pipeline1": {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
bothJobAndPipelineBundle := bundle.Bundle{Config: bothJobAndPipeline}
|
|
|
|
|
|
|
|
twoJobs := config.Root{
|
|
|
|
Resources: config.Resources{
|
|
|
|
Jobs: map[string]*resources.Job{
|
|
|
|
"job1": {},
|
|
|
|
"job2": {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
twoJobsBundle := bundle.Bundle{Config: twoJobs}
|
|
|
|
|
|
|
|
twoJobsTopLevelAndTarget := config.Root{
|
|
|
|
Resources: config.Resources{
|
|
|
|
Jobs: map[string]*resources.Job{
|
|
|
|
"job1": {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Targets: map[string]*config.Target{
|
|
|
|
"target1": {
|
|
|
|
Resources: &config.Resources{
|
|
|
|
Jobs: map[string]*resources.Job{
|
|
|
|
"job2": {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
twoJobsTopLevelAndTargetBundle := bundle.Bundle{Config: twoJobsTopLevelAndTarget}
|
|
|
|
|
|
|
|
twoJobsInTarget := config.Root{
|
|
|
|
Targets: map[string]*config.Target{
|
|
|
|
"target1": {
|
|
|
|
Resources: &config.Resources{
|
|
|
|
Jobs: map[string]*resources.Job{
|
|
|
|
"job1": {},
|
|
|
|
"job2": {},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
twoJobsInTargetBundle := bundle.Bundle{Config: twoJobsInTarget}
|
|
|
|
|
|
|
|
tcases := []struct {
|
|
|
|
name string
|
|
|
|
bundle *bundle.Bundle
|
|
|
|
expected diag.Diagnostics
|
|
|
|
fileName string
|
|
|
|
locations map[string]dyn.Location
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "single job",
|
|
|
|
bundle: &onlyJobBundle,
|
|
|
|
expected: nil,
|
|
|
|
fileName: "foo.job.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "single pipeline",
|
|
|
|
bundle: &onlyPipelineBundle,
|
|
|
|
expected: nil,
|
|
|
|
fileName: "foo.pipeline.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.pipelines.pipeline1": {File: "foo.pipeline.yaml", Line: 1, Column: 1},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "single job but extension is pipeline",
|
|
|
|
bundle: &onlyJobBundle,
|
|
|
|
expected: diag.Diagnostics{
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
2024-09-26 14:03:49 +00:00
|
|
|
Summary: "We recommend only defining a single pipeline in a file with the .pipeline.yml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - job1 (job)\n",
|
2024-09-24 14:39:02 +00:00
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: "foo.pipeline.yml", Line: 1, Column: 1},
|
|
|
|
{File: "foo.pipeline.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job1"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
fileName: "foo.pipeline.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.pipeline.yml", Line: 1, Column: 1},
|
|
|
|
"targets.target1.resources.jobs.job1": {File: "foo.pipeline.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "job and pipeline",
|
|
|
|
bundle: &bothJobAndPipelineBundle,
|
|
|
|
expected: nil,
|
|
|
|
fileName: "foo.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.yml", Line: 1, Column: 1},
|
|
|
|
"targets.target1.resources.pipelines.pipeline1": {File: "foo.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "job and pipeline but extension is job",
|
|
|
|
bundle: &bothJobAndPipelineBundle,
|
|
|
|
expected: diag.Diagnostics{
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
2024-09-26 14:03:49 +00:00
|
|
|
Summary: "We recommend only defining a single job in a file with the .job.yml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - job1 (job)\n - pipeline1 (pipeline)\n",
|
2024-09-24 14:39:02 +00:00
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
{File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.pipelines.pipeline1"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
fileName: "foo.job.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
"targets.target1.resources.pipelines.pipeline1": {File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "job and pipeline but extension is experiment",
|
|
|
|
bundle: &bothJobAndPipelineBundle,
|
|
|
|
expected: diag.Diagnostics{
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
2024-09-26 14:03:49 +00:00
|
|
|
Summary: "We recommend only defining a single experiment in a file with the .experiment.yml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - job1 (job)\n - pipeline1 (pipeline)\n",
|
2024-09-24 14:39:02 +00:00
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: "foo.experiment.yml", Line: 1, Column: 1},
|
|
|
|
{File: "foo.experiment.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.pipelines.pipeline1"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
fileName: "foo.experiment.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.experiment.yml", Line: 1, Column: 1},
|
|
|
|
"targets.target1.resources.pipelines.pipeline1": {File: "foo.experiment.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "two jobs",
|
|
|
|
bundle: &twoJobsBundle,
|
|
|
|
expected: diag.Diagnostics{
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
2024-09-26 14:03:49 +00:00
|
|
|
Summary: "We recommend only defining a single job in a file with the .job.yml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - job1 (job)\n - job2 (job)\n",
|
2024-09-24 14:39:02 +00:00
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
{File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("resources.jobs.job2"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
fileName: "foo.job.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
"resources.jobs.job2": {File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "two jobs but extension is simple yaml",
|
|
|
|
bundle: &twoJobsBundle,
|
|
|
|
expected: nil,
|
|
|
|
fileName: "foo.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.yml", Line: 1, Column: 1},
|
|
|
|
"resources.jobs.job2": {File: "foo.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "two jobs in top level and target",
|
|
|
|
bundle: &twoJobsTopLevelAndTargetBundle,
|
|
|
|
expected: diag.Diagnostics{
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
2024-09-26 14:03:49 +00:00
|
|
|
Summary: "We recommend only defining a single job in a file with the .job.yml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - job1 (job)\n - job2 (job)\n",
|
2024-09-24 14:39:02 +00:00
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
{File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job2"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
fileName: "foo.job.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"resources.jobs.job1": {File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
"targets.target1.resources.jobs.job2": {File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "two jobs in target",
|
|
|
|
bundle: &twoJobsInTargetBundle,
|
|
|
|
expected: diag.Diagnostics{
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
2024-09-26 14:03:49 +00:00
|
|
|
Summary: "We recommend only defining a single job in a file with the .job.yml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - job1 (job)\n - job2 (job)\n",
|
2024-09-24 14:39:02 +00:00
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
{File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString(("targets.target1.resources.jobs.job1")),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job2"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
fileName: "foo.job.yml",
|
|
|
|
locations: map[string]dyn.Location{
|
|
|
|
"targets.target1.resources.jobs.job1": {File: "foo.job.yml", Line: 1, Column: 1},
|
|
|
|
"targets.target1.resources.jobs.job2": {File: "foo.job.yml", Line: 2, Column: 2},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2024-09-19 09:41:40 +00:00
|
|
|
|
2024-09-24 14:39:02 +00:00
|
|
|
for _, tc := range tcases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
for k, v := range tc.locations {
|
|
|
|
bundletest.SetLocation(tc.bundle, k, []dyn.Location{v})
|
|
|
|
}
|
|
|
|
|
2024-09-26 14:07:32 +00:00
|
|
|
diags := validateFileFormat(tc.bundle.Config.Value(), tc.fileName)
|
2024-09-24 14:39:02 +00:00
|
|
|
assert.Equal(t, tc.expected, diags)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|