From 9f24330c3a7f8cb8bba08ac0782849204443526c Mon Sep 17 00:00:00 2001 From: Valentin Kiselev Date: Mon, 29 Aug 2022 22:24:50 +0300 Subject: [PATCH 1/4] fix: Pass STDIN by default Signed-off-by: Valentin Kiselev --- go.mod | 1 + go.sum | 2 ++ internal/lefthook/runner/execute_unix.go | 11 +++++++++++ internal/lefthook/runner/execute_windows.go | 1 + 4 files changed, 15 insertions(+) diff --git a/go.mod b/go.mod index 0a80a6ce..8053fd72 100644 --- a/go.mod +++ b/go.mod @@ -32,6 +32,7 @@ require ( github.com/spf13/pflag v1.0.5 // indirect github.com/subosito/gotenv v1.3.0 // indirect golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a // indirect + golang.org/x/term v0.0.0-20220722155259-a9ba230a4035 // indirect golang.org/x/text v0.3.7 // indirect gopkg.in/ini.v1 v1.66.4 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect diff --git a/go.sum b/go.sum index 5363d9f9..94e4ddd5 100644 --- a/go.sum +++ b/go.sum @@ -337,6 +337,8 @@ golang.org/x/sys v0.0.0-20220412211240-33da011f77ad/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a h1:dGzPydgVsqGcTRVwiLJ1jVbufYwmzD3LfVPLKsKg+0k= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/term v0.0.0-20220722155259-a9ba230a4035 h1:Q5284mrmYTpACcm+eAKjKJH48BBwSyfJqmmGDTtT8Vc= +golang.org/x/term v0.0.0-20220722155259-a9ba230a4035/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= diff --git a/internal/lefthook/runner/execute_unix.go b/internal/lefthook/runner/execute_unix.go index 1f4d2370..9ff5d345 100644 --- a/internal/lefthook/runner/execute_unix.go +++ b/internal/lefthook/runner/execute_unix.go @@ -12,6 +12,7 @@ import ( "strings" "github.com/creack/pty" + "golang.org/x/term" ) type CommandExecutor struct{} @@ -25,6 +26,15 @@ func (e CommandExecutor) Execute(root string, args []string) (*bytes.Buffer, err if err != nil { return nil, err } + + // Pass raw STDIN + oldState, err := term.MakeRaw(int(os.Stdin.Fd())) + if err != nil { + return nil, err + } + defer func() { _ = term.Restore(int(os.Stdin.Fd()), oldState) }() + go func() { _, _ = io.Copy(ptyOut, os.Stdin) }() + defer func() { _ = ptyOut.Close() }() out := bytes.NewBuffer(make([]byte, 0)) _, _ = io.Copy(out, ptyOut) @@ -36,6 +46,7 @@ func (e CommandExecutor) RawExecute(command string, args ...string) error { cmd := exec.Command(command, args...) cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr + cmd.Stdin = os.Stdin return cmd.Run() } diff --git a/internal/lefthook/runner/execute_windows.go b/internal/lefthook/runner/execute_windows.go index f695d9bc..8d6f7b0b 100644 --- a/internal/lefthook/runner/execute_windows.go +++ b/internal/lefthook/runner/execute_windows.go @@ -36,6 +36,7 @@ func (e CommandExecutor) RawExecute(command string, args ...string) error { cmd := exec.Command(command, args...) cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr + cmd.Stdin = os.Stdin return cmd.Run() } From 9ab7966e3e9804a150dede25f4d8b8e65b177cf4 Mon Sep 17 00:00:00 2001 From: Valentin Kiselev Date: Sat, 3 Sep 2022 12:35:59 +0300 Subject: [PATCH 2/4] fix: Copy STDIN to PTY when there is something Signed-off-by: Valentin Kiselev --- go.mod | 1 - go.sum | 2 -- internal/lefthook/runner/execute_unix.go | 18 ++++++------------ internal/lefthook/runner/execute_windows.go | 4 +++- 4 files changed, 9 insertions(+), 16 deletions(-) diff --git a/go.mod b/go.mod index 8053fd72..0a80a6ce 100644 --- a/go.mod +++ b/go.mod @@ -32,7 +32,6 @@ require ( github.com/spf13/pflag v1.0.5 // indirect github.com/subosito/gotenv v1.3.0 // indirect golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a // indirect - golang.org/x/term v0.0.0-20220722155259-a9ba230a4035 // indirect golang.org/x/text v0.3.7 // indirect gopkg.in/ini.v1 v1.66.4 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect diff --git a/go.sum b/go.sum index 94e4ddd5..5363d9f9 100644 --- a/go.sum +++ b/go.sum @@ -337,8 +337,6 @@ golang.org/x/sys v0.0.0-20220412211240-33da011f77ad/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a h1:dGzPydgVsqGcTRVwiLJ1jVbufYwmzD3LfVPLKsKg+0k= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/term v0.0.0-20220722155259-a9ba230a4035 h1:Q5284mrmYTpACcm+eAKjKJH48BBwSyfJqmmGDTtT8Vc= -golang.org/x/term v0.0.0-20220722155259-a9ba230a4035/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= diff --git a/internal/lefthook/runner/execute_unix.go b/internal/lefthook/runner/execute_unix.go index 9ff5d345..d1d9c26e 100644 --- a/internal/lefthook/runner/execute_unix.go +++ b/internal/lefthook/runner/execute_unix.go @@ -12,7 +12,6 @@ import ( "strings" "github.com/creack/pty" - "golang.org/x/term" ) type CommandExecutor struct{} @@ -22,22 +21,18 @@ func (e CommandExecutor) Execute(root string, args []string) (*bytes.Buffer, err rootDir, _ := filepath.Abs(root) command.Dir = rootDir - ptyOut, err := pty.Start(command) + p, err := pty.Start(command) if err != nil { return nil, err } - // Pass raw STDIN - oldState, err := term.MakeRaw(int(os.Stdin.Fd())) - if err != nil { - return nil, err - } - defer func() { _ = term.Restore(int(os.Stdin.Fd()), oldState) }() - go func() { _, _ = io.Copy(ptyOut, os.Stdin) }() + defer func() { _ = p.Close() }() + defer command.Process.Kill() + + go func() { io.Copy(p, os.Stdin) }() - defer func() { _ = ptyOut.Close() }() out := bytes.NewBuffer(make([]byte, 0)) - _, _ = io.Copy(out, ptyOut) + _, _ = io.Copy(out, p) return out, command.Wait() } @@ -46,7 +41,6 @@ func (e CommandExecutor) RawExecute(command string, args ...string) error { cmd := exec.Command(command, args...) cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr - cmd.Stdin = os.Stdin return cmd.Run() } diff --git a/internal/lefthook/runner/execute_windows.go b/internal/lefthook/runner/execute_windows.go index 8d6f7b0b..d6024416 100644 --- a/internal/lefthook/runner/execute_windows.go +++ b/internal/lefthook/runner/execute_windows.go @@ -29,6 +29,9 @@ func (e CommandExecutor) Execute(root string, args []string) (*bytes.Buffer, err if err != nil { return nil, err } + + defer command.Process.Kill() + return &out, command.Wait() } @@ -36,7 +39,6 @@ func (e CommandExecutor) RawExecute(command string, args ...string) error { cmd := exec.Command(command, args...) cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr - cmd.Stdin = os.Stdin return cmd.Run() } From c6e66ba66b363a32a610d62883d2b33aeeb67bf7 Mon Sep 17 00:00:00 2001 From: Valentin Kiselev Date: Mon, 5 Sep 2022 18:41:08 +0300 Subject: [PATCH 3/4] fix: Linter complains Signed-off-by: Valentin Kiselev --- internal/lefthook/runner/execute_unix.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/internal/lefthook/runner/execute_unix.go b/internal/lefthook/runner/execute_unix.go index d1d9c26e..e6265884 100644 --- a/internal/lefthook/runner/execute_unix.go +++ b/internal/lefthook/runner/execute_unix.go @@ -27,9 +27,9 @@ func (e CommandExecutor) Execute(root string, args []string) (*bytes.Buffer, err } defer func() { _ = p.Close() }() - defer command.Process.Kill() + defer func() { _ = command.Process.Kill() }() - go func() { io.Copy(p, os.Stdin) }() + go func() { _, _ = io.Copy(p, os.Stdin) }() out := bytes.NewBuffer(make([]byte, 0)) _, _ = io.Copy(out, p) From a09ecde4bf6c34259708b102456bebc5270ae410 Mon Sep 17 00:00:00 2001 From: Valentin Kiselev Date: Thu, 15 Sep 2022 11:12:14 +0300 Subject: [PATCH 4/4] fix: Remove tty hack, add interactive option --- internal/config/command.go | 3 +- internal/config/script.go | 17 +++++---- internal/lefthook/runner/execute_unix.go | 17 +++++++-- internal/lefthook/runner/execute_windows.go | 2 +- internal/lefthook/runner/executor.go | 2 +- internal/lefthook/runner/runner.go | 39 ++++++++++++++++----- internal/lefthook/runner/runner_test.go | 2 +- internal/templates/hook.tmpl | 4 --- 8 files changed, 61 insertions(+), 25 deletions(-) diff --git a/internal/config/command.go b/internal/config/command.go index 8d4a73ed..8d618424 100644 --- a/internal/config/command.go +++ b/internal/config/command.go @@ -22,7 +22,8 @@ type Command struct { Root string `mapstructure:"root"` Exclude string `mapstructure:"exclude"` - FailText string `mapstructure:"fail_text"` + FailText string `mapstructure:"fail_text"` + Interactive bool `mapstructure:"interactive"` // DEPRECATED, will be deleted in 1.2.0 Runner string `mapstructure:"runner"` diff --git a/internal/config/script.go b/internal/config/script.go index ca774824..45186c7c 100644 --- a/internal/config/script.go +++ b/internal/config/script.go @@ -15,7 +15,8 @@ type Script struct { Skip interface{} `mapstructure:"skip"` Tags []string `mapstructure:"tags"` - FailText string `mapstructure:"fail_text"` + FailText string `mapstructure:"fail_text"` + Interactive bool `mapstructure:"interactive"` // DEPRECATED Run string `mapstructure:"run"` @@ -116,17 +117,21 @@ func unmarshalScripts(s map[string]interface{}) (map[string]*Script, error) { // // ```yaml // scripts: -// "example.sh": -// runner: bash +// +// "example.sh": +// runner: bash +// // ``` // // Unmarshals into this: // // ```yaml // scripts: -// example: -// sh: -// runner: bash +// +// example: +// sh: +// runner: bash +// // ``` // // This is not an expected behavior and cannot be controlled yet diff --git a/internal/lefthook/runner/execute_unix.go b/internal/lefthook/runner/execute_unix.go index e6265884..c4dd1752 100644 --- a/internal/lefthook/runner/execute_unix.go +++ b/internal/lefthook/runner/execute_unix.go @@ -12,11 +12,24 @@ import ( "strings" "github.com/creack/pty" + "github.com/mattn/go-isatty" + + "github.com/evilmartians/lefthook/internal/log" ) type CommandExecutor struct{} -func (e CommandExecutor) Execute(root string, args []string) (*bytes.Buffer, error) { +func (e CommandExecutor) Execute(root string, args []string, interactive bool) (*bytes.Buffer, error) { + stdin := os.Stdin + if interactive && !isatty.IsTerminal(os.Stdin.Fd()) { + tty, err := os.Open("/dev/tty") + if err == nil { + defer tty.Close() + stdin = tty + } else { + log.Errorf("Couldn't enable TTY input: %s\n", err) + } + } command := exec.Command("sh", "-c", strings.Join(args, " ")) rootDir, _ := filepath.Abs(root) command.Dir = rootDir @@ -29,7 +42,7 @@ func (e CommandExecutor) Execute(root string, args []string) (*bytes.Buffer, err defer func() { _ = p.Close() }() defer func() { _ = command.Process.Kill() }() - go func() { _, _ = io.Copy(p, os.Stdin) }() + go func() { _, _ = io.Copy(p, stdin) }() out := bytes.NewBuffer(make([]byte, 0)) _, _ = io.Copy(out, p) diff --git a/internal/lefthook/runner/execute_windows.go b/internal/lefthook/runner/execute_windows.go index d6024416..48a37988 100644 --- a/internal/lefthook/runner/execute_windows.go +++ b/internal/lefthook/runner/execute_windows.go @@ -11,7 +11,7 @@ import ( type CommandExecutor struct{} -func (e CommandExecutor) Execute(root string, args []string) (*bytes.Buffer, error) { +func (e CommandExecutor) Execute(root string, args []string, _ bool) (*bytes.Buffer, error) { command := exec.Command(args[0]) command.SysProcAttr = &syscall.SysProcAttr{ CmdLine: strings.Join(args, " "), diff --git a/internal/lefthook/runner/executor.go b/internal/lefthook/runner/executor.go index 4e10ba67..0cb10d2a 100644 --- a/internal/lefthook/runner/executor.go +++ b/internal/lefthook/runner/executor.go @@ -7,6 +7,6 @@ import ( // Executor provides an interface for command execution. // It is used here for testing purpose mostly. type Executor interface { - Execute(root string, args []string) (*bytes.Buffer, error) + Execute(root string, args []string, interactive bool) (*bytes.Buffer, error) RawExecute(command string, args ...string) error } diff --git a/internal/lefthook/runner/runner.go b/internal/lefthook/runner/runner.go index ac2e6a2f..dac53898 100644 --- a/internal/lefthook/runner/runner.go +++ b/internal/lefthook/runner/runner.go @@ -27,6 +27,14 @@ const ( var surroundingQuotesRegexp = regexp.MustCompile(`^'(.*)'$`) +// RunOptions contains the options that control the execution. +type RunOptions struct { + name, root, failText string + args []string + interactive bool +} + +// Runner responds for actual execution and handling the results. type Runner struct { fs afero.Fs repo *git.Repository @@ -57,6 +65,8 @@ func NewRunner( } } +// RunAll runs scripts and commands. +// LFS hook is executed at first if needed. func (r *Runner) RunAll(hookName string, sourceDirs []string) { if err := r.runLFSHook(hookName); err != nil { log.Error(err) @@ -201,7 +211,13 @@ func (r *Runner) runScript(script *config.Script, path string, file os.FileInfo) args = append(args, path) args = append(args, r.args[:]...) - r.run(file.Name(), r.repo.RootPath, script.FailText, args) + r.run(RunOptions{ + name: file.Name(), + root: r.repo.RootPath, + args: args, + failText: script.FailText, + interactive: script.Interactive, + }) } func (r *Runner) runCommands() { @@ -260,8 +276,13 @@ func (r *Runner) runCommand(name string, command *config.Command) { return } - root := filepath.Join(r.repo.RootPath, command.Root) - r.run(name, root, command.FailText, args) + r.run(RunOptions{ + name: name, + root: filepath.Join(r.repo.RootPath, command.Root), + args: args, + failText: command.FailText, + interactive: command.Interactive, + }) } func (r *Runner) buildCommandArgs(command *config.Command) ([]string, error) { @@ -369,16 +390,16 @@ func replaceQuoted(source, substitution string, files []string) string { return source } -func (r *Runner) run(name, root, failText string, args []string) { - out, err := r.exec.Execute(root, args) +func (r *Runner) run(opts RunOptions) { + out, err := r.exec.Execute(opts.root, opts.args, opts.interactive) var execName string if err != nil { - r.fail(name, failText) - execName = fmt.Sprint(log.Red("\n EXECUTE >"), log.Bold(name)) + r.fail(opts.name, opts.failText) + execName = fmt.Sprint(log.Red("\n EXECUTE >"), log.Bold(opts.name)) } else { - r.success(name) - execName = fmt.Sprint(log.Cyan("\n EXECUTE >"), log.Bold(name)) + r.success(opts.name) + execName = fmt.Sprint(log.Cyan("\n EXECUTE >"), log.Bold(opts.name)) } if out != nil { diff --git a/internal/lefthook/runner/runner_test.go b/internal/lefthook/runner/runner_test.go index 3f9b1d74..6673664c 100644 --- a/internal/lefthook/runner/runner_test.go +++ b/internal/lefthook/runner/runner_test.go @@ -15,7 +15,7 @@ import ( type TestExecutor struct{} -func (e TestExecutor) Execute(root string, args []string) (out *bytes.Buffer, err error) { +func (e TestExecutor) Execute(root string, args []string, interactive bool) (out *bytes.Buffer, err error) { out = bytes.NewBuffer(make([]byte, 0)) if args[0] == "success" { diff --git a/internal/templates/hook.tmpl b/internal/templates/hook.tmpl index 4511c18d..cc351d43 100644 --- a/internal/templates/hook.tmpl +++ b/internal/templates/hook.tmpl @@ -4,10 +4,6 @@ if [ "$LEFTHOOK" = "0" ]; then exit 0 fi -if [ -t 1 ] ; then - exec < /dev/tty ; # <- enables interactive shell -fi - call_lefthook() { dir="$(git rev-parse --show-toplevel)"