diff --git a/NEXT_CHANGELOG.md b/NEXT_CHANGELOG.md index e738352986..9a711f6151 100644 --- a/NEXT_CHANGELOG.md +++ b/NEXT_CHANGELOG.md @@ -8,5 +8,6 @@ * Processing 'artifacts' section is now done in "bundle validate" (adding defaults, inferring "build", asserting required fields) ([#2526])(https://github.com/databricks/cli/pull/2526)) * When uploading artifacts, include relative path in log message ([#2539])(https://github.com/databricks/cli/pull/2539)) +* Add support for clusters in deployment bind/unbind commands ([#2536](https://github.com/databricks/cli/pull/2536)) ### API Changes diff --git a/acceptance/acceptance_test.go b/acceptance/acceptance_test.go index ff36085b30..d1d4c96df0 100644 --- a/acceptance/acceptance_test.go +++ b/acceptance/acceptance_test.go @@ -290,6 +290,10 @@ func getSkipReason(config *internal.TestConfig, configPath string) string { return fmt.Sprintf("Disabled via RequiresUnityCatalog setting in %s (TEST_METASTORE_ID=%s)", configPath, os.Getenv("TEST_METASTORE_ID")) } + if isTruePtr(config.RequiresCluster) && os.Getenv("TEST_DEFAULT_CLUSTER_ID") == "" { + return fmt.Sprintf("Disabled via RequiresCluster setting in %s (TEST_DEFAULT_CLUSTER_ID=%s)", configPath, os.Getenv("TEST_DEFAULT_CLUSTER_ID")) + } + } else { // Local run if !isTruePtr(config.Local) { diff --git a/acceptance/bundle/deployment/bind/cluster/databricks.yml.tmpl b/acceptance/bundle/deployment/bind/cluster/databricks.yml.tmpl new file mode 100644 index 0000000000..6206f0ad31 --- /dev/null +++ b/acceptance/bundle/deployment/bind/cluster/databricks.yml.tmpl @@ -0,0 +1,9 @@ +bundle: + name: bind-cluster-test-$UNIQUE_NAME + +resources: + clusters: + cluster1: + cluster_name: "DEFAULT Test Cluster" + spark_version: '16.2.x-scala2.12' + diff --git a/acceptance/bundle/deployment/bind/cluster/output.txt b/acceptance/bundle/deployment/bind/cluster/output.txt new file mode 100644 index 0000000000..00f5169436 --- /dev/null +++ b/acceptance/bundle/deployment/bind/cluster/output.txt @@ -0,0 +1,23 @@ + +>>> cat databricks.yml +bundle: + name: bind-cluster-test-[UNIQUE_NAME] + +resources: + clusters: + cluster1: + cluster_name: "DEFAULT Test Cluster" + spark_version: '16.2.x-scala2.12' + + +>>> [CLI] clusters get [CLUSTER-ID] +{ + "cluster_name": "DEFAULT Test Cluster" +} + +>>> [CLI] bundle deployment bind cluster1 [CLUSTER-ID] --auto-approve +Updating deployment state... +Successfully bound databricks_cluster with an id '[CLUSTER-ID]'. Run 'bundle deploy' to deploy changes to your workspace + +>>> [CLI] bundle deployment unbind cluster1 +Updating deployment state... diff --git a/acceptance/bundle/deployment/bind/cluster/script b/acceptance/bundle/deployment/bind/cluster/script new file mode 100644 index 0000000000..a0683a3d8c --- /dev/null +++ b/acceptance/bundle/deployment/bind/cluster/script @@ -0,0 +1,13 @@ +CLUSTER_ID="0123-456789-cluster0" +if [ -n "${TEST_DEFAULT_CLUSTER_ID:-}" ]; then + CLUSTER_ID=${TEST_DEFAULT_CLUSTER_ID} +fi +export CLUSTER_ID +envsubst < databricks.yml.tmpl > databricks.yml +trace cat databricks.yml + +trace $CLI clusters get ${CLUSTER_ID} | jq '{cluster_name}' + +trace $CLI bundle deployment bind cluster1 "${CLUSTER_ID}" --auto-approve + +trace $CLI bundle deployment unbind cluster1 diff --git a/acceptance/bundle/deployment/bind/cluster/test.toml b/acceptance/bundle/deployment/bind/cluster/test.toml new file mode 100644 index 0000000000..17358b5e7f --- /dev/null +++ b/acceptance/bundle/deployment/bind/cluster/test.toml @@ -0,0 +1,19 @@ +Local = true +Cloud = true +RequiresCluster = true + +Ignore = [ + "databricks.yml", +] + +[[Repls]] +Old = "[0-9]{4}-[0-9]{6}-[0-9a-z]{8}" +New = "[CLUSTER-ID]" + +[[Server]] +Pattern = "GET /api/2.1/clusters/get" +Response.Body = ''' +{ + "cluster_name": "DEFAULT Test Cluster" +} +''' diff --git a/acceptance/internal/config.go b/acceptance/internal/config.go index 24ffbe74ab..6627de21da 100644 --- a/acceptance/internal/config.go +++ b/acceptance/internal/config.go @@ -39,6 +39,9 @@ type TestConfig struct { // If true and Cloud=true, run this test only if unity catalog is available in the cloud environment RequiresUnityCatalog *bool + // If true and Cloud=true, run this test only if a default test cluster is available in the cloud environment + RequiresCluster *bool + // List of additional replacements to apply on this test. // Old is a regexp, New is a replacement expression. Repls []testdiff.Replacement diff --git a/bundle/config/resources.go b/bundle/config/resources.go index ab5c8f27f2..b6dda99ec8 100644 --- a/bundle/config/resources.go +++ b/bundle/config/resources.go @@ -124,6 +124,12 @@ func (r *Resources) FindResourceByConfigKey(key string) (ConfigResource, error) } } + for k := range r.Clusters { + if k == key { + found = append(found, r.Clusters[k]) + } + } + if len(found) == 0 { return nil, fmt.Errorf("no such resource: %s", key) }