UMN-Virt.psm1
#requires -Modules VMware.VimAutomation.Core function Remove-StuckSnapshot { <# .DESCRIPTION This script will remove a stuck snapshot from a VM or list of VMs .PARAMETER Name The name of the VM to remove the snapshot from .PARAMETER VM The VM object to remove the snapshot from .EXAMPLE Remove-StuckSnapshot -Name VM1 This will remove the stuck snapshot from VM1 .EXAMPLE Get-VM | Where-Object { $_.Extensiondata.Runtime.ConsolidationNeeded }| Remove-StuckSnapshot This will remove the stuck snapshot from all VMs #> [CmdletBinding(SupportsShouldProcess = $true)] param( [string[]]$Name, [Parameter(ValueFromPipeline)] [VMware.VimAutomation.ViCore.Types.V1.Inventory.VirtualMachine[]]$VM ) begin { if ($Name) { $VM = Get-VM $Name } } process { if (-not $VM) { Write-Error -ErrorAction Stop "No VMs found" } if ( -not $vm.ExtensionData.Runtime.ConsolidationNeeded) { Write-Warning "No stuck snapshots found, consolidation is not needed" return } foreach ($VMName in $VM) { Write-Warning "Removing stuck snapshot from $VMName" $snapshot = $VM | New-Snapshot -Name "StuckSnapshot removal" -Description "StuckSnapshot removal" $snapshot | Remove-Snapshot -RemoveChildren -Confirm:$false } } end {} } function Set-ClusterSSH { # enables or disables ssh on an entire cluster # can pipe cluster object in, or can pass the cluster name param( [Parameter (Mandatory = $true, ValueFromPipeline = $true)] [String]$ClusterName, [parameter(Mandatory = $true, ParameterSetName = 'enableSet')] [Switch]$Enable, [parameter(Mandatory = $true, ParameterSetName = 'disableSet')] [Switch]$Disable ) $cluster = Get-Cluster $ClusterName if ($cluster) { if ($Enable) { $cluster | Get-VMHost | Get-VMHostService | Where-Object { ($_.Key -eq 'TSM-SSH') } | Start-VMHostService -Confirm:$false } elseif ($Disable) { $cluster | Get-VMHost | Get-VMHostService | Where-Object { ($_.Key -eq 'TSM-SSH') -and ($_.running) } | Stop-VMHostService -Confirm:$false } } } function Get-VGPUVM { Get-VM | Select-Object name, @{n = 'cluster'; e = { $_.vmhost.parent } }, @{n = 'gpu'; e = { $_.ExtensionData.Config.Hardware.Device.Backing.vgpu } } | Where-Object { $_.gpu } } function Get-ActiveMigrations { # get active vmotions based on active tasks $runningTasks = Get-Task -Status Running | Where-Object { $_.Name -eq 'RelocateVM_Task' } $runningTasks | ForEach-Object -Process { $vm = Get-VM -Id $_.ObjectId $runningTasks | Select-Object @{N = 'Name'; E = { $vm.Name } }, @{N = 'Task'; E = { $_.Name } }, @{N = 'Progress'; E = { $_.PercentComplete } }, @{N = 'VMotionTime'; E = { Get-Date -Date $_.StartTime } }, @{N = 'VMotionMinutes'; E = { ((Get-Date) - (Get-Date -Date $_.startTime)).Minutes } } } } function Get-RunningSSH { Get-Cluster | Get-VMHost | Get-VMHostService | Where-Object { ($_.Key -eq 'TSM-SSH') -and ($_.running) } | Format-Table -Property vmhost, label, running } function Stop-AllSSH { Get-Cluster | Get-VMHost | Get-VMHostService | Where-Object { ($_.Key -eq 'TSM-SSH') -and ($_.running) } | Stop-VMHostService } function Get-AllSnapshotsInfo { Get-VM | Get-Snapshot | Select-Object VM, Name, Description, Created, SizeMB } function Get-AllSnapshots { Get-VM | Get-Snapshot } function Get-AssetInfo { param( [Parameter (ValueFromPipeline = $true)] [String]$ClusterName = '*' ) begin {} process { Get-VMHost -Location $ClusterName | Select-Object name, @{n = 'UMN_Asset_Tag'; e = { $_.CustomFields['UMNAssetTag'] } }, @{N = 'Service_Tag'; E = { ($_ | Get-View).Summary.Hardware.OtherIdentifyingInfo[3].IdentifierValue } } } end {} } function Get-ClusterCores { Get-Cluster | Select-Object Name,@{N="TotalSockets";E={($_.ExtensionData.Summary.NumHosts * 2)};},@{N="TotalCores";E={$_.ExtensionData.Summary.NumCpuCores}} } function Get-DeletedVMEvents { Get-VIEventPlus -Start ((Get-Date).adddays(-30)) -EventType 'VmRemovedEvent' | Select-Object @{Name = 'VMName'; Expression = { $_.vm.name } }, CreatedTime, UserName, fullFormattedMessage } function Get-VMByMac { param( [Parameter (Mandatory = $true)] [String]$MacAddress ) (Get-VM | Get-NetworkAdapter | Where-Object {$_.MacAddress -eq $MacAddress }).Parent } Function Get-VMByIP { param( [Parameter (Mandatory = $true)] [String]$IPAddress ) (Get-VM | Where-Object {$_.Guest.IPAddress -contains $IPAddress }) } function Get-OldHardwareVM { <# .DESCRIPTION This script will return a list of VMs with the oldest hardware version .PARAMETER Cluster The cluster/s to search for VMs [default: Hosting-01, XTR] .PARAMETER OldestCount The number of oldest hardware versions to return [default: 2] .PARAMETER Department Limit the search for VMs to a specific department .EXAMPLE Get-OldHardwareVM -Cluster Hosting-01 Department 'Virt' This will return a list of VMs with the 2 oldest hardware versions in the Hosting-01 cluster #> param( [string[]]$Cluster = @('Hosting-01', 'XTR'), [int]$OldestCount = 2, [string]$Department ) if ($Department){ $TagName = Get-Tag -Category 'Department' | Where-Object {$_.name -like $Department} $AllVMs = Get-Cluster $Cluster | Get-VM -Tag $TagName } else { $AllVMs = get-cluster $Cluster | Get-VM } $HardwareVersion = $AllVMs.HardwareVersion | Sort-Object -Unique | Sort-Object -Top $OldestCount $VMs = $AllVMs | Where-Object { $HardwareVersion -contains $_.HardwareVersion } | Sort-Object -Property Name $List = @() Write-Host "Gathering Information for $($VMs.Count) VMs with hardware version $($HardwareVersion -join ', ')" foreach ($VM in $VMs){ Write-Host -NoNewline "." $Cesi = Get-Cesi -VM $VM $Info = [PSCustomObject]@{ Name = $VM.Name HardwareVersion = $VM.HardwareVersion Unit = $Cesi.DepartmentId Contacts = $Cesi.Contacts WinRequestors = $Cesi.WinRequestors LinuxRequestors = $Cesi.LinuxRequestors SMERequestors = $Cesi.SMERequestors } $List += $Info } Write-Host "`nFound $($List.Count) VMs with hardware version $($HardwareVersion -join ', ')" return $List } function Invoke-RemediateCluster { <# .SYNOPSIS Performs parallel updates on a specified VMware cluster. .DESCRIPTION This script stages a specified number of non-compliant hosts in a VMware cluster into maintenance mode and optionally remediates them. It ensures that no more than 20% of the hosts are in maintenance mode at any given time unless the -Force parameter is used. .PARAMETER Cluster The target VMware cluster to update. This can be a cluster name (string) or a ClusterImpl object. .PARAMETER Limit The maximum number of hosts to stage into maintenance mode simultaneously. Default is 5. .PARAMETER Remediate Switch parameter to indicate if the cluster should be remediated after staging the hosts into maintenance mode. .PARAMETER Force Switch parameter to bypass the 20% maintenance mode limit check. .EXAMPLE .\parallel-updates.ps1 -Cluster "MyCluster" -Limit 3 -Remediate This command stages up to 3 non-compliant hosts in the "MyCluster" into maintenance mode and remediates them. .EXAMPLE .\parallel-updates.ps1 -Cluster "MyCluster" -Limit 3 -Remediate -Force This command stages up to 3 non-compliant hosts in the "MyCluster" into maintenance mode and remediates them, bypassing the 20% maintenance mode limit check. .NOTES Author: Justin Keppers Date: 3/2025 #> [CmdletBinding( SupportsShouldProcess )] param ( [Parameter(Mandatory = $true, ValueFromPipeline = $true)] $Cluster, [parameter()] [int]$Limit = 5, [parameter()] [switch]$Remediate, [parameter()] [switch]$Force ) process { $ErrorActionPreference = "Stop" if ($Cluster -is [string]) { $Cluster = Get-Cluster $Cluster } elseif ($cluster -is [VMware.VimAutomation.ViCore.Impl.V1.Inventory.ClusterImpl]) { # do nothing } else { throw "Invalid cluster type" } if ($Remediate) { # enable parallel remediation #$ParallelRemediationSetting = Initialize-SettingsDefaultsClustersPoliciesParallelRemediationAction -Enabled $true -MaxHosts $limit #$PolicySpec = Initialize-SettingsDefaultsClustersPoliciesApplyConfiguredPolicySpec -ParallelRemediationAction $ParallelRemediationSetting #Invoke-SetDefaultsClustersPoliciesApply -Cluster $cluster -SettingsClustersPoliciesApplyConfiguredPolicySpec $PolicySpec } # get hosts requiring updates $updates = $cluster | Test-LcmClusterCompliance $noncompliantHosts = $updates.NonCompliantHosts.vmhost | Sort-Object -Property Name # sanity check: no more than 20% of hosts can be in maintenance mode if (-not ($force)) { $totalHosts = $cluster | Get-VMHost $totalHostsCount = $totalHosts.Count $maintenanceHostsCount = ($totalHosts | Where-Object { $_.ConnectionState -eq 'Maintenance' }).Count $maintenanceHostsLimit = [math]::Ceiling($totalHostsCount * 0.2) if ($maintenanceHostsCount + $limit -gt $maintenanceHostsLimit) { throw "Cannot stage more than $maintenanceHostsLimit hosts (20%) into maintenance mode. If you're sure, use -Force" } } # pick 5 of them $remediateHosts = $noncompliantHosts | Select-Object -First $limit if ($remediateHosts) { # enter maintenance mode and disable alarms $alarmMgr = Get-View AlarmManager foreach ($vmhost in $remediateHosts) { if ($vmhost.ConnectionState -eq 'Connected') { Write-Host "$vmhost Entering Maintence mode" $vmhost | Set-VMHost -State Maintenance } elseif ($vmhost.ConnectionState -eq 'Maintenance') { Write-Host "$vmhost Already in Maintence mode" } Write-Host "$vmhost Disabling alarms" $alarmMgr.EnableAlarmActions($vmhost.ExtensionData.MoRef, $false) } if ($remediate) { #remediate cluster $cluster | Set-Cluster -Remediate -AcceptEULA # exit maintenance mode foreach ($vmhost in $remediateHosts) { if ($vmhost.ConnectionState -eq 'Maintenance') { Write-Host "$vmhost Exiting Maintence mode" $vmhost | Set-VMHost -State Connected } } # disable parallel remediation #$ParallelRemediationSetting = Initialize-SettingsDefaultsClustersPoliciesParallelRemediationAction -Enabled $false -MaxHosts $limit #$PolicySpec = Initialize-SettingsDefaultsClustersPoliciesApplyConfiguredPolicySpec -ParallelRemediationAction $ParallelRemediationSetting #Invoke-SetDefaultsClustersPoliciesApply -Cluster $cluster -SettingsClustersPoliciesApplyConfiguredPolicySpec $PolicySpec } else { Write-Host "Remediation not requested, $($remediateHosts.Count) hosts are staged in maintenance mode" } } else { Write-Host "Cluster $cluster is compliant" } } } |