Commit aefb947f authored by Mitchell Hashimoto's avatar Mitchell Hashimoto

builder/vmware: Support <wait>, send proper keycodes

parent 6d610f1c
package vmware package vmware
import ( import (
"bytes"
"fmt" "fmt"
"github.com/mitchellh/go-vnc" "github.com/mitchellh/go-vnc"
"github.com/mitchellh/multistep" "github.com/mitchellh/multistep"
...@@ -8,6 +9,7 @@ import ( ...@@ -8,6 +9,7 @@ import (
"log" "log"
"net" "net"
"strings" "strings"
"text/template"
"time" "time"
"unicode" "unicode"
"unicode/utf8" "unicode/utf8"
...@@ -15,10 +17,17 @@ import ( ...@@ -15,10 +17,17 @@ import (
const KeyLeftShift uint32 = 0xFFE1 const KeyLeftShift uint32 = 0xFFE1
type bootCommandTemplateData struct {
HTTPIP string
HTTPPort int
Name string
}
// This step "types" the boot command into the VM over VNC. // This step "types" the boot command into the VM over VNC.
// //
// Uses: // Uses:
// config *config // config *config
// http_port int
// ui packer.Ui // ui packer.Ui
// vnc_port uint // vnc_port uint
// //
...@@ -28,6 +37,7 @@ type stepTypeBootCommand struct{} ...@@ -28,6 +37,7 @@ type stepTypeBootCommand struct{}
func (s *stepTypeBootCommand) Run(state map[string]interface{}) multistep.StepAction { func (s *stepTypeBootCommand) Run(state map[string]interface{}) multistep.StepAction {
config := state["config"].(*config) config := state["config"].(*config)
httpPort := state["http_port"].(int)
ui := state["ui"].(packer.Ui) ui := state["ui"].(packer.Ui)
vncPort := state["vnc_port"].(uint) vncPort := state["vnc_port"].(uint)
...@@ -46,10 +56,21 @@ func (s *stepTypeBootCommand) Run(state map[string]interface{}) multistep.StepAc ...@@ -46,10 +56,21 @@ func (s *stepTypeBootCommand) Run(state map[string]interface{}) multistep.StepAc
} }
defer c.Close() defer c.Close()
log.Printf("Connecting to VNC desktop: %s", c.DesktopName) log.Printf("Connected to VNC desktop: %s", c.DesktopName)
tplData := &bootCommandTemplateData{
"127.0.0.1",
httpPort,
config.VMName,
}
ui.Say("Typing the boot command over VNC...") ui.Say("Typing the boot command over VNC...")
for _, command := range config.BootCommand { for _, command := range config.BootCommand {
vncSendString(c, command) var buf bytes.Buffer
t := template.Must(template.New("boot").Parse(command))
t.Execute(&buf, tplData)
vncSendString(c, buf.String())
} }
return multistep.ActionContinue return multistep.ActionContinue
...@@ -63,11 +84,20 @@ func vncSendString(c *vnc.ClientConn, original string) { ...@@ -63,11 +84,20 @@ func vncSendString(c *vnc.ClientConn, original string) {
special["<return>"] = 0xFF0D special["<return>"] = 0xFF0D
special["<esc>"] = 0xFF1B special["<esc>"] = 0xFF1B
shiftedChars := "~!@#$%^&*()_+{}|:\"<>?"
// TODO(mitchellh): Ripe for optimizations of some point, perhaps. // TODO(mitchellh): Ripe for optimizations of some point, perhaps.
for len(original) > 0 { for len(original) > 0 {
var keyCode uint32 var keyCode uint32
keyShift := false keyShift := false
if strings.HasPrefix(original, "<wait>") {
log.Printf("Special code '<wait>' found, sleeping one second")
time.Sleep(1 * time.Second)
original = original[len("<wait>"):]
continue
}
for specialCode, specialValue := range special { for specialCode, specialValue := range special {
if strings.HasPrefix(original, specialCode) { if strings.HasPrefix(original, specialCode) {
log.Printf("Special code '%s' found, replacing with: %d", specialCode, specialValue) log.Printf("Special code '%s' found, replacing with: %d", specialCode, specialValue)
...@@ -81,7 +111,7 @@ func vncSendString(c *vnc.ClientConn, original string) { ...@@ -81,7 +111,7 @@ func vncSendString(c *vnc.ClientConn, original string) {
r, size := utf8.DecodeRuneInString(original) r, size := utf8.DecodeRuneInString(original)
original = original[size:] original = original[size:]
keyCode = uint32(r) keyCode = uint32(r)
keyShift = unicode.IsUpper(r) keyShift = unicode.IsUpper(r) || strings.ContainsRune(shiftedChars, r)
log.Printf("Sending char '%c', code %d, shift %v", r, keyCode, keyShift) log.Printf("Sending char '%c', code %d, shift %v", r, keyCode, keyShift)
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment