package run import ( "bytes" "context" "errors" "os" "path/filepath" "testing" "time" "github.com/databricks/cli/bundle" "github.com/databricks/cli/bundle/config" "github.com/databricks/cli/bundle/config/mutator" "github.com/databricks/cli/bundle/config/resources" "github.com/databricks/cli/bundle/internal/bundletest" "github.com/databricks/cli/libs/cmdio" "github.com/databricks/cli/libs/dyn" "github.com/databricks/cli/libs/flags" "github.com/databricks/cli/libs/vfs" "github.com/databricks/databricks-sdk-go/experimental/mocks" "github.com/databricks/databricks-sdk-go/service/apps" "github.com/stretchr/testify/mock" "github.com/stretchr/testify/require" ) type testAppRunner struct { m *mocks.MockWorkspaceClient b *bundle.Bundle ctx context.Context } func (ta *testAppRunner) run(t *testing.T) { r := appRunner{ key: "my_app", bundle: ta.b, app: ta.b.Config.Resources.Apps["my_app"], } _, err := r.Run(ta.ctx, &Options{}) require.NoError(t, err) } func setupBundle(t *testing.T) (context.Context, *bundle.Bundle, *mocks.MockWorkspaceClient) { root := t.TempDir() err := os.MkdirAll(filepath.Join(root, "my_app"), 0o700) require.NoError(t, err) b := &bundle.Bundle{ BundleRootPath: root, SyncRoot: vfs.MustNew(root), Config: config.Root{ Workspace: config.Workspace{ RootPath: "/Workspace/Users/foo@bar.com/", }, Resources: config.Resources{ Apps: map[string]*resources.App{ "my_app": { App: &apps.App{ Name: "my_app", }, SourceCodePath: "./my_app", Config: map[string]any{ "command": []string{"echo", "hello"}, "env": []map[string]string{ {"name": "MY_APP", "value": "my value"}, }, }, }, }, }, }, } mwc := mocks.NewMockWorkspaceClient(t) b.SetWorkpaceClient(mwc.WorkspaceClient) bundletest.SetLocation(b, "resources.apps.my_app", []dyn.Location{{File: "./databricks.yml"}}) ctx := context.Background() ctx = cmdio.InContext(ctx, cmdio.NewIO(ctx, flags.OutputText, &bytes.Buffer{}, &bytes.Buffer{}, &bytes.Buffer{}, "", "...")) ctx = cmdio.NewContext(ctx, cmdio.NewLogger(flags.ModeAppend)) diags := bundle.Apply(ctx, b, bundle.Seq( mutator.DefineDefaultWorkspacePaths(), mutator.TranslatePaths(), )) require.Empty(t, diags) return ctx, b, mwc } func setupTestApp(t *testing.T, initialAppState apps.ApplicationState, initialComputeState apps.ComputeState) *testAppRunner { ctx, b, mwc := setupBundle(t) appApi := mwc.GetMockAppsAPI() appApi.EXPECT().Get(mock.Anything, apps.GetAppRequest{ Name: "my_app", }).Return(&apps.App{ Name: "my_app", AppStatus: &apps.ApplicationStatus{ State: initialAppState, }, ComputeStatus: &apps.ComputeStatus{ State: initialComputeState, }, }, nil) wait := &apps.WaitGetDeploymentAppSucceeded[apps.AppDeployment]{ Poll: func(_ time.Duration, _ func(*apps.AppDeployment)) (*apps.AppDeployment, error) { return nil, nil }, } appApi.EXPECT().Deploy(mock.Anything, apps.CreateAppDeploymentRequest{ AppName: "my_app", AppDeployment: &apps.AppDeployment{ Mode: apps.AppDeploymentModeSnapshot, SourceCodePath: "/Workspace/Users/foo@bar.com/files/my_app", }, }).Return(wait, nil) return &testAppRunner{ m: mwc, b: b, ctx: ctx, } } func TestAppRunStartedApp(t *testing.T) { r := setupTestApp(t, apps.ApplicationStateRunning, apps.ComputeStateActive) r.run(t) } func TestAppRunStoppedApp(t *testing.T) { r := setupTestApp(t, apps.ApplicationStateCrashed, apps.ComputeStateStopped) appsApi := r.m.GetMockAppsAPI() appsApi.EXPECT().Start(mock.Anything, apps.StartAppRequest{ Name: "my_app", }).Return(&apps.WaitGetAppActive[apps.App]{ Poll: func(_ time.Duration, _ func(*apps.App)) (*apps.App, error) { return &apps.App{ Name: "my_app", AppStatus: &apps.ApplicationStatus{ State: apps.ApplicationStateRunning, }, ComputeStatus: &apps.ComputeStatus{ State: apps.ComputeStateActive, }, }, nil }, }, nil) r.run(t) } func TestAppRunWithAnActiveDeploymentInProgress(t *testing.T) { r := setupTestApp(t, apps.ApplicationStateCrashed, apps.ComputeStateStopped) appsApi := r.m.GetMockAppsAPI() appsApi.EXPECT().Start(mock.Anything, apps.StartAppRequest{ Name: "my_app", }).Return(&apps.WaitGetAppActive[apps.App]{ Poll: func(_ time.Duration, _ func(*apps.App)) (*apps.App, error) { return &apps.App{ Name: "my_app", AppStatus: &apps.ApplicationStatus{ State: apps.ApplicationStateRunning, }, ComputeStatus: &apps.ComputeStatus{ State: apps.ComputeStateActive, }, ActiveDeployment: &apps.AppDeployment{ DeploymentId: "active_deployment_id", Status: &apps.AppDeploymentStatus{ State: apps.AppDeploymentStateInProgress, }, }, PendingDeployment: &apps.AppDeployment{ DeploymentId: "pending_deployment_id", Status: &apps.AppDeploymentStatus{ State: apps.AppDeploymentStateCancelled, }, }, }, nil }, }, nil) appsApi.EXPECT().WaitGetDeploymentAppSucceeded(mock.Anything, "my_app", "active_deployment_id", mock.Anything, mock.Anything).Return(nil, nil) r.run(t) } func TestAppDeployWithDeploymentInProgress(t *testing.T) { ctx, b, mwc := setupBundle(t) appApi := mwc.GetMockAppsAPI() appApi.EXPECT().Get(mock.Anything, apps.GetAppRequest{ Name: "my_app", }).Return(&apps.App{ Name: "my_app", AppStatus: &apps.ApplicationStatus{ State: apps.ApplicationStateRunning, }, ComputeStatus: &apps.ComputeStatus{ State: apps.ComputeStateActive, }, }, nil).Once() wait := &apps.WaitGetDeploymentAppSucceeded[apps.AppDeployment]{ Poll: func(_ time.Duration, _ func(*apps.AppDeployment)) (*apps.AppDeployment, error) { return nil, nil }, } // First deployment fails appApi.EXPECT().Deploy(mock.Anything, apps.CreateAppDeploymentRequest{ AppName: "my_app", AppDeployment: &apps.AppDeployment{ Mode: apps.AppDeploymentModeSnapshot, SourceCodePath: "/Workspace/Users/foo@bar.com/files/my_app", }, }).Return(nil, errors.New("deployment in progress")).Once() // After first deployment fails, we should get the app and wait for the deployment to complete appApi.EXPECT().Get(mock.Anything, apps.GetAppRequest{ Name: "my_app", }).Return(&apps.App{ Name: "my_app", ActiveDeployment: &apps.AppDeployment{ DeploymentId: "active_deployment_id", Status: &apps.AppDeploymentStatus{ State: apps.AppDeploymentStateInProgress, }, }, }, nil).Once() appApi.EXPECT().WaitGetDeploymentAppSucceeded(mock.Anything, "my_app", "active_deployment_id", mock.Anything, mock.Anything).Return(nil, nil) // Second one should succeeed appApi.EXPECT().Deploy(mock.Anything, apps.CreateAppDeploymentRequest{ AppName: "my_app", AppDeployment: &apps.AppDeployment{ Mode: apps.AppDeploymentModeSnapshot, SourceCodePath: "/Workspace/Users/foo@bar.com/files/my_app", }, }).Return(wait, nil).Once() r := &testAppRunner{ m: mwc, b: b, ctx: ctx, } r.run(t) } func TestStopApp(t *testing.T) { ctx, b, mwc := setupBundle(t) appsApi := mwc.GetMockAppsAPI() appsApi.EXPECT().Stop(mock.Anything, apps.StopAppRequest{ Name: "my_app", }).Return(&apps.WaitGetAppStopped[apps.App]{ Poll: func(_ time.Duration, _ func(*apps.App)) (*apps.App, error) { return &apps.App{ Name: "my_app", AppStatus: &apps.ApplicationStatus{ State: apps.ApplicationStateUnavailable, }, }, nil }, }, nil) r := appRunner{ key: "my_app", bundle: b, app: b.Config.Resources.Apps["my_app"], } err := r.Cancel(ctx) require.NoError(t, err) }