Categories
Sponsors
Archive
Blogroll
Badges
Community
|
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" |
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" |
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' |
$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" |
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 |
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} |
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."
}
}
} |
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"
} |
$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
} |
$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
}
}
} |
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:
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" |
Install-MicrosoftUpdate -ComputerName "MyHyperVHost01"
If you use this function in a script, then you can suppress write-host outputs:
Install-MicrosoftUpdate -ComputerName "MyHyperVHost01" -SuppressMode |
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
}
} |
##################################################
# 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
} |
##################################################
# 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."
}
} |
# 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"
}
}
}
} |
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."
}
}
} |
# 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.
|