2023-09-05 11:10:37 +00:00
|
|
|
package root
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-09-11 15:32:24 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2023-09-05 11:10:37 +00:00
|
|
|
"testing"
|
2023-09-11 15:32:24 +00:00
|
|
|
"time"
|
2023-09-05 11:10:37 +00:00
|
|
|
|
2023-09-12 13:28:53 +00:00
|
|
|
"github.com/databricks/cli/internal/testutil"
|
2023-09-11 15:32:24 +00:00
|
|
|
"github.com/databricks/cli/libs/cmdio"
|
|
|
|
"github.com/databricks/databricks-sdk-go/config"
|
2023-09-05 11:10:37 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2023-09-11 15:32:24 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2023-09-05 11:10:37 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestEmptyHttpRequest(t *testing.T) {
|
2024-12-11 16:42:03 +00:00
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
2023-09-05 11:10:37 +00:00
|
|
|
req := emptyHttpRequest(ctx)
|
|
|
|
assert.Equal(t, req.Context(), ctx)
|
|
|
|
}
|
2023-09-11 15:32:24 +00:00
|
|
|
|
|
|
|
type promptFn func(ctx context.Context, cfg *config.Config, retry bool) (any, error)
|
|
|
|
|
|
|
|
var accountPromptFn = func(ctx context.Context, cfg *config.Config, retry bool) (any, error) {
|
|
|
|
return accountClientOrPrompt(ctx, cfg, retry)
|
|
|
|
}
|
|
|
|
|
|
|
|
var workspacePromptFn = func(ctx context.Context, cfg *config.Config, retry bool) (any, error) {
|
|
|
|
return workspaceClientOrPrompt(ctx, cfg, retry)
|
|
|
|
}
|
|
|
|
|
|
|
|
func expectPrompts(t *testing.T, fn promptFn, config *config.Config) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
// Channel to pass errors from the prompting function back to the test.
|
|
|
|
errch := make(chan error, 1)
|
|
|
|
|
|
|
|
ctx, io := cmdio.SetupTest(ctx)
|
|
|
|
go func() {
|
|
|
|
defer close(errch)
|
|
|
|
defer cancel()
|
|
|
|
_, err := fn(ctx, config, true)
|
|
|
|
errch <- err
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Expect a prompt
|
|
|
|
line, _, err := io.Stderr.ReadLine()
|
|
|
|
if assert.NoError(t, err, "Expected to read a line from stderr") {
|
|
|
|
assert.Contains(t, string(line), "Search:")
|
|
|
|
} else {
|
|
|
|
// If there was an error reading from stderr, the prompting function must have terminated early.
|
|
|
|
assert.NoError(t, <-errch)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func expectReturns(t *testing.T, fn promptFn, config *config.Config) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
ctx, _ = cmdio.SetupTest(ctx)
|
|
|
|
client, err := fn(ctx, config, true)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, client)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAccountClientOrPrompt(t *testing.T) {
|
2023-09-12 13:28:53 +00:00
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
2023-09-11 15:32:24 +00:00
|
|
|
dir := t.TempDir()
|
|
|
|
configFile := filepath.Join(dir, ".databrickscfg")
|
|
|
|
err := os.WriteFile(
|
|
|
|
configFile,
|
|
|
|
[]byte(`
|
|
|
|
[account-1111]
|
|
|
|
host = https://accounts.azuredatabricks.net/
|
|
|
|
account_id = 1111
|
|
|
|
token = foobar
|
|
|
|
|
|
|
|
[account-1112]
|
|
|
|
host = https://accounts.azuredatabricks.net/
|
|
|
|
account_id = 1112
|
|
|
|
token = foobar
|
|
|
|
`),
|
2024-12-12 09:28:42 +00:00
|
|
|
0o755)
|
2023-09-11 15:32:24 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
t.Setenv("DATABRICKS_CONFIG_FILE", configFile)
|
|
|
|
t.Setenv("PATH", "/nothing")
|
|
|
|
|
|
|
|
t.Run("Prompt if nothing is specified", func(t *testing.T) {
|
|
|
|
expectPrompts(t, accountPromptFn, &config.Config{})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Prompt if a workspace host is specified", func(t *testing.T) {
|
|
|
|
expectPrompts(t, accountPromptFn, &config.Config{
|
|
|
|
Host: "https://adb-1234567.89.azuredatabricks.net/",
|
|
|
|
AccountID: "1234",
|
|
|
|
Token: "foobar",
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Prompt if account ID is not specified", func(t *testing.T) {
|
|
|
|
expectPrompts(t, accountPromptFn, &config.Config{
|
|
|
|
Host: "https://accounts.azuredatabricks.net/",
|
|
|
|
Token: "foobar",
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Prompt if no credential provider can be configured", func(t *testing.T) {
|
|
|
|
expectPrompts(t, accountPromptFn, &config.Config{
|
|
|
|
Host: "https://accounts.azuredatabricks.net/",
|
|
|
|
AccountID: "1234",
|
2024-08-15 13:23:07 +00:00
|
|
|
|
|
|
|
// Force SDK to not try and lookup the tenant ID from the host.
|
|
|
|
// The host above is invalid and will not be reachable.
|
|
|
|
AzureTenantID: "nonempty",
|
2023-09-11 15:32:24 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Returns if configuration is valid", func(t *testing.T) {
|
|
|
|
expectReturns(t, accountPromptFn, &config.Config{
|
|
|
|
Host: "https://accounts.azuredatabricks.net/",
|
|
|
|
AccountID: "1234",
|
|
|
|
Token: "foobar",
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Returns if a valid profile is specified", func(t *testing.T) {
|
|
|
|
expectReturns(t, accountPromptFn, &config.Config{
|
|
|
|
Profile: "account-1111",
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWorkspaceClientOrPrompt(t *testing.T) {
|
2023-09-12 13:28:53 +00:00
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
2023-09-11 15:32:24 +00:00
|
|
|
dir := t.TempDir()
|
|
|
|
configFile := filepath.Join(dir, ".databrickscfg")
|
|
|
|
err := os.WriteFile(
|
|
|
|
configFile,
|
|
|
|
[]byte(`
|
|
|
|
[workspace-1111]
|
|
|
|
host = https://adb-1111.11.azuredatabricks.net/
|
|
|
|
token = foobar
|
|
|
|
|
|
|
|
[workspace-1112]
|
|
|
|
host = https://adb-1112.12.azuredatabricks.net/
|
|
|
|
token = foobar
|
|
|
|
`),
|
2024-12-12 09:28:42 +00:00
|
|
|
0o755)
|
2023-09-11 15:32:24 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
t.Setenv("DATABRICKS_CONFIG_FILE", configFile)
|
|
|
|
t.Setenv("PATH", "/nothing")
|
|
|
|
|
|
|
|
t.Run("Prompt if nothing is specified", func(t *testing.T) {
|
|
|
|
expectPrompts(t, workspacePromptFn, &config.Config{})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Prompt if an account host is specified", func(t *testing.T) {
|
|
|
|
expectPrompts(t, workspacePromptFn, &config.Config{
|
|
|
|
Host: "https://accounts.azuredatabricks.net/",
|
|
|
|
AccountID: "1234",
|
|
|
|
Token: "foobar",
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Prompt if no credential provider can be configured", func(t *testing.T) {
|
|
|
|
expectPrompts(t, workspacePromptFn, &config.Config{
|
|
|
|
Host: "https://adb-1111.11.azuredatabricks.net/",
|
2024-08-15 13:23:07 +00:00
|
|
|
|
|
|
|
// Force SDK to not try and lookup the tenant ID from the host.
|
|
|
|
// The host above is invalid and will not be reachable.
|
|
|
|
AzureTenantID: "nonempty",
|
2023-09-11 15:32:24 +00:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Returns if configuration is valid", func(t *testing.T) {
|
|
|
|
expectReturns(t, workspacePromptFn, &config.Config{
|
|
|
|
Host: "https://adb-1111.11.azuredatabricks.net/",
|
|
|
|
Token: "foobar",
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Returns if a valid profile is specified", func(t *testing.T) {
|
|
|
|
expectReturns(t, workspacePromptFn, &config.Config{
|
|
|
|
Profile: "workspace-1111",
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
2024-01-02 15:34:43 +00:00
|
|
|
|
|
|
|
func TestMustAccountClientWorksWithDatabricksCfg(t *testing.T) {
|
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
|
|
|
dir := t.TempDir()
|
|
|
|
configFile := filepath.Join(dir, ".databrickscfg")
|
|
|
|
err := os.WriteFile(
|
|
|
|
configFile,
|
|
|
|
[]byte(`
|
|
|
|
[account-1111]
|
|
|
|
host = https://accounts.azuredatabricks.net/
|
|
|
|
account_id = 1111
|
|
|
|
token = foobar
|
|
|
|
`),
|
2024-12-12 09:28:42 +00:00
|
|
|
0o755)
|
2024-01-02 15:34:43 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
cmd := New(context.Background())
|
|
|
|
|
|
|
|
t.Setenv("DATABRICKS_CONFIG_FILE", configFile)
|
|
|
|
err = MustAccountClient(cmd, []string{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMustAccountClientWorksWithNoDatabricksCfgButEnvironmentVariables(t *testing.T) {
|
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
|
|
|
ctx, tt := cmdio.SetupTest(context.Background())
|
|
|
|
t.Cleanup(tt.Done)
|
|
|
|
cmd := New(ctx)
|
|
|
|
t.Setenv("DATABRICKS_HOST", "https://accounts.azuredatabricks.net/")
|
|
|
|
t.Setenv("DATABRICKS_TOKEN", "foobar")
|
|
|
|
t.Setenv("DATABRICKS_ACCOUNT_ID", "1111")
|
|
|
|
|
|
|
|
err := MustAccountClient(cmd, []string{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMustAccountClientErrorsWithNoDatabricksCfg(t *testing.T) {
|
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
|
|
|
ctx, tt := cmdio.SetupTest(context.Background())
|
|
|
|
t.Cleanup(tt.Done)
|
|
|
|
cmd := New(ctx)
|
|
|
|
|
|
|
|
err := MustAccountClient(cmd, []string{})
|
|
|
|
require.ErrorContains(t, err, "no configuration file found at")
|
|
|
|
}
|
2024-04-03 08:14:04 +00:00
|
|
|
|
|
|
|
func TestMustAnyClientCanCreateWorkspaceClient(t *testing.T) {
|
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
|
|
|
dir := t.TempDir()
|
|
|
|
configFile := filepath.Join(dir, ".databrickscfg")
|
|
|
|
err := os.WriteFile(
|
|
|
|
configFile,
|
|
|
|
[]byte(`
|
|
|
|
[workspace-1111]
|
|
|
|
host = https://adb-1111.11.azuredatabricks.net/
|
|
|
|
token = foobar
|
|
|
|
`),
|
2024-12-12 09:28:42 +00:00
|
|
|
0o755)
|
2024-04-03 08:14:04 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
ctx, tt := cmdio.SetupTest(context.Background())
|
|
|
|
t.Cleanup(tt.Done)
|
|
|
|
cmd := New(ctx)
|
|
|
|
|
|
|
|
t.Setenv("DATABRICKS_CONFIG_FILE", configFile)
|
|
|
|
isAccount, err := MustAnyClient(cmd, []string{})
|
|
|
|
require.False(t, isAccount)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
w := WorkspaceClient(cmd.Context())
|
|
|
|
require.NotNil(t, w)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMustAnyClientCanCreateAccountClient(t *testing.T) {
|
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
|
|
|
dir := t.TempDir()
|
|
|
|
configFile := filepath.Join(dir, ".databrickscfg")
|
|
|
|
err := os.WriteFile(
|
|
|
|
configFile,
|
|
|
|
[]byte(`
|
|
|
|
[account-1111]
|
|
|
|
host = https://accounts.azuredatabricks.net/
|
|
|
|
account_id = 1111
|
|
|
|
token = foobar
|
|
|
|
`),
|
2024-12-12 09:28:42 +00:00
|
|
|
0o755)
|
2024-04-03 08:14:04 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
ctx, tt := cmdio.SetupTest(context.Background())
|
|
|
|
t.Cleanup(tt.Done)
|
|
|
|
cmd := New(ctx)
|
|
|
|
|
|
|
|
t.Setenv("DATABRICKS_CONFIG_FILE", configFile)
|
|
|
|
isAccount, err := MustAnyClient(cmd, []string{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.True(t, isAccount)
|
|
|
|
|
|
|
|
a := AccountClient(cmd.Context())
|
|
|
|
require.NotNil(t, a)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMustAnyClientWithEmptyDatabricksCfg(t *testing.T) {
|
|
|
|
testutil.CleanupEnvironment(t)
|
|
|
|
|
|
|
|
dir := t.TempDir()
|
|
|
|
configFile := filepath.Join(dir, ".databrickscfg")
|
|
|
|
err := os.WriteFile(
|
|
|
|
configFile,
|
|
|
|
[]byte(""), // empty file
|
2024-12-12 09:28:42 +00:00
|
|
|
0o755)
|
2024-04-03 08:14:04 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
ctx, tt := cmdio.SetupTest(context.Background())
|
|
|
|
t.Cleanup(tt.Done)
|
|
|
|
cmd := New(ctx)
|
|
|
|
|
|
|
|
t.Setenv("DATABRICKS_CONFIG_FILE", configFile)
|
|
|
|
|
|
|
|
_, err = MustAnyClient(cmd, []string{})
|
|
|
|
require.ErrorContains(t, err, "does not contain account profiles")
|
|
|
|
}
|