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) } }