Skip to content

Commit

Permalink
Merge pull request hashicorp#141 from terraform-providers/f-vmfs-disk…
Browse files Browse the repository at this point in the history
…s-data-source

New data source: vsphere_vmfs_disks
  • Loading branch information
vancluever authored Sep 1, 2017
2 parents 17c4453 + b5407a9 commit 966ba86
Show file tree
Hide file tree
Showing 7 changed files with 294 additions and 0 deletions.
2 changes: 2 additions & 0 deletions tf-vsphere-devrc.mk.example
Original file line number Diff line number Diff line change
Expand Up @@ -38,5 +38,7 @@ export VSPHERE_DC_FOLDER ?= dc-folder # DC resource test folder
export VSPHERE_ESXI_HOST ?= esxi1 # ESXi host to work with
export VSPHERE_HOST_NIC0 ?= vmnic0 # NIC0 for host net tests
export VSPHERE_HOST_NIC1 ?= vmnic1 # NIC1 for host net tests
export VSPHERE_VMFS_EXPECTED ?= scsi-name # Name of expected SCSI disk
export VSPHERE_VMFS_REGEXP ?= expr # Regexp for SCSI disk search

# vi: filetype=make
89 changes: 89 additions & 0 deletions vsphere/data_source_vsphere_vmfs_disks.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
package vsphere

import (
"context"
"fmt"
"regexp"
"sort"
"time"

"github.com/hashicorp/terraform/helper/schema"
"github.com/hashicorp/terraform/helper/validation"
"github.com/vmware/govmomi"
"github.com/vmware/govmomi/vim25/mo"
"github.com/vmware/govmomi/vim25/types"
)

func dataSourceVSphereVmfsDisks() *schema.Resource {
return &schema.Resource{
Read: dataSourceVSphereVmfsDisksRead,

Schema: map[string]*schema.Schema{
"host_system_id": &schema.Schema{
Type: schema.TypeString,
Description: "The managed object ID of the host to search for disks on.",
Required: true,
},
"rescan": &schema.Schema{
Type: schema.TypeBool,
Description: "Rescan the system for disks before querying. This may lengthen the time it takes to gather information.",
Optional: true,
},
"filter": &schema.Schema{
Type: schema.TypeString,
Description: "A regular expression to filter the disks against. Only disks with canonical names that match will be included.",
Optional: true,
ValidateFunc: validation.ValidateRegexp,
},
"disks": &schema.Schema{
Type: schema.TypeList,
Description: "The names of the disks discovered by the search.",
Computed: true,
Elem: &schema.Schema{Type: schema.TypeString},
},
},
}
}

func dataSourceVSphereVmfsDisksRead(d *schema.ResourceData, meta interface{}) error {
client := meta.(*govmomi.Client)
hsID := d.Get("host_system_id").(string)
ss, err := hostStorageSystemFromHostSystemID(client, hsID)
if err != nil {
return fmt.Errorf("error loading host storage system: %s", err)
}

if d.Get("rescan").(bool) {
ctx, cancel := context.WithTimeout(context.Background(), defaultAPITimeout)
defer cancel()
if err := ss.RescanAllHba(ctx); err != nil {
return err
}
}

var hss mo.HostStorageSystem
ctx, cancel := context.WithTimeout(context.Background(), defaultAPITimeout)
defer cancel()
if err := ss.Properties(ctx, ss.Reference(), nil, &hss); err != nil {
return fmt.Errorf("error querying storage system properties: %s", err)
}

d.SetId(time.Now().UTC().String())

var disks []string
for _, sl := range hss.StorageDeviceInfo.ScsiLun {
if hsd, ok := sl.(*types.HostScsiDisk); ok {
if matched, _ := regexp.MatchString(d.Get("filter").(string), hsd.CanonicalName); matched {
disks = append(disks, hsd.CanonicalName)
}
}
}

sort.Strings(disks)

if err := d.Set("disks", disks); err != nil {
return fmt.Errorf("error saving results to state: %s", err)
}

return nil
}
123 changes: 123 additions & 0 deletions vsphere/data_source_vsphere_vmfs_disks_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
package vsphere

import (
"fmt"
"os"
"testing"

"github.com/hashicorp/terraform/helper/resource"
)

func TestAccDataSourceVSphereVmfsDisks(t *testing.T) {
var tp *testing.T
testAccDataSourceVSphereVmfsDisksCases := []struct {
name string
testCase resource.TestCase
}{
{
"basic",
resource.TestCase{
PreCheck: func() {
testAccPreCheck(tp)
testAccDataSourceVSphereVmfsDisksPreCheck(tp)
},
Providers: testAccProviders,
Steps: []resource.TestStep{
{
Config: testAccDataSourceVSphereVmfsDisksConfig(),
Check: resource.ComposeTestCheckFunc(
resource.TestCheckOutput("found", "true"),
),
},
},
},
},
{
"with regular expression",
resource.TestCase{
PreCheck: func() {
testAccPreCheck(tp)
testAccDataSourceVSphereVmfsDisksPreCheck(tp)
},
Providers: testAccProviders,
Steps: []resource.TestStep{
{
Config: testAccDataSourceVSphereVmfsDisksConfigRegexp(),
Check: resource.ComposeTestCheckFunc(
resource.TestCheckOutput("expected_length", "true"),
),
},
},
},
},
}

for _, tc := range testAccDataSourceVSphereVmfsDisksCases {
t.Run(tc.name, func(t *testing.T) {
tp = t
resource.Test(t, tc.testCase)
})
}
}

func testAccDataSourceVSphereVmfsDisksPreCheck(t *testing.T) {
if os.Getenv("VSPHERE_ESXI_HOST") == "" {
t.Skip("set VSPHERE_ESXI_HOST to run vsphere_vmfs_disks acceptance tests")
}
if os.Getenv("VSPHERE_VMFS_EXPECTED") == "" {
t.Skip("set VSPHERE_VMFS_EXPECTED to run vsphere_vmfs_disks acceptance tests")
}
if os.Getenv("VSPHERE_VMFS_REGEXP") == "" {
t.Skip("set VSPHERE_VMFS_REGEXP to run vsphere_vmfs_disks acceptance tests")
}
}

func testAccDataSourceVSphereVmfsDisksConfig() string {
return fmt.Sprintf(`
data "vsphere_datacenter" "datacenter" {
name = "%s"
}
data "vsphere_host" "esxi_host" {
name = "%s"
datacenter_id = "${data.vsphere_datacenter.datacenter.id}"
}
data "vsphere_vmfs_disks" "available" {
host_system_id = "${data.vsphere_host.esxi_host.id}"
rescan = true
}
output "found" {
value = "${contains(data.vsphere_vmfs_disks.available.disks, "%s")}"
}
`, os.Getenv("VSPHERE_DATACENTER"), os.Getenv("VSPHERE_ESXI_HOST"), os.Getenv("VSPHERE_VMFS_EXPECTED"))
}

func testAccDataSourceVSphereVmfsDisksConfigRegexp() string {
return fmt.Sprintf(`
variable "regexp" {
type = "string"
default = "%s"
}
data "vsphere_datacenter" "datacenter" {
name = "%s"
}
data "vsphere_host" "esxi_host" {
name = "%s"
datacenter_id = "${data.vsphere_datacenter.datacenter.id}"
}
data "vsphere_vmfs_disks" "available" {
host_system_id = "${data.vsphere_host.esxi_host.id}"
rescan = true
filter = "${var.regexp}"
}
output "expected_length" {
value = "${length(data.vsphere_vmfs_disks.available.disks) == 2 ? "true" : "false" }"
}
`, os.Getenv("VSPHERE_VMFS_REGEXP"), os.Getenv("VSPHERE_DATACENTER"), os.Getenv("VSPHERE_ESXI_HOST"))
}
20 changes: 20 additions & 0 deletions vsphere/host_storage_system_helper.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
package vsphere

import (
"context"

"github.com/vmware/govmomi"
"github.com/vmware/govmomi/object"
)

// hostStorageSystemFromHostSystemID locates a HostStorageSystem from a
// specified HostSystem managed object ID.
func hostStorageSystemFromHostSystemID(client *govmomi.Client, hsID string) (*object.HostStorageSystem, error) {
hs, err := hostSystemFromID(client, hsID)
if err != nil {
return nil, err
}
ctx, cancel := context.WithTimeout(context.Background(), defaultAPITimeout)
defer cancel()
return hs.ConfigManager().StorageSystem(ctx)
}
1 change: 1 addition & 0 deletions vsphere/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ func Provider() terraform.ResourceProvider {
DataSourcesMap: map[string]*schema.Resource{
"vsphere_datacenter": dataSourceVSphereDatacenter(),
"vsphere_host": dataSourceVSphereHost(),
"vsphere_vmfs_disks": dataSourceVSphereVmfsDisks(),
},

ConfigureFunc: providerConfigure,
Expand Down
56 changes: 56 additions & 0 deletions website/docs/d/vmfs_disks.html.markdown
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
---
layout: "vsphere"
page_title: "VMware vSphere: vsphere_vmfs_disks"
sidebar_current: "docs-vsphere-data-source-vmfs-disks"
description: |-
A data source that can be used to discover storage devices that can be used for VMFS datastores.
---

# vsphere\_vmfs\_disks

The `vsphere_vmfs_disks` data source can be used to discover the storage
devices available on an ESXi host. This data source can be combined with the
[`vsphere_vmfs_datastore`][data-source-vmfs-datastore] resource to create VMFS
datastores based off a set of discovered disks.

[data-source-vmfs-datastore]: /docs/providers/vsphere/r/vmfs_datastore.html

## Example Usage

```hcl
data "vsphere_datacenter" "datacenter" {
name = "dc1"
}
data "vsphere_host" "host" {
name = "esxi1"
datacenter_id = "${data.vsphere_datacenter.datacenter.id}"
}
data "vsphere_vmfs_disks" "available" {
host_system_id = "${data.vsphere_host.host.id}"
rescan = true
filter = "mpx.vmhba1:C0:T[12]:L0"
}
```

## Argument Reference

The following arguments are supported:

* `host_system_id` - (String, required) The managed object ID of the host to
look for disks on.
* `rescan` - (Boolean, optional) Whether or not to rescan storage adapters
before searching for disks. This may lengthen the time it takes to perform
the search. Default: `false`.
* `filter` - (String, optional) A regular expression to filter the disks
against. Only disks with canonical names that match will be included.

~> **NOTE:** Using a `filter` is recommended if there is any chance the host
will have any specific storage devices added to it that may affect the order of
the output `disks` attribute below, which is lexicographically sorted.

## Attribute Reference

* `disks` - (List of strings) A lexicographically sorted list of devices
discovered by the operation, matching the supplied `filter`, if provided.
3 changes: 3 additions & 0 deletions website/vsphere.erb
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,9 @@
<li<%= sidebar_current("docs-vsphere-data-source-host") %>>
<a href="/docs/providers/vsphere/d/host.html">vsphere_host</a>
</li>
<li<%= sidebar_current("docs-vsphere-data-source-vmfs-disks") %>>
<a href="/docs/providers/vsphere/d/vmfs_disks.html">vsphere_vmfs_disks</a>
</li>
</ul>
</li>

Expand Down

0 comments on commit 966ba86

Please sign in to comment.