SDC End to End Example
Overview
This guide will give the steps in order to install and being to use a PowerFlex SDC.
It will give examples of the following:
- Deploy the SDC Linux Virtual Machine (vSphere)
- Install the SDC
- Create and Attach a Volume to an SDCs
Step 1: VM Deployment
There are many way in which you can deploy a Virtual Machine. Below is an approch using the vSphere terraform provider. If your team use a different hypervisor, or has a differnt process for creating the VMs this step can just be skipped. The examples below show how to use the vSphere terraform provider but for more information you can check out there documentaion: https://registry.terraform.io/providers/hashicorp/vsphere/latest
1.1 Use the vSphere provider and Setup provider.tf
Provider Example:
Fill in username, password, vsphere_server ip/hostname
provider.tf
terraform {
required_providers {
vsphere = {
source = "hashicorp/vsphere"
version = "2.8.2"
}
}
}
provider "vsphere" {
user = var.vsphere_user
password = var.vsphere_password
vsphere_server = var.vsphere_server
allow_unverified_ssl = var.allow_unverified_ssl
api_timeout = 10
}
// Variables
variable "vsphere_user" {
type = string
description = "Stores the username of vsphere_user."
}
variable "vsphere_password" {
type = string
description = "Stores the password of vsphere_password."
}
variable "vsphere_server" {
type = string
description = "Stores the host ip/fqdn of the vsphere_server."
}
variable "allow_unverified_ssl" {
type = string
description = "Allow unverified ssl connection"
default = true
}
1.2 Gather Needed Data
Using vSphere Terraform Datasources for datacenter, datastore, resource_pool, host and network, we need to gather all of the information needed to create the virtual machine
Example:
datasources.tf
data "vsphere_datacenter" "datacenter" {
name = var.vsphere_datacenter_name
}
data "vsphere_datastore" "datastore" {
name = var.vsphere_datastore_name
datacenter_id = data.vsphere_datacenter.datacenter.id
}
data "vsphere_resource_pool" "pool" {
name = var.vsphere_resource_pool_name
datacenter_id = data.vsphere_datacenter.datacenter.id
}
data "vsphere_host" "host" {
name = var.vsphere_host_name
datacenter_id = data.vsphere_datacenter.datacenter.id
}
data "vsphere_network" "network" {
name = var.vsphere_network_name
datacenter_id = data.vsphere_datacenter.datacenter.id
}
// Variables
variable "vsphere_datacenter_name" {
type = string
description = "The name of the datacenter in which to deploy the virtual machines."
}
variable "vsphere_datastore_name" {
type = string
description = "The name of the vSphere datastore in which to deploy the virtual machines."
}
variable "vsphere_network_name" {
type = string
description = "The name of the vSphere datastore in which to deploy the virtual machines."
}
variable "vsphere_resource_pool_name" {
type = string
description = "The name of the vSphere resource pool in which to deploy the virtual machines."
}
variable "vsphere_host_name" {
type = string
description = "The name of the vSphere host in which to deploy the virtual machines."
}
1.3 Deploy the Virtual Machine
You can deploy one of two ways using a .ova image or using a vsphere template. Here is an examples of both.
Note: VM should have the following packages installed fio sshpass unzip yum-utils wget
.ova Example:
Note: the vapp properties will need to be filled out based on each usecase as it will be different in each enviorment
resource "vsphere_virtual_machine" "vm-installer" {
name = var.vm_name
datacenter_id = data.vsphere_datacenter.datacenter.id
datastore_id = data.vsphere_datastore.datastore.id
host_system_id = data.vsphere_host.host.id
resource_pool_id = data.vsphere_resource_pool.pool.id
num_cpus = var.num_cpus
memory = var.memory
wait_for_guest_net_timeout = 0
wait_for_guest_ip_timeout = 0
network_interface {
network_id = data.vsphere_network.network.id
adapter_type = var.adapter_type
}
ovf_deploy {
allow_unverified_ssl_cert = var.allow_unverified_ssl
remote_ovf_url = var.vm_ova_path
disk_provisioning = var.disk_provisioning
ip_protocol = var.ip_protocol
ip_allocation_policy = var.ip_allocation_policy
ovf_network_map = {
network_id = data.vsphere_network.network.id
}
}
// Depends on the image you need to fill out for your specific case/ova image you are using
vapp {
properties = {
"guestinfo.hostname" = "remote-foo.example.com",
"guestinfo.ipaddress" = "172.16.11.101",
"guestinfo.netmask" = "255.255.255.0",
"guestinfo.gateway" = "172.16.11.1",
"guestinfo.dns" = "172.16.11.4",
"guestinfo.domain" = "example.com",
"guestinfo.ntp" = "ntp.example.com",
"guestinfo.password" = "VMware1!",
"guestinfo.ssh" = "True"
}
}
}
// Variables
variable "vm_name" {
type = string
description = "Name of the PFMP installer VM, default to `new_terraform_vm`"
default = "new_terraform_sdc_vm"
}
variable "disk_provisioning" {
type = string
description = "The disk provisioning type for the virtual machine. Options (thin, flat, think, sameAsSource) defaults to `thin`"
default = "thin"
}
variable "ip_protocol" {
type = string
description = "The IP protocol for the virtual machine. Defaults to `IPV4`"
default = "IPV4"
}
variable "ip_allocation_policy" {
type = string
description = "The IP allocation policy for the virtual machine. Defaults to `STATIC_MANUAL`"
default = "STATIC_MANUAL"
}
variable "num_cpus" {
type = number
description = "Number of CPUs for the virtual machine. Defaults to `1`"
default = 1
}
variable "memory" {
type = number
description = "Memory for the virtual machine. Defaults to `4060`"
default = 4060
}
variable "adapter_type" {
type = string
description = "Network Adapter Type for the virtual machine. Defaults to `vmxnet3` Options can be found here: https://docs.vmware.com/en/VMware-vSphere/7.0/com.vmware.vsphere.vm_admin.doc/GUID-AF9E24A8-2CFA-447B-AC83-35D563119667.html"
default = "vmxnet3"
}
Template Example:
data "vsphere_virtual_machine" "template_example" {
name = "linux-template"
datacenter_id = data.vsphere_datacenter.datacenter.id
}
resource "vsphere_virtual_machine" "template" {
name = var.vm_name
resource_pool_id = data.vsphere_resource_pool.pool.id
datastore_id = data.vsphere_datastore.primary_datastore.id
num_cpus = var.num_cpus
memory = var.memory
folder = var.vmfolder
guest_id = data.vsphere_virtual_machine.template_example.guest_id
scsi_type = data.vsphere_virtual_machine.template_example.scsi_type
scsi_controller_count = 4
firmware = "efi"
network_interface {
network_id = data.vsphere_network.VM_Network.id
adapter_type = data.vsphere_virtual_machine.template_example.network_interface_types[0]
}
wait_for_guest_net_timeout = 10000000000000000
wait_for_guest_net_routable = false
wait_for_guest_ip_timeout = 10000000000000000
shutdown_wait_timeout = 10
migrate_wait_timeout = 10000000000000000
force_power_off = false
disk {
label = "OS"
size = 40
thin_provisioned = data.vsphere_virtual_machine.template_example.disks.0.thin_provisioned
unit_number = 0
}
clone {
template_uuid = data.vsphere_virtual_machine.template_example.id
customize {
linux_options {
host_name = var.hostname
domain = var.dns[0]
}
network_interface {
ipv4_address = var.mgmt_ip
ipv4_netmask = 22
}
ipv4_gateway = var.gw
dns_server_list = var.dns
dns_suffix_list = var.search_domains
}
}
// Variables
variable "vm_name" {
type = string
description = "Name of the PFMP installer VM, default to `new_terraform_vm`"
default = "new_terraform_sdc_vm"
}
variable "num_cpus" {
type = number
description = "Number of CPUs for the virtual machine. Defaults to `1`"
default = 1
}
variable "memory" {
type = number
description = "Memory for the virtual machine. Defaults to `4060`"
default = 4060
}
variable "search_domains" {
type = list(string)
description = "List of DNS search domains"
}
variable "dns" {
type = list(string)
}
variable "gw" {
type = string
}
variable "vmfolder" {
description = "Name of Folder to Deploy VM to"
}
variable "mgmt_ip" {
description = "IP on Management Network"
type = string
}
Step 2: SDC Install
We support installing SDC on Windows EXSi and Linux Hosts. In this example we will use the linux sdc terraform module to do the install on the linux VM you have created above. For more examples including EXSi and Windows look here.
2.1 Download the SDC Packages
Download the SDC Packages from Dell Support. Once downloaded add the sdc package to an http or ftp site for easy access when deploying sdcs
2.2 Install the SDC
Fill in username, password, and endpoint ip/hostname
provider.tf
terraform {
required_providers {
powerflex = {
version = ">=1.6.0"
source = "registry.terraform.io/dell/powerflex"
}
}
}
provider "powerflex" {
username = var.username
password = var.password
endpoint = var.endpoint
insecure = true
timeout = 120
}
// Variables
variable "username" {
type = string
description = "Stores the username of PowerFlex host."
}
variable "password" {
type = string
description = "Stores the password of PowerFlex host."
}
variable "endpoint" {
type = string
description = "Stores the endpoint of PowerFlex host. eg: https://10.1.1.1:443, here 443 is port where API requests are getting accepted"
}
Example Module:
module "modules_example_sdc_host_linux" {
source = "dell/modules/powerflex//examples/sdc_host_linux"
version = "1.2.0"
remote_host = {
user = var.remote_user
password = var.remote_password
// If you have a ssh key and certificate.
// You are able to use those instead of user/pass
//private_key = ""
//certificate = ""
}
ip = var.ip
versions = versions={
pflex = var.sdc_pflex_version //"4.5.3000.118"
kernel = var.linux_kernal_version //"5.15.0-1-generic"
}
scini = {
linux_distro = "RHEL9" #"Ubuntu"
autobuild_scini = true
// If autobuild is equal to false, give the path to your scini.ko or scini.rpm else this is not needed
//url = "http://example.com/release/5.15.0-1-generic"
}
sdc_pkg = {
url = var.package_url
// Do not change this value
remote_pkg_name = var.remote_pkg_name
remote_file = var.package_name
use_remote_path = true
skip_download_sdc = true
}
mdm_ips = var.mdm_ips
}
// Variables
variable "ip" {
type = string
description = "Stores the IP address of the remote Linux host."
}
variable "package_url" {
type = string
description = "URL to where the SDC Package is located for download. This package will be downloaded directly onto the virtual machine. Example: http://example.com/EMC-ScaleIO-sdc-3.6-700.103.Ubuntu.22.04.x86_64.tar, ftp://username:password@ftpserver/path/to/file"
}
variable "remote_pkg_name" {
type = string
description = "The name of the package. Example: emc-sdc-package.tar (for Ubuntu) emc-sdc-package.rpm (for RHEL)"
default = "emc-sdc-package.tar"
}
variable "package_name" {
type = string
description = "The name of the package. Example: EMC-ScaleIO-sdc-4.5-3000.118.Ubuntu.22.04.x86_64.tar (for Ubuntu) EMC-ScaleIO-sdc-4.5-3000.118.Ubuntu.22.04.x86_64.rpm (for RHEL)"
}
variable "remote_user" {
type = string
description = "Stores the username of the remote Linux host."
}
variable "remote_password" {
type = string
description = "Stores the password of the remote Linux host."
}
variable "sdc_pflex_version" {
type = string
description = "The version of the powerflex sdc you are installing, this can be grabbed from the sdc package name i.e: PowerFlex_4.5.2100.105_Ubuntu18.04_SDC.tar the version would be 4.5.2100.105"
}
variable "linux_kernal_version" {
type = string
description = "The version of linux we are using"
}
variable "mdm_ips" {
description = "all the mdms (either primary,secondary or virtual ips) in a comma separated list by cluster. If sdc is only being connected to one powerflex leave this unset and it will use the mdms of the cluster set of the provider by default"
type = list(string)
default = []
}
Step 3: Create and Attach Volume
3.1 Fill in the provider config
Fill in username, password, and endpoint ip/hostname
provider.tf
terraform {
required_providers {
powerflex = {
version = ">=1.6.0"
source = "registry.terraform.io/dell/powerflex"
}
}
}
provider "powerflex" {
username = var.username
password = var.password
endpoint = var.endpoint
insecure = true
timeout = 120
}
// Variables
variable "username" {
type = string
description = "Stores the username of PowerFlex host."
}
variable "password" {
type = string
description = "Stores the password of PowerFlex host."
}
variable "endpoint" {
type = string
description = "Stores the endpoint of PowerFlex host. eg: https://10.1.1.1:443, here 443 is port where API requests are getting accepted"
}
3.2 Create the Volume
resource "powerflex_volume" "volume-create" {
name = var.name
# To create / update, either protection_domain_id or protection_domain_name must be provided
protection_domain_name = var.protection_domain_name
# To create / update, either storage_pool_id or storage_pool_name must be provided
storage_pool_name = var.storage_pool_name
# The unit of size of the volume is defined by capacity_unit whose default value is "GB".
size = var.size
capacity_unit = var.capacity_unit # GB/TB
use_rm_cache = var.use_rm_cache # true/false
volume_type = var.volume_type # ThickProvisioned/ThinProvisioned volume type
access_mode = var.access_mode # ReadWrite/ReadOnly volume access mode
remove_mode = var.remove_mode # INCLUDING_DESCENDANTS/ONLY_ME remove mode
}
// Variables
variable "name" {
type = string
description = "name of the Powerflex Volume"
}
variable "protection_domain_name" {
type = string
description = "protection_domain name of the Powerflex Volume"
}
variable "storage_pool_name" {
type = string
description = "storage_pool name of the Powerflex Volume"
}
variable "size" {
type = number
description = "size of the Powerflex Volume"
default = 8
}
variable "capacity_unit" {
type = string
description = "Capacity Unit of the Powerflex Volume Options:(GB/TB)"
default = "GB"
}
variable "use_rm_cache" {
type = bool
description = "Sets the rm_cache option of the Powerflex Volume"
default = true
}
variable "volume_type" {
type = string
description = "Volume Type of the Powerflex Volume Options:(ThickProvisioned/ThinProvisioned)"
default = "ThickProvisioned"
}
variable "access_mode" {
type = string
description = "Access Mode of the Powerflex Volume Options:(ReadWrite/ReadOnly)"
default = "ReadWrite"
}
variable "remove_mode" {
type = string
description = "Remove Mode of the Powerflex Volume Options:(INCLUDING_DESCENDANTS/ONLY_ME)"
default = "INCLUDING_DESCENDANTS"
}
3.3 Attach the Volume to the SDC
Grab the SDC ID using the sdc datasource, then map the newly created volume to the SDC
data "powerflex_sdc" "filtered" {
filter {
name = [var.sdc_name]
}
}
resource "powerflex_sdc_volumes_mapping" "mapping-test" {
# SDC id
id = data.powerflex_sdc.filtered.sdcs[0].id
volume_list = [
{
# id of the volume which needs to be mapped.
# either volume_id or volume_name can be used.
volume_id = resource.powerflex_volume.volume-create.id
# Valid values are 0 or integers greater than 10
limit_iops = var.limit_iops
# Default value is 0
limit_bw_in_mbps = var.limit_bw_in_mbps
access_mode = var.access_mode # ReadOnly/ReadWrite/NoAccess
}
]
}
// Variables
variable "sdc_name" {
type = string
description = "name of the sdc"
}
variable "limit_iops" {
type = number
description = "limit_iops Valid values are 0 or integers greater than 10"
default = 140
}
variable "limit_bw_in_mbps" {
type = number
description = "limit_bw_in_mbps Default value is 0"
default = 0
}
variable "access_mode" {
type = string
description = "access_mode Options(ReadOnly/ReadWrite/NoAccess)"
default = "ReadOnly"
}