From afc1125bd9964d27b56fc51eaeac808d5183bdd8 Mon Sep 17 00:00:00 2001 From: Valentin Kiselev Date: Thu, 30 May 2024 11:10:41 +0300 Subject: [PATCH] fix: share STDIN across different commands on pre-push hook (#732) * fix: error and bail if multiple commands useStdin Since Stdin from Git is not intercepted and forwared to each commands/scripts, this could lead to hang. * fix: CommandExecutor.RawExecute forward os.Stdin This is only used by LFS pre-push Hook which require Stdin. * fix: use cached reader for stdin when use_stdin is specified * chore: remove special checks for pre-push hook * chore: fix typos * fix: fail the hook if git-lfs command failed * chore: add more verbose error --------- Co-authored-by: Thomas Desveaux --- internal/lefthook/run.go | 5 +- internal/lefthook/runner/cached_reader.go | 49 +++++++++++++++++++ .../lefthook/runner/cached_reader_test.go | 24 +++++++++ internal/lefthook/runner/exec/execute_unix.go | 9 ++-- .../lefthook/runner/exec/execute_windows.go | 11 ++--- internal/lefthook/runner/exec/executor.go | 4 +- internal/lefthook/runner/exec/nullReader.go | 10 ---- internal/lefthook/runner/null_reader.go | 15 ++++++ internal/lefthook/runner/null_reader_test.go | 20 ++++++++ internal/lefthook/runner/runner.go | 28 ++++++++--- internal/lefthook/runner/runner_test.go | 9 ++-- 11 files changed, 148 insertions(+), 36 deletions(-) create mode 100644 internal/lefthook/runner/cached_reader.go create mode 100644 internal/lefthook/runner/cached_reader_test.go delete mode 100644 internal/lefthook/runner/exec/nullReader.go create mode 100644 internal/lefthook/runner/null_reader.go create mode 100644 internal/lefthook/runner/null_reader_test.go diff --git a/internal/lefthook/run.go b/internal/lefthook/run.go index 59b526fb..4d2b58fa 100644 --- a/internal/lefthook/run.go +++ b/internal/lefthook/run.go @@ -169,7 +169,10 @@ Run 'lefthook install' manually.`, ) startTime := time.Now() - results := r.RunAll(ctx, sourceDirs) + results, runErr := r.RunAll(ctx, sourceDirs) + if runErr != nil { + return fmt.Errorf("failed to run the hook: %w", runErr) + } if ctx.Err() != nil { return errors.New("Interrupted") diff --git a/internal/lefthook/runner/cached_reader.go b/internal/lefthook/runner/cached_reader.go new file mode 100644 index 00000000..b2bfbb65 --- /dev/null +++ b/internal/lefthook/runner/cached_reader.go @@ -0,0 +1,49 @@ +package runner + +import ( + "bytes" + "io" +) + +// cachedReader reads from the provided `io.Reader` until `io.EOF` and saves +// the read content into the inner buffer. +// +// After `io.EOF` it will be providing the read data again and again. +type cachedReader struct { + in io.Reader + useBuffer bool + buf []byte + reader *bytes.Reader +} + +func NewCachedReader(in io.Reader) *cachedReader { + return &cachedReader{ + in: in, + buf: []byte{}, + reader: bytes.NewReader([]byte{}), + } +} + +func (r *cachedReader) Read(p []byte) (int, error) { + if r.useBuffer { + n, err := r.reader.Read(p) + if err == io.EOF { + _, seekErr := r.reader.Seek(0, io.SeekStart) + if seekErr != nil { + panic(seekErr) + } + + return n, err + } + + return n, err + } + + n, err := r.in.Read(p) + r.buf = append(r.buf, p[:n]...) + if err == io.EOF { + r.useBuffer = true + r.reader = bytes.NewReader(r.buf) + } + return n, err +} diff --git a/internal/lefthook/runner/cached_reader_test.go b/internal/lefthook/runner/cached_reader_test.go new file mode 100644 index 00000000..7c6d3056 --- /dev/null +++ b/internal/lefthook/runner/cached_reader_test.go @@ -0,0 +1,24 @@ +package runner + +import ( + "bytes" + "io" + "testing" +) + +func TestCachedReader(t *testing.T) { + testSlice := []byte("Some example string\nMultiline") + + cachedReader := NewCachedReader(bytes.NewReader(testSlice)) + + for range 5 { + res, err := io.ReadAll(cachedReader) + if err != nil { + t.Errorf("unexpected err: %s", err) + } + + if !bytes.Equal(res, testSlice) { + t.Errorf("expected %v to be equal to %v", res, testSlice) + } + } +} diff --git a/internal/lefthook/runner/exec/execute_unix.go b/internal/lefthook/runner/exec/execute_unix.go index 9958e8b7..874aa169 100644 --- a/internal/lefthook/runner/exec/execute_unix.go +++ b/internal/lefthook/runner/exec/execute_unix.go @@ -28,11 +28,7 @@ type executeArgs struct { interactive, useStdin bool } -func (e CommandExecutor) Execute(ctx context.Context, opts Options, out io.Writer) error { - var in io.Reader = nullReader{} - if opts.UseStdin { - in = os.Stdin - } +func (e CommandExecutor) Execute(ctx context.Context, opts Options, in io.Reader, out io.Writer) error { if opts.Interactive && !isatty.IsTerminal(os.Stdin.Fd()) { tty, err := os.Open("/dev/tty") if err == nil { @@ -72,9 +68,10 @@ func (e CommandExecutor) Execute(ctx context.Context, opts Options, out io.Write return nil } -func (e CommandExecutor) RawExecute(ctx context.Context, command []string, out io.Writer) error { +func (e CommandExecutor) RawExecute(ctx context.Context, command []string, in io.Reader, out io.Writer) error { cmd := exec.CommandContext(ctx, command[0], command[1:]...) + cmd.Stdin = in cmd.Stdout = out cmd.Stderr = os.Stderr diff --git a/internal/lefthook/runner/exec/execute_windows.go b/internal/lefthook/runner/exec/execute_windows.go index e911121b..752a264f 100644 --- a/internal/lefthook/runner/exec/execute_windows.go +++ b/internal/lefthook/runner/exec/execute_windows.go @@ -23,11 +23,7 @@ type executeArgs struct { root string } -func (e CommandExecutor) Execute(ctx context.Context, opts Options, out io.Writer) error { - var in io.Reader = nullReader{} - if opts.UseStdin { - in = os.Stdin - } +func (e CommandExecutor) Execute(ctx context.Context, opts Options, in io.Reader, out io.Writer) error { if opts.Interactive && !isatty.IsTerminal(os.Stdin.Fd()) { tty, err := tty.Open() if err == nil { @@ -63,9 +59,10 @@ func (e CommandExecutor) Execute(ctx context.Context, opts Options, out io.Write return nil } -func (e CommandExecutor) RawExecute(ctx context.Context, command []string, out io.Writer) error { - cmd := exec.Command(command[0], command[1:]...) +func (e CommandExecutor) RawExecute(ctx context.Context, command []string, in io.Reader, out io.Writer) error { + cmd := exec.CommandContext(ctx, command[0], command[1:]...) + cmd.Stdin = in cmd.Stdout = out cmd.Stderr = os.Stderr diff --git a/internal/lefthook/runner/exec/executor.go b/internal/lefthook/runner/exec/executor.go index d4e67628..12ec9a39 100644 --- a/internal/lefthook/runner/exec/executor.go +++ b/internal/lefthook/runner/exec/executor.go @@ -16,6 +16,6 @@ type Options struct { // Executor provides an interface for command execution. // It is used here for testing purpose mostly. type Executor interface { - Execute(ctx context.Context, opts Options, out io.Writer) error - RawExecute(ctx context.Context, command []string, out io.Writer) error + Execute(ctx context.Context, opts Options, in io.Reader, out io.Writer) error + RawExecute(ctx context.Context, command []string, in io.Reader, out io.Writer) error } diff --git a/internal/lefthook/runner/exec/nullReader.go b/internal/lefthook/runner/exec/nullReader.go deleted file mode 100644 index dd774b5e..00000000 --- a/internal/lefthook/runner/exec/nullReader.go +++ /dev/null @@ -1,10 +0,0 @@ -package exec - -import "io" - -// nullReader always returns EOF. -type nullReader struct{} - -func (nullReader) Read(b []byte) (int, error) { - return 0, io.EOF -} diff --git a/internal/lefthook/runner/null_reader.go b/internal/lefthook/runner/null_reader.go new file mode 100644 index 00000000..caf5a568 --- /dev/null +++ b/internal/lefthook/runner/null_reader.go @@ -0,0 +1,15 @@ +package runner + +import "io" + +// nullReader always returns `io.EOF`. +type nullReader struct{} + +func NewNullReader() io.Reader { + return nullReader{} +} + +// Implements io.Reader interface. +func (nullReader) Read(b []byte) (int, error) { + return 0, io.EOF +} diff --git a/internal/lefthook/runner/null_reader_test.go b/internal/lefthook/runner/null_reader_test.go new file mode 100644 index 00000000..df4fd06e --- /dev/null +++ b/internal/lefthook/runner/null_reader_test.go @@ -0,0 +1,20 @@ +package runner + +import ( + "bytes" + "io" + "testing" +) + +func TestNullReader(t *testing.T) { + nullReader := NewNullReader() + + res, err := io.ReadAll(nullReader) + if err != nil { + t.Errorf("unexpected err: %s", err) + } + + if !bytes.Equal(res, []byte{}) { + t.Errorf("expected %v to be equal to %v", res, []byte{}) + } +} diff --git a/internal/lefthook/runner/runner.go b/internal/lefthook/runner/runner.go index 80c1467c..e1095f4e 100644 --- a/internal/lefthook/runner/runner.go +++ b/internal/lefthook/runner/runner.go @@ -51,6 +51,7 @@ type Options struct { type Runner struct { Options + stdin io.Reader partiallyStagedFiles []string failed atomic.Bool executor exec.Executor @@ -58,7 +59,11 @@ type Runner struct { func New(opts Options) *Runner { return &Runner{ - Options: opts, + Options: opts, + + // Some hooks use STDIN for parsing data from Git. To allow multiple commands + // and scripts access the same Git data STDIN is cached via cachedReader. + stdin: NewCachedReader(os.Stdin), executor: exec.CommandExecutor{}, } } @@ -79,16 +84,16 @@ type executable interface { // RunAll runs scripts and commands. // LFS hook is executed at first if needed. -func (r *Runner) RunAll(ctx context.Context, sourceDirs []string) []Result { +func (r *Runner) RunAll(ctx context.Context, sourceDirs []string) ([]Result, error) { results := make([]Result, 0, len(r.Hook.Commands)+len(r.Hook.Scripts)) if err := r.runLFSHook(ctx); err != nil { - log.Error(err) + return results, err } if r.Hook.DoSkip(r.Repo.State()) { r.logSkip(r.HookName, "hook setting") - return results + return results, nil } if !r.DisableTTY && !r.Hook.Follow { @@ -113,7 +118,7 @@ func (r *Runner) RunAll(ctx context.Context, sourceDirs []string) []Result { r.postHook() - return results + return results, nil } func (r *Runner) runLFSHook(ctx context.Context) error { @@ -144,6 +149,7 @@ func (r *Runner) runLFSHook(ctx context.Context) error { []string{"git", "lfs", r.HookName}, r.GitArgs..., ), + r.stdin, out, ) @@ -490,6 +496,12 @@ func (r *Runner) run(ctx context.Context, opts exec.Options, follow bool) bool { log.SetName(opts.Name) defer log.UnsetName(opts.Name) + // If the command does not explicitly `use_stdin` no input will be provided. + var in io.Reader = NewNullReader() + if opts.UseStdin { + in = r.stdin + } + if (follow || opts.Interactive) && r.LogSettings.LogExecution() { r.logExecute(opts.Name, nil, nil) @@ -500,12 +512,14 @@ func (r *Runner) run(ctx context.Context, opts exec.Options, follow bool) bool { out = io.Discard } - err := r.executor.Execute(ctx, opts, out) + err := r.executor.Execute(ctx, opts, in, out) + return err == nil } out := bytes.NewBuffer(make([]byte, 0)) - err := r.executor.Execute(ctx, opts, out) + + err := r.executor.Execute(ctx, opts, in, out) r.logExecute(opts.Name, err, out) diff --git a/internal/lefthook/runner/runner_test.go b/internal/lefthook/runner/runner_test.go index 8536de55..16888a81 100644 --- a/internal/lefthook/runner/runner_test.go +++ b/internal/lefthook/runner/runner_test.go @@ -21,7 +21,7 @@ import ( type TestExecutor struct{} -func (e TestExecutor) Execute(_ctx context.Context, opts exec.Options, _out io.Writer) (err error) { +func (e TestExecutor) Execute(_ctx context.Context, opts exec.Options, _in io.Reader, _out io.Writer) (err error) { if strings.HasPrefix(opts.Commands[0], "success") { err = nil } else { @@ -31,7 +31,7 @@ func (e TestExecutor) Execute(_ctx context.Context, opts exec.Options, _out io.W return } -func (e TestExecutor) RawExecute(_ctx context.Context, _command []string, _out io.Writer) error { +func (e TestExecutor) RawExecute(_ctx context.Context, _command []string, _in io.Reader, _out io.Writer) error { return nil } @@ -766,7 +766,10 @@ func TestRunAll(t *testing.T) { } t.Run(fmt.Sprintf("%d: %s", i, tt.name), func(t *testing.T) { - results := runner.RunAll(context.Background(), tt.sourceDirs) + results, err := runner.RunAll(context.Background(), tt.sourceDirs) + if err != nil { + t.Errorf("unexpected error %s", err) + } var success, fail []Result for _, result := range results {