Skip to content

Commit

Permalink
Apply logical_port_tags to all VM ports
Browse files Browse the repository at this point in the history
When VM is attached to multiple NSX switches, nsxt_vm_tags resource
needs to update logical port tags on all relevant logical ports.
We assume tags are fully managed by terraform and thus are the same
for all ports.
  • Loading branch information
annakhm committed Feb 13, 2020
1 parent a4ec8c0 commit 19afbe4
Show file tree
Hide file tree
Showing 2 changed files with 39 additions and 18 deletions.
55 changes: 38 additions & 17 deletions nsxt/resource_nsxt_vm_tags.go
Original file line number Diff line number Diff line change
Expand Up @@ -118,40 +118,45 @@ func findVMByExternalID(nsxClient *api.APIClient, instanceID string) (*manager.V
return nil, fmt.Errorf("Failed to find Virtual Machine with id %s in inventory", instanceID)
}

func findVIFByExternalID(nsxClient *api.APIClient, instanceID string) (*manager.VirtualNetworkInterface, error) {
func findVIFsByExternalID(nsxClient *api.APIClient, instanceID string) ([]manager.VirtualNetworkInterface, error) {

vifList, err := getVIFList(nsxClient)
var resultList []manager.VirtualNetworkInterface
if err != nil {
return nil, err
}

for _, vif := range vifList.Results {
if vif.OwnerVmId == instanceID {
return &vif, nil
resultList = append(resultList, vif)
}
}

return nil, fmt.Errorf("Failed to find VIF for VM id %s in inventory", instanceID)
return resultList, nil
}

func findPortByExternalID(nsxClient *api.APIClient, instanceID string) (*manager.LogicalPort, error) {
func findPortsByExternalID(nsxClient *api.APIClient, instanceID string) ([]manager.LogicalPort, error) {

vif, err := findVIFByExternalID(nsxClient, instanceID)
var resultList []manager.LogicalPort
vifs, err := findVIFsByExternalID(nsxClient, instanceID)
if err != nil {
return nil, err
return resultList, err
}

portList, err := getPortList(nsxClient)
if err != nil {
return nil, err
return resultList, err
}
for _, elem := range portList.Results {
if elem.Attachment != nil && elem.Attachment.Id == vif.LportAttachmentId {
return &elem, nil
for _, vif := range vifs {
if elem.Attachment != nil && elem.Attachment.Id == vif.LportAttachmentId {
resultList = append(resultList, elem)
continue
}
}
}

return nil, fmt.Errorf("Failed to find logical port with attachment id %s", vif.LportAttachmentId)
return resultList, nil
}

func updateTags(nsxClient *api.APIClient, id string, tags []common.Tag) error {
Expand All @@ -178,18 +183,26 @@ func updateTags(nsxClient *api.APIClient, id string, tags []common.Tag) error {
func updatePortTags(nsxClient *api.APIClient, id string, tags []common.Tag) error {
log.Printf("[DEBUG] Updating logical port tags for %s", id)

port, err := findPortByExternalID(nsxClient, id)
ports, err := findPortsByExternalID(nsxClient, id)
if err != nil {
return err
}

port.Tags = tags
_, resp, err := nsxClient.LogicalSwitchingApi.UpdateLogicalPort(nsxClient.Context, port.Id, *port)
portsUpdated := 0

if err != nil || resp.StatusCode == http.StatusNotFound {
return fmt.Errorf("Error while updating logical port %s: %v", id, err)
for _, port := range ports {
port.Tags = tags
log.Printf("[DEBUG] Applying %d tags on logical port %s", len(tags), port.Id)
_, resp, err := nsxClient.LogicalSwitchingApi.UpdateLogicalPort(nsxClient.Context, port.Id, port)

if err != nil || (resp != nil && resp.StatusCode == http.StatusNotFound) {
return fmt.Errorf("Error while updating tags on logical port %s: %v", port.Id, err)
}
portsUpdated++
}

log.Printf("[INFO] Applied %d tags on %d logical ports for VM %s", len(tags), portsUpdated, id)

return nil
}

Expand Down Expand Up @@ -239,13 +252,21 @@ func resourceNsxtVMTagsRead(d *schema.ResourceData, m interface{}) error {
return fmt.Errorf("Error during VM retrieval: %v", err)
}

port, err := findPortByExternalID(nsxClient, id)
ports, err := findPortsByExternalID(nsxClient, id)
if err != nil {
return fmt.Errorf("Error during logical port retrieval: %v", err)
}

setTagsInSchema(d, vm.Tags)
setCustomizedTagsInSchema(d, port.Tags, "logical_port_tag")
// assuming all ports have same tags
// note - more flexible implementation will be provided with policy resource
if len(ports) > 0 {
setCustomizedTagsInSchema(d, ports[0].Tags, "logical_port_tag")
} else {
// assign empty list
var tagList []map[string]string
d.Set("logical_port_tag", tagList)
}

return nil
}
Expand Down
2 changes: 1 addition & 1 deletion website/docs/r/vm_tags.html.markdown
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ The following arguments are supported:

* `instance_id` - (Required) BIOS Id of the Virtual Machine.
* `tag` - (Optional) A list of scope + tag pairs to associate with this VM.
* `logical_port_tag` - (Optional) A list of scope + tag pairs to associate with logical port that is automatically created for this VM.
* `logical_port_tag` - (Optional) A list of scope + tag pairs to associate with all logical ports that are automatically created for this VM.

## Importing

Expand Down

0 comments on commit 19afbe4

Please sign in to comment.