![Por que o tamanho do arquivo Get-VHD retorna 0 para VMs hospedadas no Server 2012R2 no Powershell Runspace Job?](https://rvso.com/image/718157/Por%20que%20o%20tamanho%20do%20arquivo%20Get-VHD%20retorna%200%20para%20VMs%20hospedadas%20no%20Server%202012R2%20no%20Powershell%20Runspace%20Job%3F.png)
Eu tenho esse script para fazer um CSV de VMs hospedadas em nossos clusters Hyper-V. Está funcionando muito bem e até consegui separar cada VM em um trabalho!
Nota lateral: eu não me importaria com conselhos sobre como fazer meu agendamento de maneira mais eficiente. O principal problema é que alguns clusters são WS 2016 e outros são WS 2012R2, que requerem um módulo Hyper-V PowerShell mais antigo. Não tenho certeza de como paralelizar módulos diferentes, então apenas obtenho a lista de VMs para cada nó do Hyper-V, paralelizo-os e altero o módulo se o próximo nó exigir.
Enfim, meu problema. Aqui está meu roteiro.
$numThreads = 4
$ScriptBlock = {
Param (
[string]$Cluster,
[string]$Node,
$VM
)
If ($ADComputer = (Get-ADComputer $VM.Name -Properties OperatingSystem)) {
$OS = $ADComputer.OperatingSystem
}
Else {
$OS = 'Unknown (not on domain)'
}
try {
Return [pscustomobject] @{
Cluster = $Cluster
Node = $Node
Name = $VM.Name
OS = $OS
State = $VM.State
Status = $VM.Status
Uptime = "{0:dd} days {0:hh} hours" -f $VM.Uptime
CPUUsage = $VM.CPUUsage
ProcessorCount = $VM.ProcessorCount
'MemoryDemand (GB)' = [math]::Round( $VM.MemoryDemand / 1GB, 2 )
'MemoryAssigned (GB)' = [math]::Round( $VM.MemoryAssigned / 1GB, 2 )
'VHD Size (GB)' = [math]::Round( ((Get-VHD -ComputerName $Node -VMId $VM.Id).FileSize | Measure-Object -Sum).Sum / 1GB, 2 )
Version = $VM.Version
'VLAN IDs' = ($VM | Select-Object -ExpandProperty NetworkAdapters | Select-Object -ExpandProperty VlanSetting).AccessVlanId -Join ", "
'IP Addresses' = ($VM | Select-Object -ExpandProperty NetworkAdapters).IPAddresses -Join ", "
}
}
catch {
Return [pscustomobject] @{
Cluster = $Cluster
Node = $Node
Name = $VM.Name
State = "Script Fail"
Status = $_.Exception.Message
}
}
}
$RunspacePool = [RunspaceFactory]::CreateRunspacePool(1, $numThreads)
$RunspacePool.Open()
$Jobs = @()
$VMInfoList = @()
ForEach ($Cluster in 'USHOMECLU0', 'USHOMECLU1') {
If ((Get-ADComputer $Cluster -Property OperatingSystem).OperatingSystem -Match '2016') {
Remove-Module Hyper-V
Import-Module Hyper-V -RequiredVersion 2.0.0.0
}
Else {
Remove-Module Hyper-V
Import-Module Hyper-V -RequiredVersion 1.1
}
ForEach ($Node in Get-ClusterNode -Cluster $Cluster) {
Get-VM -ComputerName $Node | ForEach-Object {
$Job = [powershell]::Create().AddScript($ScriptBlock).AddParameter("Cluster", $Cluster).AddParameter("Node", $Node).AddParameter("VM", $_)
$Job.RunspacePool = $RunspacePool
$Jobs += New-Object PSObject -Property @{
Job = $Job
Result = $Job.BeginInvoke()
}
}
}
}
# EndInvoke returns the objects from the background threads
ForEach ($Job in $Jobs) {
$VMInfoList += $Job.Job.EndInvoke($Job.Result)
}
$VMInfoList | Export-Csv -NoTypeInformation C:\PSReports_vms.csv
O problema que tenho é que os trabalhos para VMs nos nós do Windows Server 2012R2 sempre retornam 0 para isso:
'VHD Size' = ((Get-VHD -ComputerName $Node -VMId $VM.Id).FileSize | Measure-Object -Sum).Sum
No entanto, para nós do Windows Server 2016, todas as VMs retornam o valor apropriado. O que também é interessante é que se eu capturar a expressão exata que está sendo atribuída ao tamanho do VHD para as VMs hospedadas no Server 2012R2 (substituir as variáveis pela string direta) e executá-la fora do bloco de script do trabalho, obtenho o valor preciso.