2024-12-13 14:38:58 +00:00
package bundle_test
2024-02-14 18:04:45 +00:00
import (
"os"
"path/filepath"
2025-01-07 10:49:23 +00:00
"strconv"
2024-02-14 18:04:45 +00:00
"testing"
2024-12-17 07:45:58 +00:00
"github.com/databricks/cli/integration/internal/acc"
2024-12-12 16:48:51 +00:00
"github.com/databricks/cli/internal/testcli"
2024-12-12 12:35:38 +00:00
"github.com/databricks/cli/internal/testutil"
2024-12-16 11:34:37 +00:00
"github.com/databricks/cli/libs/env"
2024-02-14 18:04:45 +00:00
"github.com/databricks/databricks-sdk-go"
"github.com/databricks/databricks-sdk-go/service/jobs"
"github.com/google/uuid"
2024-09-02 13:43:17 +00:00
"github.com/stretchr/testify/assert"
2024-02-14 18:04:45 +00:00
"github.com/stretchr/testify/require"
)
2024-12-13 14:47:50 +00:00
func TestBindJobToExistingJob ( t * testing . T ) {
2024-02-14 18:04:45 +00:00
ctx , wt := acc . WorkspaceTest ( t )
2024-12-12 21:28:04 +00:00
gt := & generateJobTest { T : wt , w : wt . W }
2024-02-14 18:04:45 +00:00
2024-12-12 21:28:04 +00:00
nodeTypeId := testutil . GetCloud ( t ) . NodeTypeID ( )
2024-02-14 18:04:45 +00:00
uniqueId := uuid . New ( ) . String ( )
2024-12-16 12:41:32 +00:00
bundleRoot := initTestTemplate ( t , ctx , "basic" , map [ string ] any {
2024-02-14 18:04:45 +00:00
"unique_id" : uniqueId ,
"spark_version" : "13.3.x-scala2.12" ,
"node_type_id" : nodeTypeId ,
} )
jobId := gt . createTestJob ( ctx )
t . Cleanup ( func ( ) {
gt . destroyJob ( ctx , jobId )
} )
2024-12-16 11:34:37 +00:00
ctx = env . Set ( ctx , "BUNDLE_ROOT" , bundleRoot )
2025-01-07 10:49:23 +00:00
c := testcli . NewRunner ( t , ctx , "bundle" , "deployment" , "bind" , "foo" , strconv . FormatInt ( jobId , 10 ) , "--auto-approve" )
2024-12-16 12:41:32 +00:00
_ , _ , err := c . Run ( )
2024-02-14 18:04:45 +00:00
require . NoError ( t , err )
// Remove .databricks directory to simulate a fresh deployment
err = os . RemoveAll ( filepath . Join ( bundleRoot , ".databricks" ) )
require . NoError ( t , err )
2024-12-16 12:41:32 +00:00
deployBundle ( t , ctx , bundleRoot )
2024-02-14 18:04:45 +00:00
w , err := databricks . NewWorkspaceClient ( )
require . NoError ( t , err )
// Check that job is bound and updated with config from bundle
job , err := w . Jobs . Get ( ctx , jobs . GetJobRequest {
JobId : jobId ,
} )
require . NoError ( t , err )
2025-01-07 10:49:23 +00:00
require . Equal ( t , job . Settings . Name , "test-job-basic-" + uniqueId )
2024-02-14 18:04:45 +00:00
require . Contains ( t , job . Settings . Tasks [ 0 ] . SparkPythonTask . PythonFile , "hello_world.py" )
2024-12-16 11:34:37 +00:00
c = testcli . NewRunner ( t , ctx , "bundle" , "deployment" , "unbind" , "foo" )
2024-02-14 18:04:45 +00:00
_ , _ , err = c . Run ( )
require . NoError ( t , err )
// Remove .databricks directory to simulate a fresh deployment
err = os . RemoveAll ( filepath . Join ( bundleRoot , ".databricks" ) )
require . NoError ( t , err )
2024-12-16 12:41:32 +00:00
destroyBundle ( t , ctx , bundleRoot )
2024-02-14 18:04:45 +00:00
// Check that job is unbound and exists after bundle is destroyed
job , err = w . Jobs . Get ( ctx , jobs . GetJobRequest {
JobId : jobId ,
} )
require . NoError ( t , err )
2025-01-07 10:49:23 +00:00
require . Equal ( t , job . Settings . Name , "test-job-basic-" + uniqueId )
2024-02-14 18:04:45 +00:00
require . Contains ( t , job . Settings . Tasks [ 0 ] . SparkPythonTask . PythonFile , "hello_world.py" )
}
2024-12-13 14:47:50 +00:00
func TestAbortBind ( t * testing . T ) {
2024-02-14 18:04:45 +00:00
ctx , wt := acc . WorkspaceTest ( t )
2024-12-12 21:28:04 +00:00
gt := & generateJobTest { T : wt , w : wt . W }
2024-02-14 18:04:45 +00:00
2024-12-12 21:28:04 +00:00
nodeTypeId := testutil . GetCloud ( t ) . NodeTypeID ( )
2024-02-14 18:04:45 +00:00
uniqueId := uuid . New ( ) . String ( )
2024-12-16 12:41:32 +00:00
bundleRoot := initTestTemplate ( t , ctx , "basic" , map [ string ] any {
2024-02-14 18:04:45 +00:00
"unique_id" : uniqueId ,
"spark_version" : "13.3.x-scala2.12" ,
"node_type_id" : nodeTypeId ,
} )
jobId := gt . createTestJob ( ctx )
t . Cleanup ( func ( ) {
gt . destroyJob ( ctx , jobId )
2024-12-16 12:41:32 +00:00
destroyBundle ( t , ctx , bundleRoot )
2024-02-14 18:04:45 +00:00
} )
2024-09-02 13:43:17 +00:00
// Bind should fail because prompting is not possible.
2024-12-16 11:34:37 +00:00
ctx = env . Set ( ctx , "BUNDLE_ROOT" , bundleRoot )
ctx = env . Set ( ctx , "TERM" , "dumb" )
2025-01-07 10:49:23 +00:00
c := testcli . NewRunner ( t , ctx , "bundle" , "deployment" , "bind" , "foo" , strconv . FormatInt ( jobId , 10 ) )
2024-02-14 18:04:45 +00:00
2024-09-02 13:43:17 +00:00
// Expect error suggesting to use --auto-approve
2024-12-16 12:41:32 +00:00
_ , _ , err := c . Run ( )
2024-09-02 13:43:17 +00:00
assert . ErrorContains ( t , err , "failed to bind the resource" )
assert . ErrorContains ( t , err , "This bind operation requires user confirmation, but the current console does not support prompting. Please specify --auto-approve if you would like to skip prompts and proceed" )
2024-02-14 18:04:45 +00:00
2024-12-16 12:41:32 +00:00
deployBundle ( t , ctx , bundleRoot )
2024-02-14 18:04:45 +00:00
w , err := databricks . NewWorkspaceClient ( )
require . NoError ( t , err )
// Check that job is not bound and not updated with config from bundle
job , err := w . Jobs . Get ( ctx , jobs . GetJobRequest {
JobId : jobId ,
} )
require . NoError ( t , err )
2025-01-07 10:49:23 +00:00
require . NotEqual ( t , job . Settings . Name , "test-job-basic-" + uniqueId )
2024-02-14 18:04:45 +00:00
require . Contains ( t , job . Settings . Tasks [ 0 ] . NotebookTask . NotebookPath , "test" )
}
2024-12-13 14:47:50 +00:00
func TestGenerateAndBind ( t * testing . T ) {
2024-02-14 18:04:45 +00:00
ctx , wt := acc . WorkspaceTest ( t )
2024-12-12 21:28:04 +00:00
gt := & generateJobTest { T : wt , w : wt . W }
2024-02-14 18:04:45 +00:00
uniqueId := uuid . New ( ) . String ( )
2024-12-16 12:41:32 +00:00
bundleRoot := initTestTemplate ( t , ctx , "with_includes" , map [ string ] any {
2024-02-14 18:04:45 +00:00
"unique_id" : uniqueId ,
} )
w , err := databricks . NewWorkspaceClient ( )
require . NoError ( t , err )
jobId := gt . createTestJob ( ctx )
t . Cleanup ( func ( ) {
_ , err = w . Jobs . Get ( ctx , jobs . GetJobRequest {
JobId : jobId ,
} )
if err == nil {
gt . destroyJob ( ctx , jobId )
}
} )
2024-12-16 11:34:37 +00:00
ctx = env . Set ( ctx , "BUNDLE_ROOT" , bundleRoot )
c := testcli . NewRunner ( t , ctx , "bundle" , "generate" , "job" ,
2024-02-14 18:04:45 +00:00
"--key" , "test_job_key" ,
2025-01-07 10:49:23 +00:00
"--existing-job-id" , strconv . FormatInt ( jobId , 10 ) ,
2024-02-14 18:04:45 +00:00
"--config-dir" , filepath . Join ( bundleRoot , "resources" ) ,
"--source-dir" , filepath . Join ( bundleRoot , "src" ) )
_ , _ , err = c . Run ( )
require . NoError ( t , err )
_ , err = os . Stat ( filepath . Join ( bundleRoot , "src" , "test.py" ) )
require . NoError ( t , err )
2024-11-20 12:53:25 +00:00
matches , err := filepath . Glob ( filepath . Join ( bundleRoot , "resources" , "test_job_key.job.yml" ) )
2024-02-14 18:04:45 +00:00
require . NoError ( t , err )
require . Len ( t , matches , 1 )
2025-01-07 10:49:23 +00:00
c = testcli . NewRunner ( t , ctx , "bundle" , "deployment" , "bind" , "test_job_key" , strconv . FormatInt ( jobId , 10 ) , "--auto-approve" )
2024-02-14 18:04:45 +00:00
_ , _ , err = c . Run ( )
require . NoError ( t , err )
2024-12-16 12:41:32 +00:00
deployBundle ( t , ctx , bundleRoot )
2024-02-14 18:04:45 +00:00
2024-12-16 12:41:32 +00:00
destroyBundle ( t , ctx , bundleRoot )
2024-02-14 18:04:45 +00:00
// Check that job is bound and does not extsts after bundle is destroyed
_ , err = w . Jobs . Get ( ctx , jobs . GetJobRequest {
JobId : jobId ,
} )
require . ErrorContains ( t , err , "does not exist." )
}