File: step_configure_qmp.go

package info (click to toggle)
packer 1.6.6%2Bds1-2
  • links: PTS, VCS
  • area: main
  • in suites: bullseye
  • size: 32,016 kB
  • sloc: sh: 1,154; python: 619; makefile: 251; ruby: 205; xml: 97
file content (92 lines) | stat: -rw-r--r-- 2,394 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
package qemu

import (
	"context"
	"fmt"
	"log"
	"os"
	"time"

	"github.com/digitalocean/go-qemu/qmp"
	"github.com/hashicorp/packer/packer-plugin-sdk/multistep"
	packersdk "github.com/hashicorp/packer/packer-plugin-sdk/packer"
)

// This step configures the VM to enable the QMP listener.
//
// Uses:
//   config *config
//   ui     packersdk.Ui
//
// Produces:
type stepConfigureQMP struct {
	monitor       *qmp.SocketMonitor
	QMPSocketPath string
}

func (s *stepConfigureQMP) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction {
	config := state.Get("config").(*Config)
	ui := state.Get("ui").(packersdk.Ui)

	if !config.QMPEnable {
		return multistep.ActionContinue
	}

	ui.Say(fmt.Sprintf("QMP socket at: %s", s.QMPSocketPath))

	// Only initialize and open QMP when we have a use for it.
	// Open QMP socket
	var err error
	var cmd []byte
	var result []byte
	s.monitor, err = qmp.NewSocketMonitor("unix", s.QMPSocketPath, 2*time.Second)
	if err != nil {
		err := fmt.Errorf("Error opening QMP socket: %s", err)
		state.Put("error", err)
		ui.Error(err.Error())
		return multistep.ActionHalt
	}

	// Connect to QMP
	// function automatically calls capabilities so is immediately ready for commands
	err = s.monitor.Connect()
	if err != nil {
		err := fmt.Errorf("Error connecting to QMP socket: %s", err)
		state.Put("error", err)
		ui.Error(err.Error())
		return multistep.ActionHalt
	}
	log.Printf("QMP socket open SUCCESS")

	vncPassword := state.Get("vnc_password")
	if vncPassword != "" {
		cmd = []byte(fmt.Sprintf("{ \"execute\": \"change-vnc-password\", \"arguments\": { \"password\": \"%s\" } }",
			vncPassword))
		result, err = s.monitor.Run(cmd)
		if err != nil {
			err := fmt.Errorf("Error connecting to QMP socket: %s", err)
			state.Put("error", err)
			ui.Error(err.Error())
			return multistep.ActionHalt
		}
		log.Printf("QMP Command: %s\nResult: %s", cmd, result)
	}

	// make the qmp_monitor available to other steps.
	state.Put("qmp_monitor", s.monitor)

	return multistep.ActionContinue
}

func (s *stepConfigureQMP) Cleanup(multistep.StateBag) {
	if s.monitor != nil {
		err := s.monitor.Disconnect()
		if err != nil {
			log.Printf("failed to disconnect QMP: %v", err)
		}
		// Delete file associated with qmp socket.
		if err := os.Remove(s.QMPSocketPath); err != nil {
			log.Printf("Failed to delete the qmp socket file: %s", err)
		}
	}
}