[ VMware ] [ PowerCLI ] [ PowerShell ]
Contents
Overview
A storage migration is eventually going to get to a phase where the old datastores need to be unmounted and detached from hosts. I found a script showing how to do this via PowerCLI, but I wanted to expand on it to better serve my use case.
What I came up with is below and also on my github.
When running the script you must specify the vCenter name and the folder containing the datastores that will be unmounted. A regular expression is used to specify the names of the datastores you want to unmount. For example, “Datastore\d\d” would unmount datastores named DATASTORE01 and DATASTORE99, but not DMZ_STORAGE01 (the PowerShell -match operator is not case sensitive). Also, the script will only work on datastores that have been put into maintenance mode.
Prerequisites
- The VMware PowerCLI PowerShell module must be installed.
- You must be connected via
Connect-VIServer
to the vCenter containing the datastores you intend to unmount. - Datastores must be removed from datastore clusters.
- Datastores must be in maintenance mode.
- Datastores must be moved to the same datastore folder.
The Script
<#
.SYNOPSIS
A script to unmount and detach datastores from VMWare vCenter.
.DESCRIPTION
This script unmounts datastores from vCenter if they are in maintenance mode, reside in the specified folder, and the name matches the specified regular expression.
The VMware PowerCLI module is required and a connection to vCenter (Connect-VIServer) must be established before running the script.
.EXAMPLE
Unmount-Datastores.ps1 -VIServer vcenter.domain.local -NameRegex "Datastore\d\d" -Folder "Unmount"
This command will unmount and detach datastores in the vcenter.domain.local vCenter that reside in the datastore folder named Unmount with names that -match the regular expression "Datastore\d\d".
.NOTES
Author: Luke Arntz
Date: May 13, 2019
.Link
https://blue42.net
.Link
https://github.com/larntz/PowerCLI
#>
Param(
# Specifies a vCenter server name.
[Parameter(Position=0,mandatory=$true)]
[string]$VIServer,
# Specifies a datastore folder name.
[Parameter(Position=1,mandatory=$true)]
[string]$Folder,
# Specifies a regular expression to -match datastore names.
[Parameter(Position=2,mandatory=$true)]
[string]$NameRegex
)
$MaintenanceModeDatastores = Get-Datastore -Location (Get-Folder -Name $Folder) -Server $ViServer | where {$_.State -eq "Maintenance" -And $_.Name -match $NameRegex}
$UnMounted = @()
foreach ($MaintenanceModeDatastore in $MaintenanceModeDatastores)
{
$datastoreHostInfo = Get-VmHost -Server $viserver -Id ($MaintenanceModeDatastore.ExtensionData.Host.Key)
$CanonicalName = $MaintenanceModeDatastore.ExtensionData.Info.Vmfs.Extent[0].Diskname
Write-Host "Processing $($MaintenanceModeDatastore)..."
foreach ($vmHost in $datastoreHostInfo)
{
$unmount = [PSCustomObject]@{
Datastore = $MaintenanceModeDatastore.Name
DsUuid = $MaintenanceModeDatastore.ExtensionData.info.Vmfs.uuid
LunUuid = (Get-ScsiLun -Server $viserver -VmHost $vmHost | where {$_.CanonicalName -eq $CanonicalName}).ExtensionData.Uuid
Host = $vmHost.name
CanonicalName = $CanonicalName
UnmountTime = $null
}
$hostMountInfo = $MaintenanceModeDatastore.ExtensionData.Host | Where {$_.Key -eq $VMHost.Id}
if($hostMountInfo.MountInfo.Mounted -eq "True")
{
Write-Host "Unmounting $($unmount.Datastore) from host $($VMHost.Name)"
$hostStorageSystem = Get-View $VMHost.Extensiondata.ConfigManager.StorageSystem
$hostSTorageSystem.UnmountVmfsVolume($unmount.DsUuid)
$hostStorageSystem.DetachScsiLun($unmount.LunUuid)
} else {
write-host "$($unmount.Datastore) not mounted on $($VMhost.Name)"
}
$unmount.UnmountTime = Get-Date -Format "s"
$UnMounted += $unmount
}
}
$UnMounted