PSWsusSpringClean.psm1
Function Invoke-WsusSpringClean { <# .SYNOPSIS Performs additional WSUS server clean-up beyond the capabilities of the built-in tools. .DESCRIPTION Adds the ability to decline numerous additional commonly unneeded updates as well as discover potentially incorrectly declined updates. .PARAMETER SynchroniseServer Perform a synchronisation against the upstream server before running cleanup. .PARAMETER DeclineCategoriesExclude Array of update categories in the bundled updates catalog to not decline. .PARAMETER DeclineCategoriesInclude Array of update categories in the bundled updates catalog to decline. .PARAMETER DeclineClusterUpdates Decline any updates which are exclusively for failover clustering installations. .PARAMETER DeclineFarmUpdates Decline any updates which are exclusively for farm deployment installations. .PARAMETER DeclineItaniumUpdates Decline any updates which are exclusively for Itanium architecture installations. .PARAMETER DeclineUnneededUpdates Decline any updates in the bundled updates catalog filtered against the provided inclusion or exclusion categories. .PARAMETER FindSuspectDeclines Scan all declined updates for any that may have been inadvertently declined. The returned suspect updates are those which: - Are not superseded or expired - Are not cluster, farm or Itanium updates (if set to decline) - Are not in the filtered list of updates to decline from the bundled catalog .PARAMETER CleanupObsoleteComputers Specifies that the cmdlet deletes obsolete computers from the database. .PARAMETER CleanupObsoleteUpdates Specifies that the cmdlet deletes obsolete updates from the database. .PARAMETER CleanupUnneededContentFiles Specifies that the cmdlet deletes unneeded update files. .PARAMETER CompressUpdates Specifies that the cmdlet deletes obsolete revisions to updates from the database. .PARAMETER DeclineExpiredUpdates Specifies that the cmdlet declines expired updates. .PARAMETER DeclineSupersededUpdates Specifies that the cmdlet declines superseded updates. .EXAMPLE PS C:\>Invoke-WsusSpringClean -DeclineClusterUpdates -DeclineFarmUpdates -DeclineItaniumUpdates Declines all failover clustering, farm server/deployment & Itanium updates. .EXAMPLE PS C:\>Invoke-WsusSpringClean -DeclineUnneededUpdates -DeclineCategoriesInclude @('Superseded', 'Pre-release') Declines all unneeded updates in the Superseded & Pre-release categories. .NOTES The script intentionally avoids usage of most WSUS cmdlets provided by the UpdateServices module as many are extremely slow. This is particularly true of the Get-WsusUpdate cmdlet. The efficiency of the update declining logic could be substantially improved. That said, this script is not typically run frequently (~monthly), so this isn't a major priority. .LINK https://github.com/ralish/PSWsusSpringClean #> [CmdletBinding(DefaultParameterSetName='Include',SupportsShouldProcess=$true)] Param( [Parameter(Mandatory=$false)] [Switch]$SynchroniseServer, [Parameter(ParameterSetName='Exclude',Mandatory=$false)] [AllowEmptyCollection()] [String[]]$DeclineCategoriesExclude=@(), [Parameter(ParameterSetName='Include',Mandatory=$false)] [AllowEmptyCollection()] [String[]]$DeclineCategoriesInclude=@(), [Parameter(Mandatory=$false)] [Switch]$DeclineClusterUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineFarmUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineItaniumUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineUnneededUpdates, [Parameter(Mandatory=$false)] [Switch]$FindSuspectDeclines, # Wrapping of Invoke-WsusServerCleanup [Parameter(Mandatory=$false)] [Switch]$CleanupObsoleteComputers, [Parameter(Mandatory=$false)] [Switch]$CleanupObsoleteUpdates, [Parameter(Mandatory=$false)] [Switch]$CleanupUnneededContentFiles, [Parameter(Mandatory=$false)] [Switch]$CompressUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineExpiredUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineSupersededUpdates ) # Ensure that any errors we receive are considered fatal $ErrorActionPreference = 'Stop' if ($SynchroniseServer) { Write-Host -ForegroundColor Green "`r`nStarting WSUS server synchronisation ..." Invoke-WsusServerSynchronisation } Write-Host -ForegroundColor Green "`r`nBeginning WSUS server cleanup (Phase 1) ..." Invoke-WsusServerCleanupWrapper -CleanupObsoleteUpdates:$CleanupObsoleteUpdates ` -CompressUpdates:$CompressUpdates ` -DeclineExpiredUpdates:$DeclineExpiredUpdates ` -DeclineSupersededUpdates:$DeclineSupersededUpdates Write-Host -ForegroundColor Green "`r`nBeginning WSUS server cleanup (Phase 2) ..." if ($PSCmdlet.ParameterSetName -eq 'Exclude') { $SuspectDeclines = Invoke-WsusServerExtraCleanup -DeclineCategoriesExclude $DeclineCategoriesExclude ` -DeclineClusterUpdates:$DeclineClusterUpdates ` -DeclineFarmUpdates:$DeclineFarmUpdates ` -DeclineItaniumUpdates:$DeclineItaniumUpdates ` -DeclineUnneededUpdates:$DeclineUnneededUpdates ` -FindSuspectDeclines:$FindSuspectDeclines } else { $SuspectDeclines = Invoke-WsusServerExtraCleanup -DeclineCategoriesInclude $DeclineCategoriesInclude ` -DeclineClusterUpdates:$DeclineClusterUpdates ` -DeclineFarmUpdates:$DeclineFarmUpdates ` -DeclineItaniumUpdates:$DeclineItaniumUpdates ` -DeclineUnneededUpdates:$DeclineUnneededUpdates ` -FindSuspectDeclines:$FindSuspectDeclines } Write-Host -ForegroundColor Green "`r`nBeginning WSUS server cleanup (Phase 3) ..." Invoke-WsusServerCleanupWrapper -CleanupObsoleteComputers:$CleanupObsoleteComputers ` -CleanupUnneededContentFiles:$CleanupUnneededContentFiles if ($FindSuspectDeclines) { return $SuspectDeclines } } Function ConvertTo-WsusSpringCleanCatalog { [CmdletBinding()] Param( [Parameter(Mandatory=$true)] [Microsoft.UpdateServices.Internal.BaseApi.Update[]]$Updates ) foreach ($Update in $Updates) { $ProductTitles = New-Object -TypeName String[] -ArgumentList $Update.ProductTitles.Count $Update.ProductTitles.CopyTo($ProductTitles, 0) [PSCustomObject]@{ Category = 'Unknown' Title = $Update.Title ProductTitles = [String]::Join(', ', $ProductTitles) } } } Function Invoke-WsusServerCleanupWrapper { [CmdletBinding(SupportsShouldProcess=$true)] Param( [Parameter(Mandatory=$false)] [Switch]$CleanupObsoleteComputers, [Parameter(Mandatory=$false)] [Switch]$CleanupObsoleteUpdates, [Parameter(Mandatory=$false)] [Switch]$CleanupUnneededContentFiles, [Parameter(Mandatory=$false)] [Switch]$CompressUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineExpiredUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineSupersededUpdates ) if ($CleanupObsoleteComputers) { Write-Host -ForegroundColor Green "[*] Deleting obsolete computers ..." Write-Host (Invoke-WsusServerCleanup -CleanupObsoleteComputers) } if ($CleanupObsoleteUpdates) { Write-Host -ForegroundColor Green "[*] Deleting obsolete updates ..." Write-Host (Invoke-WsusServerCleanup -CleanupObsoleteUpdates) } if ($CleanupUnneededContentFiles) { Write-Host -ForegroundColor Green "[*] Deleting unneeded update files ..." Write-Host (Invoke-WsusServerCleanup -CleanupUnneededContentFiles) } if ($CompressUpdates) { Write-Host -ForegroundColor Green "[*] Deleting obsolete update revisions ..." Write-Host (Invoke-WsusServerCleanup -CompressUpdates) } if ($DeclineExpiredUpdates) { Write-Host -ForegroundColor Green "[*] Declining expired updates ..." Write-Host (Invoke-WsusServerCleanup -DeclineExpiredUpdates) } if ($DeclineSupersededUpdates) { Write-Host -ForegroundColor Green "[*] Declining superseded updates ..." Write-Host (Invoke-WsusServerCleanup -DeclineSupersededUpdates) } } Function Invoke-WsusServerExtraCleanup { [CmdletBinding(SupportsShouldProcess=$true)] Param( [Parameter(ParameterSetName='Exclude',Mandatory=$true)] [AllowEmptyCollection()] [String[]]$DeclineCategoriesExclude, [Parameter(ParameterSetName='Include',Mandatory=$true)] [AllowEmptyCollection()] [String[]]$DeclineCategoriesInclude, [Parameter(Mandatory=$false)] [Switch]$DeclineClusterUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineFarmUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineItaniumUpdates, [Parameter(Mandatory=$false)] [Switch]$DeclineUnneededUpdates, [Parameter(Mandatory=$false)] [Switch]$FindSuspectDeclines ) $WsusServer = Get-WsusServer $UpdateScope = New-Object Microsoft.UpdateServices.Administration.UpdateScope Write-Host -ForegroundColor Green "[*] Importing update catalog ..." $Updates = Import-Csv (Join-Path $PSScriptRoot 'PSWsusSpringClean.csv') $Categories = $Updates.Category | Sort-Object | Get-Unique if ($PSCmdlet.ParameterSetName -eq 'Exclude') { $FilteredCategories = $Categories | ? { $_ -notin $DeclineCategoriesExclude } } else { $FilteredCategories = $Categories | ? { $_ -in $DeclineCategoriesInclude } } if ($DeclineItaniumUpdates -or $DeclineUnneededUpdates) { Write-Host -ForegroundColor Green "[*] Retrieving approved updates ..." $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::LatestRevisionApproved $WsusApproved = $WsusServer.GetUpdates($UpdateScope) Write-Host -ForegroundColor Green "[*] Retrieving unapproved updates ..." $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::NotApproved $WsusUnapproved = $WsusServer.GetUpdates($UpdateScope) $WsusAnyExceptDeclined = $WsusApproved + $WsusUnapproved } if ($DeclineClusterUpdates) { Write-Host -ForegroundColor Green "[*] Declining cluster updates ..." foreach ($Update in $WsusAnyExceptDeclined) { if ($Update.Title -match 'Failover Clustering') { if ($PSCmdlet.ShouldProcess($Update.Title, 'Decline')) { Write-Host -ForegroundColor Cyan ("[-] Declining update: " + $Update.Title) $Update.Decline() } } } } if ($DeclineFarmUpdates) { Write-Host -ForegroundColor Green "[*] Declining farm updates ..." foreach ($Update in $WsusAnyExceptDeclined) { if ($Update.Title -match '(Farm( |-deployment))') { if ($PSCmdlet.ShouldProcess($Update.Title, 'Decline')) { Write-Host -ForegroundColor Cyan ("[-] Declining update: " + $Update.Title) $Update.Decline() } } } } if ($DeclineItaniumUpdates) { Write-Host -ForegroundColor Green "[*] Declining Itanium updates ..." foreach ($Update in $WsusAnyExceptDeclined) { if ($Update.Title -match '(IA64|Itanium)') { if ($PSCmdlet.ShouldProcess($Update.Title, 'Decline')) { Write-Host -ForegroundColor Cyan ("[-] Declining update: " + $Update.Title) $Update.Decline() } } } } if ($DeclineUnneededUpdates) { foreach ($Category in $FilteredCategories) { Write-Host -ForegroundColor Green "[*] Declining updates in category: $Category" $UpdatesToDecline = $Updates | ? { $_.Category -eq $Category } foreach ($Update in $UpdatesToDecline) { $MatchingUpdates = $WsusAnyExceptDeclined | ? { $_.Title -eq $Update.Title } foreach ($Update in $MatchingUpdates) { if ($PSCmdlet.ShouldProcess($Update.Title, 'Decline')) { Write-Host -ForegroundColor Cyan ("[-] Declining update: " + $Update.Title) $Update.Decline() } } } } } if ($FindSuspectDeclines) { Write-Host -ForegroundColor Green "[*] Retrieving declined updates ..." $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::Declined $WsusDeclined = $WsusServer.GetUpdates($UpdateScope) $IgnoredDeclines = $Updates | ? { $_.Category -in $FilteredCategories } Write-Host -ForegroundColor Green "[*] Finding suspect declined updates ..." $SuspectUpdates = @() foreach ($Update in $WsusDeclined) { # Ignore superseded and expired updates if ($Update.IsSuperseded -or $Update.PublicationState -eq 'Expired') { continue } # Ignore cluster updates if they were declined if ($DeclineClusterUpdates -and $Update.Title -match 'Failover Clustering') { continue } # Ignore farm updates if they were declined if ($DeclineFarmUpdates -and $Update.Title -match '(Farm( |-deployment))') { continue } # Ignore Itanium updates if they were declined if ($DeclineItaniumUpdates -and $Update.Title -match '(IA64|Itanium)') { continue } # Ignore any update categories which were declined if ($Update.Title -in $IgnoredDeclines.Title) { continue } $SuspectUpdates += $Update } return $SuspectUpdates } } Function Invoke-WsusServerSynchronisation { [CmdletBinding(SupportsShouldProcess=$true)] Param() $WsusServer = Get-WsusServer if ($PSCmdlet.ShouldProcess($env:COMPUTERNAME, 'Server synchronization')) { $SyncStatus = $WsusServer.GetSubscription().GetSynchronizationStatus() if ($SyncStatus -eq 'NotProcessing') { $WsusServer.GetSubscription().StartSynchronization() } elseif ($SyncStatus -eq 'Running') { Write-Warning "[!] A synchronisation appears to already be running! We'll wait for this one to complete ..." } else { throw "WSUS server returned unknown synchronisation status: $SyncStatus" } do { #$WsusServer.GetSubscription().GetSynchronizationProgress() Start-Sleep -Seconds 5 } while ($WsusServer.GetSubscription().GetSynchronizationStatus() -eq 'Running') $SyncResult = $WsusServer.GetSubscription().GetLastSynchronizationInfo().Result if ($SyncResult -ne 'Succeeded') { throw "WSUS server synchronisation completed with unexpected result: $SyncResult" } } } Function Test-WsusSpringCleanCatalog { [CmdletBinding()] Param() $WsusServer = Get-WsusServer $UpdateScope = New-Object Microsoft.UpdateServices.Administration.UpdateScope Write-Host -ForegroundColor Green "[*] Importing update catalog ..." $Updates = Import-Csv (Join-Path $PSScriptRoot 'PSWsusSpringClean.csv') Write-Host -ForegroundColor Green "[*] Retrieving all updates ..." $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::Any $WsusUpdates = $WsusServer.GetUpdates($UpdateScope) Write-Host -ForegroundColor Green "[*] Scanning for updates only present in catalog ..." $CatalogOnly = @() foreach ($Update in $Updates) { if ($Update.Title -notin $WsusUpdates.Title) { $CatalogOnly += $Update } } return $CatalogOnly } |