Skip to content

Commit

Permalink
Implement Host Transport Node
Browse files Browse the repository at this point in the history
Signed-off-by: graysonwu <[email protected]>
  • Loading branch information
GraysonWu committed Sep 19, 2023
1 parent 2e54c46 commit 488baf8
Show file tree
Hide file tree
Showing 5 changed files with 488 additions and 7 deletions.
1 change: 1 addition & 0 deletions nsxt/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -428,6 +428,7 @@ func Provider() *schema.Provider {
"nsxt_transport_node": resourceNsxtTransportNode(),
"nsxt_failure_domain": resourceNsxtFailureDomain(),
"nsxt_cluster_virtual_ip": resourceNsxtClusterVirualIP(),
"nsxt_policy_host_transport_node": resourceNsxtPolicyHostTransportNode(),
},

ConfigureFunc: providerConfigure,
Expand Down
308 changes: 308 additions & 0 deletions nsxt/resource_nsxt_policy_host_transport_node.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,308 @@
/* Copyright © 2020 VMware, Inc. All Rights Reserved.
SPDX-License-Identifier: MPL-2.0 */

package nsxt

import (
"fmt"
"log"

"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
"github.com/vmware/vsphere-automation-sdk-go/runtime/protocol/client"
"github.com/vmware/vsphere-automation-sdk-go/services/nsxt/infra"
"github.com/vmware/vsphere-automation-sdk-go/services/nsxt/infra/sites/enforcement_points"
model2 "github.com/vmware/vsphere-automation-sdk-go/services/nsxt/model"
"golang.org/x/exp/maps"
)

func resourceNsxtPolicyHostTransportNode() *schema.Resource {
return &schema.Resource{
Create: resourceNsxtPolicyHostTransportNodeCreate,
Read: resourceNsxtPolicyHostTransportNodeRead,
Update: resourceNsxtPolicyHostTransportNodeUpdate,
Delete: resourceNsxtPolicyHostTransportNodeDelete,
Importer: &schema.ResourceImporter{
State: resourceNsxtPolicyHostTransportNodeImporter,
},

Schema: map[string]*schema.Schema{
"nsx_id": getNsxIDSchema(),
"path": getPathSchema(),
"display_name": getDisplayNameSchema(),
"description": getDescriptionSchema(),
"revision": getRevisionSchema(),
"tag": getTagsSchema(),
"site_path": {
Type: schema.TypeString,
Description: "Path to the site this Host Transport Node belongs to",
Optional: true,
ForceNew: true,
Default: defaultInfraSitePath,
ValidateFunc: validatePolicyPath(),
},
"enforcement_point": {
Type: schema.TypeString,
Description: "ID of the enforcement point this Host Transport Node belongs to",
Optional: true,
ForceNew: true,
Computed: true,
},
"discovered_node_id": {
Type: schema.TypeString,
Optional: true,
Description: "Discovered node id to create Host Transport Node",
},
"node_deployment_info": getFabricHostNodeSchema(),
// host_switch_spec
"standard_host_switch": getStandardHostSwitchSchema(),
"preconfigured_host_switch": getPreconfiguredHostSwitchSchema(),
},
}
}

func getFabricHostNodeSchema() *schema.Schema {
elemSchema := map[string]*schema.Schema{
"fqdn": {
Type: schema.TypeString,
Computed: true,
Description: "Fully qualified domain name of the fabric node",
},
"ip_addresses": {
Type: schema.TypeList,
Optional: true,
Description: "IP Addresses of the Node, version 4 or 6",
Elem: &schema.Schema{
Type: schema.TypeString,
},
},
}
maps.Copy(elemSchema, getHostNodeSchemaAddlElements())
s := schema.Schema{
Type: schema.TypeList,
MaxItems: 1,
Optional: true,
Elem: &schema.Resource{
Schema: elemSchema,
},
}
return &s
}

func resourceNsxtPolicyHostTransportNodeRead(d *schema.ResourceData, m interface{}) error {
connector := getPolicyConnector(m)
htnClient := enforcement_points.NewHostTransportNodesClient(connector)

id, siteID, epID, err := policyIDSiteEPTuple(d, m)
if err != nil {
return err
}

obj, err := htnClient.Get(siteID, epID, id)
if err != nil {
return handleReadError(d, "HostTransportNode", id, err)
}
sitePath, err := getSitePathFromChildResourcePath(*obj.ParentPath)
if err != nil {
return handleReadError(d, "HostTransportNode", id, err)
}

d.Set("site_path", sitePath)
d.Set("enforcement_point", epID)
d.Set("display_name", obj.DisplayName)
d.Set("description", obj.Description)
setPolicyTagsInSchema(d, obj.Tags)
d.Set("nsx_id", id)
d.Set("path", obj.Path)
d.Set("revision", obj.Revision)

err = setHostSwitchSpecInSchema(d, obj.HostSwitchSpec)
if err != nil {
return err
}

fabricHostNode := obj.NodeDeploymentInfo
elem := make(map[string]interface{})
elem["fqdn"] = fabricHostNode.Fqdn
elem["ip_addresses"] = fabricHostNode.IpAddresses

elem["os_type"] = fabricHostNode.OsType
elem["os_version"] = fabricHostNode.OsVersion
elem["windows_install_location"] = fabricHostNode.WindowsInstallLocation

d.Set("node_deployment_info", []map[string]interface{}{elem})

return nil
}

func resourceNsxtPolicyHostTransportNodeExists(siteID, epID, tzID string, connector client.Connector) (bool, error) {
var err error

// Check site existence first
siteClient := infra.NewSitesClient(connector)
_, err = siteClient.Get(siteID)
if err != nil {
msg := fmt.Sprintf("failed to read site %s", siteID)
return false, logAPIError(msg, err)
}

// Check (ep, htn) existence. In case of ep not found, NSX returns BAD_REQUEST
htnClient := enforcement_points.NewHostTransportNodesClient(connector)
_, err = htnClient.Get(siteID, epID, tzID)
if err == nil {
return true, nil
}

if isNotFoundError(err) {
return false, nil
}

return false, logAPIError("Error retrieving resource", err)
}

func getFabricHostNodeFromSchema(d *schema.ResourceData) *model2.FabricHostNode {
for _, ni := range d.Get("node_deployment_info").([]interface{}) {
nodeInfo := ni.(map[string]interface{})
ipAddresses := interfaceListToStringList(nodeInfo["ip_addresses"].([]interface{}))

var hostCredential *model2.HostNodeLoginCredential
for _, hci := range nodeInfo["host_credential"].([]interface{}) {
hc := hci.(map[string]interface{})
password := hc["password"].(string)
thumbprint := hc["thumbprint"].(string)
username := hc["username"].(string)
hostCredential = &model2.HostNodeLoginCredential{
Password: &password,
Thumbprint: &thumbprint,
Username: &username,
}
}
osType := nodeInfo["os_type"].(string)
osVersion := nodeInfo["os_version"].(string)
windowsInstallLocation := nodeInfo["windows_install_location"].(string)

fabricHostNode := model2.FabricHostNode{
IpAddresses: ipAddresses,
HostCredential: hostCredential,
OsType: &osType,
OsVersion: &osVersion,
WindowsInstallLocation: &windowsInstallLocation,
}
return &fabricHostNode
}
return nil
}

func policyHostTransportNodePatch(siteID, epID, htnID string, d *schema.ResourceData, m interface{}) error {
connector := getPolicyConnector(m)
htnClient := enforcement_points.NewHostTransportNodesClient(connector)

description := d.Get("description").(string)
displayName := d.Get("display_name").(string)
tags := getPolicyTagsFromSchema(d)
discoveredNodeID := d.Get("discovered_node_id").(string)
nodeDeploymentInfo := getFabricHostNodeFromSchema(d)
hostSwitchSpec, err := getHostSwitchSpecFromSchema(d)
if err != nil {
return fmt.Errorf("failed to create hostSwitchSpec of HostTransportNode: %v", err)
}

obj := model2.HostTransportNode{
Description: &description,
DisplayName: &displayName,
Tags: tags,
HostSwitchSpec: hostSwitchSpec,
NodeDeploymentInfo: nodeDeploymentInfo,
DiscoveredNodeIdForCreate: &discoveredNodeID,
}

return htnClient.Patch(siteID, epID, htnID, obj, nil, nil, nil, nil, nil, nil, nil)
}

func resourceNsxtPolicyHostTransportNodeCreate(d *schema.ResourceData, m interface{}) error {
connector := getPolicyConnector(m)
id := d.Get("nsx_id").(string)
if id == "" {
id = newUUID()
}
sitePath := d.Get("site_path").(string)
siteID := getResourceIDFromResourcePath(sitePath, "sites")
if siteID == "" {
return fmt.Errorf("error obtaining Site ID from site path %s", sitePath)
}
epID := d.Get("enforcement_point").(string)
if epID == "" {
epID = getPolicyEnforcementPoint(m)
}
exists, err := resourceNsxtPolicyHostTransportNodeExists(siteID, epID, id, connector)
if err != nil {
return err
}
if exists {
return fmt.Errorf("resource with ID %s already exists", id)
}

// Create the resource using PATCH
log.Printf("[INFO] Creating HostTransportNode with ID %s under site %s enforcement point %s", id, siteID, epID)
err = policyHostTransportNodePatch(siteID, epID, id, d, m)
if err != nil {
return handleCreateError("HostTransportNode", id, err)
}

d.SetId(id)
d.Set("nsx_id", id)

return resourceNsxtPolicyHostTransportNodeRead(d, m)
}

func resourceNsxtPolicyHostTransportNodeUpdate(d *schema.ResourceData, m interface{}) error {
id, siteID, epID, err := policyIDSiteEPTuple(d, m)
if err != nil {
return err
}

log.Printf("[INFO] Updateing HostTransportNode with ID %s", id)
err = policyHostTransportNodePatch(siteID, epID, id, d, m)
if err != nil {
return handleUpdateError("HostTransportNode", id, err)
}

return resourceNsxtPolicyHostTransportNodeRead(d, m)
}

func resourceNsxtPolicyHostTransportNodeDelete(d *schema.ResourceData, m interface{}) error {
connector := getPolicyConnector(m)
htnClient := enforcement_points.NewHostTransportNodesClient(connector)

id, siteID, epID, err := policyIDSiteEPTuple(d, m)
if err != nil {
return err
}

log.Printf("[INFO] Deleting HostTransportNode with ID %s", id)
err = htnClient.Delete(siteID, epID, id, nil, nil)
if err != nil {
return handleDeleteError("HostTransportNode", id, err)
}

return nil
}

func resourceNsxtPolicyHostTransportNodeImporter(d *schema.ResourceData, m interface{}) ([]*schema.ResourceData, error) {
importID := d.Id()
rd, err := nsxtPolicyPathResourceImporterHelper(d, m)
if err != nil {
return rd, err
}

epID, err := getParameterFromPolicyPath("/enforcement-points/", "/host-transport-nodes/", importID)
if err != nil {
return nil, err
}
d.Set("enforcement_point", epID)
sitePath, err := getSitePathFromChildResourcePath(importID)
if err != nil {
return rd, err
}
d.Set("site_path", sitePath)
return rd, nil
}
8 changes: 4 additions & 4 deletions nsxt/resource_nsxt_policy_transport_zone.go
Original file line number Diff line number Diff line change
Expand Up @@ -147,7 +147,7 @@ func policyTransportZonePatch(siteID, epID, tzID string, d *schema.ResourceData,
return err
}

func policyTransportZoneIDTuple(d *schema.ResourceData, m interface{}) (id, siteID, epID string, err error) {
func policyIDSiteEPTuple(d *schema.ResourceData, m interface{}) (id, siteID, epID string, err error) {
id = d.Id()
if id == "" {
err = fmt.Errorf("error obtaining TransportZone ID")
Expand Down Expand Up @@ -207,7 +207,7 @@ func resourceNsxtPolicyTransportZoneRead(d *schema.ResourceData, m interface{})
connector := getPolicyConnector(m)
tzClient := enforcement_points.NewTransportZonesClient(connector)

id, siteID, epID, err := policyTransportZoneIDTuple(d, m)
id, siteID, epID, err := policyIDSiteEPTuple(d, m)
if err != nil {
return err
}
Expand Down Expand Up @@ -238,7 +238,7 @@ func resourceNsxtPolicyTransportZoneRead(d *schema.ResourceData, m interface{})
}

func resourceNsxtPolicyTransportZoneUpdate(d *schema.ResourceData, m interface{}) error {
id, siteID, epID, err := policyTransportZoneIDTuple(d, m)
id, siteID, epID, err := policyIDSiteEPTuple(d, m)
if err != nil {
return err
}
Expand All @@ -256,7 +256,7 @@ func resourceNsxtPolicyTransportZoneDelete(d *schema.ResourceData, m interface{}
connector := getPolicyConnector(m)
tzClient := enforcement_points.NewTransportZonesClient(connector)

id, siteID, epID, err := policyTransportZoneIDTuple(d, m)
id, siteID, epID, err := policyIDSiteEPTuple(d, m)
if err != nil {
return err
}
Expand Down
14 changes: 11 additions & 3 deletions nsxt/resource_nsxt_transport_node.go
Original file line number Diff line number Diff line change
Expand Up @@ -489,8 +489,8 @@ func getEdgeNodeSchema() *schema.Schema {
return getNodeSchema(s, false)
}

func getHostNodeSchema() *schema.Schema {
s := map[string]*schema.Schema{
func getHostNodeSchemaAddlElements() map[string]*schema.Schema {
return map[string]*schema.Schema{
"host_credential": {
Type: schema.TypeList,
MaxItems: 1,
Expand Down Expand Up @@ -535,6 +535,10 @@ func getHostNodeSchema() *schema.Schema {
Description: "Install location of Windows Server on baremetal being managed by NSX",
},
}
}

func getHostNodeSchema() *schema.Schema {
s := getHostNodeSchemaAddlElements()
return getNodeSchema(s, false)
}

Expand Down Expand Up @@ -1393,6 +1397,10 @@ func getHostSwitchSpecFromSchema(d *schema.ResourceData) (*data.StructValue, err
return nil, fmt.Errorf("error parsing HostSwitchSpec schema %v", err)
}
portGroupTZID := swData["portgroup_transport_zone_id"].(string)
portGroupTZIDPtr := &portGroupTZID
if portGroupTZID == "" {
portGroupTZIDPtr = nil
}
transportNodeSubProfileCfg := getTransportNodeSubProfileCfg(swData["transport_node_profile_sub_configs"])
transportZoneEndpoints := getTransportZoneEndpointsFromSchema(swData["transport_zone_endpoint"].([]interface{}))
uplinks := getUplinksFromSchema(swData["uplink"].([]interface{}))
Expand All @@ -1406,7 +1414,7 @@ func getHostSwitchSpecFromSchema(d *schema.ResourceData) (*data.StructValue, err
IpAssignmentSpec: iPAssignmentSpec,
IsMigratePnics: &isMigratePNics,
Pnics: pNics,
PortgroupTransportZoneId: &portGroupTZID,
PortgroupTransportZoneId: portGroupTZIDPtr,
TransportNodeProfileSubConfigs: transportNodeSubProfileCfg,
TransportZoneEndpoints: transportZoneEndpoints,
Uplinks: uplinks,
Expand Down
Loading

0 comments on commit 488baf8

Please sign in to comment.