From d7859d63309774887b77d1efb84e438eaeb467fb Mon Sep 17 00:00:00 2001 From: Andrew Nester Date: Fri, 15 Dec 2023 17:49:19 +0100 Subject: [PATCH 1/4] Do not allow inout prompts in Git Bash terminal --- cmd/bundle/init.go | 2 +- cmd/bundle/run.go | 2 +- cmd/labs/project/installer.go | 2 +- cmd/labs/project/login.go | 8 ++++---- cmd/root/auth.go | 4 ++-- libs/cmdio/io.go | 24 ++++++++++++++++++++++++ libs/template/config.go | 2 +- 7 files changed, 34 insertions(+), 10 deletions(-) diff --git a/cmd/bundle/init.go b/cmd/bundle/init.go index ac6f49de38..18d76db11f 100644 --- a/cmd/bundle/init.go +++ b/cmd/bundle/init.go @@ -131,7 +131,7 @@ See https://docs.databricks.com/en/dev-tools/bundles/templates.html for more inf templatePath = args[0] } else { var err error - if !cmdio.IsOutTTY(ctx) || !cmdio.IsInTTY(ctx) { + if !cmdio.IsPromptSupported(ctx) { return errors.New("please specify a template") } templatePath, err = cmdio.AskSelect(ctx, "Template to use", nativeTemplateOptions()) diff --git a/cmd/bundle/run.go b/cmd/bundle/run.go index b2766b205e..c9e35aa3ba 100644 --- a/cmd/bundle/run.go +++ b/cmd/bundle/run.go @@ -45,7 +45,7 @@ func newRunCommand() *cobra.Command { } // If no arguments are specified, prompt the user to select something to run. - if len(args) == 0 && cmdio.IsInteractive(ctx) { + if len(args) == 0 && cmdio.IsPromptSupported(ctx) { // Invert completions from KEY -> NAME, to NAME -> KEY. inv := make(map[string]string) for k, v := range run.ResourceCompletionMap(b) { diff --git a/cmd/labs/project/installer.go b/cmd/labs/project/installer.go index fa6768196c..7ba2830e5a 100644 --- a/cmd/labs/project/installer.go +++ b/cmd/labs/project/installer.go @@ -157,7 +157,7 @@ func (i *installer) recordVersion(ctx context.Context) error { } func (i *installer) login(ctx context.Context) (*databricks.WorkspaceClient, error) { - if !cmdio.IsInteractive(ctx) { + if !cmdio.IsPromptSupported(ctx) { log.Debugf(ctx, "Skipping workspace profile prompts in non-interactive mode") return nil, nil } diff --git a/cmd/labs/project/login.go b/cmd/labs/project/login.go index dd2350642e..fc872bcfd1 100644 --- a/cmd/labs/project/login.go +++ b/cmd/labs/project/login.go @@ -50,7 +50,7 @@ func (lc *loginConfig) askWorkspaceProfile(ctx context.Context, cfg *config.Conf lc.WorkspaceProfile = cfg.Profile return } - if !cmdio.IsInteractive(ctx) { + if !cmdio.IsPromptSupported(ctx) { return ErrNotInTTY } lc.WorkspaceProfile, err = root.AskForWorkspaceProfile(ctx) @@ -66,7 +66,7 @@ func (lc *loginConfig) askCluster(ctx context.Context, w *databricks.WorkspaceCl lc.ClusterID = w.Config.ClusterID return } - if !cmdio.IsInteractive(ctx) { + if !cmdio.IsPromptSupported(ctx) { return ErrNotInTTY } clusterID, err := cfgpickers.AskForCluster(ctx, w, @@ -87,7 +87,7 @@ func (lc *loginConfig) askWarehouse(ctx context.Context, w *databricks.Workspace lc.WarehouseID = w.Config.WarehouseID return } - if !cmdio.IsInteractive(ctx) { + if !cmdio.IsPromptSupported(ctx) { return ErrNotInTTY } lc.WarehouseID, err = cfgpickers.AskForWarehouse(ctx, w, @@ -99,7 +99,7 @@ func (lc *loginConfig) askAccountProfile(ctx context.Context, cfg *config.Config if !lc.HasAccountLevelCommands() { return nil } - if !cmdio.IsInteractive(ctx) { + if !cmdio.IsPromptSupported(ctx) { return ErrNotInTTY } lc.AccountProfile, err = root.AskForAccountProfile(ctx) diff --git a/cmd/root/auth.go b/cmd/root/auth.go index 33f80e1fe2..2a0cb22e66 100644 --- a/cmd/root/auth.go +++ b/cmd/root/auth.go @@ -41,7 +41,7 @@ func accountClientOrPrompt(ctx context.Context, cfg *config.Config, allowPrompt } prompt := false - if allowPrompt && err != nil && cmdio.IsInteractive(ctx) { + if allowPrompt && err != nil && cmdio.IsPromptSupported(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. @@ -109,7 +109,7 @@ func workspaceClientOrPrompt(ctx context.Context, cfg *config.Config, allowPromp } prompt := false - if allowPrompt && err != nil && cmdio.IsInteractive(ctx) { + if allowPrompt && err != nil && cmdio.IsPromptSupported(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. diff --git a/libs/cmdio/io.go b/libs/cmdio/io.go index cf405a7a49..28f4ab42a7 100644 --- a/libs/cmdio/io.go +++ b/libs/cmdio/io.go @@ -88,6 +88,30 @@ func (c *cmdIO) IsTTY() bool { return isatty.IsTerminal(fd) || isatty.IsCygwinTerminal(fd) } +func IsPromptSupported(ctx context.Context) bool { + // We do not allow prompting in non-interactive mode and in Git Bash. + // Likely due to fact that Git Bash does not (correctly support ANSI escape sequences, + // we cannot use promptui package there. + // See known issues: + // - https://github.com/manifoldco/promptui/issues/208 + // - https://github.com/chzyer/readline/issues/191 + // We also do not allow prompting in non-interactive mode, + // because it's not possible to read from stdin in non-interactive mode. + return IsInteractive(ctx) && IsOutTTY(ctx) && IsInTTY(ctx) && !IsGitBash(ctx) +} + +func IsGitBash(ctx context.Context) bool { + // Check if the MSYSTEM environment variable is set to "MINGW64" + msystem := os.Getenv("MSYSTEM") + if strings.EqualFold(msystem, "MINGW64") { + // Check for typical Git Bash prompt patterns + prompt := os.Getenv("PS1") + return strings.Contains(prompt, "MINGW") + } + + return false +} + func Render(ctx context.Context, v any) error { c := fromContext(ctx) return RenderWithTemplate(ctx, v, c.template) diff --git a/libs/template/config.go b/libs/template/config.go index 2b4d19d14a..b52c0ee8bd 100644 --- a/libs/template/config.go +++ b/libs/template/config.go @@ -212,7 +212,7 @@ func (c *config) promptForValues(r *renderer) error { // Prompt user for any missing config values. Assign default values if // terminal is not TTY func (c *config) promptOrAssignDefaultValues(r *renderer) error { - if cmdio.IsOutTTY(c.ctx) && cmdio.IsInTTY(c.ctx) { + if cmdio.IsPromptSupported(c.ctx) { return c.promptForValues(r) } return c.assignDefaultValues(r) From be4420d57f114c260b88fdeccd78673b91bdaf24 Mon Sep 17 00:00:00 2001 From: Andrew Nester Date: Fri, 15 Dec 2023 17:53:15 +0100 Subject: [PATCH 2/4] updated git bash check --- libs/cmdio/io.go | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/libs/cmdio/io.go b/libs/cmdio/io.go index 28f4ab42a7..2c64d733d9 100644 --- a/libs/cmdio/io.go +++ b/libs/cmdio/io.go @@ -104,9 +104,8 @@ func IsGitBash(ctx context.Context) bool { // Check if the MSYSTEM environment variable is set to "MINGW64" msystem := os.Getenv("MSYSTEM") if strings.EqualFold(msystem, "MINGW64") { - // Check for typical Git Bash prompt patterns - prompt := os.Getenv("PS1") - return strings.Contains(prompt, "MINGW") + // Check for typical Git Bash env variable for prompts + return os.Getenv("PS1") != "" } return false From cab054541574b8671d1d3b056d442869686cae3b Mon Sep 17 00:00:00 2001 From: Andrew Nester Date: Mon, 18 Dec 2023 15:41:04 +0100 Subject: [PATCH 3/4] addressed feedback --- libs/cmdio/io.go | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/libs/cmdio/io.go b/libs/cmdio/io.go index 2c64d733d9..40d832ef61 100644 --- a/libs/cmdio/io.go +++ b/libs/cmdio/io.go @@ -10,6 +10,7 @@ import ( "time" "github.com/briandowns/spinner" + "github.com/databricks/cli/libs/env" "github.com/databricks/cli/libs/flags" "github.com/manifoldco/promptui" "github.com/mattn/go-isatty" @@ -89,7 +90,7 @@ func (c *cmdIO) IsTTY() bool { } func IsPromptSupported(ctx context.Context) bool { - // We do not allow prompting in non-interactive mode and in Git Bash. + // We do not allow prompting in non-interactive mode and in Git Bash on Windows. // Likely due to fact that Git Bash does not (correctly support ANSI escape sequences, // we cannot use promptui package there. // See known issues: @@ -97,7 +98,7 @@ func IsPromptSupported(ctx context.Context) bool { // - https://github.com/chzyer/readline/issues/191 // We also do not allow prompting in non-interactive mode, // because it's not possible to read from stdin in non-interactive mode. - return IsInteractive(ctx) && IsOutTTY(ctx) && IsInTTY(ctx) && !IsGitBash(ctx) + return (IsInteractive(ctx) || (IsOutTTY(ctx) && IsInTTY(ctx))) && !IsGitBash(ctx) } func IsGitBash(ctx context.Context) bool { @@ -105,7 +106,8 @@ func IsGitBash(ctx context.Context) bool { msystem := os.Getenv("MSYSTEM") if strings.EqualFold(msystem, "MINGW64") { // Check for typical Git Bash env variable for prompts - return os.Getenv("PS1") != "" + ps1 := env.Get(ctx, "PS1") + return strings.Contains(ps1, "MINGW") || strings.Contains(ps1, "MSYSTEM") } return false From f2e16d19a6bdb09923b9b888c79ce8f4e9ea0e3c Mon Sep 17 00:00:00 2001 From: Andrew Nester Date: Mon, 18 Dec 2023 15:49:47 +0100 Subject: [PATCH 4/4] added test --- libs/cmdio/io.go | 2 +- libs/cmdio/io_test.go | 21 +++++++++++++++++++++ 2 files changed, 22 insertions(+), 1 deletion(-) create mode 100644 libs/cmdio/io_test.go diff --git a/libs/cmdio/io.go b/libs/cmdio/io.go index 40d832ef61..8b421ef5e4 100644 --- a/libs/cmdio/io.go +++ b/libs/cmdio/io.go @@ -103,7 +103,7 @@ func IsPromptSupported(ctx context.Context) bool { func IsGitBash(ctx context.Context) bool { // Check if the MSYSTEM environment variable is set to "MINGW64" - msystem := os.Getenv("MSYSTEM") + msystem := env.Get(ctx, "MSYSTEM") if strings.EqualFold(msystem, "MINGW64") { // Check for typical Git Bash env variable for prompts ps1 := env.Get(ctx, "PS1") diff --git a/libs/cmdio/io_test.go b/libs/cmdio/io_test.go new file mode 100644 index 0000000000..1e4742043b --- /dev/null +++ b/libs/cmdio/io_test.go @@ -0,0 +1,21 @@ +package cmdio + +import ( + "context" + "testing" + + "github.com/databricks/cli/libs/env" + "github.com/stretchr/testify/assert" +) + +func TestIsPromptSupportedFalseForGitBash(t *testing.T) { + ctx := context.Background() + ctx, _ = SetupTest(ctx) + + assert.True(t, IsPromptSupported(ctx)) + + ctx = env.Set(ctx, "MSYSTEM", "MINGW64") + ctx = env.Set(ctx, "TERM", "xterm") + ctx = env.Set(ctx, "PS1", "\\[\033]0;$TITLEPREFIX:$PWD\007\\]\n\\[\033[32m\\]\\u@\\h \\[\033[35m\\]$MSYSTEM \\[\033[33m\\]\\w\\[\033[36m\\]`__git_ps1`\\[\033[0m\\]\n$") + assert.False(t, IsPromptSupported(ctx)) +}