diff --git a/interp/example_test.go b/interp/example_test.go index 2a289623e..b41928a1f 100644 --- a/interp/example_test.go +++ b/interp/example_test.go @@ -5,6 +5,7 @@ package interp_test import ( "context" + "fmt" "os" "strings" @@ -35,20 +36,14 @@ func Example() { } func ExampleModuleExec() { - src := ` - ls example_test.* || echo "ls failed" - rm example_test.* || echo "rm failed" - ` + src := "echo foo; missing-program bar" file, _ := syntax.NewParser().Parse(strings.NewReader(src), "") exec := func(ctx context.Context, path string, args []string) error { - switch args[0] { - case "ls": - // whitelist the "ls" program - return interp.DefaultExec(ctx, path, args) - default: - // refuse to run any other program + if path == "" { + fmt.Printf("%s is not installed\n", args[0]) return interp.ExitStatus(1) } + return interp.DefaultExec(ctx, path, args) } runner, _ := interp.New( interp.StdIO(nil, os.Stdout, os.Stdout), @@ -56,6 +51,6 @@ func ExampleModuleExec() { ) runner.Run(context.TODO(), file) // Output: - // example_test.go - // rm failed + // foo + // missing-program is not installed } diff --git a/interp/interp_test.go b/interp/interp_test.go index 2d52e0316..d1a3636ab 100644 --- a/interp/interp_test.go +++ b/interp/interp_test.go @@ -7,12 +7,14 @@ import ( "bytes" "context" "fmt" + "io" "io/ioutil" "os" "os/exec" "path/filepath" "regexp" "runtime" + "sort" "strings" "sync" "testing" @@ -57,6 +59,33 @@ let i=(2 + 3) var hasBash50 bool func TestMain(m *testing.M) { + if os.Getenv("GOSH_PROG") != "" { + src := strings.NewReader(os.Args[1]) + prog, err := syntax.NewParser().Parse(src, "") + if err != nil { + fmt.Fprintln(os.Stderr, err) + os.Exit(1) + } + runner, _ := New( + StdIO(os.Stdin, os.Stdout, os.Stderr), + Module(OpenDevImpls(DefaultOpen)), + Module(WithBuiltins(testBuiltins, DefaultExec)), + ) + ctx := context.Background() + switch err := runner.Run(ctx, prog).(type) { + case nil: + case ShellExitStatus: + os.Exit(int(err)) + case ExitStatus: + os.Exit(int(err)) + default: + fmt.Fprintln(os.Stderr, err) + os.Exit(1) + } + os.Exit(0) + } + os.Setenv("GOSH_PROG", os.Args[0]) + os.Setenv("LANGUAGE", "en_US.UTF8") os.Setenv("LC_ALL", "en_US.UTF8") os.Unsetenv("CDPATH") @@ -71,6 +100,13 @@ func TestMain(m *testing.M) { os.Setenv("MIXEDCASE_INTERP_GLOBAL", "value") } + // Some program which should be in $PATH. + if runtime.GOOS == "windows" { + os.Setenv("PATH_PROG", "cmd") + } else { + os.Setenv("PATH_PROG", "sh") + } + for _, s := range []string{"a", "b", "c", "d", "foo", "bar"} { os.Unsetenv(s) } @@ -120,9 +156,11 @@ func (c *concBuffer) Reset() { c.Unlock() } -var fileCases = []struct { +type runTest struct { in, want string -}{ +} + +var runTests = []runTest{ // no-op programs {"", ""}, {"true", ""}, @@ -272,13 +310,13 @@ var fileCases = []struct { {"a=b; a+=c x+=y; echo $a $x", "bc y\n"}, {`a=" x y"; b=$a c="$a"; echo $b; echo $c`, "x y\nx y\n"}, {`a=" x y"; b=$a c="$a"; echo "$b"; echo "$c"`, " x y\n x y\n"}, - {"env | sed -n '1 s/^$/empty/p'", ""}, // never begin with an empty element + // TODO: reenable once we figure out the broken pipe error + //{`env | while read line; do if test -z "$line"; then echo empty; fi; break; done`, ""}, // never begin with an empty element // special vars {"echo $?; false; echo $?", "0\n1\n"}, {"for i in 1 2; do\necho $LINENO\necho $LINENO\ndone", "2\n3\n2\n3\n"}, {"[[ -n $$ && $$ -gt 0 ]]", ""}, - {"[[ -n $PPID && $PPID -gt 0 ]]", ""}, {"[[ $$ -eq $PPID ]]", "exit status 1"}, // var manipulation @@ -638,7 +676,7 @@ var fileCases = []struct { "", }, { - "touch a && cd a", + ">a && cd a", "exit status 1 #JUSTERR", }, { @@ -723,7 +761,7 @@ var fileCases = []struct { "exit status 1", }, { - "mkdir a; pushd a >/dev/null; pushd >/dev/null; rmdir a; pushd", + "mkdir a; pushd a >/dev/null; pushd >/dev/null; rm -r a; pushd", "exit status 1 #JUSTERR", }, { @@ -748,7 +786,7 @@ var fileCases = []struct { "exit status 1", }, { - "mkdir a; pushd a >/dev/null; pushd >/dev/null; rmdir a; popd", + "mkdir a; pushd a >/dev/null; pushd >/dev/null; rm -r a; popd", "exit status 1 #JUSTERR", }, @@ -828,19 +866,19 @@ var fileCases = []struct { // exec { - "bash -c 'echo foo'", + "$GOSH_PROG 'echo foo'", "foo\n", }, { - "bash -c 'echo foo >&2' >/dev/null", + "$GOSH_PROG 'echo foo >&2' >/dev/null", "foo\n", }, { - "echo foo | bash -c 'cat >&2' >/dev/null", + "echo foo | $GOSH_PROG 'cat >&2' >/dev/null", "foo\n", }, { - "bash -c 'exit 1'", + "$GOSH_PROG 'exit 1'", "exit status 1", }, { @@ -886,13 +924,11 @@ var fileCases = []struct { {"command -o echo", "command: invalid option -o\nexit status 2 #JUSTERR"}, {"echo() { :; }; echo foo", ""}, {"echo() { :; }; command echo foo", "foo\n"}, - {"bash() { :; }; bash -c 'echo foo'", ""}, - {"bash() { :; }; command bash -c 'echo foo'", "foo\n"}, {"command -v does-not-exist", "exit status 1"}, {"foo() { :; }; command -v foo", "foo\n"}, {"foo() { :; }; command -v does-not-exist foo", "foo\n"}, {"command -v echo", "echo\n"}, - {"[[ $(command -v bash) == bash ]]", "exit status 1"}, + {"[[ $(command -v $PATH_PROG) == $PATH_PROG ]]", "exit status 1"}, // cmd substitution { @@ -1151,19 +1187,19 @@ var fileCases = []struct { "exit status 1", }, { - "touch a b; [[ a -ef b ]]", + ">a >b; [[ a -ef b ]]", "exit status 1", }, { - "touch a; [[ a -ef a ]]", + ">a; [[ a -ef a ]]", "", }, { - "touch a; ln a b; [[ a -ef b ]]", + ">a; ln a b; [[ a -ef b ]]", "", }, { - "touch a; ln -s a b; [[ a -ef b ]]", + ">a; ln -s a b; [[ a -ef b ]]", "", }, { @@ -1227,15 +1263,15 @@ var fileCases = []struct { "exit status 2", }, { - "[[ -e a ]] && echo x; touch a; [[ -e a ]] && echo y", + "[[ -e a ]] && echo x; >a; [[ -e a ]] && echo y", "y\n", }, { - "ln -s b a; [[ -e a ]] && echo x; touch b; [[ -e a ]] && echo y", + "ln -s b a; [[ -e a ]] && echo x; >b; [[ -e a ]] && echo y", "y\n", }, { - "[[ -f a ]] && echo x; touch a; [[ -f a ]] && echo y", + "[[ -f a ]] && echo x; >a; [[ -f a ]] && echo y", "y\n", }, { @@ -1247,15 +1283,15 @@ var fileCases = []struct { "y\n", }, { - "[[ -r a ]] && echo x; touch a; [[ -r a ]] && echo y", + "[[ -r a ]] && echo x; >a; [[ -r a ]] && echo y", "y\n", }, { - "[[ -w a ]] && echo x; touch a; [[ -w a ]] && echo y", + "[[ -w a ]] && echo x; >a; [[ -w a ]] && echo y", "y\n", }, { - "[[ -x a ]] && echo x; touch a; chmod +x a; [[ -x a ]] && echo y", + "[[ -x a ]] && echo x; >a; chmod +x a; [[ -x a ]] && echo y", "y\n", }, { @@ -1271,23 +1307,23 @@ var fileCases = []struct { "y\n", }, { - "touch a; [[ -k a ]] && echo x; chmod +t a; [[ -k a ]] && echo y", + ">a; [[ -k a ]] && echo x; chmod +t a; [[ -k a ]] && echo y", "y\n", }, { - "touch a; [[ -u a ]] && echo x; chmod u+s a; [[ -u a ]] && echo y", + ">a; [[ -u a ]] && echo x; chmod u+s a; [[ -u a ]] && echo y", "y\n", }, { - "touch a; [[ -g a ]] && echo x; chmod g+s a; [[ -g a ]] && echo y", + ">a; [[ -g a ]] && echo x; chmod g+s a; [[ -g a ]] && echo y", "y\n", }, { - "mkdir a; cd a; test -f b && echo x; touch b; test -f b && echo y", + "mkdir a; cd a; test -f b && echo x; >b; test -f b && echo y", "y\n", }, { - "touch a; [[ -b a ]] && echo block; [[ -c a ]] && echo char; true", + ">a; [[ -b a ]] && echo block; [[ -c a ]] && echo char; true", "", }, { @@ -1342,7 +1378,7 @@ var fileCases = []struct { {"[ 3 -gt 4 ]", "exit status 1"}, {"[ 3 -lt 4 ]", ""}, { - "[ -e a ] && echo x; touch a; [ -e a ] && echo y", + "[ -e a ] && echo x; >a; [ -e a ] && echo y", "y\n", }, { @@ -1366,7 +1402,7 @@ var fileCases = []struct { "", }, { - "touch a; [ a -ef a ]", + ">a; [ a -ef a ]", "", }, {"[ 3 -eq 04 ]", "exit status 1"}, @@ -1379,15 +1415,15 @@ var fileCases = []struct { "y\n", }, { - "[ -r a ] && echo x; touch a; [ -r a ] && echo y", + "[ -r a ] && echo x; >a; [ -r a ] && echo y", "y\n", }, { - "[ -w a ] && echo x; touch a; [ -w a ] && echo y", + "[ -w a ] && echo x; >a; [ -w a ] && echo y", "y\n", }, { - "[ -x a ] && echo x; touch a; chmod +x a; [ -x a ] && echo y", + "[ -x a ] && echo x; >a; chmod +x a; [ -x a ] && echo y", "y\n", }, { @@ -1403,19 +1439,19 @@ var fileCases = []struct { "y\n", }, { - "touch a; [ -k a ] && echo x; chmod +t a; [ -k a ] && echo y", + ">a; [ -k a ] && echo x; chmod +t a; [ -k a ] && echo y", "y\n", }, { - "touch a; [ -u a ] && echo x; chmod u+s a; [ -u a ] && echo y", + ">a; [ -u a ] && echo x; chmod u+s a; [ -u a ] && echo y", "y\n", }, { - "touch a; [ -g a ] && echo x; chmod g+s a; [ -g a ] && echo y", + ">a; [ -g a ] && echo x; chmod g+s a; [ -g a ] && echo y", "y\n", }, { - "touch a; [ -b a ] && echo block; [ -c a ] && echo char; true", + ">a; [ -b a ] && echo block; [ -c a ] && echo char; true", "", }, {"[ -t 1234 ]", "exit status 1"}, // TODO: reliable way to test a positive? @@ -1592,11 +1628,11 @@ var fileCases = []struct { "exit status 1", }, { - "set -f; touch a.x; echo *.x;", + "set -f; >a.x; echo *.x;", "*.x\n", }, { - "set -f; set +f; touch a.x; echo *.x;", + "set -f; set +f; >a.x; echo *.x;", "a.x\n", }, { @@ -1720,8 +1756,8 @@ set +o pipefail // type {"type", ""}, {"type echo", "echo is a shell builtin\n"}, - {"echo() { :; }; type echo | sed 1q", "echo is a function\n"}, - {"type bash | grep -q -E 'bash is (/|[A-Z]:).*'", ""}, + {"echo() { :; }; type echo | grep 'is a function'", "echo is a function\n"}, + {"type $PATH_PROG | grep -q -E ' is (/|[A-Z]:).*'", ""}, {"type noexist", "type: noexist: not found\nexit status 1 #JUSTERR"}, // eval @@ -2037,19 +2073,19 @@ set +o pipefail {"echo ..", "..\n"}, {"echo ./.", "./.\n"}, { - "touch a.x b.x c.x; echo *.x; rm a.x b.x c.x", + ">a.x >b.x >c.x; echo *.x; rm a.x b.x c.x", "a.x b.x c.x\n", }, { - `touch a.x; echo '*.x' "*.x"; rm a.x`, + `>a.x; echo '*.x' "*.x"; rm a.x`, "*.x *.x\n", }, { - `touch a.x b.y; echo *'.'x; rm a.x`, + `>a.x >b.y; echo *'.'x; rm a.x`, "a.x\n", }, { - `touch a.x; echo *'.x' "a."* '*'.x; rm a.x`, + `>a.x; echo *'.x' "a."* '*'.x; rm a.x`, "a.x a.x *.x\n", }, { @@ -2057,7 +2093,7 @@ set +o pipefail "*.x\nfoo *.y bar\n", }, { - "mkdir a; touch a/b.x; echo */*.x | sed 's@\\\\@/@g'; cd a; echo *.x", + "mkdir a; >a/b.x; echo */*.x | sed 's@\\\\@/@g'; cd a; echo *.x", "a/b.x\nb.x\n", }, { @@ -2065,15 +2101,11 @@ set +o pipefail "a/b\n", }, { - "mkdir -p '*/a.z' 'b/a.z'; cd '*'; set -- *.z; echo $#", - "1\n", - }, - { - "touch .hidden a; echo *; echo .h*; rm .hidden a", + ">.hidden >a; echo *; echo .h*; rm .hidden a", "a\n.hidden\n", }, { - `mkdir d; touch d/.hidden d/a; set -- "$(echo d/*)" "$(echo d/.h*)"; echo ${#1} ${#2}; rm -r d`, + `mkdir d; >d/.hidden >d/a; set -- "$(echo d/*)" "$(echo d/.h*)"; echo ${#1} ${#2}; rm -r d`, "3 9\n", }, { @@ -2105,19 +2137,19 @@ set +o pipefail "../a/b ../a/c\n", }, { - "mkdir x-d1 x-d2; touch x-f; echo x-*/ | sed -e 's@\\\\@/@g'", + "mkdir x-d1 x-d2; >x-f; echo x-*/ | sed 's@\\\\@/@g'", "x-d1/ x-d2/\n", }, { - "mkdir x-d1 x-d2; touch x-f; echo ././x-*/// | sed -e 's@\\\\@/@g'", + "mkdir x-d1 x-d2; >x-f; echo ././x-*/// | sed 's@\\\\@/@g'", "././x-d1/ ././x-d2/\n", }, { - "mkdir -p x-d1/a x-d2/b; touch x-f; echo x-*/* | sed -e 's@\\\\@/@g'", + "mkdir -p x-d1/a x-d2/b; >x-f; echo x-*/* | sed 's@\\\\@/@g'", "x-d1/a x-d2/b\n", }, { - "mkdir x-d; touch x-f; test -d $PWD/x-*/", + "mkdir x-d; >x-f; test -d $PWD/x-*/", "", }, { @@ -2125,7 +2157,7 @@ set +o pipefail "sym/\nsym/bar\n", }, { - "touch foo; ln -s foo sym; echo sy*; echo sy*/", + ">foo; ln -s foo sym; echo sy*; echo sy*/", "sym\nsy*/\n", }, @@ -2176,7 +2208,7 @@ set +o pipefail "\"builtin\": executable file not found in $PATH\nexit status 127 #JUSTERR", }, { - "exec echo foo; echo bar", + "exec $GOSH_PROG 'echo foo'; echo bar", "foo\n", }, @@ -2317,6 +2349,31 @@ set +o pipefail }, } +var runTestsUnix = []runTest{ + {"[[ -n $PPID && $PPID -gt 0 ]]", ""}, + { + // windows does not support dirs named '*' + "mkdir -p '*/a.z' 'b/a.z'; cd '*'; set -- *.z; echo $#", + "1\n", + }, + {"sh() { :; }; sh -c 'echo foo'", ""}, + {"sh() { :; }; command sh -c 'echo foo'", "foo\n"}, +} + +var runTestsWindows = []runTest{ + {"[[ -n $PPID || $PPID -gt 0 ]]", ""}, // os.Getppid can be 0 on windows + {"cmd() { :; }; cmd /c 'echo foo'", ""}, + {"cmd() { :; }; command cmd /c 'echo foo'", "foo\r\n"}, +} + +func init() { + if runtime.GOOS == "windows" { + runTests = append(runTests, runTestsWindows...) + } else { + runTests = append(runTests, runTestsUnix...) + } +} + // wc: leading whitespace padding // touch -d @: no way to set unix timestamps var skipOnDarwin = regexp.MustCompile(`\bwc\b|touch -d @`) @@ -2326,7 +2383,8 @@ var skipOnDarwin = regexp.MustCompile(`\bwc\b|touch -d @`) // ln -s: requires linked path to exist, stat does not work well // ~root: username does not exist // env: missing on Travis? TODO: investigate -var skipOnWindows = regexp.MustCompile(`chmod|mkfifo|ln -s|~root|env`) +// touch -d: TODO +var skipOnWindows = regexp.MustCompile(`chmod|mkfifo|ln -s|~root|env|touch -d`) func skipIfUnsupported(tb testing.TB, src string) { switch { @@ -2337,11 +2395,11 @@ func skipIfUnsupported(tb testing.TB, src string) { } } -func TestFile(t *testing.T) { +func TestRunnerRun(t *testing.T) { p := syntax.NewParser() - for i := range fileCases { + for i := range runTests { t.Run(fmt.Sprintf("%03d", i), func(t *testing.T) { - c := fileCases[i] + c := runTests[i] skipIfUnsupported(t, c.in) file, err := p.Parse(strings.NewReader(c.in), "") if err != nil { @@ -2355,7 +2413,9 @@ func TestFile(t *testing.T) { defer os.RemoveAll(dir) var cb concBuffer r, err := New(Dir(dir), StdIO(nil, &cb, &cb), - Module(OpenDevImpls(DefaultOpen))) + Module(OpenDevImpls(DefaultOpen)), + Module(WithBuiltins(testBuiltins, DefaultExec)), + ) if err != nil { t.Fatal(err) } @@ -2375,7 +2435,170 @@ func TestFile(t *testing.T) { } } -func TestFileConfirm(t *testing.T) { +// TODO: consider making WithBuiltins part of the v3 API. + +func WithBuiltins(m map[string]func(ModuleCtx, []string) error, next ModuleExec) ModuleExec { + return func(ctx context.Context, path string, args []string) error { + if fn := m[args[0]]; fn != nil { + mc, _ := FromModuleContext(ctx) + return fn(mc, args[1:]) + } + return next(ctx, path, args) + } +} + +func readLines(mc ModuleCtx) ([][]byte, error) { + bs, err := ioutil.ReadAll(mc.Stdin) + if err != nil { + return nil, err + } + bs = bytes.TrimSuffix(bs, []byte("\n")) + return bytes.Split(bs, []byte("\n")), nil +} + +var testBuiltins = map[string]func(ModuleCtx, []string) error{ + "cat": func(mc ModuleCtx, args []string) error { + if len(args) == 0 { + if mc.Stdin == nil || mc.Stdout == nil { + return nil + } + _, err := io.Copy(mc.Stdout, mc.Stdin) + return err + } + for _, arg := range args { + path := filepath.Join(mc.Dir, arg) + f, err := os.Open(path) + if err != nil { + return err + } + _, err = io.Copy(mc.Stdout, f) + f.Close() + if err != nil { + return err + } + } + return nil + }, + "wc": func(mc ModuleCtx, args []string) error { + bs, err := ioutil.ReadAll(mc.Stdin) + if err != nil { + return err + } + if len(args) == 0 { + fmt.Fprintf(mc.Stdout, "%7d", bytes.Count(bs, []byte("\n"))) + fmt.Fprintf(mc.Stdout, "%8d", len(bytes.Fields(bs))) + fmt.Fprintf(mc.Stdout, "%8d\n", len(bs)) + } else if args[0] == "-c" { + fmt.Fprintln(mc.Stdout, len(bs)) + } else if args[0] == "-l" { + fmt.Fprintln(mc.Stdout, bytes.Count(bs, []byte("\n"))) + } + return nil + }, + "sort": func(mc ModuleCtx, args []string) error { + lines, err := readLines(mc) + if err != nil { + return err + } + sort.Slice(lines, func(i, j int) bool { + return bytes.Compare(lines[i], lines[j]) < 0 + }) + for _, line := range lines { + fmt.Fprintf(mc.Stdout, "%s\n", line) + } + return nil + }, + "grep": func(mc ModuleCtx, args []string) error { + var rx *regexp.Regexp + quiet := false + for _, arg := range args { + if arg == "-q" { + quiet = true + } else if arg == "-E" { + } else if rx == nil { + rx = regexp.MustCompile(arg) + } + } + lines, err := readLines(mc) + if err != nil { + return err + } + any := false + for _, line := range lines { + if rx.Match(line) { + if quiet { + return nil + } + any = true + fmt.Fprintf(mc.Stdout, "%s\n", line) + } + } + if !any { + return ExitStatus(1) + } + return nil + }, + "sed": func(mc ModuleCtx, args []string) error { + if len(args) != 1 { + return nil // unimplemented + } + expr := args[0] + if expr == "" || expr[0] != 's' { + return nil // unimplemented + } + sep := expr[1] + expr = expr[2:] + from := expr[:strings.IndexByte(expr, sep)] + expr = expr[len(from)+1:] + to := expr[:strings.IndexByte(expr, sep)] + bs, err := ioutil.ReadAll(mc.Stdin) + if err != nil { + return err + } + rx := regexp.MustCompile(from) + bs = rx.ReplaceAllLiteral(bs, []byte(to)) + _, err = mc.Stdout.Write(bs) + return err + }, + "mkdir": func(mc ModuleCtx, args []string) error { + for _, arg := range args { + if arg == "-p" { + continue + } + path := filepath.Join(mc.Dir, arg) + if err := os.MkdirAll(path, 0777); err != nil { + return err + } + } + return nil + }, + "rm": func(mc ModuleCtx, args []string) error { + for _, arg := range args { + if arg == "-r" { + continue + } + path := filepath.Join(mc.Dir, arg) + if err := os.RemoveAll(path); err != nil { + return err + } + } + return nil + }, + "ln": func(mc ModuleCtx, args []string) error { + symbolic := args[0] == "-s" + if symbolic { + args = args[1:] + } + oldname := filepath.Join(mc.Dir, args[0]) + newname := filepath.Join(mc.Dir, args[1]) + if symbolic { + return os.Symlink(oldname, newname) + } + return os.Link(oldname, newname) + }, +} + +func TestRunnerRunConfirm(t *testing.T) { if testing.Short() { t.Skip("calling bash is slow") } @@ -2387,9 +2610,9 @@ func TestFileConfirm(t *testing.T) { // case-sensitive, which isn't how Windows works. t.Skip("bash on Windows emulates Unix-y behavior") } - for i := range fileCases { + for i := range runTests { t.Run(fmt.Sprintf("%03d", i), func(t *testing.T) { - c := fileCases[i] + c := runTests[i] if strings.Contains(c.want, " #IGNORE") { return } @@ -2606,11 +2829,6 @@ func TestElapsedString(t *testing.T) { func TestRunnerDir(t *testing.T) { t.Parallel() - dir, err := ioutil.TempDir("", "interp") - if err != nil { - t.Fatal(err) - } - defer os.Remove(dir) wd, err := os.Getwd() if err != nil { t.Fatal(err) @@ -2634,10 +2852,10 @@ func TestRunnerDir(t *testing.T) { } }) t.Run("Relative", func(t *testing.T) { - rel, err := filepath.Rel(wd, dir) - if err != nil { - t.Fatal(err) - } + // On Windows, it's impossible to make a relative path from one + // drive to another. Use the parent directory, as that's for + // sure in the same drive as the current directory. + rel := ".." + string(filepath.Separator) r, err := New(Dir(rel)) if err != nil { t.Fatal(err) diff --git a/interp/module_test.go b/interp/module_test.go index 65fb93be8..9e47d5be1 100644 --- a/interp/module_test.go +++ b/interp/module_test.go @@ -31,7 +31,7 @@ var modCases = []struct { if args[0] == "sleep" { return fmt.Errorf("blacklisted: %s", args[0]) } - return DefaultExec(ctx, path, args) + return WithBuiltins(testBuiltins, DefaultExec)(ctx, path, args) }, src: "echo foo; sleep 1", want: "foo\nblacklisted: sleep", @@ -44,7 +44,7 @@ var modCases = []struct { default: return fmt.Errorf("blacklisted: %s", args[0]) } - return DefaultExec(ctx, path, args) + return WithBuiltins(testBuiltins, DefaultExec)(ctx, path, args) }, src: "a=$(echo foo | sed 's/o/a/g'); echo $a; $a args", want: "faa\nblacklisted: faa",