system/command.go
package system
import (
"bytes"
"fmt"
"io"
"os/exec"
"time"
"github.com/aelsabbahy/goss/util"
)
type Command interface {
Command() string
Exists() (bool, error)
ExitStatus() (int, error)
Stdout() (io.Reader, error)
Stderr() (io.Reader, error)
}
type DefCommand struct {
command string
exitStatus int
stdout io.Reader
stderr io.Reader
loaded bool
Timeout int
err error
}
func NewDefCommand(command string, system *System, config util.Config) Command {
return &DefCommand{
command: command,
Timeout: config.Timeout,
}
}
func (c *DefCommand) setup() error {
if c.loaded {
return c.err
}
c.loaded = true
cmd := util.NewCommand("sh", "-c", c.command)
err := runCommand(cmd, c.Timeout)
// We don't care about ExitError since it's covered by status
if _, ok := err.(*exec.ExitError); !ok {
c.err = err
}
c.exitStatus = cmd.Status
c.stdout = bytes.NewReader(cmd.Stdout.Bytes())
c.stderr = bytes.NewReader(cmd.Stderr.Bytes())
return c.err
}
func (c *DefCommand) Command() string {
return c.command
}
func (c *DefCommand) ExitStatus() (int, error) {
err := c.setup()
return c.exitStatus, err
}
func (c *DefCommand) Stdout() (io.Reader, error) {
err := c.setup()
return c.stdout, err
}
func (c *DefCommand) Stderr() (io.Reader, error) {
err := c.setup()
return c.stderr, err
}
// Stub out
func (c *DefCommand) Exists() (bool, error) {
return false, nil
}
func runCommand(cmd *util.Command, timeout int) error {
c1 := make(chan bool, 1)
e1 := make(chan error, 1)
timeoutD := time.Duration(timeout) * time.Millisecond
go func() {
err := cmd.Run()
if err != nil {
e1 <- err
}
c1 <- true
}()
select {
case <-c1:
return nil
case err := <-e1:
return err
case <-time.After(timeoutD):
return fmt.Errorf("Command execution timed out (%s)", timeoutD)
}
}