packer-cn/communicator/ssh/communicator.go

162 lines
3.7 KiB
Go
Raw Normal View History

2013-05-20 18:47:41 -04:00
package ssh
import (
"bytes"
"code.google.com/p/go.crypto/ssh"
"fmt"
2013-05-20 18:47:41 -04:00
"github.com/mitchellh/packer/packer"
"io"
2013-05-20 18:47:41 -04:00
"log"
"net"
"path/filepath"
2013-05-20 18:47:41 -04:00
)
type comm struct {
client *ssh.ClientConn
}
// Creates a new packer.Communicator implementation over SSH. This takes
// an already existing TCP connection and SSH configuration.
func New(c net.Conn, config *ssh.ClientConfig) (result *comm, err error) {
client, err := ssh.Client(c, config)
result = &comm{client}
return
}
func (c *comm) Start(cmd string) (remote *packer.RemoteCommand, err error) {
session, err := c.client.NewSession()
if err != nil {
return
}
// Create the buffers to store our stdin/stdout/stderr
stdin := new(bytes.Buffer)
stdout := new(bytes.Buffer)
stderr := new(bytes.Buffer)
// Setup our session
session.Stdin = stdin
session.Stdout = stdout
session.Stderr = stderr
// Setup the remote command
remote = &packer.RemoteCommand{
stdin,
stdout,
stderr,
false,
-1,
}
log.Printf("starting remote command: %s", cmd)
err = session.Start(cmd + "\n")
if err != nil {
return
}
// Start a goroutine to wait for the session to end and set the
// exit boolean and status.
go func() {
defer session.Close()
err := session.Wait()
remote.ExitStatus = 0
if err != nil {
exitErr, ok := err.(*ssh.ExitError)
if ok {
remote.ExitStatus = exitErr.ExitStatus()
}
}
remote.Exited = true
}()
return
}
func (c *comm) Upload(path string, input io.Reader) error {
session, err := c.client.NewSession()
if err != nil {
return err
}
defer session.Close()
// Get a pipe to stdin so that we can send data down
w, err := session.StdinPipe()
if err != nil {
return err
}
2013-05-24 12:06:13 -04:00
// Set stderr/stdout to a bytes buffer
stderr := new(bytes.Buffer)
stdout := new(bytes.Buffer)
session.Stderr = stderr
session.Stdout = stdout
// We only want to close once, so we nil w after we close it,
// and only close in the defer if it hasn't been closed already.
defer func() {
if w != nil {
w.Close()
}
}()
// The target directory and file for talking the SCP protocol
target_dir := filepath.Dir(path)
target_file := filepath.Base(path)
// Start the sink mode on the other side
// TODO(mitchellh): There are probably issues with shell escaping the path
log.Println("Starting remote scp process in sink mode")
2013-05-24 12:06:13 -04:00
if err = session.Start("scp -vt " + target_dir); err != nil {
return err
}
// Determine the length of the upload content by copying it
// into an in-memory buffer. Note that this means what we upload
// must fit into memory.
log.Println("Copying input data into in-memory buffer so we can get the length")
input_memory := new(bytes.Buffer)
if _, err = io.Copy(input_memory, input); err != nil {
return err
}
// Start the protocol
fmt.Fprintln(w, "C0644", input_memory.Len(), target_file)
io.Copy(w, input_memory)
2013-05-24 12:06:13 -04:00
fmt.Fprint(w, "\x00") // XXX(mitchellh): WHY?
// Close the stdin, which sends an EOF, and then set w to nil so that
// our defer func doesn't close it again since that is unsafe with
// the Go SSH package.
w.Close()
w = nil
// Wait for the SCP connection to close, meaning it has consumed all
// our data and has completed. Or has errored.
2013-05-24 12:06:13 -04:00
err = session.Wait()
if err != nil {
exitErr, ok := err.(*ssh.ExitError)
if !ok {
// This wasn't an exit error, so something fatal happened
return err
}
// Otherwise, we have an ExitErorr, meaning we can just read
// the exit status
log.Printf("exit status: %d", exitErr.ExitStatus())
}
// TODO(mitchellh): Check for return data (expect a 0 or error)
2013-05-24 12:06:13 -04:00
log.Printf("scp stdout: %s", stdout.String())
log.Printf("scp stderr: %s", stderr.String())
return nil
}
func (c *comm) Download(string, io.Writer) error {
return nil
}