packer-cn/common/shell-local/communicator.go

73 lines
1.7 KiB
Go
Raw Normal View History

package shell_local
2015-06-19 18:06:06 -04:00
import (
"fmt"
"io"
2018-03-01 13:56:30 -05:00
"log"
2015-06-19 18:06:06 -04:00
"os"
"os/exec"
"syscall"
2017-04-04 16:39:01 -04:00
"github.com/hashicorp/packer/packer"
2015-06-19 18:06:06 -04:00
)
type Communicator struct {
ExecuteCommand []string
}
func (c *Communicator) Start(cmd *packer.RemoteCmd) error {
if len(c.ExecuteCommand) == 0 {
2018-02-28 17:35:42 -05:00
return fmt.Errorf("Error launching command via shell-local communicator: No ExecuteCommand provided")
2015-06-19 18:06:06 -04:00
}
// Build the local command to execute
2018-05-21 14:25:51 -04:00
log.Printf("[INFO] (shell-local communicator): Executing local shell command %s", c.ExecuteCommand)
2015-06-19 18:06:06 -04:00
localCmd := exec.Command(c.ExecuteCommand[0], c.ExecuteCommand[1:]...)
localCmd.Stdin = cmd.Stdin
localCmd.Stdout = cmd.Stdout
localCmd.Stderr = cmd.Stderr
// Start it. If it doesn't work, then error right away.
if err := localCmd.Start(); err != nil {
return err
}
// We've started successfully. Start a goroutine to wait for
// it to complete and track exit status.
go func() {
var exitStatus int
err := localCmd.Wait()
if err != nil {
if exitErr, ok := err.(*exec.ExitError); ok {
exitStatus = 1
// There is no process-independent way to get the REAL
// exit status so we just try to go deeper.
if status, ok := exitErr.Sys().(syscall.WaitStatus); ok {
exitStatus = status.ExitStatus()
}
}
}
cmd.SetExited(exitStatus)
}()
return nil
}
func (c *Communicator) Upload(string, io.Reader, *os.FileInfo) error {
return fmt.Errorf("upload not supported")
}
func (c *Communicator) UploadDir(string, string, []string) error {
return fmt.Errorf("uploadDir not supported")
}
func (c *Communicator) Download(string, io.Writer) error {
return fmt.Errorf("download not supported")
}
func (c *Communicator) DownloadDir(string, string, []string) error {
return fmt.Errorf("downloadDir not supported")
}