![Warum gibt Get-VHD filesize für auf Server 2012R2 gehostete VMs im Powershell Runspace Job 0 zurück?](https://rvso.com/image/718157/Warum%20gibt%20Get-VHD%20filesize%20f%C3%BCr%20auf%20Server%202012R2%20gehostete%20VMs%20im%20Powershell%20Runspace%20Job%200%20zur%C3%BCck%3F.png)
Ich habe dieses Skript, um eine CSV-Datei der VMs zu erstellen, die auf unseren Hyper-V-Clustern gehostet werden. Es funktioniert großartig und ich habe es sogar geschafft, jede VM in einen Job aufzuteilen!
Randbemerkung: Ich hätte nichts gegen Ratschläge, wie ich meine Planung effizienter gestalten kann. Das Hauptproblem ist, dass einige Cluster WS 2016 und andere WS 2012R2 sind, die ein älteres Hyper-V PowerShell-Modul erfordern. Ich bin nicht sicher, wie ich verschiedene Module parallelisieren soll, also hole ich mir einfach die VM-Liste für jeden Hyper-V-Knoten, parallelisiere sie und ändere dann das Modul, wenn der nächste Knoten es erfordert.
Wie auch immer, mein Problem. Hier ist mein Skript.
$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
Mein Problem besteht darin, dass Jobs für VMs auf Windows Server 2012R2-Knoten hierfür immer 0 zurückgeben:
'VHD Size' = ((Get-VHD -ComputerName $Node -VMId $VM.Id).FileSize | Measure-Object -Sum).Sum
Für Windows Server 2016-Knoten geben jedoch alle VMs den entsprechenden Wert zurück. Interessant ist auch, dass ich den genauen Wert erhalte, wenn ich den genauen Ausdruck erfasse, der der VHD-Größe für diese auf Server 2012R2 gehosteten VMs zugewiesen wird (Variablen durch die direkte Zeichenfolge ersetzen) und ihn außerhalb des Skriptblocks für den Job ausführe.