Skip to content

Commit

Permalink
Add xenstore support to the xenorchestra_vm resource
Browse files Browse the repository at this point in the history
Signed-off-by: Dom Del Nano <ddelnano@gmail.com>
  • Loading branch information
ddelnano committed Jan 25, 2024
1 parent 526f26d commit bc7dea4
Show file tree
Hide file tree
Showing 4 changed files with 192 additions and 6 deletions.
26 changes: 21 additions & 5 deletions client/vm.go
Original file line number Diff line number Diff line change
Expand Up @@ -125,11 +125,12 @@ type Vm struct {
ResourceSet *FlatResourceSet `json:"resourceSet"`
// TODO: (#145) Uncomment this once issues with secure_boot have been figured out
// SecureBoot bool `json:"secureBoot,omitempty"`
Tags []string `json:"tags"`
Videoram Videoram `json:"videoram,omitempty"`
Vga string `json:"vga,omitempty"`
StartDelay int `json:startDelay,omitempty"`
Host string `json:"$container"`
Tags []string `json:"tags"`
Videoram Videoram `json:"videoram,omitempty"`
Vga string `json:"vga,omitempty"`
StartDelay int `json:startDelay,omitempty"`
Host string `json:"$container"`
XenstoreData map[string]interface{} `json:"xenStoreData,omitempty"`

// These fields are used for passing in disk inputs when
// creating Vms, however, this is not a real field as far
Expand Down Expand Up @@ -357,6 +358,17 @@ func (c *Client) CreateVm(vmReq Vm, createTime time.Duration) (*Vm, error) {
return nil, err
}

xsParams := map[string]interface{}{
"id": vmId,
"xenStoreData": vmReq.XenstoreData,
}
var success bool
err = c.Call("vm.set", xsParams, &success)

if err != nil {
return nil, err
}

bootAfterCreate := params["bootAfterCreate"].(bool)
if !bootAfterCreate && vmReq.PowerState == RunningPowerState {
err = c.StartVm(vmId)
Expand Down Expand Up @@ -431,6 +443,10 @@ func (c *Client) UpdateVm(vmReq Vm) (*Vm, error) {
params["resourceSet"] = vmReq.ResourceSet
}

if len(vmReq.XenstoreData) > 0 {
params["xenStoreData"] = vmReq.XenstoreData
}

vga := vmReq.Vga
if vga != "" {
params["vga"] = vga
Expand Down
7 changes: 7 additions & 0 deletions examples/resources/xenorchestra_vm/resource.tf
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,13 @@ resource "xenorchestra_vm" "bar" {
timeouts {
create = "20m"
}

// Note: Xen Orchestra populates values within Xenstore and will need ignored via
// lifecycle ignore_changes or modeled in your terraform code
xenstore = {
key1 = "val1"
key2 = "val2"
}
}

# vm resource that uses wait_for_ip
Expand Down
48 changes: 47 additions & 1 deletion xoa/resource_xenorchestra_vm.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import (
"regexp"
"sort"
"strconv"
"strings"
"time"

"github.com/ddelnano/terraform-provider-xenorchestra/client"
Expand Down Expand Up @@ -408,6 +409,14 @@ $ xo-cli xo.getAllObjects filter='json:{"id": "cf7b5d7d-3cd5-6b7c-5025-5c935c8cd
},
},
},
"xenstore": &schema.Schema{
Type: schema.TypeMap,
Optional: true,
Description: "The key value pairs to be populated in xenstore.",
Elem: &schema.Schema{
Type: schema.TypeString,
},
},
"tags": resourceTags(),
}
}
Expand Down Expand Up @@ -566,7 +575,8 @@ func resourceVmCreate(d *schema.ResourceData, m interface{}) error {
Videoram: client.Videoram{
Value: d.Get("videoram").(int),
},
Vga: d.Get("vga").(string),
XenstoreData: d.Get("xenstore").(map[string]interface{}),
Vga: d.Get("vga").(string),
}

affinityHost := d.Get("affinity_host").(string)
Expand Down Expand Up @@ -927,6 +937,23 @@ func resourceVmUpdate(d *schema.ResourceData, m interface{}) error {
vmReq.AffinityHost = &affinityHost
}

if d.HasChange("xenstore") {
xenstoreParams := map[string]interface{}{}
o, n := d.GetChange("xenstore")
oXs := o.(map[string]interface{})
nXs := n.(map[string]interface{})

for k, _ := range oXs {
xenstoreParams[k] = nil
}

for k, v := range nXs {
xenstoreParams[k] = v
}

vmReq.XenstoreData = xenstoreParams
}

haltPerformed := false

if haltForUpdates {
Expand Down Expand Up @@ -1185,10 +1212,29 @@ func recordToData(resource client.Vm, vifs []client.VIF, disks []client.Disk, cd
return err
}
}
filtered := filterXenstoreDataToVmData(resource.XenstoreData)
log.Printf("[DEBUG] Found the following xenstore data : %v after=%v\n", resource.XenstoreData, filtered)
if err := d.Set("xenstore", filtered); err != nil {
return err
}

return nil
}

func filterXenstoreDataToVmData(xenstore map[string]interface{}) map[string]interface{} {
filtered := map[string]interface{}{}
for key, value := range xenstore {
if strings.HasPrefix(key, "vm-data/") {
pieces := strings.SplitAfterN(key, "vm-data/", 2)
if len(pieces) != 2 {
continue
}
filtered[pieces[1]] = value
}
}
return filtered
}

func vmBlockedOperationsToList(v client.Vm) []string {
blockedOperations := []string{}
for k, _ := range v.BlockedOperations {
Expand Down
117 changes: 117 additions & 0 deletions xoa/resource_xenorchestra_vm_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -622,6 +622,46 @@ func TestAccXenorchestraVm_createWhenWaitingForIp(t *testing.T) {
})
}

func TestAccXenorchestraVm_createAndUpdateXenstoreData(t *testing.T) {
resourceName := "xenorchestra_vm.bar"
vmName := fmt.Sprintf("%s - %s", accTestPrefix, t.Name())
resource.ParallelTest(t, resource.TestCase{
PreCheck: func() { testAccPreCheck(t) },
Providers: testAccProviders,
CheckDestroy: testAccCheckXenorchestraVmDestroy,
Steps: []resource.TestStep{
{
Config: testAccVmConfigWithSingleXenstoreData(vmName),
Check: resource.ComposeAggregateTestCheckFunc(
testAccVmExists(resourceName),
resource.TestCheckResourceAttrSet(resourceName, "id"),
resource.TestCheckResourceAttr(resourceName, "xenstore.%", "2"),
resource.TestCheckResourceAttr(resourceName, "xenstore.first", "value"),
),
},
{
Config: testAccVmConfigWithMultipleXenstoreData(vmName),
Check: resource.ComposeAggregateTestCheckFunc(
testAccVmExists(resourceName),
resource.TestCheckResourceAttrSet(resourceName, "id"),
resource.TestCheckResourceAttr(resourceName, "xenstore.first", "value"),
resource.TestCheckResourceAttr(resourceName, "xenstore.second", "value"),
),
},
{
Config: testAccVmConfigWithSingleXenstoreData(vmName),
Check: resource.ComposeAggregateTestCheckFunc(
testAccVmExists(resourceName),
resource.TestCheckResourceAttrSet(resourceName, "id"),
resource.TestCheckResourceAttr(resourceName, "xenstore.%", "2"),
resource.TestCheckResourceAttr(resourceName, "xenstore.first", "value"),
resource.TestCheckNoResourceAttr(resourceName, "xenstore.second"),
),
},
},
})
}

func TestAccXenorchestraVm_ensureVmsInResourceSetsCanBeUpdatedByNonAdminUsers(t *testing.T) {
vmName := fmt.Sprintf("%s - %s", accTestPrefix, t.Name())
adminUser := os.Getenv("XOA_USER")
Expand Down Expand Up @@ -2261,6 +2301,83 @@ resource "xenorchestra_vm" "bar" {
`, accDefaultNetwork.NameLabel, accTestPool.Id, vmName, accDefaultSr.Id)
}

func testAccVmConfigWithSingleXenstoreData(vmName string) string {
return testAccCloudConfigConfig(fmt.Sprintf("vm-template-%s", vmName), "template") + testAccTemplateConfig() + fmt.Sprintf(`
data "xenorchestra_network" "network" {
name_label = "%s"
pool_id = "%s"
}
resource "xenorchestra_vm" "bar" {
memory_max = 4295000000
wait_for_ip = true
cpus = 1
cloud_config = xenorchestra_cloud_config.bar.template
name_label = "%s"
name_description = "description"
template = data.xenorchestra_template.template.id
network {
network_id = data.xenorchestra_network.network.id
}
disk {
sr_id = "%s"
name_label = "disk 1"
size = 10001317888
}
xenstore = {
first = "value"
}
lifecycle {
ignore_changes = [
xenstore["mmio-hole-size"],
]
}
}
`, accDefaultNetwork.NameLabel, accTestPool.Id, vmName, accDefaultSr.Id)
}

func testAccVmConfigWithMultipleXenstoreData(vmName string) string {
return testAccCloudConfigConfig(fmt.Sprintf("vm-template-%s", vmName), "template") + testAccTemplateConfig() + fmt.Sprintf(`
data "xenorchestra_network" "network" {
name_label = "%s"
pool_id = "%s"
}
resource "xenorchestra_vm" "bar" {
memory_max = 4295000000
wait_for_ip = true
cpus = 1
cloud_config = xenorchestra_cloud_config.bar.template
name_label = "%s"
name_description = "description"
template = data.xenorchestra_template.template.id
network {
network_id = data.xenorchestra_network.network.id
}
disk {
sr_id = "%s"
name_label = "disk 1"
size = 10001317888
}
xenstore = {
first = "value"
second = "value"
}
lifecycle {
ignore_changes = [
xenstore["mmio-hole-size"],
]
}
}
`, accDefaultNetwork.NameLabel, accTestPool.Id, vmName, accDefaultSr.Id)
}

func testAccVmConfigWithDiskNameLabelAndNameDescription(vmName, nameLabel, description string) string {
return testAccCloudConfigConfig(fmt.Sprintf("vm-template-%s", vmName), "template") + testAccTemplateConfig() + fmt.Sprintf(`
data "xenorchestra_network" "network" {
Expand Down

0 comments on commit bc7dea4

Please sign in to comment.