2023-12-21 15:45:23 +00:00
|
|
|
package exec
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-01-11 12:26:31 +00:00
|
|
|
"errors"
|
2023-12-21 15:45:23 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
2024-01-11 12:26:31 +00:00
|
|
|
"os"
|
|
|
|
osexec "os/exec"
|
2023-12-21 15:45:23 +00:00
|
|
|
"runtime"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestExecutorWithSimpleInput(t *testing.T) {
|
|
|
|
executor, err := NewCommandExecutor(".")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
out, err := executor.Exec(context.Background(), "echo 'Hello'")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NotNil(t, out)
|
|
|
|
assert.Equal(t, "Hello\n", string(out))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestExecutorWithComplexInput(t *testing.T) {
|
|
|
|
executor, err := NewCommandExecutor(".")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
out, err := executor.Exec(context.Background(), "echo 'Hello' && echo 'World'")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NotNil(t, out)
|
|
|
|
assert.Equal(t, "Hello\nWorld\n", string(out))
|
|
|
|
}
|
|
|
|
|
2024-02-12 15:04:14 +00:00
|
|
|
func TestExecutorWithStderr(t *testing.T) {
|
|
|
|
executor, err := NewCommandExecutor(".")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
out, err := executor.Exec(context.Background(), "echo 'Hello' && >&2 echo 'Error'")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NotNil(t, out)
|
|
|
|
assert.Equal(t, "Hello\nError\n", string(out))
|
|
|
|
}
|
|
|
|
|
2023-12-21 15:45:23 +00:00
|
|
|
func TestExecutorWithInvalidCommand(t *testing.T) {
|
|
|
|
executor, err := NewCommandExecutor(".")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
out, err := executor.Exec(context.Background(), "invalid-command")
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Contains(t, string(out), "invalid-command: command not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestExecutorWithInvalidCommandWithWindowsLikePath(t *testing.T) {
|
|
|
|
if runtime.GOOS != "windows" {
|
|
|
|
t.SkipNow()
|
|
|
|
}
|
|
|
|
|
|
|
|
executor, err := NewCommandExecutor(".")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
out, err := executor.Exec(context.Background(), `"C:\Program Files\invalid-command.exe"`)
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Contains(t, string(out), "C:\\Program Files\\invalid-command.exe: No such file or directory")
|
|
|
|
}
|
|
|
|
|
2024-01-11 12:26:31 +00:00
|
|
|
func testExecutorWithShell(t *testing.T, shell string) {
|
|
|
|
p, err := osexec.LookPath(shell)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Is(err, osexec.ErrNotFound) {
|
|
|
|
switch runtime.GOOS {
|
|
|
|
case "windows":
|
|
|
|
if shell == "cmd" {
|
|
|
|
// We must find `cmd.exe` on Windows.
|
|
|
|
t.Fatal("cmd.exe not found")
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
if shell == "bash" || shell == "sh" {
|
|
|
|
// We must find `bash` or `sh` on other operating systems.
|
|
|
|
t.Fatal("bash or sh not found")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
t.Skipf("shell %s not found", shell)
|
|
|
|
}
|
|
|
|
t.Fatal(err)
|
2023-12-21 15:45:23 +00:00
|
|
|
}
|
|
|
|
|
2024-01-11 12:26:31 +00:00
|
|
|
// Create temporary directory with only the shell executable in the PATH.
|
|
|
|
tmpDir := t.TempDir()
|
2025-01-15 11:05:46 +00:00
|
|
|
t.Setenv("PATH", fmt.Sprintf("%s%c%s", tmpDir, os.PathListSeparator, os.Getenv("PATH")))
|
2024-01-11 12:26:31 +00:00
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
err = os.Symlink(p, fmt.Sprintf("%s/%s.exe", tmpDir, shell))
|
2023-12-21 15:45:23 +00:00
|
|
|
require.NoError(t, err)
|
2024-01-11 12:26:31 +00:00
|
|
|
} else {
|
|
|
|
err = os.Symlink(p, fmt.Sprintf("%s/%s", tmpDir, shell))
|
2023-12-21 15:45:23 +00:00
|
|
|
require.NoError(t, err)
|
2024-01-11 12:26:31 +00:00
|
|
|
}
|
2023-12-21 15:45:23 +00:00
|
|
|
|
2024-01-11 12:26:31 +00:00
|
|
|
executor, err := NewCommandExecutor(".")
|
2023-12-21 15:45:23 +00:00
|
|
|
assert.NoError(t, err)
|
2024-01-11 12:26:31 +00:00
|
|
|
out, err := executor.Exec(context.Background(), "echo 'Hello from shell'")
|
2023-12-21 15:45:23 +00:00
|
|
|
assert.NoError(t, err)
|
2024-01-11 12:26:31 +00:00
|
|
|
assert.NotNil(t, out)
|
|
|
|
assert.Contains(t, string(out), "Hello from shell")
|
2023-12-21 15:45:23 +00:00
|
|
|
}
|
|
|
|
|
2024-01-11 12:26:31 +00:00
|
|
|
func TestExecutorWithDifferentShells(t *testing.T) {
|
|
|
|
for _, shell := range []string{"bash", "sh", "cmd"} {
|
|
|
|
t.Run(shell, func(t *testing.T) {
|
|
|
|
testExecutorWithShell(t, shell)
|
|
|
|
})
|
2023-12-21 15:45:23 +00:00
|
|
|
}
|
2024-01-11 12:26:31 +00:00
|
|
|
}
|
2023-12-21 15:45:23 +00:00
|
|
|
|
2024-01-11 12:26:31 +00:00
|
|
|
func TestExecutorNoShellFound(t *testing.T) {
|
|
|
|
t.Setenv("PATH", "")
|
|
|
|
_, err := NewCommandExecutor(".")
|
|
|
|
assert.ErrorContains(t, err, "no shell found")
|
2023-12-21 15:45:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestExecutorCleanupsTempFiles(t *testing.T) {
|
|
|
|
executor, err := NewCommandExecutor(".")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
2024-02-12 15:04:14 +00:00
|
|
|
cmd, ec, err := executor.prepareCommand(context.Background(), "echo 'Hello'")
|
2023-12-21 15:45:23 +00:00
|
|
|
assert.NoError(t, err)
|
|
|
|
|
2024-02-12 15:04:14 +00:00
|
|
|
command, err := executor.start(context.Background(), cmd, ec)
|
2023-12-21 15:45:23 +00:00
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
fileName := ec.args[1]
|
|
|
|
assert.FileExists(t, fileName)
|
|
|
|
|
2024-02-12 15:04:14 +00:00
|
|
|
err = command.Wait()
|
2023-12-21 15:45:23 +00:00
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NoFileExists(t, fileName)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMultipleCommandsRunInParrallel(t *testing.T) {
|
|
|
|
executor, err := NewCommandExecutor(".")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
const count = 5
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
2025-01-03 09:25:07 +00:00
|
|
|
for i := range count {
|
2023-12-21 15:45:23 +00:00
|
|
|
wg.Add(1)
|
|
|
|
cmd, err := executor.StartCommand(context.Background(), fmt.Sprintf("echo 'Hello %d'", i))
|
|
|
|
go func(cmd Command, i int) {
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
stdout := cmd.Stdout()
|
|
|
|
out, err := io.ReadAll(stdout)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
err = cmd.Wait()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
assert.Equal(t, fmt.Sprintf("Hello %d\n", i), string(out))
|
|
|
|
}(cmd, i)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
}
|