More on the accessing the Hyper-V API from Powershell
... In which we find VMs, them choose one, start them, stop them , and connect to them.
I spent more of the last week than I planned looking at Hyper-V and Powershell, and I'm getting dangerously close to calling myself an expert.
There are two WMI objects which do most of the work "Msvm_ImageManagementService" and "Msvm_virtualSystemManagementService" and I've got posts to write about things you can do with both of them, and some of the tricks of Wbemtest as well.
I also found that in several places I needed the WMI object representing the VM, or the object representing its status. Cue two functions
function Get-VM
{Param ($machineName="%")
Get-WmiObject -Namespace "root\virtualization"
-Query "SELECT * FROM Msvm_ComputerSystem
WHERE ElementName like '$machineName' AND caption like 'Virtual%' "
}
#example 1: Get-VM
# Returns WMI Msvm_ComputerSystem objects for all Virtual Machines
# (n.b. Parent Partition is filtered out)
#Example 2: Get-VM "%2008%"
# Returns WMI Msvm_ComputerSystem objects for machines containing 2008 in their name
# (n.b.Wild card is % sign not *)
I re-wrote my function for displaying VM status as a format function and this little bit of code
function Format-VMStatus
{param ($VM)
if ($VM -eq $null) {$VM=$input}
$global:Counter=-1
$VM | Format-Table -autoSize -properties as in the old post}
Function Get-VMStatus
{Param ($machineName="%")
Get-VM $MachineName | Format-VmStatus
}
and Choosing a VM turned into
function Choose-VM
{$VMs = Get-VM
$VMs| Format-VmStatus | out-host
$VMs[ [int[]](Read-Host "Which one(s) ?").Split(",")]
}
I wanted to be able to start a VM (multiple VMs), stop it, pause it or connect to it using either it's name or the Msvm_ComputerSystem object so I wrote 4 little functions which will do just that. (Yes I should have 3 of them calling a single "change state" function !)
function Start-VM
{Param ($vm)
if ($vm -is [array]) {$vm | forEach-object {Start-VM $_ } }
if ($vm -is [string]) {$vm=(Get-VM $vm) }
if ($vm -is [System.Management.ManagementObject]) {$vm.requestStateChange(2) }
}
#Example Start-vm (choose-vm) - prompts the user to select one or more VMs and starts them
function Suspend-VM
{Param ($vm) if ($vm -is [array]) {$vm | forEach-object {Suspend-VM $_ } }
if ($vm -is [string]) {$vm=(Get-VM $vm) } if ($vm -is [System.Management.ManagementObject]) {$vm.requestStateChange(32769) }
}
function Stop-VM
{Param ($vm)
if ($vm -is [array]) {$vm | forEach-object {Stop-VM $_ } }
if ($vm -is [string]) {$vm=(Get-VM $vm) }
if ($vm -is [System.Management.ManagementObject]) {$vm.requestStateChange(3) }
}
function Get-VMConnectSession
{Param ($vm)
if ($vm -is [string]) {& 'C:\Program Files\Hyper-V\vmconnect.exe' $VSMgtSvc.SystemName $vm }
if ($vm -is [System.Management.ManagementObject]) {& 'C:\Program Files\Hyper-V\vmconnect.exe' $VSMgtSvc.SystemName $vm.elementName }
}
#Example: Get-VMconnectSession $tenby
# Launches a Terminal connection to the server pointed to by $tenby.
In the next post I'll look at things we can do with virtual disks using the "Msvm_ImageManagementService object
Technorati Tags: Microsoft,Windows Server 2008,Hyper-v,Virtulization,Powershell
Comments
Anonymous
January 01, 2003
The machine is running quite happily and has never been snapped. Late on Monday make a snapshot. ThisAnonymous
January 01, 2003
It is possible open the Hyper-V console for an individual VM without opening the full mmc.  YouAnonymous
January 01, 2003
In my last post I explained how snapshots work and gave a little bit of PowerShell for creating a oneAnonymous
January 01, 2003
In which we see how to set the number of CPUs I started with getting MSVM Computer System objects - whichAnonymous
January 01, 2003
Ibrahim, sorry I should have said that $vsMgtSvc = Get-wmiObject -nameSpace rootvirtualization -class Msvm_virtualSystemManagementService __server is the name of the machine where the WMI command executed and $VsMgtSvc.SystemName is the name of the hyper-V machine. In practice they will usually be the same.Anonymous
April 24, 2008
Nice post. I can’t find where you get $VSMgtSvc from... and it seems to work better if you change $VSMgtSvc.SystemName to $vm.__SERVER Cheers, Ibrahim