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-24 14:39:02 +00:00
|
|
|
"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-26 14:51:46 +00:00
|
|
|
func TestProcessIncludeFormatPass(t *testing.T) {
|
|
|
|
for _, fileName := range []string{
|
|
|
|
"one_job.job.yml",
|
|
|
|
"one_pipeline.pipeline.yaml",
|
|
|
|
"two_job.yml",
|
|
|
|
"job_and_pipeline.yml",
|
|
|
|
} {
|
|
|
|
t.Run(fileName, func(t *testing.T) {
|
|
|
|
b := &bundle.Bundle{
|
|
|
|
RootPath: "testdata/format_pass",
|
|
|
|
Config: config.Root{
|
|
|
|
Bundle: config.Bundle{
|
|
|
|
Name: "format_test",
|
2024-09-24 12:39:28 +00:00
|
|
|
},
|
2024-09-19 09:41:40 +00:00
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
}
|
2024-09-24 14:39:02 +00:00
|
|
|
|
2024-09-26 14:51:46 +00:00
|
|
|
m := ProcessInclude(filepath.Join(b.RootPath, fileName), fileName)
|
|
|
|
diags := bundle.Apply(context.Background(), b, m)
|
|
|
|
assert.Empty(t, diags)
|
|
|
|
})
|
2024-09-24 14:39:02 +00:00
|
|
|
}
|
2024-09-26 14:51:46 +00:00
|
|
|
}
|
2024-09-24 14:39:02 +00:00
|
|
|
|
2024-09-26 14:51:46 +00:00
|
|
|
func TestProcessIncludeFormatFail(t *testing.T) {
|
|
|
|
for fileName, expectedDiags := range map[string]diag.Diagnostics{
|
|
|
|
"single_job.pipeline.yaml": {
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
|
|
|
Summary: "We recommend only defining a single pipeline in a file with the .pipeline.yaml extension.",
|
|
|
|
Detail: "The following resources are defined or configured in this file:\n - job1 (job)\n",
|
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/single_job.pipeline.yaml"), Line: 11, Column: 11},
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/single_job.pipeline.yaml"), Line: 4, Column: 7},
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job1"),
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
"job_and_pipeline.job.yml": {
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
|
|
|
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",
|
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/job_and_pipeline.job.yml"), Line: 12, Column: 11},
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/job_and_pipeline.job.yml"), Line: 5, Column: 7},
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.pipelines.pipeline1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job1"),
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
"job_and_pipeline.experiment.yml": {
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
|
|
|
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",
|
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/job_and_pipeline.experiment.yml"), Line: 12, Column: 11},
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/job_and_pipeline.experiment.yml"), Line: 5, Column: 7},
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.pipelines.pipeline1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job1"),
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
"two_jobs.job.yml": {
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
|
|
|
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",
|
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/two_jobs.job.yml"), Line: 5, Column: 7},
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/two_jobs.job.yml"), Line: 8, Column: 7},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("resources.jobs.job2"),
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
"second_job_in_target.job.yml": {
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
|
|
|
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",
|
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/second_job_in_target.job.yml"), Line: 12, Column: 11},
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/second_job_in_target.job.yml"), Line: 5, Column: 7},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job2"),
|
|
|
|
},
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
"two_jobs_in_target.job.yml": {
|
|
|
|
{
|
|
|
|
Severity: diag.Info,
|
|
|
|
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",
|
|
|
|
Locations: []dyn.Location{
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/two_jobs_in_target.job.yml"), Line: 6, Column: 11},
|
|
|
|
{File: filepath.FromSlash("testdata/format_fail/two_jobs_in_target.job.yml"), Line: 8, Column: 11},
|
|
|
|
},
|
|
|
|
Paths: []dyn.Path{
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job1"),
|
|
|
|
dyn.MustPathFromString("targets.target1.resources.jobs.job2"),
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
} {
|
|
|
|
b := &bundle.Bundle{
|
|
|
|
RootPath: "testdata/format_fail",
|
|
|
|
Config: config.Root{
|
|
|
|
Bundle: config.Bundle{
|
|
|
|
Name: "format_test",
|
2024-09-24 14:39:02 +00:00
|
|
|
},
|
|
|
|
},
|
2024-09-26 14:51:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
m := ProcessInclude(filepath.Join(b.RootPath, fileName), fileName)
|
|
|
|
diags := bundle.Apply(context.Background(), b, m)
|
|
|
|
require.Len(t, diags, 1)
|
|
|
|
assert.Equal(t, expectedDiags, diags)
|
2024-09-24 14:39:02 +00:00
|
|
|
}
|
2024-09-26 14:51:46 +00:00
|
|
|
}
|
2024-09-19 09:41:40 +00:00
|
|
|
|
2024-09-26 14:51:46 +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)
|
|
|
|
}
|
2024-09-24 14:39:02 +00:00
|
|
|
|
2024-09-26 14:51:46 +00:00
|
|
|
// 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 14:39:02 +00:00
|
|
|
}
|
|
|
|
}
|