2024-12-13 14:38:58 +00:00
|
|
|
package bundle_test
|
2023-09-07 14:08:16 +00:00
|
|
|
|
|
|
|
import (
|
2024-08-22 15:04:26 +00:00
|
|
|
"bytes"
|
2023-09-07 14:08:16 +00:00
|
|
|
"context"
|
|
|
|
"encoding/json"
|
2024-03-18 14:41:58 +00:00
|
|
|
"fmt"
|
2023-09-07 14:08:16 +00:00
|
|
|
"os"
|
2024-08-22 15:04:26 +00:00
|
|
|
"os/exec"
|
2023-09-07 14:08:16 +00:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
2024-10-29 09:11:08 +00:00
|
|
|
"github.com/databricks/cli/bundle"
|
2023-09-07 14:08:16 +00:00
|
|
|
"github.com/databricks/cli/cmd/root"
|
2024-12-12 16:48:51 +00:00
|
|
|
"github.com/databricks/cli/internal/testcli"
|
2024-12-12 14:42:15 +00:00
|
|
|
"github.com/databricks/cli/internal/testutil"
|
2023-09-07 14:08:16 +00:00
|
|
|
"github.com/databricks/cli/libs/cmdio"
|
2024-08-09 15:13:31 +00:00
|
|
|
"github.com/databricks/cli/libs/env"
|
2023-09-07 14:08:16 +00:00
|
|
|
"github.com/databricks/cli/libs/flags"
|
2024-12-11 08:44:22 +00:00
|
|
|
"github.com/databricks/cli/libs/folders"
|
2023-09-07 14:08:16 +00:00
|
|
|
"github.com/databricks/cli/libs/template"
|
2024-03-18 14:41:58 +00:00
|
|
|
"github.com/databricks/databricks-sdk-go"
|
|
|
|
"github.com/stretchr/testify/require"
|
2023-09-07 14:08:16 +00:00
|
|
|
)
|
|
|
|
|
2024-04-19 11:31:54 +00:00
|
|
|
const defaultSparkVersion = "13.3.x-snapshot-scala2.12"
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
func initTestTemplate(t testutil.TestingT, ctx context.Context, templateName string, config map[string]any) string {
|
2024-07-16 08:57:04 +00:00
|
|
|
bundleRoot := t.TempDir()
|
|
|
|
return initTestTemplateWithBundleRoot(t, ctx, templateName, config, bundleRoot)
|
|
|
|
}
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
func initTestTemplateWithBundleRoot(t testutil.TestingT, ctx context.Context, templateName string, config map[string]any, bundleRoot string) string {
|
2023-09-07 14:08:16 +00:00
|
|
|
templateRoot := filepath.Join("bundles", templateName)
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
configFilePath := writeConfigFile(t, config)
|
2023-09-07 14:08:16 +00:00
|
|
|
|
2024-02-07 11:18:56 +00:00
|
|
|
ctx = root.SetWorkspaceClient(ctx, nil)
|
2024-12-16 11:34:37 +00:00
|
|
|
cmd := cmdio.NewIO(ctx, flags.OutputJSON, strings.NewReader(""), os.Stdout, os.Stderr, "", "bundles")
|
2023-09-07 14:08:16 +00:00
|
|
|
ctx = cmdio.InContext(ctx, cmd)
|
|
|
|
|
2025-01-20 12:09:28 +00:00
|
|
|
r := template.Resolver{
|
|
|
|
TemplatePathOrUrl: templateRoot,
|
|
|
|
ConfigFile: configFilePath,
|
|
|
|
OutputDir: bundleRoot,
|
|
|
|
}
|
|
|
|
|
|
|
|
tmpl, err := r.Resolve(ctx)
|
2024-11-20 11:42:23 +00:00
|
|
|
require.NoError(t, err)
|
2025-01-20 12:09:28 +00:00
|
|
|
defer tmpl.Reader.Cleanup(ctx)
|
|
|
|
|
|
|
|
err = tmpl.Writer.Materialize(ctx, tmpl.Reader)
|
2024-12-16 12:41:32 +00:00
|
|
|
require.NoError(t, err)
|
2025-01-20 12:09:28 +00:00
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
return bundleRoot
|
2023-09-07 14:08:16 +00:00
|
|
|
}
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
func writeConfigFile(t testutil.TestingT, config map[string]any) string {
|
2023-09-07 14:08:16 +00:00
|
|
|
bytes, err := json.Marshal(config)
|
2024-12-16 12:41:32 +00:00
|
|
|
require.NoError(t, err)
|
2023-09-07 14:08:16 +00:00
|
|
|
|
|
|
|
dir := t.TempDir()
|
|
|
|
filepath := filepath.Join(dir, "config.json")
|
|
|
|
t.Log("Configuration for template: ", string(bytes))
|
|
|
|
|
|
|
|
err = os.WriteFile(filepath, bytes, 0o644)
|
2024-12-16 12:41:32 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
return filepath
|
2023-09-07 14:08:16 +00:00
|
|
|
}
|
|
|
|
|
2024-12-12 14:42:15 +00:00
|
|
|
func validateBundle(t testutil.TestingT, ctx context.Context, path string) ([]byte, error) {
|
2024-08-09 15:13:31 +00:00
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
2024-12-16 11:34:37 +00:00
|
|
|
c := testcli.NewRunner(t, ctx, "bundle", "validate", "--output", "json")
|
2024-06-18 15:04:20 +00:00
|
|
|
stdout, _, err := c.Run()
|
|
|
|
return stdout.Bytes(), err
|
|
|
|
}
|
|
|
|
|
2024-12-12 14:42:15 +00:00
|
|
|
func mustValidateBundle(t testutil.TestingT, ctx context.Context, path string) []byte {
|
2024-10-29 09:11:08 +00:00
|
|
|
data, err := validateBundle(t, ctx, path)
|
|
|
|
require.NoError(t, err)
|
|
|
|
return data
|
|
|
|
}
|
|
|
|
|
2024-12-12 14:42:15 +00:00
|
|
|
func unmarshalConfig(t testutil.TestingT, data []byte) *bundle.Bundle {
|
2024-10-29 09:11:08 +00:00
|
|
|
bundle := &bundle.Bundle{}
|
|
|
|
err := json.Unmarshal(data, &bundle.Config)
|
|
|
|
require.NoError(t, err)
|
|
|
|
return bundle
|
|
|
|
}
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
func deployBundle(t testutil.TestingT, ctx context.Context, path string) {
|
2024-08-09 15:13:31 +00:00
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
2024-12-16 11:34:37 +00:00
|
|
|
c := testcli.NewRunner(t, ctx, "bundle", "deploy", "--force-lock", "--auto-approve")
|
2023-09-07 14:08:16 +00:00
|
|
|
_, _, err := c.Run()
|
2024-12-16 12:41:32 +00:00
|
|
|
require.NoError(t, err)
|
2023-09-07 14:08:16 +00:00
|
|
|
}
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
func deployBundleWithArgsErr(t testutil.TestingT, ctx context.Context, path string, args ...string) (string, string, error) {
|
2024-10-29 09:11:08 +00:00
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
|
|
|
args = append([]string{"bundle", "deploy"}, args...)
|
2024-12-16 11:34:37 +00:00
|
|
|
c := testcli.NewRunner(t, ctx, args...)
|
2024-10-29 09:11:08 +00:00
|
|
|
stdout, stderr, err := c.Run()
|
|
|
|
return stdout.String(), stderr.String(), err
|
|
|
|
}
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
func deployBundleWithArgs(t testutil.TestingT, ctx context.Context, path string, args ...string) (string, string) {
|
|
|
|
stdout, stderr, err := deployBundleWithArgsErr(t, ctx, path, args...)
|
|
|
|
require.NoError(t, err)
|
|
|
|
return stdout, stderr
|
|
|
|
}
|
|
|
|
|
|
|
|
func deployBundleWithFlags(t testutil.TestingT, ctx context.Context, path string, flags []string) {
|
2024-08-09 15:13:31 +00:00
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
2024-03-18 15:39:18 +00:00
|
|
|
args := []string{"bundle", "deploy", "--force-lock"}
|
|
|
|
args = append(args, flags...)
|
2024-12-16 11:34:37 +00:00
|
|
|
c := testcli.NewRunner(t, ctx, args...)
|
2024-03-18 15:39:18 +00:00
|
|
|
_, _, err := c.Run()
|
2024-12-16 12:41:32 +00:00
|
|
|
require.NoError(t, err)
|
2024-03-18 15:39:18 +00:00
|
|
|
}
|
|
|
|
|
2024-12-12 14:42:15 +00:00
|
|
|
func runResource(t testutil.TestingT, ctx context.Context, path, key string) (string, error) {
|
2024-08-09 15:13:31 +00:00
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
2023-09-07 14:08:16 +00:00
|
|
|
ctx = cmdio.NewContext(ctx, cmdio.Default())
|
|
|
|
|
2024-12-16 11:34:37 +00:00
|
|
|
c := testcli.NewRunner(t, ctx, "bundle", "run", key)
|
2023-09-07 14:08:16 +00:00
|
|
|
stdout, _, err := c.Run()
|
|
|
|
return stdout.String(), err
|
|
|
|
}
|
|
|
|
|
2025-01-13 16:43:48 +00:00
|
|
|
func runResourceWithStderr(t testutil.TestingT, ctx context.Context, path, key string) (string, string) {
|
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
|
|
|
ctx = cmdio.NewContext(ctx, cmdio.Default())
|
|
|
|
|
|
|
|
c := testcli.NewRunner(t, ctx, "bundle", "run", key)
|
|
|
|
stdout, stderr, err := c.Run()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
return stdout.String(), stderr.String()
|
|
|
|
}
|
|
|
|
|
2024-12-12 14:42:15 +00:00
|
|
|
func runResourceWithParams(t testutil.TestingT, ctx context.Context, path, key string, params ...string) (string, error) {
|
2024-08-09 15:13:31 +00:00
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
2023-12-01 10:35:20 +00:00
|
|
|
ctx = cmdio.NewContext(ctx, cmdio.Default())
|
|
|
|
|
|
|
|
args := make([]string, 0)
|
|
|
|
args = append(args, "bundle", "run", key)
|
|
|
|
args = append(args, params...)
|
2024-12-16 11:34:37 +00:00
|
|
|
c := testcli.NewRunner(t, ctx, args...)
|
2023-12-01 10:35:20 +00:00
|
|
|
stdout, _, err := c.Run()
|
|
|
|
return stdout.String(), err
|
|
|
|
}
|
|
|
|
|
2024-12-16 12:41:32 +00:00
|
|
|
func destroyBundle(t testutil.TestingT, ctx context.Context, path string) {
|
2024-08-09 15:13:31 +00:00
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", path)
|
2024-12-16 11:34:37 +00:00
|
|
|
c := testcli.NewRunner(t, ctx, "bundle", "destroy", "--auto-approve")
|
2023-09-07 14:08:16 +00:00
|
|
|
_, _, err := c.Run()
|
2024-12-16 12:41:32 +00:00
|
|
|
require.NoError(t, err)
|
2023-09-07 14:08:16 +00:00
|
|
|
}
|
2024-03-18 14:41:58 +00:00
|
|
|
|
2024-12-12 14:42:15 +00:00
|
|
|
func getBundleRemoteRootPath(w *databricks.WorkspaceClient, t testutil.TestingT, uniqueId string) string {
|
2024-03-18 14:41:58 +00:00
|
|
|
// Compute root path for the bundle deployment
|
|
|
|
me, err := w.CurrentUser.Me(context.Background())
|
|
|
|
require.NoError(t, err)
|
2024-10-02 15:34:00 +00:00
|
|
|
root := fmt.Sprintf("/Workspace/Users/%s/.bundle/%s", me.UserName, uniqueId)
|
2024-03-18 14:41:58 +00:00
|
|
|
return root
|
|
|
|
}
|
2024-08-22 15:04:26 +00:00
|
|
|
|
2024-12-16 11:34:37 +00:00
|
|
|
func blackBoxRun(t testutil.TestingT, ctx context.Context, root string, args ...string) (stdout, stderr string) {
|
2024-12-11 08:44:22 +00:00
|
|
|
gitRoot, err := folders.FindDirWithLeaf(".", ".git")
|
2024-08-22 15:04:26 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Create the command
|
|
|
|
cmd := exec.Command("go", append([]string{"run", "main.go"}, args...)...)
|
2024-12-11 08:44:22 +00:00
|
|
|
cmd.Dir = gitRoot
|
2024-08-22 15:04:26 +00:00
|
|
|
|
2024-12-16 11:34:37 +00:00
|
|
|
// Configure the environment
|
|
|
|
ctx = env.Set(ctx, "BUNDLE_ROOT", root)
|
|
|
|
for key, value := range env.All(ctx) {
|
|
|
|
cmd.Env = append(cmd.Env, fmt.Sprintf("%s=%s", key, value))
|
|
|
|
}
|
|
|
|
|
2024-08-22 15:04:26 +00:00
|
|
|
// Create buffers to capture output
|
|
|
|
var outBuffer, errBuffer bytes.Buffer
|
|
|
|
cmd.Stdout = &outBuffer
|
|
|
|
cmd.Stderr = &errBuffer
|
|
|
|
|
|
|
|
// Run the command
|
|
|
|
err = cmd.Run()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Get the output
|
|
|
|
stdout = outBuffer.String()
|
|
|
|
stderr = errBuffer.String()
|
|
|
|
return
|
|
|
|
}
|