1
0
mirror of https://github.com/go-task/task.git synced 2025-01-12 04:34:11 +02:00
task/internal/execext/exec.go

114 lines
2.4 KiB
Go
Raw Normal View History

2017-03-12 22:18:59 +02:00
package execext
import (
"context"
"errors"
"io"
"os"
"path/filepath"
"strings"
"time"
2019-09-09 02:55:02 +02:00
"mvdan.cc/sh/v3/expand"
"mvdan.cc/sh/v3/interp"
"mvdan.cc/sh/v3/shell"
"mvdan.cc/sh/v3/syntax"
2017-03-12 22:18:59 +02:00
)
2017-04-24 15:25:38 +02:00
// RunCommandOptions is the options for the RunCommand func
type RunCommandOptions struct {
Command string
Dir string
Env []string
Stdin io.Reader
Stdout io.Writer
Stderr io.Writer
}
2017-03-12 22:18:59 +02:00
var (
// ErrNilOptions is returned when a nil options is given
ErrNilOptions = errors.New("execext: nil options given")
2017-03-12 22:18:59 +02:00
)
// RunCommand runs a shell command
2018-09-01 16:02:23 +02:00
func RunCommand(ctx context.Context, opts *RunCommandOptions) error {
if opts == nil {
return ErrNilOptions
}
p, err := syntax.NewParser().Parse(strings.NewReader(opts.Command), "")
if err != nil {
return err
}
2017-03-12 22:18:59 +02:00
environ := opts.Env
if len(environ) == 0 {
environ = os.Environ()
}
2018-09-01 16:02:23 +02:00
r, err := interp.New(
interp.Params("-e"),
2018-12-15 19:43:40 +02:00
interp.Env(expand.ListEnviron(environ...)),
interp.ExecHandler(interp.DefaultExecHandler(15*time.Second)),
2020-12-27 21:49:51 +02:00
interp.OpenHandler(openHandler),
2018-09-01 16:02:23 +02:00
interp.StdIO(opts.Stdin, opts.Stdout, opts.Stderr),
dirOption(opts.Dir),
2018-09-01 16:02:23 +02:00
)
if err != nil {
2017-08-05 19:20:44 +02:00
return err
}
2022-05-22 21:54:35 +02:00
return r.Run(ctx, p)
2018-09-01 16:02:23 +02:00
}
// IsExitError returns true the given error is an exis status error
func IsExitError(err error) bool {
2019-11-25 00:17:09 +02:00
if _, ok := interp.IsExitStatus(err); ok {
2018-09-01 16:02:23 +02:00
return true
}
2019-11-25 00:17:09 +02:00
return false
2017-03-12 22:18:59 +02:00
}
// Expand is a helper to mvdan.cc/shell.Fields that returns the first field
// if available.
func Expand(s string) (string, error) {
s = filepath.ToSlash(s)
2022-08-17 19:37:58 +02:00
s = strings.ReplaceAll(s, " ", `\ `)
fields, err := shell.Fields(s, nil)
if err != nil {
return "", err
}
if len(fields) > 0 {
return fields[0], nil
}
return "", nil
}
2020-12-27 21:49:51 +02:00
func openHandler(ctx context.Context, path string, flag int, perm os.FileMode) (io.ReadWriteCloser, error) {
if path == "/dev/null" {
return devNull{}, nil
}
return interp.DefaultOpenHandler()(ctx, path, flag, perm)
}
func dirOption(path string) interp.RunnerOption {
return func(r *interp.Runner) error {
err := interp.Dir(path)(r)
if err == nil {
return nil
}
// If the specified directory doesn't exist, it will be created later.
// Therefore, even if `interp.Dir` method returns an error, the
// directory path should be set only when the directory cannot be found.
if absPath, _ := filepath.Abs(path); absPath != "" {
if _, err := os.Stat(absPath); os.IsNotExist(err) {
r.Dir = absPath
return nil
}
}
return err
}
}