2023-12-21 15:45:23 +00:00
|
|
|
package exec
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-02-01 14:10:04 +00:00
|
|
|
"fmt"
|
2023-12-21 15:45:23 +00:00
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
osexec "os/exec"
|
|
|
|
)
|
|
|
|
|
2024-02-01 14:10:04 +00:00
|
|
|
type ExecutableType string
|
|
|
|
|
|
|
|
const (
|
|
|
|
BashExecutable ExecutableType = `bash`
|
|
|
|
ShExecutable ExecutableType = `sh`
|
|
|
|
CmdExecutable ExecutableType = `cmd`
|
2024-12-12 09:28:42 +00:00
|
|
|
)
|
2024-02-01 14:10:04 +00:00
|
|
|
|
|
|
|
var finders map[ExecutableType](func() (shell, error)) = map[ExecutableType](func() (shell, error)){
|
|
|
|
BashExecutable: newBashShell,
|
|
|
|
ShExecutable: newShShell,
|
|
|
|
CmdExecutable: newCmdShell,
|
|
|
|
}
|
|
|
|
|
2023-12-21 15:45:23 +00:00
|
|
|
type Command interface {
|
|
|
|
// Wait for command to terminate. It must have been previously started.
|
|
|
|
Wait() error
|
|
|
|
|
|
|
|
// StdinPipe returns a pipe that will be connected to the command's standard input when the command starts.
|
|
|
|
Stdout() io.ReadCloser
|
|
|
|
|
|
|
|
// StderrPipe returns a pipe that will be connected to the command's standard error when the command starts.
|
|
|
|
Stderr() io.ReadCloser
|
|
|
|
}
|
|
|
|
|
|
|
|
type command struct {
|
|
|
|
cmd *osexec.Cmd
|
|
|
|
execContext *execContext
|
|
|
|
stdout io.ReadCloser
|
|
|
|
stderr io.ReadCloser
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *command) Wait() error {
|
|
|
|
// After the command has finished (cmd.Wait call), remove the temporary script file
|
|
|
|
defer os.Remove(c.execContext.scriptFile)
|
|
|
|
|
|
|
|
err := c.cmd.Wait()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *command) Stdout() io.ReadCloser {
|
|
|
|
return c.stdout
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *command) Stderr() io.ReadCloser {
|
|
|
|
return c.stderr
|
|
|
|
}
|
|
|
|
|
|
|
|
type Executor struct {
|
2024-01-11 12:26:31 +00:00
|
|
|
shell shell
|
|
|
|
dir string
|
2023-12-21 15:45:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewCommandExecutor(dir string) (*Executor, error) {
|
2024-01-11 12:26:31 +00:00
|
|
|
shell, err := findShell()
|
2023-12-21 15:45:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &Executor{
|
2024-01-11 12:26:31 +00:00
|
|
|
shell: shell,
|
|
|
|
dir: dir,
|
2023-12-21 15:45:23 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2024-02-01 14:10:04 +00:00
|
|
|
func NewCommandExecutorWithExecutable(dir string, execType ExecutableType) (*Executor, error) {
|
|
|
|
f, ok := finders[execType]
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("%s is not supported as an artifact executable, options are: %s, %s or %s", execType, BashExecutable, ShExecutable, CmdExecutable)
|
|
|
|
}
|
|
|
|
shell, err := f()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Executor{
|
|
|
|
shell: shell,
|
|
|
|
dir: dir,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2024-02-12 15:04:14 +00:00
|
|
|
func (e *Executor) prepareCommand(ctx context.Context, command string) (*osexec.Cmd, *execContext, error) {
|
2024-01-11 12:26:31 +00:00
|
|
|
ec, err := e.shell.prepare(command)
|
2023-12-21 15:45:23 +00:00
|
|
|
if err != nil {
|
2024-02-12 15:04:14 +00:00
|
|
|
return nil, nil, err
|
2023-12-21 15:45:23 +00:00
|
|
|
}
|
|
|
|
cmd := osexec.CommandContext(ctx, ec.executable, ec.args...)
|
|
|
|
cmd.Dir = e.dir
|
2024-02-12 15:04:14 +00:00
|
|
|
return cmd, ec, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Executor) StartCommand(ctx context.Context, command string) (Command, error) {
|
|
|
|
cmd, ec, err := e.prepareCommand(ctx, command)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return e.start(ctx, cmd, ec)
|
|
|
|
}
|
2023-12-21 15:45:23 +00:00
|
|
|
|
2024-02-12 15:04:14 +00:00
|
|
|
func (e *Executor) start(ctx context.Context, cmd *osexec.Cmd, ec *execContext) (Command, error) {
|
2023-12-21 15:45:23 +00:00
|
|
|
stdout, err := cmd.StdoutPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
stderr, err := cmd.StderrPipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &command{cmd, ec, stdout, stderr}, cmd.Start()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Executor) Exec(ctx context.Context, command string) ([]byte, error) {
|
2024-02-12 15:04:14 +00:00
|
|
|
cmd, ec, err := e.prepareCommand(ctx, command)
|
2023-12-21 15:45:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-02-12 15:04:14 +00:00
|
|
|
defer os.Remove(ec.scriptFile)
|
|
|
|
return cmd.CombinedOutput()
|
2023-12-21 15:45:23 +00:00
|
|
|
}
|
2024-02-01 14:10:04 +00:00
|
|
|
|
|
|
|
func (e *Executor) ShellType() ExecutableType {
|
|
|
|
return e.shell.getType()
|
|
|
|
}
|