mirror of https://github.com/databricks/cli.git
378 lines
11 KiB
Go
378 lines
11 KiB
Go
package mutator
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/databricks/cli/bundle"
|
|
"github.com/databricks/cli/bundle/config"
|
|
"github.com/databricks/cli/bundle/config/resources"
|
|
"github.com/databricks/cli/bundle/config/variable"
|
|
"github.com/databricks/cli/libs/diag"
|
|
"github.com/databricks/cli/libs/dyn"
|
|
"github.com/databricks/databricks-sdk-go/service/compute"
|
|
"github.com/databricks/databricks-sdk-go/service/jobs"
|
|
"github.com/databricks/databricks-sdk-go/service/pipelines"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestResolveVariableReferencesToBundleVariables(t *testing.T) {
|
|
b := &bundle.Bundle{
|
|
Config: config.Root{
|
|
Bundle: config.Bundle{
|
|
Name: "example",
|
|
},
|
|
Workspace: config.Workspace{
|
|
RootPath: "${bundle.name}/${var.foo}",
|
|
},
|
|
Variables: map[string]*variable.Variable{
|
|
"foo": {
|
|
Value: "bar",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
// Apply with a valid prefix. This should change the workspace root path.
|
|
diags := bundle.Apply(context.Background(), b, ResolveVariableReferences("bundle", "variables"))
|
|
require.NoError(t, diags.Error())
|
|
require.Equal(t, "example/bar", b.Config.Workspace.RootPath)
|
|
}
|
|
|
|
func TestResolveVariableReferencesForPrimitiveNonStringFields(t *testing.T) {
|
|
var diags diag.Diagnostics
|
|
|
|
b := &bundle.Bundle{
|
|
Config: config.Root{
|
|
Variables: map[string]*variable.Variable{
|
|
"no_alert_for_canceled_runs": {},
|
|
"no_alert_for_skipped_runs": {},
|
|
"min_workers": {},
|
|
"max_workers": {},
|
|
"spot_bid_max_price": {},
|
|
},
|
|
Resources: config.Resources{
|
|
Jobs: map[string]*resources.Job{
|
|
"job1": {
|
|
JobSettings: &jobs.JobSettings{
|
|
NotificationSettings: &jobs.JobNotificationSettings{
|
|
NoAlertForCanceledRuns: false,
|
|
NoAlertForSkippedRuns: false,
|
|
},
|
|
Tasks: []jobs.Task{
|
|
{
|
|
NewCluster: &compute.ClusterSpec{
|
|
Autoscale: &compute.AutoScale{
|
|
MinWorkers: 0,
|
|
MaxWorkers: 0,
|
|
},
|
|
AzureAttributes: &compute.AzureAttributes{
|
|
SpotBidMaxPrice: 0.0,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
// Initialize the variables.
|
|
diags = bundle.ApplyFunc(ctx, b, func(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
err := b.Config.InitializeVariables([]string{
|
|
"no_alert_for_canceled_runs=true",
|
|
"no_alert_for_skipped_runs=true",
|
|
"min_workers=1",
|
|
"max_workers=2",
|
|
"spot_bid_max_price=0.5",
|
|
})
|
|
return diag.FromErr(err)
|
|
})
|
|
require.NoError(t, diags.Error())
|
|
|
|
// Assign the variables to the dynamic configuration.
|
|
diags = bundle.ApplyFunc(ctx, b, func(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
err := b.Config.Mutate(func(v dyn.Value) (dyn.Value, error) {
|
|
var p dyn.Path
|
|
var err error
|
|
|
|
// Set the notification settings.
|
|
p = dyn.MustPathFromString("resources.jobs.job1.notification_settings")
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("no_alert_for_canceled_runs")), dyn.V("${var.no_alert_for_canceled_runs}"))
|
|
require.NoError(t, err)
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("no_alert_for_skipped_runs")), dyn.V("${var.no_alert_for_skipped_runs}"))
|
|
require.NoError(t, err)
|
|
|
|
// Set the min and max workers.
|
|
p = dyn.MustPathFromString("resources.jobs.job1.tasks[0].new_cluster.autoscale")
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("min_workers")), dyn.V("${var.min_workers}"))
|
|
require.NoError(t, err)
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("max_workers")), dyn.V("${var.max_workers}"))
|
|
require.NoError(t, err)
|
|
|
|
// Set the spot bid max price.
|
|
p = dyn.MustPathFromString("resources.jobs.job1.tasks[0].new_cluster.azure_attributes")
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("spot_bid_max_price")), dyn.V("${var.spot_bid_max_price}"))
|
|
require.NoError(t, err)
|
|
|
|
return v, nil
|
|
})
|
|
return diag.FromErr(err)
|
|
})
|
|
require.NoError(t, diags.Error())
|
|
|
|
// Apply for the variable prefix. This should resolve the variables to their values.
|
|
diags = bundle.Apply(context.Background(), b, ResolveVariableReferences("variables"))
|
|
require.NoError(t, diags.Error())
|
|
assert.True(t, b.Config.Resources.Jobs["job1"].JobSettings.NotificationSettings.NoAlertForCanceledRuns)
|
|
assert.True(t, b.Config.Resources.Jobs["job1"].JobSettings.NotificationSettings.NoAlertForSkippedRuns)
|
|
assert.Equal(t, 1, b.Config.Resources.Jobs["job1"].JobSettings.Tasks[0].NewCluster.Autoscale.MinWorkers)
|
|
assert.Equal(t, 2, b.Config.Resources.Jobs["job1"].JobSettings.Tasks[0].NewCluster.Autoscale.MaxWorkers)
|
|
assert.InDelta(t, 0.5, b.Config.Resources.Jobs["job1"].JobSettings.Tasks[0].NewCluster.AzureAttributes.SpotBidMaxPrice, 0.0001)
|
|
}
|
|
|
|
func TestResolveComplexVariable(t *testing.T) {
|
|
b := &bundle.Bundle{
|
|
Config: config.Root{
|
|
Bundle: config.Bundle{
|
|
Name: "example",
|
|
},
|
|
Variables: map[string]*variable.Variable{
|
|
"cluster": {
|
|
Value: map[string]any{
|
|
"node_type_id": "Standard_DS3_v2",
|
|
"num_workers": 2,
|
|
},
|
|
Type: variable.VariableTypeComplex,
|
|
},
|
|
},
|
|
|
|
Resources: config.Resources{
|
|
Jobs: map[string]*resources.Job{
|
|
"job1": {
|
|
JobSettings: &jobs.JobSettings{
|
|
JobClusters: []jobs.JobCluster{
|
|
{
|
|
NewCluster: compute.ClusterSpec{
|
|
NodeTypeId: "random",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
// Assign the variables to the dynamic configuration.
|
|
diags := bundle.ApplyFunc(ctx, b, func(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
err := b.Config.Mutate(func(v dyn.Value) (dyn.Value, error) {
|
|
var p dyn.Path
|
|
var err error
|
|
|
|
p = dyn.MustPathFromString("resources.jobs.job1.job_clusters[0]")
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("new_cluster")), dyn.V("${var.cluster}"))
|
|
require.NoError(t, err)
|
|
|
|
return v, nil
|
|
})
|
|
return diag.FromErr(err)
|
|
})
|
|
require.NoError(t, diags.Error())
|
|
|
|
diags = bundle.Apply(ctx, b, ResolveVariableReferences("bundle", "workspace", "variables"))
|
|
require.NoError(t, diags.Error())
|
|
require.Equal(t, "Standard_DS3_v2", b.Config.Resources.Jobs["job1"].JobSettings.JobClusters[0].NewCluster.NodeTypeId)
|
|
require.Equal(t, 2, b.Config.Resources.Jobs["job1"].JobSettings.JobClusters[0].NewCluster.NumWorkers)
|
|
}
|
|
|
|
func TestResolveComplexVariableReferencesWithComplexVariablesError(t *testing.T) {
|
|
b := &bundle.Bundle{
|
|
Config: config.Root{
|
|
Bundle: config.Bundle{
|
|
Name: "example",
|
|
},
|
|
Variables: map[string]*variable.Variable{
|
|
"cluster": {
|
|
Value: map[string]any{
|
|
"node_type_id": "Standard_DS3_v2",
|
|
"num_workers": 2,
|
|
"spark_conf": "${var.spark_conf}",
|
|
},
|
|
Type: variable.VariableTypeComplex,
|
|
},
|
|
"spark_conf": {
|
|
Value: map[string]any{
|
|
"spark.executor.memory": "4g",
|
|
"spark.executor.cores": "2",
|
|
},
|
|
Type: variable.VariableTypeComplex,
|
|
},
|
|
},
|
|
|
|
Resources: config.Resources{
|
|
Jobs: map[string]*resources.Job{
|
|
"job1": {
|
|
JobSettings: &jobs.JobSettings{
|
|
JobClusters: []jobs.JobCluster{
|
|
{
|
|
NewCluster: compute.ClusterSpec{
|
|
NodeTypeId: "random",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
// Assign the variables to the dynamic configuration.
|
|
diags := bundle.ApplyFunc(ctx, b, func(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
err := b.Config.Mutate(func(v dyn.Value) (dyn.Value, error) {
|
|
var p dyn.Path
|
|
var err error
|
|
|
|
p = dyn.MustPathFromString("resources.jobs.job1.job_clusters[0]")
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("new_cluster")), dyn.V("${var.cluster}"))
|
|
require.NoError(t, err)
|
|
|
|
return v, nil
|
|
})
|
|
return diag.FromErr(err)
|
|
})
|
|
require.NoError(t, diags.Error())
|
|
|
|
diags = bundle.Apply(ctx, b, bundle.Seq(ResolveVariableReferencesInComplexVariables(), ResolveVariableReferences("bundle", "workspace", "variables")))
|
|
require.ErrorContains(t, diags.Error(), "complex variables cannot contain references to another complex variables")
|
|
}
|
|
|
|
func TestResolveComplexVariableWithVarReference(t *testing.T) {
|
|
b := &bundle.Bundle{
|
|
Config: config.Root{
|
|
Bundle: config.Bundle{
|
|
Name: "example",
|
|
},
|
|
Variables: map[string]*variable.Variable{
|
|
"package_version": {
|
|
Value: "1.0.0",
|
|
},
|
|
"cluster_libraries": {
|
|
Value: [](map[string]any){
|
|
{
|
|
"pypi": map[string]string{
|
|
"package": "cicd_template==${var.package_version}",
|
|
},
|
|
},
|
|
},
|
|
Type: variable.VariableTypeComplex,
|
|
},
|
|
},
|
|
|
|
Resources: config.Resources{
|
|
Jobs: map[string]*resources.Job{
|
|
"job1": {
|
|
JobSettings: &jobs.JobSettings{
|
|
Tasks: []jobs.Task{
|
|
{
|
|
Libraries: []compute.Library{},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
ctx := context.Background()
|
|
|
|
// Assign the variables to the dynamic configuration.
|
|
diags := bundle.ApplyFunc(ctx, b, func(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
err := b.Config.Mutate(func(v dyn.Value) (dyn.Value, error) {
|
|
var p dyn.Path
|
|
var err error
|
|
|
|
p = dyn.MustPathFromString("resources.jobs.job1.tasks[0]")
|
|
v, err = dyn.SetByPath(v, p.Append(dyn.Key("libraries")), dyn.V("${var.cluster_libraries}"))
|
|
require.NoError(t, err)
|
|
|
|
return v, nil
|
|
})
|
|
return diag.FromErr(err)
|
|
})
|
|
require.NoError(t, diags.Error())
|
|
|
|
diags = bundle.Apply(ctx, b, bundle.Seq(
|
|
ResolveVariableReferencesInComplexVariables(),
|
|
ResolveVariableReferences("bundle", "workspace", "variables"),
|
|
))
|
|
require.NoError(t, diags.Error())
|
|
require.Equal(t, "cicd_template==1.0.0", b.Config.Resources.Jobs["job1"].JobSettings.Tasks[0].Libraries[0].Pypi.Package)
|
|
}
|
|
|
|
func TestResolveVariableReferencesWithSourceLinkedDeployment(t *testing.T) {
|
|
testCases := []struct {
|
|
enabled bool
|
|
assert func(t *testing.T, b *bundle.Bundle)
|
|
}{
|
|
{
|
|
true,
|
|
func(t *testing.T, b *bundle.Bundle) {
|
|
// Variables that use workspace file path should have SyncRootValue during resolution phase
|
|
require.Equal(t, "sync/root/path", b.Config.Resources.Pipelines["pipeline1"].PipelineSpec.Configuration["source"])
|
|
|
|
// The file path itself should remain the same
|
|
require.Equal(t, "file/path", b.Config.Workspace.FilePath)
|
|
},
|
|
},
|
|
{
|
|
false,
|
|
func(t *testing.T, b *bundle.Bundle) {
|
|
require.Equal(t, "file/path", b.Config.Resources.Pipelines["pipeline1"].PipelineSpec.Configuration["source"])
|
|
require.Equal(t, "file/path", b.Config.Workspace.FilePath)
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, testCase := range testCases {
|
|
b := &bundle.Bundle{
|
|
SyncRootPath: "sync/root/path",
|
|
Config: config.Root{
|
|
Presets: config.Presets{
|
|
SourceLinkedDeployment: &testCase.enabled,
|
|
},
|
|
Workspace: config.Workspace{
|
|
FilePath: "file/path",
|
|
},
|
|
Resources: config.Resources{
|
|
Pipelines: map[string]*resources.Pipeline{
|
|
"pipeline1": {
|
|
PipelineSpec: &pipelines.PipelineSpec{
|
|
Configuration: map[string]string{
|
|
"source": "${workspace.file_path}",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
diags := bundle.Apply(context.Background(), b, ResolveVariableReferences("workspace"))
|
|
require.NoError(t, diags.Error())
|
|
testCase.assert(t, b)
|
|
}
|
|
}
|