mirror of https://github.com/databricks/cli.git
292 lines
9.0 KiB
Go
292 lines
9.0 KiB
Go
package root
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
"os"
|
|
|
|
"github.com/databricks/cli/bundle"
|
|
"github.com/databricks/cli/libs/cmdio"
|
|
"github.com/databricks/cli/libs/databrickscfg"
|
|
"github.com/databricks/databricks-sdk-go"
|
|
"github.com/databricks/databricks-sdk-go/config"
|
|
"github.com/manifoldco/promptui"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
// Placeholders to use as unique keys in context.Context.
|
|
var workspaceClient int
|
|
var accountClient int
|
|
|
|
func initProfileFlag(cmd *cobra.Command) {
|
|
cmd.PersistentFlags().StringP("profile", "p", "", "~/.databrickscfg profile")
|
|
cmd.RegisterFlagCompletionFunc("profile", databrickscfg.ProfileCompletion)
|
|
}
|
|
|
|
func profileFlagValue(cmd *cobra.Command) (string, bool) {
|
|
profileFlag := cmd.Flag("profile")
|
|
if profileFlag == nil {
|
|
return "", false
|
|
}
|
|
value := profileFlag.Value.String()
|
|
return value, value != ""
|
|
}
|
|
|
|
// Helper function to create an account client or prompt once if the given configuration is not valid.
|
|
func accountClientOrPrompt(ctx context.Context, cfg *config.Config, allowPrompt bool) (*databricks.AccountClient, error) {
|
|
a, err := databricks.NewAccountClient((*databricks.Config)(cfg))
|
|
if err == nil {
|
|
err = a.Config.Authenticate(emptyHttpRequest(ctx))
|
|
}
|
|
|
|
prompt := false
|
|
if allowPrompt && err != nil && cmdio.IsInteractive(ctx) {
|
|
// Prompt to select a profile if the current configuration is not an account client.
|
|
prompt = prompt || errors.Is(err, databricks.ErrNotAccountClient)
|
|
// Prompt to select a profile if the current configuration doesn't resolve to a credential provider.
|
|
prompt = prompt || errors.Is(err, config.ErrCannotConfigureAuth)
|
|
}
|
|
|
|
if !prompt {
|
|
// If we are not prompting, we can return early.
|
|
return a, err
|
|
}
|
|
|
|
// Try picking a profile dynamically if the current configuration is not valid.
|
|
profile, err := askForAccountProfile(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
a, err = databricks.NewAccountClient(&databricks.Config{Profile: profile})
|
|
if err == nil {
|
|
err = a.Config.Authenticate(emptyHttpRequest(ctx))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return a, nil
|
|
}
|
|
|
|
func MustAccountClient(cmd *cobra.Command, args []string) error {
|
|
cfg := &config.Config{}
|
|
|
|
// The command-line profile flag takes precedence over DATABRICKS_CONFIG_PROFILE.
|
|
profile, hasProfileFlag := profileFlagValue(cmd)
|
|
if hasProfileFlag {
|
|
cfg.Profile = profile
|
|
}
|
|
|
|
if cfg.Profile == "" {
|
|
// account-level CLI was not really done before, so here are the assumptions:
|
|
// 1. only admins will have account configured
|
|
// 2. 99% of admins will have access to just one account
|
|
// hence, we don't need to create a special "DEFAULT_ACCOUNT" profile yet
|
|
_, profiles, err := databrickscfg.LoadProfiles(databrickscfg.MatchAccountProfiles)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(profiles) == 1 {
|
|
cfg.Profile = profiles[0].Name
|
|
}
|
|
}
|
|
|
|
noPrompt, ok := cmd.Context().Value(noPromptKey).(bool)
|
|
allowPrompt := !hasProfileFlag && (!ok || !noPrompt)
|
|
a, err := accountClientOrPrompt(cmd.Context(), cfg, allowPrompt)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
cmd.SetContext(context.WithValue(cmd.Context(), &accountClient, a))
|
|
return nil
|
|
}
|
|
|
|
type noPrompt int
|
|
|
|
var noPromptKey noPrompt
|
|
|
|
// NoPrompt allows to skip prompt for profile configuration in MustWorkspaceClient.
|
|
//
|
|
// When calling MustWorkspaceClient we want to be able to customise if to show prompt or not.
|
|
// Since we can't change function interface, in the code we only have an access to `cmd“ object.
|
|
// Command struct does not have any state flag which indicates that it's being called in completion mode and
|
|
// thus the Context object seems to be the only viable option for us to configure prompt behaviour based on
|
|
// the context it's executed from.
|
|
func NoPrompt(ctx context.Context) context.Context {
|
|
return context.WithValue(ctx, noPromptKey, true)
|
|
}
|
|
|
|
// Helper function to create a workspace client or prompt once if the given configuration is not valid.
|
|
func workspaceClientOrPrompt(ctx context.Context, cfg *config.Config, allowPrompt bool) (*databricks.WorkspaceClient, error) {
|
|
w, err := databricks.NewWorkspaceClient((*databricks.Config)(cfg))
|
|
if err == nil {
|
|
err = w.Config.Authenticate(emptyHttpRequest(ctx))
|
|
}
|
|
|
|
prompt := false
|
|
if allowPrompt && err != nil && cmdio.IsInteractive(ctx) {
|
|
// Prompt to select a profile if the current configuration is not a workspace client.
|
|
prompt = prompt || errors.Is(err, databricks.ErrNotWorkspaceClient)
|
|
// Prompt to select a profile if the current configuration doesn't resolve to a credential provider.
|
|
prompt = prompt || errors.Is(err, config.ErrCannotConfigureAuth)
|
|
}
|
|
|
|
if !prompt {
|
|
// If we are not prompting, we can return early.
|
|
return w, err
|
|
}
|
|
|
|
// Try picking a profile dynamically if the current configuration is not valid.
|
|
profile, err := askForWorkspaceProfile(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
w, err = databricks.NewWorkspaceClient(&databricks.Config{Profile: profile})
|
|
if err == nil {
|
|
err = w.Config.Authenticate(emptyHttpRequest(ctx))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return w, nil
|
|
}
|
|
|
|
func MustWorkspaceClient(cmd *cobra.Command, args []string) error {
|
|
cfg := &config.Config{}
|
|
|
|
// The command-line profile flag takes precedence over DATABRICKS_CONFIG_PROFILE.
|
|
profile, hasProfileFlag := profileFlagValue(cmd)
|
|
if hasProfileFlag {
|
|
cfg.Profile = profile
|
|
}
|
|
|
|
// try configuring a bundle
|
|
err := TryConfigureBundle(cmd, args)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// and load the config from there
|
|
currentBundle := bundle.GetOrNil(cmd.Context())
|
|
if currentBundle != nil {
|
|
cfg = currentBundle.WorkspaceClient().Config
|
|
}
|
|
|
|
allowPrompt := !hasProfileFlag
|
|
w, err := workspaceClientOrPrompt(cmd.Context(), cfg, allowPrompt)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ctx := cmd.Context()
|
|
ctx = context.WithValue(ctx, &workspaceClient, w)
|
|
cmd.SetContext(ctx)
|
|
return nil
|
|
}
|
|
|
|
func SetWorkspaceClient(ctx context.Context, w *databricks.WorkspaceClient) context.Context {
|
|
return context.WithValue(ctx, &workspaceClient, w)
|
|
}
|
|
|
|
func transformLoadError(path string, err error) error {
|
|
if os.IsNotExist(err) {
|
|
return fmt.Errorf("no configuration file found at %s; please create one first", path)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func askForWorkspaceProfile(ctx context.Context) (string, error) {
|
|
path, err := databrickscfg.GetPath()
|
|
if err != nil {
|
|
return "", fmt.Errorf("cannot determine Databricks config file path: %w", err)
|
|
}
|
|
file, profiles, err := databrickscfg.LoadProfiles(databrickscfg.MatchWorkspaceProfiles)
|
|
if err != nil {
|
|
return "", transformLoadError(path, err)
|
|
}
|
|
switch len(profiles) {
|
|
case 0:
|
|
return "", fmt.Errorf("%s does not contain workspace profiles; please create one first", path)
|
|
case 1:
|
|
return profiles[0].Name, nil
|
|
}
|
|
i, _, err := cmdio.RunSelect(ctx, &promptui.Select{
|
|
Label: fmt.Sprintf("Workspace profiles defined in %s", file),
|
|
Items: profiles,
|
|
Searcher: profiles.SearchCaseInsensitive,
|
|
StartInSearchMode: true,
|
|
Templates: &promptui.SelectTemplates{
|
|
Label: "{{ . | faint }}",
|
|
Active: `{{.Name | bold}} ({{.Host|faint}})`,
|
|
Inactive: `{{.Name}}`,
|
|
Selected: `{{ "Using workspace profile" | faint }}: {{ .Name | bold }}`,
|
|
},
|
|
})
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return profiles[i].Name, nil
|
|
}
|
|
|
|
func askForAccountProfile(ctx context.Context) (string, error) {
|
|
path, err := databrickscfg.GetPath()
|
|
if err != nil {
|
|
return "", fmt.Errorf("cannot determine Databricks config file path: %w", err)
|
|
}
|
|
file, profiles, err := databrickscfg.LoadProfiles(databrickscfg.MatchAccountProfiles)
|
|
if err != nil {
|
|
return "", transformLoadError(path, err)
|
|
}
|
|
switch len(profiles) {
|
|
case 0:
|
|
return "", fmt.Errorf("%s does not contain account profiles; please create one first", path)
|
|
case 1:
|
|
return profiles[0].Name, nil
|
|
}
|
|
i, _, err := cmdio.RunSelect(ctx, &promptui.Select{
|
|
Label: fmt.Sprintf("Account profiles defined in %s", file),
|
|
Items: profiles,
|
|
Searcher: profiles.SearchCaseInsensitive,
|
|
StartInSearchMode: true,
|
|
Templates: &promptui.SelectTemplates{
|
|
Label: "{{ . | faint }}",
|
|
Active: `{{.Name | bold}} ({{.AccountID|faint}} {{.Cloud|faint}})`,
|
|
Inactive: `{{.Name}}`,
|
|
Selected: `{{ "Using account profile" | faint }}: {{ .Name | bold }}`,
|
|
},
|
|
})
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return profiles[i].Name, nil
|
|
}
|
|
|
|
// To verify that a client is configured correctly, we pass an empty HTTP request
|
|
// to a client's `config.Authenticate` function. Note: this functionality
|
|
// should be supported by the SDK itself.
|
|
func emptyHttpRequest(ctx context.Context) *http.Request {
|
|
req, err := http.NewRequestWithContext(ctx, "", "", nil)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return req
|
|
}
|
|
|
|
func WorkspaceClient(ctx context.Context) *databricks.WorkspaceClient {
|
|
w, ok := ctx.Value(&workspaceClient).(*databricks.WorkspaceClient)
|
|
if !ok {
|
|
panic("cannot get *databricks.WorkspaceClient. Please report it as a bug")
|
|
}
|
|
return w
|
|
}
|
|
|
|
func AccountClient(ctx context.Context) *databricks.AccountClient {
|
|
a, ok := ctx.Value(&accountClient).(*databricks.AccountClient)
|
|
if !ok {
|
|
panic("cannot get *databricks.AccountClient. Please report it as a bug")
|
|
}
|
|
return a
|
|
}
|