search
Categories
Sponsors
VirtualMetric Hyper-V Monitoring, Hyper-V Reporting
Archive
Blogroll

Badges
MCSE
Community

Cozumpark Bilisim Portali
Posted in Virtual Machine Manager, Windows Powershell, Windows Server | 1 Comment | 3,351 views | 20/07/2014 09:31

Lets assume that you have a 3 years old storage and you are using it to place your virtual machines on Hyper-V Cluster. Then you bought a new storage box, you created LUNs on it and assign them as a CSV volumes on your existing Hyper-V Cluster. So next achievement would be migrating your virtual machines from old storage to new one.

That migration could be a headache if you have many VMs and many storage LUNs. So in that case, you can use following script to migrate Virtual Machines per CSV Volume. Just you need to specify which CSV volume you would like to drain. So let’s assume that volume is Volume1. I will specify that volume like following:

Start-CSVMigration -TargetVolume "Volume1"

Now you need to specify other CSV Luns as well to filter them. Otherwise, you can migrate a VM to old storage again. So we are filtering them like following:

Start-CSVMigration -TargetVolume "Volume1" -FilteredVolumes "Volume2","Volume3","Volume4"

So how does it work?

We are creating an array called FilteredVolumes then adding our old storage luns to this array. You will see that I’m also adding $TargetVolume to same array. Because there is a chance to try to migrate a VM to same volume. So that will prevent that kind of issues.

So how can I get destination volume?

$FullQuery = '((Get-ClusterSharedVolume |' + $Query + ' Select -ExpandProperty SharedVolumeInfo | Select @{label="Name";expression={(($_.FriendlyVolumeName).Split("\"))[-1]}},@{label="FreeSpace";expression={($_ | Select -Expand Partition).FreeSpace}} | Sort FreeSpace -Descending)[0]).Name'

As you can notice from the code below, i’m getting all CSV volumes, filtering old storage volumes, sorting new storage CSVs by their free space and selecting the storage which has biggest available disk space.

So I’ll place VM into that volume:

Move-VMStorage -ComputerName "$ClusterNode" -VMName "$VMName" -DestinationStoragePath "C:\ClusterStorage\$Volume\$VMName"

You can also allow VMs with passthrough disks and vHBA. You just need to add -AllowPT switch.

Start-CSVMigration -TargetVolume "Volume1" -FilteredVolumes "Volume2","Volume3","Volume4" -AllowPT

This is what I use to filter VMs with vHBA:

Where {($_ | Select -expand FibreChannelHostBusAdapters) -eq $Null}

If you get the idea, now I can share full script:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
function Start-CSVMigration {
 
<#
    .SYNOPSIS
 
        Function to migrate all virtual machines placed on a CSV volume by using Hyper-V Live Storage Migration.
 
    .DESCRIPTION
 
        Lets assume that you have a 3 years old storage and you are using it to place your virtual machines on Hyper-V Cluster.
		Then you bought a new storage box, you created LUNs on it and assign them as a CSV volumes on your existing Hyper-V Cluster.
		So next achievement would be migrating your virtual machines from old storage to new one.
		This script moves virtual machines into different CSV volumes by using Hyper-V Live Storage Migration.
 
    .PARAMETER  WhatIf
 
        Display what would happen if you would run the function with given parameters.
 
    .PARAMETER  Confirm
 
        Prompts for confirmation for each operation. Allow user to specify Yes/No to all option to stop prompting.
 
    .EXAMPLE
 
        Start-CSVMigration -TargetVolume "Volume1"
 
    .EXAMPLE
 
        Start-CSVMigration -TargetVolume "Volume1" -FilteredVolumes "Volume2","Volume3","Volume4"
 
    .EXAMPLE
 
        Start-CSVMigration -TargetVolume "Volume1" -FilteredVolumes "Volume2","Volume3","Volume4" -AllowPT
 
    .INPUTS
 
        None
 
    .OUTPUTS
 
        None
 
    .NOTES
 
        Author: Yusuf Ozturk
        Website: http://www.yusufozturk.info
        Email: ysfozy@gmail.com
        Date created: 05-July-2014
        Last modified: 20-July-2014
        Version: 1.4
 
    .LINK
 
        http://www.yusufozturk.info
        http://twitter.com/yusufozturk
 
#>
 
[CmdletBinding(SupportsShouldProcess = $true)]
param (
 
	# Target Volume
	[Parameter(
		Mandatory = $true,
		HelpMessage = 'The CSV Volume name that you want to drain, like: Volume1')]
	$TargetVolume,
 
	# Filtered Volumes
	[Parameter(
		Mandatory = $false,
		HelpMessage = 'The CSV Volume names that you want to filter, like: Volume2')]
	[array]$FilteredVolumes,
 
	# Allow Passthrough Disks
	[Parameter(
		Mandatory = $false,
		HelpMessage = 'Allow Passthrough Disks')]
	[switch]$AllowPT = $false,
 
	# Debug Mode
	[Parameter(
		Mandatory = $false,
		HelpMessage = 'Debug Mode')]
	[switch]$DebugMode = $false
)
	# Enable Debug Mode
	if ($DebugMode)
	{
		$DebugPreference = "Continue"
	}
	else
	{
		$ErrorActionPreference = "silentlycontinue"
	}
 
	# Filtered Volumes
	$FilteredVolumes += $TargetVolume
 
	# Clear Query
	$Query = $Null;
 
	# Create Query
	foreach ($FilteredVolume in $FilteredVolumes)
	{
		$Query = $Query + ' Where {$_.SharedVolumeInfo.FriendlyVolumeName -notlike "*' + $FilteredVolume + '"} |'
	}
 
	# Full Query
	$FullQuery = '((Get-ClusterSharedVolume |' + $Query + ' Select -ExpandProperty SharedVolumeInfo | Select @{label="Name";expression={(($_.FriendlyVolumeName).Split("\"))[-1]}},@{label="FreeSpace";expression={($_ | Select -Expand Partition).FreeSpace}} | Sort FreeSpace -Descending)[0]).Name'
 
	# Get Cluster Nodes
	$ClusterNodes = Get-Cluster | Get-ClusterNode
 
	foreach ($ClusterNode in $ClusterNodes)
	{
		# Clear Variables
		$VMs = $Null;
 
		# Create VM Array
		$VMArray = @()
 
		if ($AllowPT)
		{
			# Get All Virtual Machines on Target Volume
			$VMs = Get-VM -ComputerName "$ClusterNode" | Where ConfigurationLocation -like "C:\ClusterStorage\$TargetVolume\*"
 
			foreach ($VM in $VMs)
			{
				# Get VM Information
				$VMName = $VM.Name
 
				# Add VM to VMArray
				$VMArray += $VMName
			}
		}
		else
		{
			# Get All Virtual Machines on Target Volume
			$VMs = Get-VM -ComputerName "$ClusterNode" | Where ConfigurationLocation -like "C:\ClusterStorage\$TargetVolume\*" | Where {($_ | Select -expand FibreChannelHostBusAdapters) -eq $Null}
 
			foreach ($VM in $VMs)
			{
				# Get VM Information
				$VMName = $VM.Name
 
				if ($VM.HardDrives.Path -like "Disk*")
				{
					# Skipping VM
				}
				else
				{
					# Add VM to VMArray
					$VMArray += $VMName
				}
			}
		}
 
		foreach ($VMName in $VMArray)
		{ 
			Write-Host " "
			Write-Host "Working on $VMName .."
			Write-Host "Hyper-V Host: $ClusterNode"
 
			# Get Volume Information
			$Volume = Invoke-Expression $FullQuery
 
			# Move Virtual Machine
			Move-VMStorage -ComputerName "$ClusterNode" -VMName "$VMName" -DestinationStoragePath "C:\ClusterStorage\$Volume\$VMName"
 
			Write-Host "Done."
		}
	}
}

It’s enough to run this script on one of the Cluster Node, so that will start migrating virtual machines. That will also give an output, so you will be able to see which VM you are migrating.

This is also an example how you can automate storage migrations:

1
2
3
4
5
6
7
8
9
10
11
$Volumes = @()
$Volumes += "Volume1"
$Volumes += "Volume2"
$Volumes += "Volume3"
$Volumes += "Volume4"
$Volumes += "Volume5"
 
foreach ($Volume in $Volumes)
{
	Start-CSVMigration -TargetVolume "$Volume" -FilteredVolumes "Volume1","Volume2","Volume3","Volume4","Volume5","Volume6","Volume7","Volume8","Volume9","Volume10","Volume11","Volume12","Volume13","Volume14","Volume15"
}

I hope you will find it useful. See you!


Posted in Virtual Machine Manager, Windows Powershell, Windows Server | No Comment | 2,044 views | 24/06/2014 09:42

You can update all SCVMM agents with following script:

1
2
3
4
5
6
7
$Credential = Get-SCRunAsAccount -Name "Domain_Account"
$HyperVHosts = Get-SCVMMManagedComputer
foreach ($HyperVHost in $HyperVHosts)
{
	$ManagedComputer = Get-SCVMMManagedComputer -ComputerName $HyperVHost
	Update-SCVMMManagedComputer -Credential $Credential -VMMManagedComputer $HyperVHost
}

That will update all Hyper-V hosts agents as soon as possible.


Posted in Virtual Machine Manager, Windows Powershell, Windows Server | No Comment | 2,612 views | 19/06/2014 02:26

You can install Microsoft Updates on your server by using following PowerShell script.

This script doesn’t reboot your server after Microsoft Update, even if patch requires reboot.
Instead, it gives you True/False result as an output, so you can reboot server by checking Update results.
So you can use this script to create your own Cluster Aware Update procedure.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
Function Install-MicrosoftUpdate {
 
<#
    .SYNOPSIS
 
        Function to install Microsoft Updates on Windows Servers.
 
    .DESCRIPTION
 
        If you have large number of Windows Servers, Microsoft Updates may not be an easy job.
		This script could start Microsoft Update installation on remote servers by using PowerShell.
 
    .PARAMETER  WhatIf
 
        Display what would happen if you would run the function with given parameters.
 
    .PARAMETER  Confirm
 
        Prompts for confirmation for each operation. Allow user to specify Yes/No to all option to stop prompting.
 
    .EXAMPLE
 
        Install-MicrosoftUpdate
 
    .EXAMPLE
 
        Install-MicrosoftUpdate -ComputerName Server01
 
    .EXAMPLE
 
        Install-MicrosoftUpdate -ComputerName Server01 -SuppressMode
 
    .INPUTS
 
        None
 
    .OUTPUTS
 
        None
 
    .NOTES
 
        Author: Yusuf Ozturk
        Website: http://www.yusufozturk.info
        Email: ysfozy@gmail.com
        Date created: 19-June-2014
        Last modified: 22-June-2014
        Version: 1.2
 
    .LINK
 
        http://www.yusufozturk.info
        http://twitter.com/yusufozturk
 
#>
 
	[CmdletBinding(SupportsShouldProcess = $true)]
	param (
 
		# ComputerName
		[Parameter(
			Mandatory = $false,
			HelpMessage = 'Computer Name')]
		$ComputerName = "localhost",
 
		# Suppress Mode
		[Parameter(
			Mandatory = $false,
			HelpMessage = 'Suppress Mode')]
		[switch]$SuppressMode = $false,		
 
		# Debug Mode
		[Parameter(
			Mandatory = $false,
			HelpMessage = 'Debug Mode')]
		[switch]$DebugMode = $false
	)
 
	# Enable Debug Mode
	if ($DebugMode)
	{
		# Set Error Action Preference
		$ErrorActionPreference = "Stop"
 
		# Set Debug Preference
		$DebugPreference = "Continue"
	}
	else
	{
		# Set Error Action Preference
		$ErrorActionPreference = "silentlycontinue"
	}
 
	if ($ComputerName -ne "localhost")
	{
		if (!$SuppressMode)
		{
			# Informational Output
			Write-Host "Starting remote patching on $ComputerName.." -ForegroundColor Cyan
			Write-Host "Please wait.." -ForegroundColor Gray
			Write-Host " "
		}
 
		# Invoke Script on Remote Computer
		$InvokeCommand = Invoke-Command -ComputerName $ComputerName -ArgumentList $SuppressMode, $DebugMode -ScriptBlock {
 
			# Define Parameters
			param ($SuppressMode, $DebugMode)
 
			# Set Execution Policy
			$SetExecutionPolicy = Set-ExecutionPolicy Unrestricted -Force -Confirm:$False
 
			# Default Reboot Status
			$RebootStatus = "False"
 
			# Create Microsoft Update Session
			$UpdateSession = New-Object -ComObject "Microsoft.Update.Session"
 
			# Search Microsoft Updates
			$UpdateSearcher = $UpdateSession.CreateUpdateSearcher()
			$SearchResult = $UpdateSearcher.Search("IsInstalled=0 and IsHidden=0")
 
			# Create Microsoft Update Install Queue
			$InstallQueue = New-Object -ComObject "Microsoft.Update.UpdateColl"
 
			for ($i = 0; $i -lt $SearchResult.Updates.Count; $i++) 
			{
				# Get Current Microsoft Update
				$Update = $SearchResult.Updates.Item($i)
 
				# Get Microsoft Update Description
				$UpdateDescription = $Update.Title
 
				# Microsoft Update KB Pattern
				$KBPattern = "KB\d{1,20}"
 
				# Search for Microsoft Update KB
				$RegexSearch = $UpdateDescription -match $KBPattern
 
				# Get Microsoft Update KB
				$UpdateKB = $Null;
				$UpdateKB = $Matches[0]
 
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "-------------------------------------------------"
					Write-Host " "
					Write-Host "Working on $UpdateKB.." -ForegroundColor Cyan
					Write-Host "Description: $UpdateDescription" -ForegroundColor Gray
					Write-Host " "
				}
 
				# Skip User Input
				if ($Update.InstallationBehavior.CanRequestUserInput) { Continue }
 
				# Accept Eula
				if ($Update.EulaAccepted -eq $False) { $Update.AcceptEula() }
 
				# Get Microsoft Update Status
				if ($Update.IsDownloaded) 
				{
					# Add to Install Queue
					$InstallQueue.Add($Update) | Out-Null
 
					# Reboot Behaviour
					if ($Update.InstallationBehavior.RebootBehavior -gt 0) 
					{ 
						# Set Reboot Status
						$RebootRequired = $True
 
						if (!$SuppressMode)
						{
							# Informational Output
							Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
							Write-Host "Warning: $UpdateKB requires reboot." -ForegroundColor Yellow
							Write-Host " "
							Write-Host " "
						}
					}
					else
					{
						if (!$SuppressMode)
						{
							# Informational Output
							Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
							Write-Host " "
							Write-Host " "
						}
					}
				}
				else
				{
					# Add to Install Queue
					$InstallQueue.Add($Update) | Out-Null
 
					if (!$SuppressMode)
					{
						# Informational Output
						Write-Host "Warning: $UpdateKB is not downloaded on server yet." -ForegroundColor Yellow
						Write-Host " "
						Write-Host " "
					}
 
					# Reboot Behaviour
					if ($Update.InstallationBehavior.RebootBehavior -gt 0) 
					{ 
						# Set Reboot Status
						$RebootRequired = $True
 
						if (!$SuppressMode)
						{
							# Informational Output
							Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
							Write-Host "Warning: $UpdateKB requires reboot." -ForegroundColor Yellow
							Write-Host " "
							Write-Host " "
						}
					}
					else
					{
						if (!$SuppressMode)
						{
							# Informational Output
							Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
							Write-Host " "
							Write-Host " "
						}
					}				
				}	
			}
 
			if ($InstallQueue.Count -eq 0) 
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Warning: There are no Microsoft Updates available" -ForegroundColor Yellow
				}
			}
			else
			{
				$MicrosoftUpdateScript = @'
Function Install-MicrosoftUpdate
{
	# Default Reboot Behaviour
	$RebootRequired = $False
 
	# Create Microsoft Update Session
	$UpdateSession = New-Object -ComObject "Microsoft.Update.Session"
 
	# Search Microsoft Updates
	$UpdateSearcher = $UpdateSession.CreateUpdateSearcher()
	$SearchResult = $UpdateSearcher.Search("IsInstalled=0 and IsHidden=0")
 
	# Create Microsoft Update Install Queue
	$InstallQueue = New-Object -ComObject "Microsoft.Update.UpdateColl"
 
	for ($i = 0; $i -lt $SearchResult.Updates.Count; $i++) 
	{
		# Get Current Microsoft Update
		$Update = $SearchResult.Updates.Item($i)
 
		# Get Microsoft Update Description
		$UpdateDescription = $Update.Title
 
		# Microsoft Update KB Pattern
		$KBPattern = "KB\d{1,20}"
 
		# Search for Microsoft Update KB
		$RegexSearch = $UpdateDescription -match $KBPattern
 
		# Get Microsoft Update KB
		$UpdateKB = $Null;
		$UpdateKB = $Matches[0]
 
		if (!$SuppressMode)
		{
			# Informational Output
			Write-Host "-------------------------------------------------"
			Write-Host " "
			Write-Host "Working on $UpdateKB.." -ForegroundColor Cyan
			Write-Host "Description: $UpdateDescription" -ForegroundColor Gray
			Write-Host " "
		}
 
		# Skip User Input
		if ($Update.InstallationBehavior.CanRequestUserInput) { Continue }
 
		# Accept Eula
		if ($Update.EulaAccepted -eq $False) { $Update.AcceptEula() }
 
		# Get Microsoft Update Status
		if ($Update.IsDownloaded) 
		{
			# Add to Install Queue
			$InstallQueue.Add($Update) | Out-Null
 
			# Reboot Behaviour
			if ($Update.InstallationBehavior.RebootBehavior -gt 0) 
			{ 
				# Set Reboot Status
				$RebootRequired = $True
 
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
					Write-Host "Warning: $UpdateKB requires reboot." -ForegroundColor Yellow
					Write-Host " "
					Write-Host " "
				}
			}
			else
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
					Write-Host " "
					Write-Host " "
				}
			}
		}
		else
		{
			# Add to Install Queue
			$InstallQueue.Add($Update) | Out-Null
 
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Warning: $UpdateKB is not downloaded on server yet." -ForegroundColor Yellow
				Write-Host " "
				Write-Host " "
			}
 
			# Reboot Behaviour
			if ($Update.InstallationBehavior.RebootBehavior -gt 0) 
			{ 
				# Set Reboot Status
				$RebootRequired = $True
 
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
					Write-Host "Warning: $UpdateKB requires reboot." -ForegroundColor Yellow
					Write-Host " "
					Write-Host " "
				}
			}
			else
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
					Write-Host " "
					Write-Host " "
				}
			}				
		}	
	}
 
	if ($InstallQueue.Count -eq 0) 
	{
		if (!$SuppressMode)
		{
			# Informational Output
			Write-Host "Warning: There are no Microsoft Updates available." -ForegroundColor Yellow
		}
	}
	else
	{
		# Download Microsoft Updates
		$Downloader = $UpdateSession.CreateUpdateDownloader()
		$Downloader.Updates = $InstallQueue
		$DownloadResults = $Downloader.Download()
 
		# Install Microsoft Updates
		$UpdateInstaller = $UpdateSession.CreateUpdateInstaller()
		$UpdateInstaller.Updates = $InstallQueue
		$Results = $UpdateInstaller.Install()
		$ResultCode = $Results.ResultCode
 
		if ($ResultCode -eq "0")
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Result: Microsoft Update is not started yet." -ForegroundColor Yellow
			}
 
			# Set Reboot Status
			$RebootStatus =	"False"
		}
		elseif ($ResultCode -eq "1")
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Result: Another installation is in progress." -ForegroundColor Yellow
			}
 
			# Set Reboot Status
			$RebootStatus =	"False"
		}
		elseif ($ResultCode -eq "2")
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Result: Update installation is succeeded." -ForegroundColor Green
			}
 
			# Set Reboot Status
			if ($RebootRequired -eq $True) { $RebootStatus = "True" } else { $RebootStatus = "False" }
		}
		elseif ($ResultCode -eq "3")
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Result: Update installation is succeeded with errors." -ForegroundColor Yellow
			}
 
			# Set Reboot Status
			if ($RebootRequired -eq $True) { $RebootStatus = "True" } else { $RebootStatus = "False" }				
		}
		elseif ($ResultCode -eq "4")
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Result: Update installation is failed." -ForegroundColor Red
			}
 
			# Set Reboot Status
			$RebootStatus =	"False"
		}
		elseif ($ResultCode -eq "5")
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Result: Update installation is aborted." -ForegroundColor Red
			}
 
			# Set Reboot Status
			$RebootStatus =	"False"
		}
		else
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Result: Unknown Error" -ForegroundColor Red
			}
 
			# Set Reboot Status
			$RebootStatus = "False"
		}
 
		# Get Date Information
		$Date = (Get-Date).ToUniversalTime()
		$DateUTC = Get-Date($Date) -Format 'yyyy-MM-dd HH:mm:ss'
 
		# Microsoft Update Results
		$LastSuccessTime = $DateUTC
		$LastError = $Results.HResult
 
		# Get Registry Path of Microsoft Update Results
		$RegPath = "Registry::HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\WindowsUpdate\Auto Update\Results\Install"
 
		if (Test-Path -Path $RegPath) 
		{
			# Get Registry Key
			$RegKey = Get-ItemProperty -Path $RegPath
 
			# Set Results to Registry
			if($RegKey -match 'LastError') { Set-ItemProperty -Path $RegPath -Name 'LastError' -Value $LastError }
			if($RegKey -match 'LastSuccessTime') { Set-ItemProperty -Path $RegPath -Name 'LastSuccessTime' -Value $LastSuccessTime }
		}
	}
 
	# Reboot Status Output
	$RebootStatusOutput = Set-Content -Value $RebootStatus -Path "C:\Program Files\ClusterAwareUpdating\RebootStatusOutput.txt"
	$RebootStatus
}
 
	# Install Microsoft Updates
	Install-MicrosoftUpdate
'@
				# Get Cluster Aware Update Path
				$ClusterAwareUpdatePath = "C:\Program Files\ClusterAwareUpdating"
 
				# Get Microsoft Update Path
				$MicrosoftUpdatePath = "C:\Program Files\ClusterAwareUpdating\MicrosoftUpdate.ps1"
 
				# Test Cluster Aware Update Path
				$TestClusterAwareUpdate = Test-Path -Path $ClusterAwareUpdatePath
 
				if ($TestClusterAwareUpdate)
				{
					# Set Microsoft Update Script
					Set-Content -Value $MicrosoftUpdateScript -Path $MicrosoftUpdatePath | Out-Null
				}
				else
				{
					# Create Cluster Aware Update Folder
					New-Item -Name ClusterAwareUpdating -Path "C:\Program Files" -Type Directory | Out-Null
 
					# Set Microsoft Update Script
					Set-Content -Value $MicrosoftUpdateScript -Path $MicrosoftUpdatePath | Out-Null
				}
 
				# Get Scheduled Task
				$ScheduledTask = Get-ScheduledTask | Where {$_.TaskName -eq "VirtualMetric-ClusterAwareUpdating"}
 
				# Get Scheduled Task Name
				$ScheduledTaskName = $ScheduledTask.TaskName
 
				# Get Scheduled Task Path
				$ScheduledTaskPath = $ScheduledTask.TaskPath
 
				if ($ScheduledTaskName -ne "VirtualMetric-ClusterAwareUpdating")
				{
					# Register Scheduled Job
					$RegisterJob = Register-ScheduledJob -FilePath $MicrosoftUpdatePath -Name "VirtualMetric-ClusterAwareUpdating"
 
					# Set Elevated Access
					$SetElevatedAccess = Get-ScheduledJob -Name "VirtualMetric-ClusterAwareUpdating" | Get-ScheduledJobOption | Set-ScheduledJobOption -RunElevated
 
					# Get Scheduled Task
					$ScheduledTask = Get-ScheduledTask -TaskName "VirtualMetric-ClusterAwareUpdating"
 
					# Get Scheduled Task Name
					$ScheduledTaskName = $ScheduledTask.Name
 
					# Get Scheduled Task Path
					$ScheduledTaskPath = $ScheduledTask.TaskPath
				}
 
				# Start Scheduled Job
				$StartScheduledJob = Start-ScheduledTask -TaskName "VirtualMetric-ClusterAwareUpdating" -TaskPath $ScheduledTaskPath | Out-Null
 
				# Set Scheduled Job Status
				$JobStatus = "Running"
 
				# Wait Until Scheduled Job Finished
				while ($JobStatus -eq "Running")
				{
					# Clear Job Status
					$JobStatus = "Ready"
 
					# Get Scheduled Job Status
					$JobStatus = (Get-ScheduledTask | Where {$_.TaskName -eq "VirtualMetric-ClusterAwareUpdating"}).State
 
					# Wait Until Next Check
					if ($JobStatus -eq "Running") { Start-Sleep -s 60 }
				}
 
				# Test Reboot Status Output
				$TestRebootStatusOutput = Test-Path -Path "C:\Program Files\ClusterAwareUpdating\RebootStatusOutput.txt"
 
				if ($TestRebootStatusOutput)
				{
					# Get Reboot Status
					$RebootStatus = Get-Content -Path "C:\Program Files\ClusterAwareUpdating\RebootStatusOutput.txt"
				}
				else
				{
					if (!$SuppressMode)
					{
						# Informational Output
						Write-Host "Error: Please check last result code of task scheduler" -ForegroundColor Red
					}
 
					# Set Reboot Status
					$RebootStatus = "False"
				}
			}
 
			# Output Reboot Status
			$RebootStatus
		}
 
		if ($SuppressMode)
		{
			# Output Reboot Status
			$InvokeCommand
		}
	}
	else
	{
		# Default Reboot Behaviour
		$RebootRequired = $False
 
		# Create Microsoft Update Session
		$UpdateSession = New-Object -ComObject "Microsoft.Update.Session"
 
		# Search Microsoft Updates
		$UpdateSearcher = $UpdateSession.CreateUpdateSearcher()
		$SearchResult = $UpdateSearcher.Search("IsInstalled=0 and IsHidden=0")
 
		# Create Microsoft Update Install Queue
		$InstallQueue = New-Object -ComObject "Microsoft.Update.UpdateColl"
 
		for ($i = 0; $i -lt $SearchResult.Updates.Count; $i++) 
		{
			# Get Current Microsoft Update
			$Update = $SearchResult.Updates.Item($i)
 
			# Get Microsoft Update Description
			$UpdateDescription = $Update.Title
 
			# Microsoft Update KB Pattern
			$KBPattern = "KB\d{1,20}"
 
			# Search for Microsoft Update KB
			$RegexSearch = $UpdateDescription -match $KBPattern
 
			# Get Microsoft Update KB
			$UpdateKB = $Null;
			$UpdateKB = $Matches[0]
 
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "-------------------------------------------------"
				Write-Host " "			
				Write-Host "Working on $UpdateKB.." -ForegroundColor Cyan
				Write-Host "Description: $UpdateDescription" -ForegroundColor Gray
				Write-Host " "
			}
 
			# Skip User Input
			if ($Update.InstallationBehavior.CanRequestUserInput) { Continue }
 
			# Accept Eula
			if ($Update.EulaAccepted -eq $False) { $Update.AcceptEula() }
 
			# Get Microsoft Update Status
			if ($Update.IsDownloaded) 
			{
				# Add to Install Queue
				$InstallQueue.Add($Update) | Out-Null
 
				# Reboot Behaviour
				if ($Update.InstallationBehavior.RebootBehavior -gt 0) 
				{ 
					# Set Reboot Status
					$RebootRequired = $True
 
					if (!$SuppressMode)
					{
						# Informational Output
						Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
						Write-Host "Warning: $UpdateKB requires reboot." -ForegroundColor Yellow
						Write-Host " "
						Write-Host " "
					}
				}
				else
				{
					if (!$SuppressMode)
					{
						# Informational Output
						Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
						Write-Host " "
						Write-Host " "
					}
				}
			}
			else
			{
				# Add to Install Queue
				$InstallQueue.Add($Update) | Out-Null
 
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Warning: $UpdateKB is not downloaded on server yet." -ForegroundColor Yellow
					Write-Host " "
					Write-Host " "
				}
 
				# Reboot Behaviour
				if ($Update.InstallationBehavior.RebootBehavior -gt 0) 
				{ 
					# Set Reboot Status
					$RebootRequired = $True
 
					if (!$SuppressMode)
					{
						# Informational Output
						Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
						Write-Host "Warning: $UpdateKB requires reboot." -ForegroundColor Yellow
						Write-Host " "
						Write-Host " "
					}
				}
				else
				{
					if (!$SuppressMode)
					{
						# Informational Output
						Write-Host "Adding $UpdateKB to Update Queue.." -ForegroundColor Green
						Write-Host " "
						Write-Host " "
					}
				}				
			}	
		}
 
		if ($InstallQueue.Count -eq 0) 
		{
			if (!$SuppressMode)
			{
				# Informational Output
				Write-Host "Warning: There are no Microsoft Updates available." -ForegroundColor Yellow
			}
		}
		else
		{
			# Download Microsoft Updates
			$Downloader = $UpdateSession.CreateUpdateDownloader()
			$Downloader.Updates = $InstallQueue
			$DownloadResults = $Downloader.Download()
 
			# Install Microsoft Updates
			$UpdateInstaller = $UpdateSession.CreateUpdateInstaller()
			$UpdateInstaller.Updates = $InstallQueue
			$Results = $UpdateInstaller.Install()
			$ResultCode = $Results.ResultCode
 
			if ($ResultCode -eq "0")
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Result: Microsoft Update is not started yet." -ForegroundColor Yellow
				}
 
				# Set Reboot Status
				$RebootStatus =	"False"
			}
			elseif ($ResultCode -eq "1")
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Result: Another installation is in progress." -ForegroundColor Yellow
				}
 
				# Set Reboot Status
				$RebootStatus =	"False"
			}
			elseif ($ResultCode -eq "2")
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Result: Update installation is succeeded." -ForegroundColor Green
				}
 
				# Set Reboot Status
				if ($RebootRequired -eq $True) { $RebootStatus = "True" } else { $RebootStatus = "False" }
			}
			elseif ($ResultCode -eq "3")
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Result: Update installation is succeeded with errors." -ForegroundColor Yellow
				}
 
				# Set Reboot Status
				if ($RebootRequired -eq $True) { $RebootStatus = "True" } else { $RebootStatus = "False" }				
			}
			elseif ($ResultCode -eq "4")
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Result: Update installation is failed." -ForegroundColor Red
				}
 
				# Set Reboot Status
				$RebootStatus =	"False"
			}
			elseif ($ResultCode -eq "5")
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Result: Update installation is aborted." -ForegroundColor Red
				}
 
				# Set Reboot Status
				$RebootStatus =	"False"
			}
			else
			{
				if (!$SuppressMode)
				{
					# Informational Output
					Write-Host "Result: Unknown Error" -ForegroundColor Red
				}
 
				# Set Reboot Status
				$RebootStatus = "False"
			}
 
			# Get Date Information
			$Date = (Get-Date).ToUniversalTime()
			$DateUTC = Get-Date($Date) -Format 'yyyy-MM-dd HH:mm:ss'
 
			# Microsoft Update Results
			$LastSuccessTime = $DateUTC
			$LastError = $Results.HResult
 
			# Get Registry Path of Microsoft Update Results
			$RegPath = "Registry::HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\WindowsUpdate\Auto Update\Results\Install"
 
			if (Test-Path -Path $RegPath) 
			{
				# Get Registry Key
				$RegKey = Get-ItemProperty -Path $RegPath
 
				# Set Results to Registry
				if($RegKey -match 'LastError') { Set-ItemProperty -Path $RegPath -Name 'LastError' -Value $LastError }
				if($RegKey -match 'LastSuccessTime') { Set-ItemProperty -Path $RegPath -Name 'LastSuccessTime' -Value $LastSuccessTime }
			}
		}
 
		if ($SuppressMode)
		{
			# Output Reboot Status
			$RebootStatus
		}
	}
}

So if you just type:

Install-MicrosoftUpdate

that will install Microsoft Updates on your local server.
If you want to install Microsoft Updates on remote server, then type like:

Install-MicrosoftUpdate -ComputerName "MyHyperVHost01"

If you use this function in a script, then you can suppress write-host outputs:

Install-MicrosoftUpdate -ComputerName "MyHyperVHost01" -SuppressMode

That will give you Reboot Status as a result.


Posted in Virtual Machine Manager, Windows Server | No Comment | 1,970 views | 09/06/2014 17:54

This is my notes and scripts for a Hyper-V Disaster Recovery with Storage Replication.

First script on Main Site, creates Volume identifiers under CSV Volumes. So after a Volume restore, we can restore Volume names with that index. Also scripts exports all virtual machine configs. That will make it easier to restore all Virtual Machines on Hyper-V.

On Main Site:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
##################################################
# START ON MAIN SITE
##################################################
 
# Clear VM Config
$TestVMConfigPath = Test-Path -Path "C:\ClusterStorage\Volume1\VMExport.Config"
 
if ($TestVMConfigPath)
{
	Clear-Content -Path "C:\ClusterStorage\Volume1\VMExport.Config"
}
 
# Export VM Config
$ClusterNodes = Get-Cluster | Get-ClusterNode
 
foreach ($ClusterNode in $ClusterNodes)
{
	# Get Virtual Machines
	$VMs = Get-VM -ComputerName $ClusterNode
 
	# Export VM Config
	foreach ($VM in $VMs)
	{
		# Get Virtual Machines
		$VMConfigPath = $VM.ConfigurationLocation + "\Virtual Machines\" + $VM.Id + ".xml"
 
		# Export to Config File
		Add-Content -Value $VMConfigPath -Path "C:\ClusterStorage\Volume1\VMExport.Config"
	}
}
 
# Read All CSV Directories
$CSVVolumes = Get-ChildItem "C:\ClusterStorage"
 
foreach ($CSVVolume in $CSVVolumes)
{
	# Get CSV Volume Name
	$CSVVolumeName = $CSVVolume.Name
 
	# Get CSV Identifier Name
	$CSVIdentifierName = $CSVVolumeName + ".CSV"
 
	# Get CSV Path
	$CSVPath = $CSVVolume.FullName
 
	# Get CSV Identifier Path
	$CSVIdentifierPath = $CSVPath + "\" + $CSVIdentifierName
 
	# Test CSV Identifier Path
	$TestCSVIdentifierPath = Test-Path $CSVIdentifierPath
 
	if (!$TestCSVIdentifierPath)
	{
		# Create CSV Identifier
		New-Item -Path "$CSVPath" -Name "$CSVIdentifierName" -ItemType File
	}
}

Second Script on DR site, clears all old objects on Hyper-V and Failover Cluster. Restores volume names according to Volume Identifiers. After that imports all virtual machines by using our virtual machine exports.

On DR Site:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
##################################################
# START ON DR SITE
##################################################
 
# Clear Failover Cluster Objects
$ClusterGroup = Get-ClusterGroup | Where GroupType -eq "VirtualMachine" | Remove-ClusterGroup -RemoveResources -Force
 
# Get All Cluster Nodes
$ClusterNodes = Get-Cluster | Get-ClusterNode
 
if ($ClusterNode in $ClusterNodes)
{
	Invoke-Command -ComputerName $ClusterNode -ScriptBlock {
 
		# Stop Hyper-V Management Service
		Set-Service "vmms" -StartupType Disabled
		Stop-Service "vmms"
 
		# Clear Hyper-V Database
		Get-ChildItem -Path "C:\ProgramData\Microsoft\Windows\Hyper-V\Virtual Machines" | Remove-Item
	}
}
 
# Read All CSV Directories
$CSVVolumes = Get-ChildItem "C:\ClusterStorage"
 
# Clear Iteration
$i = 1;
 
foreach ($CSVVolume in $CSVVolumes)
{	
	# Create Time Guid
	$TimeGuid = (Get-Date).ToString("ddMMyyyyhhmmss")
 
	# Change CSV Volume Name
	$CSVVolume | Rename-Item -NewName "CSVolume$TimeGuid$i"
 
	# Update Iteration
	$i++
}
 
# Read Updated CSV Directories
$CSVVolumes = Get-ChildItem "C:\ClusterStorage"
 
foreach ($CSVVolume in $CSVVolumes)
{
	# Get CSV Volume Name
	$CSVVolumeName = $CSVVolume.Name
 
	# Get CSV Volume Filter
	$CSVVolumeFilter = "*.CSV"
 
	# Get CSV Path
	$CSVPath = $CSVVolume.FullName
 
	# Find CSV Identifier
	$CSVIdentifier = $Null;
	$CSVIdentifier = (Get-ChildItem "$CSVPath" -Filter "$CSVVolumeFilter" -Recurse).Name
	$CSVIdentifier = $CSVIdentifier.Split(".")[0]
 
	if ($CSVIdentifier)
	{
		# CSV Volume ID
		$NewCSVVolumeName = "Volume" + $CSVIdentifier.Substring("0,6")
 
		# Change CSV Volume Name
		$CSVVolume | Rename-Item -NewName "$NewCSVVolumeName"
	}
	else
	{
		Write-Host "Error: Could not find CSV Identifier for $CSVVolume"
	}
}
 
# Get All Cluster Nodes
$ClusterNodes = Get-Cluster | Get-ClusterNode
 
if ($ClusterNode in $ClusterNodes)
{
	Invoke-Command -ComputerName $ClusterNode -ScriptBlock {
 
		# Start Hyper-V Management Service
		Set-Service "vmms" -StartupType Automatic
		Start-Service "vmms"
	}
}
 
# Get VM Import Config
$VMImportConfig = Get-Content "C:\ClusterStorage\Volume1\VMExport.Config"
 
foreach ($VM in $VMImportConfig)
{
	# Import Virtual Machine
	$ImportVM = Import-VM $VM
 
	# Get Virtual Machine Name
	$VMName = $ImportVM.Name
 
	# Add VM to Failover Cluster
	Add-VMToCluster $VMName
}

After that, you will be able to start your virtual machines.


Posted in Virtual Machine Manager, Windows Powershell, Windows Server | 3 Comments | 3,062 views | 09/06/2014 01:19

Sometimes, you may need to empty your CSV volume for a maintenance operation or due to a volume corruption.
So you can use this script if you need to drain all virtual machines and virtual disks from a CSV.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
# Target Volume
$TargetVolume = "Volume8"
 
# Get Cluster Nodes
$ClusterNodes = Get-Cluster | Get-ClusterNode
 
foreach ($ClusterNode in $ClusterNodes)
{
	# Clear Variables
	$VMs = $Null;
 
	# Get All Virtual Machines on Target Volume
	$VMs = Get-VM -ComputerName "$ClusterNode" | Where ConfigurationLocation -like "C:\ClusterStorage\$TargetVolume*"
 
	foreach ($VM in $VMs)
	{
		# Get VM Information
		$VMName = $VM.Name
 
		Write-Host " "
		Write-Host "Working on $VMName .."
		Write-Host "Hyper-V Host: $ClusterNode"
 
		# Get Volume Information
		$Volume = ((Get-ClusterSharedVolume | Where {$_.SharedVolumeInfo.FriendlyVolumeName -notlike "*$TargetVolume*"} | Select -ExpandProperty SharedVolumeInfo | Select @{label="Name";expression={(($_.FriendlyVolumeName).Split("\"))[-1]}},@{label="FreeSpace";expression={($_ | Select -Expand Partition).FreeSpace}} | Sort FreeSpace -Descending)[0]).Name
 
		# Move Virtual Machine
		Move-VMStorage -ComputerName "$ClusterNode" -VMName "$VMName" -DestinationStoragePath "C:\ClusterStorage\$Volume\$VMName"
 
		Write-Host "Done."
	}
}

That will move all virtual machines via Storage Live Migration into best available CSV volume.


Posted in Virtual Machine Manager, Windows Powershell, Windows Server | No Comment | 1,947 views | 08/06/2014 21:56

Your Hyper-V storage environment crashed and you restored it from a backup or volume snapshot or you want to restore all virtual machines to the Disaster Center via Storage Replication. You added new storage luns into Failover Cluster but CSV names are changed. In that case you may not be able to import Virtual Machines into Hyper-V quickly due to incorrect virtual disk paths.

This script will check all Virtual Machine configs and shows you which configuration is not correct.
Then you can fix it manually or you can fix it via this script.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
function Repair-VMConfig {
 
<#
    .SYNOPSIS
 
        Function to repair Virtual Machine config after CSV volume path changes.
        You should put a volume identifier into CSV volumes to be able to use this script.
	Simply put a Volume1.CSV into Volume1 as a volume identifier.
 
    .DESCRIPTION
 
	In some cases, CSV volume names could be changed. So you may not start your virtual machines due to incorrect virtual disk paths.
	This function is written to fix this issue. It finds all Virtual Machines configuration problems and shows you as a output.
	If you use ApplyChanges switch, it stops all Hyper-V Management Services in all Cluster Nodes, backups VM configs and fix incorrect VM virtual disk paths. Restarts Hyper-V Management Services after changes.
	You should put a volume identifier into CSV volumes to be able to use this script. Simply put a Volume1.CSV into Volume1 as a volume identifier.
 
    .PARAMETER  WhatIf
 
        Display what would happen if you would run the function with given parameters.
 
    .PARAMETER  Confirm
 
        Prompts for confirmation for each operation. Allow user to specify Yes/No to all option to stop prompting.
 
    .EXAMPLE
 
        Repair-VMConfig
	Just shows VM configuration problems and do not take any action unless you add -ApplyChanges switch.
 
    .EXAMPLE
 
        Repair-VMConfig -ApplyChanges
        Stops all Hyper-V Management Services in all Cluster Nodes, backups VM configs and fix incorrect VM virtual disk paths. Restarts Hyper-V Management Services after changes.
 
    .INPUTS
 
        ApplyChanges switch to apply changes on Virtual Machines.
 
    .OUTPUTS
 
        None
 
    .NOTES
 
        Author: Yusuf Ozturk
        Website: http://www.yusufozturk.info
        Email: ysfozy@gmail.com
        Date created: 07-June-2014
        Last modified: 08-June-2014
        Version: 1.2
 
    .LINK
 
        http://www.yusufozturk.info
	http://twitter.com/yusufozturk
 
#>
 
[CmdletBinding(SupportsShouldProcess = $true)]
param (
 
    # Do you confirm changes on Virtual Machines?
	[Parameter(
        Mandatory = $false,
        HelpMessage = 'Do you confirm changes on Virtual Machines?')]
    [switch]$ApplyChanges = $false,
 
	# Debug Mode
	[Parameter(
        Mandatory = $false,
        HelpMessage = 'Debug Mode')]
    [switch]$DebugMode = $false
)
 
	if ($ApplyChanges -eq $True)
	{
		# Get Cluster Node List
		$ClusterNodes = Get-Cluster | Get-ClusterNode
 
		foreach ($ClusterNode in $ClusterNodes)
		{
			# Stop Hyper-V Management Service
			Invoke-Command -ComputerName $ClusterNode -ScriptBlock {
				Set-Service "vmms" -StartupType Disabled
				Stop-Service "vmms"
			}
		}
	}
 
	# Read All CSV Directories
	$CSVVolumes = Get-ChildItem "C:\ClusterStorage"
 
	foreach ($CSVVolume in $CSVVolumes)
	{
		# Informational Output
		Write-Host " "
		Write-Host " "	
		Write-Host "Working on $CSVVolume .." -ForegroundColor Cyan
		Write-Host " "
 
		# CSV Volume Name
		$CSVVolumeName = $CSVVolume.Name
 
		# Read All VM Config Files
		$VMConfigs = Get-ChildItem "C:\ClusterStorage\$CSVVolumeName" -Filter "*.xml" -Recurse
 
		foreach ($VMConfig in $VMConfigs)
		{
			# Set Replace VM Config Identifier
			$ReplaceVMConfig = "0"
 
			# Get VM Config Full Path
			$VMConfigPath = $VMConfig.FullName
 
			if ($ApplyChanges -eq $True)
			{			
				# VM Config Backup Name
				$VMConfigBackupPath = $VMConfigPath + ".backup"
 
				# Backup Old VM Config
				$BackupVMConfig = Copy-Item -Path $VMConfigPath -Destination $VMConfigBackupPath
			}
 
			# Informational Output
			Write-Host " "
			Write-Host "Working on $VMConfigPath .." -ForegroundColor Yellow
 
			# CSV Volume Regex Pattern
			$Pattern = '<pathname type="string">(.*?)</pathname>'
 
			# Get VM Config Content
			$VMConfigContent = Get-Content "$VMConfigPath"
 
			# Get CSV Volume Name
			$RegexResults = $VMConfigContent -match $Pattern
 
			foreach ($RegexResult in $RegexResults)
			{			
				# Search for Virtual Disks
				$RegexSearch = [string]$RegexResult -match $Pattern
 
				# Get Virtual Disk Path
				$VHDPath = $Null;
				$VHDPath = $Matches[1]
 
				if ($VHDPath)
				{
					# Check VHD Path
					$TestVHDPath = $Null;
					$TestVHDPath = Test-Path "$VHDPath"
 
					if (!$TestVHDPath)
					{
						Write-Host "Error: $VHDPath" -ForegroundColor Red
 
						# CSV Volume Pattern
						$CSVPattern = "volume\d{1,2}"
 
						# Search for CSV Volume Name
						$RegexSearch = [string]$RegexResult -match $CSVPattern
 
						# Get CSV Volume Name
						$CSVName = $Null;
						$CSVName = $Matches[0]
 
						if ($CSVName -like "Volume*")
						{						
							# Get CSV Volume Filter
							$CSVVolumeFilter = "*" + $CSVName + ".CSV"
 
							# Find CSV Directory
							$GetCSVDirectory = $Null;
							$GetCSVDirectory = (Get-ChildItem "C:\ClusterStorage" -Filter "$CSVVolumeFilter" -Recurse).Directory.FullName
 
							if ($GetCSVDirectory -ne $Null)
							{
								# Get Config Content
								$ConfigContent = Get-Content -Path "$VMConfigPath"
 
								# Get New CSV Volume
								$NewCSVVolume = $GetCSVDirectory.Split("\")[-1]
 
								# Check CSV Volume
								if ($NewCSVVolume -like "Volume*")
								{
									$NewCSVVolume = "TEMPORARY_VOLUME" + $NewCSVVolume.Substring("0,6")
 
									if ($ApplyChanges -eq $True)
									{
										# Replace CSV Volume
										$Regex = New-Object Text.RegularExpressions.Regex $CSVName, "None"
										$NewConfigContent = $Regex.Replace($ConfigContent, $NewCSVVolume)
 
										# Save Changes on Config
										Set-Content -Value "$NewConfigContent" -Path "$VMConfigPath" -Encoding "BigEndianUnicode"
 
										# Set Replace VM Config Identifier
										$ReplaceVMConfig = "1"
									}
								}
								else
								{
									Write-Host "Error: Could not get volume name of $VHDPath" -ForegroundColor Red
								}
							}
							else
							{
								Write-Host "Error: Could not find any volume identifier for $CSVName" -ForegroundColor Red
							}
						}
						else
						{
							Write-Host "Error: $VHDPath is not on a CSV Volume" -ForegroundColor Red
						}
					}
					else
					{
						Write-Host "OK: $VHDPath" -ForegroundColor Green
					}
				}
			}
 
			if ($ApplyChanges -eq $True)
			{			
				if ($ReplaceVMConfig -eq "1")
				{
					# Get Config Content
					$ConfigContent = Get-Content -Path "$VMConfigPath"
 
					# CSV Volume Prefix
					$CSVVolumePrefix = "Volume"
 
					# Replace CSV Volume
					$Regex = New-Object Text.RegularExpressions.Regex "TEMPORARY_VOLUME", "None"
					$NewConfigContent = $Regex.Replace($ConfigContent, $CSVVolumePrefix)
 
					# Save Changes on Config
					Set-Content -Value "$NewConfigContent" -Path "$VMConfigPath" -Encoding "BigEndianUnicode"
				}
			}
		}
	}
 
	if ($ApplyChanges -eq $True)
	{
		foreach ($ClusterNode in $ClusterNodes)
		{
			# Start Hyper-V Management Service
			Invoke-Command -ComputerName $ClusterNode -ScriptBlock {
				Set-Service "vmms" -StartupType Automatic
				Start-Service "vmms"
			}
		}
	}
}

You should add -ApplyChanges switch in order to confirm changes on virtual machine configs. That will backup all VM configs before modification. Also that will stop and start all Hyper-V Management Services in all cluster nodes due to changes on VM configs.


Posted in Virtual Machine Manager, Windows Powershell, Windows Server | No Comment | 1,907 views | 06/06/2014 08:39

You can migrate selected Virtual Machines in a Hyper-V Cluster with following script.
That will migrate all virtual machines into target volume.
If target volume is not specified, it will look for best available CSV volume, and migrate virtual machine into that.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
# Target Volume
$TargetVolume = "Volume1"
 
# Create VM Array
$VMArray = New-Object System.Collections.ArrayList
$VMArray.Clear();
 
$AddArray = $VMArray.Add("VM01")
$AddArray = $VMArray.Add("VM02")
$AddArray = $VMArray.Add("VM03")
 
# Get Cluster Nodes
$ClusterNodes = Get-Cluster | Get-ClusterNode
 
foreach ($ClusterNode in $ClusterNodes)
{
	# Get All Virtual Machines
	$VMs = Get-VM -ComputerName "$ClusterNode"
 
	foreach ($VM in $VMs)
	{
		# Get VM Information
		$VMName = $VM.Name
 
		if ($VMArray.Contains($VMName) -eq $True)
		{ 
			Write-Host " "
			Write-Host "Working on $VMName .."
			Write-Host "Hyper-V Host: $ClusterNode"
 
			if (!$TargetVolume)
			{
				# Get Volume Information
				$Volume = ((Get-ClusterSharedVolume | Select -ExpandProperty SharedVolumeInfo | Select @{label="Name";expression={(($_.FriendlyVolumeName).Split("\"))[-1]}},@{label="FreeSpace";expression={($_ | Select -Expand Partition).FreeSpace}} | Sort FreeSpace -Descending)[0]).Name
			}
			else
			{
				$Volume = $TargetVolume
			}
 
			# Move Virtual Machine
			Move-VMStorage -ComputerName "$ClusterNode" -VMName "$VMName" -DestinationStoragePath "C:\ClusterStorage\$Volume\$VMName"
 
			Write-Host "Done."
		}
	}
}

This script should work even if you have pass-through disks on virtual machine or virtual hba.