package common import ( "bytes" gossh "code.google.com/p/go.crypto/ssh" "fmt" "github.com/mitchellh/multistep" commonssh "github.com/mitchellh/packer/common/ssh" "github.com/mitchellh/packer/communicator/ssh" "io" "log" "net" "strings" ) func SSHAddress(state multistep.StateBag) (string, error) { sshIP := state.Get("ssh_address").(string) sshHostPort := 22 return fmt.Sprintf("%s:%d", sshIP, sshHostPort), nil } func SSHLocalAddress(state multistep.StateBag) (string, error) { sshLocalPort, ok := state.Get("local_ssh_port").(uint) if !ok { return "", fmt.Errorf("SSH port forwarding hasn't been set up yet") } conn_str := fmt.Sprintf("%s:%d", "127.0.0.1", sshLocalPort) return conn_str, nil } func SSHConfig(state multistep.StateBag) (*gossh.ClientConfig, error) { config := state.Get("commonconfig").(CommonConfig) auth := []gossh.AuthMethod{ gossh.Password(config.SSHPassword), gossh.KeyboardInteractive( ssh.PasswordKeyboardInteractive(config.SSHPassword)), } if config.SSHKeyPath != "" { signer, err := commonssh.FileSigner(config.SSHKeyPath) if err != nil { return nil, err } auth = append(auth, gossh.PublicKeys(signer)) } return &gossh.ClientConfig{ User: config.SSHUser, Auth: auth, }, nil } func doExecuteSSHCmd(cmd, target string, config *gossh.ClientConfig) (stdout string, err error) { client, err := gossh.Dial("tcp", target, config) if err != nil { return "", err } //Create session session, err := client.NewSession() if err != nil { return "", err } defer session.Close() var b bytes.Buffer session.Stdout = &b if err := session.Run(cmd); err != nil { return "", err } return strings.Trim(b.String(), "\n"), nil } func ExecuteHostSSHCmd(state multistep.StateBag, cmd string) (stdout string, err error) { config := state.Get("commonconfig").(CommonConfig) // Setup connection config sshConfig := &gossh.ClientConfig{ User: config.Username, Auth: []gossh.AuthMethod{ gossh.Password(config.Password), }, } return doExecuteSSHCmd(cmd, config.HostIp+":22", sshConfig) } func ExecuteGuestSSHCmd(state multistep.StateBag, cmd string) (stdout string, err error) { localAddress, err := SSHLocalAddress(state) if err != nil { return } sshConfig, err := SSHConfig(state) if err != nil { return } return doExecuteSSHCmd(cmd, localAddress, sshConfig) } func forward(local_conn net.Conn, config *gossh.ClientConfig, server, remote_dest string, remote_port uint) error { defer local_conn.Close() ssh_client_conn, err := gossh.Dial("tcp", server+":22", config) if err != nil { log.Printf("local ssh.Dial error: %s", err) return err } defer ssh_client_conn.Close() remote_loc := fmt.Sprintf("%s:%d", remote_dest, remote_port) ssh_conn, err := ssh_client_conn.Dial("tcp", remote_loc) if err != nil { log.Printf("ssh.Dial error: %s", err) return err } defer ssh_conn.Close() txDone := make(chan struct{}) rxDone := make(chan struct{}) go func() { _, err = io.Copy(ssh_conn, local_conn) if err != nil { log.Printf("io.copy failed: %v", err) } close(txDone) }() go func() { _, err = io.Copy(local_conn, ssh_conn) if err != nil { log.Printf("io.copy failed: %v", err) } close(rxDone) }() <-txDone <-rxDone return nil } func ssh_port_forward(local_port uint, remote_port uint, remote_dest, host, username, password string) error { config := &gossh.ClientConfig{ User: username, Auth: []gossh.AuthMethod{ gossh.Password(password), }, } // Listen on a local port local_listener, err := net.Listen("tcp", fmt.Sprintf("%s:%d", "127.0.0.1", local_port)) if err != nil { log.Printf("Local listen failed: %s", err) return err } for { local_connection, err := local_listener.Accept() if err != nil { log.Printf("Local accept failed: %s", err) return err } // Forward to a remote port go forward(local_connection, config, host, remote_dest, remote_port) } return nil }