Scripts/Optimize-WsusContents.ps1

#Requires -Version 5.0

<#
 .Synopsis
   Cleanup wsus contents
 
 .Description
   The task schedule automatically deletes the old version.
 
 .Parameter ConfigPath
   Set ODT config path
   Create from 'Show-WsustainableSettingsView'
 
 .Parameter FistLaunch
   FistLaunch mode
 
 .Parameter Verbose
   Display verbose message
 
 .Example
   # Simply command sample
   Optimize-WsusContents -ConfigFileName 'C:\ProgramData\Wsustainable\0.1\Config.json'
 
#>


Function Optimize-WsusContents{
    Param(
        [Parameter(Mandatory)][String]$ConfigPath,
        [Switch]$FistLaunch
    )
    Function Global:Get-SqlCmdPath{
        $Path = (Get-ChildItem "$($env:ProgramFiles)\Microsoft SQL Server\Client SDK\ODBC\" -Recurse -Filter "sqlcmd.exe" -File | Sort-Object FullName -Descending | Select-Object -First 1).FullName
        If ($Path -eq $Null){
            $Path = $DefaultConfig.MaintenanceSql.SqlCmdPath
        }
        Return $Path
    }
    Function Global:Get-WsusSqlServerName{
        If (Test-Path "HKLM:\SOFTWARE\Microsoft\Update Services\Server\Setup"){
            $ServerInstancePath = (Get-Item -Path "HKLM:\SOFTWARE\Microsoft\Update Services\Server\Setup").GetValue("SqlServerName")
            If ($ServerInstancePath -eq "MICROSOFT##WID"){
                $ServerInstancePath = "\\.\pipe\Microsoft##WID\tsql\query"
            }
            Return $ServerInstancePath
        }
    }
    Function Deny-WsusFilteredUpdates($DeclineRule, $UpdateScope, $RetryCount, $Month, $FindMode){
        $Global:UpdatesResult = @()
        Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month $FindMode

        If (-not [String]::IsNullOrEmpty($DeclineRule.Mode)){
            Write-Verbose "[Deny-WsusFilteredUpdates] Mode: $($DeclineRule.Mode)"
            switch ($DeclineRule.Mode){
                "Wildcard" {
                    $Updates = @($UpdatesResult | Where-Object Title -like $DeclineRule.Filter)
                    Write-Verbose "$(Get-Date -Format F): [Wildcard] Updates.Count: $($Updates.Count) / UpdatesResult.Count: $($UpdatesResult.Count)"
                    Write-Verbose "$(Get-Date -Format F): [Wildcard] Filter: $($DeclineRule.Filter)"
                    $DeclineUpdateCount = 0
                    $ActionDetails = ""

                    $Updates | ForEach-Object{
                        $Update = $_
                        $Update.Decline() | Out-Null
                        $DeclineUpdateCount++
                        
                        If ($CurrentConfig.Log.IsLogging){
                            If ($CurrentConfig.Log.Verbose){
                                $Update | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                            Else{
                                $Update | Select-Object Title, LegacyName, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                        }
                    }
                    Write-Verbose "$(Get-Date -Format F): [Wildcard] Denied: $DeclineUpdateCount"
                }
                "Windows 11 FU Exclude Languages" {
                    $Updates = @($UpdatesResult | Where-Object {$_.Title -match $DeclineRule.Filter} | Select-Object @{Name="Item";Expression={$_}}, @{Name="Language";Expression={$_.Title -match $DeclineRule.Filter | Out-Null; Return $Matches["Language"]}} | Where-Object Language -ne $Null)
                    Write-Verbose "$(Get-Date -Format F): [Windows 11 FU Exclude Languages] Updates.Count: $($Updates.Count) / UpdatesResult.Count: $($UpdatesResult.Count)"
                    Write-Verbose "$(Get-Date -Format F): [Windows 11 FU Exclude Languages] Filter: $($DeclineRule.Filter)"
                    Write-Verbose "$(Get-Date -Format F): [Windows 11 FU Exclude Languages] Exclude Languages:$($CurrentConfig.ChooseProducts.'Windows 11'.ExcludeLanguages)"
                    $DeclineUpdateCount = 0
                    $ActionDetails = ""

                    $Languages = ($CurrentConfig.ChooseProducts.'Windows 11'.ExcludeLanguages -split ",")
                    $Updates | Where-Object Language -notin $Languages | ForEach-Object{
                        $Update = $_.Item
                        $ActionDetails = "$($Update.Language) -notin $($CurrentConfig.ChooseProducts.'Windows 11'.ExcludeLanguages)"
                        $Update.Decline() | Out-Null
                        $DeclineUpdateCount++
                        
                        If ($CurrentConfig.Log.IsLogging){
                            If ($CurrentConfig.Log.Verbose){
                                $Update | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                            Else{
                                $Update | Select-Object Title, LegacyName, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                    }
                    }
                    Write-Verbose "$(Get-Date -Format F): [Windows 11 FU Exclude Languages] Denied: $DeclineUpdateCount"
                }
                "Windows 10 FU Exclude Languages" {
                    $Updates = @($UpdatesResult | Where-Object {$_.Title -match $DeclineRule.Filter} | Select-Object @{Name="Item";Expression={$_}}, @{Name="Language";Expression={$_.Title -match $DeclineRule.Filter | Out-Null; Return $Matches["Language"]}} | Where-Object Language -ne $Null)
                    Write-Verbose "$(Get-Date -Format F): [Windows 10 FU Exclude Languages] Updates.Count: $($Updates.Count) / UpdatesResult.Count: $($UpdatesResult.Count)"
                    Write-Verbose "$(Get-Date -Format F): [Windows 10 FU Exclude Languages] Filter: $($DeclineRule.Filter)"
                    Write-Verbose "$(Get-Date -Format F): [Windows 10 FU Exclude Languages] Exclude Languages:$($CurrentConfig.ChooseProducts.'Windows 10'.ExcludeLanguages)"
                    $DeclineUpdateCount = 0
                    $ActionDetails = ""

                    $Languages = ($CurrentConfig.ChooseProducts.'Windows 10'.ExcludeLanguages -split ",")
                    $Updates | Where-Object Language -notin $Languages | ForEach-Object{
                        $Update = $_.Item
                        $ActionDetails = "$($Update.Language) -notin $($CurrentConfig.ChooseProducts.'Windows 10'.ExcludeLanguages)"
                        $Update.Decline() | Out-Null
                        $DeclineUpdateCount++
                        
                        If ($CurrentConfig.Log.IsLogging){
                            If ($CurrentConfig.Log.Verbose){
                                $Update | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                            Else{
                                $Update | Select-Object Title, LegacyName, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                        }
                    }
                    Write-Verbose "$(Get-Date -Format F): [Windows 10 FU Exclude Languages] Denied: $DeclineUpdateCount"
                }
                "Decline Old Version" {
                    $Updates = @($UpdatesResult | Where-Object {$_.Title -match $DeclineRule.Filter} | Select-Object @{Name="Item";Expression={$_}}, @{Name="Version";Expression={$_.Title -match $DeclineRule.Filter | Out-Null; Return $Matches["Version"]}} | Where-Object Version -ne $Null)
                    $LatestVersion = @($Updates | Sort-Object {[Version]$_.Version} -Descending)[0].Version
                    Write-Verbose "$(Get-Date -Format F): [Decline Old Version] Updates.Count: $($Updates.Count) / UpdatesResult.Count: $($UpdatesResult.Count)"
                    Write-Verbose "$(Get-Date -Format F): [Decline Old Version] Filter: $($DeclineRule.Filter)"
                    Write-Verbose "$(Get-Date -Format F): [Decline Old Version] LatestVersion: $LatestVersion"
                    $DeclineUpdateCount = 0
                    $ActionDetails = ""

                    $Updates | Where-Object Version -ne $LatestVersion | ForEach-Object{
                        $Update = $_.Item
                        $ActionDetails = "$($Update.Version) < $LatestVersion"
                        $Update.Decline() | Out-Null
                        $DeclineUpdateCount++
                        
                        If ($CurrentConfig.Log.IsLogging){
                            If ($CurrentConfig.Log.Verbose){
                                $Update | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                            Else{
                                $Update | Select-Object Title, LegacyName, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                        }
                    }
                    Write-Verbose "$(Get-Date -Format F): [Decline Old Version] Denied: $DeclineUpdateCount"
                }
                "Decline Superseded" {
                    $Updates = @($UpdatesResult | Where-Object {$_.Title -match $DeclineRule.Filter -and $_.IsSuperseded} | Select-Object @{Name="Item";Expression={$_}})
                    Write-Verbose "$(Get-Date -Format F): [Decline Superseded] Updates.Count: $($Updates.Count) / UpdatesResult.Count: $($UpdatesResult.Count)"
                    Write-Verbose "$(Get-Date -Format F): [Decline Superseded] Filter: $($DeclineRule.Filter)"
                    $DeclineUpdateCount = 0
                    $ActionDetails = ""

                    $Updates | ForEach-Object{
                        $Update = $_.Item
                        $Update.Decline() | Out-Null
                        $DeclineUpdateCount++
                        
                        If ($CurrentConfig.Log.IsLogging){
                            If ($CurrentConfig.Log.Verbose){
                                $Update | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                            Else{
                                $Update | Select-Object Title, LegacyName, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                            }
                        }
                    }
                    Write-Verbose "$(Get-Date -Format F): [Decline Superseded] Denied: $DeclineUpdateCount"
                }
                default {
                    Write-Warning "[Deny-WsusFilteredUpdates] Not supported Mode: $($DeclineRule.Mode)"
                }
            }
        }
    }
    Function Get-WsusFilteredUpdates($DeclineRule, $UpdateScope, $RetryCount, $Month, $FindMode){
        $DeclineUpdateCount = 0

        If ($RetryCount -gt $CurrentConfig.UpdatesFindMode.MaximumRetry){
            Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] skiped"
            Return
        }

        If (($CurrentConfig.UpdatesFindMode | Get-Member -Name 'ForceHalfModePerMonthLength').Count -eq 1){
            If ($Month -ne $Null){
                $UpdateScope.FromCreationDate = $Month
                $UpdateScope.ToCreationDate = $Month.AddMonths(1).AddSeconds(-1)
            }

            $UpdateCount = $WsusServer.GetUpdateCount($UpdateScope)

            If ($UpdateCount -eq 0){
                Switch ($FindMode){
                    ([UpdateFindMode]::H1){
                        Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Not found. $LogFileName updates $($Month.ToString("y")) (1/5)"
                    }
                    ([UpdateFindMode]::H2){
                        Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Not found. $LogFileName updates $($Month.ToString("y")) (2/5)"
                    }
                    ([UpdateFindMode]::H3){
                        Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Not found. $LogFileName updates $($Month.ToString("y")) (3/5)"
                    }
                    ([UpdateFindMode]::H4){
                        Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Not found. $LogFileName updates $($Month.ToString("y")) (4/5)"
                    }
                    ([UpdateFindMode]::H5){
                        Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Not found. $LogFileName updates $($Month.ToString("y")) (5/5)"
                    }
                    ([UpdateFindMode]::Full){
                        Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Not found. $LogFileName updates $($Month.ToString("y"))"
                    }
                    Default{
                        Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Not found. $LogFileName"
                    }
                }
                Return
            }
            ElseIf (($FindMode -eq $Null) -and ($UpdateCount -ge $CurrentConfig.UpdatesFindMode.ForceHalfModePerMonthLength)){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Switch to split mode. $LogFileName updates count: $UpdateCount"
                ForEach($Month in $Months){
                    Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month ([UpdateFindMode]::Full)
                }
                Return
            }
            ElseIf (($Month -ne $Null) -and ($FindMode -eq ([UpdateFindMode]::Full)) -and ($UpdateCount -ge $CurrentConfig.UpdatesFindMode.ForceHalfModePerMonthLength)){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] Switch to split mode. $LogFileName updates count: $UpdateCount date: $($Month.ToString("y"))"
                Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month ([UpdateFindMode]::H1)
                Return
            }
        }

        Switch ($FindMode){
            ([UpdateFindMode]::H1){
                $UpdateScope.FromCreationDate = $Month
                $UpdateScope.ToCreationDate = $Month.AddDays(8).AddSeconds(-1)
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (1/5) Found: $($WsusServer.GetUpdateCount($UpdateScope))"
            }
            ([UpdateFindMode]::H2){
                $UpdateScope.FromCreationDate = $Month.AddDays(7)
                $UpdateScope.ToCreationDate = $Month.AddDays(15).AddSeconds(-1)
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (2/5) Found: $($WsusServer.GetUpdateCount($UpdateScope))"
            }
            ([UpdateFindMode]::H3){
                $UpdateScope.FromCreationDate = $Month.AddDays(14)
                $UpdateScope.ToCreationDate = $Month.AddDays(22).AddSeconds(-1)
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (3/5) Found: $($WsusServer.GetUpdateCount($UpdateScope))"
            }
            ([UpdateFindMode]::H4){
                $UpdateScope.FromCreationDate = $Month.AddDays(21)
                $UpdateScope.ToCreationDate = $Month.AddDays(28).AddSeconds(-1)
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (4/5) Found: $($WsusServer.GetUpdateCount($UpdateScope))"
            }
            ([UpdateFindMode]::H5){
                $UpdateScope.FromCreationDate = $Month.AddDays(28)
                $UpdateScope.ToCreationDate = $Month.AddMonths(1).AddSeconds(-1)
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (5/5) Found: $($WsusServer.GetUpdateCount($UpdateScope))"
            }
            ([UpdateFindMode]::Full){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) Found: $UpdateCount"
            }
            Default{
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates Found: $UpdateCount"
            }
        }

        Try {
            $Updates = $WsusServer.GetUpdates($UpdateScope)

            If ([String]::IsNullOrEmpty($DeclineRule.Mode)){
                Write-Verbose "[Deny-WsusFilteredUpdates] Find TextIncludes (NotMode)"
                $Updates = $Updates | ForEach-Object {
                    $ActionDetails = ""
                    $Update = $_
                    $DeclineUpdateCount++
                    $Update.Decline()
                    If ($CurrentConfig.Log.IsLogging){
                        If ($CurrentConfig.Log.Verbose){
                            $Update | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                        }
                        Else{
                            $Update | Select-Object Title, LegacyName, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "$LogFileName.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                        }
                    }
                }
            }
            $Global:UpdatesResult += $Updates
        }
        Catch{
            Write-Warning "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updatesを取得できなかったため再試行します`n$($_.Exception)"

            $RetryCount++
            Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month $FindMode
        }

        Switch ($FindMode){
            ([UpdateFindMode]::H1){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (1/5) Target: $DeclineUpdateCount"
                Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month ([UpdateFindMode]::H2)
            }
            ([UpdateFindMode]::H2){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (2/5) Target: $DeclineUpdateCount"
                Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month ([UpdateFindMode]::H3)
            }
            ([UpdateFindMode]::H3){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (3/5) Target: $DeclineUpdateCount"
                Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month ([UpdateFindMode]::H4)
            }
            ([UpdateFindMode]::H4){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (4/5) Target: $DeclineUpdateCount"
                Get-WsusFilteredUpdates $DeclineRule $UpdateScope $RetryCount $Month ([UpdateFindMode]::H5)
            }
            ([UpdateFindMode]::H5){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) (5/5) Target: $DeclineUpdateCount"
            }
            ([UpdateFindMode]::Full){
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates $($Month.ToString("y")) Target: $DeclineUpdateCount"
            }
            Default{
                Write-Verbose "$(Get-Date -Format F): [Get-WsusFilteredUpdates] $LogFileName updates Target: $DeclineUpdateCount"
            }
        }
        Return

    }
    Function Start-WsusServerSynchronization($WsusServer, $TimeOut){
        If ($TimeOut -eq $Null){
            $TimeOut = (Get-Date).AddMinutes(120)
        }
        Switch ($WsusServer.GetSubscription().GetSynchronizationStatus()){
            ([Microsoft.UpdateServices.Administration.SynchronizationStatus]::NotProcessing){
                If (-not $RunningStartWsusServerSynchronization){
                    $Global:RunningStartWsusServerSynchronization = $True
                    Write-Verbose "$(Get-Date -Format F): [Start-WsusServerSynchronization] Starting now..."
                    $WsusServer.GetSubscription().StartSynchronization()
                    Start-Sleep -Seconds 10
                    Start-WsusServerSynchronization -WsusServer $WsusServer -TimeOut $TimeOut
                }
                Else{
                    Write-Verbose "$(Get-Date -Format F): [Start-WsusServerSynchronization] Not processing."
                }
            }
            ([Microsoft.UpdateServices.Administration.SynchronizationStatus]::Running){
                $Global:RunningStartWsusServerSynchronization = $True
                If ((Get-Date) -ge $TimeOut){
                    Write-Error "$(Get-Date -Format F): [Start-WsusServerSynchronization] Timed out."
                }
                Else{
                    Write-Verbose "$(Get-Date -Format F): [Start-WsusServerSynchronization] Running now... $($WsusServer.GetSubscription().GetSynchronizationProgress().ProcessedItems) / $($WsusServer.GetSubscription().GetSynchronizationProgress().TotalItems)"
                    Start-Sleep -Seconds 10
                    Start-WsusServerSynchronization -WsusServer $WsusServer -TimeOut $TimeOut
                }
            }
            ([Microsoft.UpdateServices.Administration.SynchronizationStatus]::Stopping){
                If ((Get-Date) -ge $TimeOut){
                    Write-Error "$(Get-Date -Format F): [Start-WsusServerSynchronization] Timed out."
                }
                Else{
                    Write-Verbose "$(Get-Date -Format F): [Start-WsusServerSynchronization] Stopping now..."
                    Start-Sleep -Seconds 10
                    Start-WsusServerSynchronization -WsusServer $WsusServer -TimeOut $TimeOut
                }
            }
        }
        $Global:RunningStartWsusServerSynchronization = $False
    }
    Function Stop-WsusServerSynchronization($WsusServer, $TimeOut){
        If ($TimeOut -eq $Null){
            $TimeOut = (Get-Date).AddMinutes(10)
        }
        Switch ($WsusServer.GetSubscription().GetSynchronizationStatus()){
            ([Microsoft.UpdateServices.Administration.SynchronizationStatus]::NotProcessing){
                Write-Verbose "$(Get-Date -Format F): [Stop-WsusServerSynchronization] Already stopped."
            }
            ([Microsoft.UpdateServices.Administration.SynchronizationStatus]::Running){
                Write-Verbose "$(Get-Date -Format F): [Stop-WsusServerSynchronization] Stopping now..."
                $WsusServer.GetSubscription().StopSynchronization()
                Start-Sleep -Seconds 5
                Stop-WsusServerSynchronization -WsusServer $WsusServer -TimeOut $TimeOut
            }
            ([Microsoft.UpdateServices.Administration.SynchronizationStatus]::Stopping){
                If ((Get-Date) -ge $TimeOut){
                    Write-Error "$(Get-Date -Format F): [Stop-WsusServerSynchronization] Timed out."
                }
                Else{
                    Write-Verbose "$(Get-Date -Format F): [Stop-WsusServerSynchronization] Stopping now..."
                    Start-Sleep -Seconds 10
                    Stop-WsusServerSynchronization -WsusServer $WsusServer -TimeOut $TimeOut
                }
            }
        }
    }

    # Load config
    If ([String]::IsNullOrWhiteSpace($ConfigPath)){
        $TestConfigPathResult = $False
    }
    Else{
        $TestConfigPathResult = (Test-Path $ConfigPath -PathType Leaf)
    }
    If ($TestConfigPathResult){
        $Global:CurrentConfig = Get-Content $ConfigPath -Encoding UTF8 | ConvertFrom-Json
    }Else{
        Write-Error ([System.IO.FileNotFoundException]::new("Not found config. Check access to $ConfigPath")) -ErrorAction Stop
    }

    # Logging
    . (Join-Path (Get-Module Wsustainable).ModuleBase "Scripts\LogManager.ps1")
    Initialize-Directories
    Start-Logging

    Write-Verbose "$(Get-Date -Format F): Start Optimize-WsusContents"

    # Initialize config
    $Global:CurrentConfig = Get-DeclineRules -Config $CurrentConfig
    If ($Verbose -or $CurrentConfig.Log.Verbose){
        $CurrentConfig | ConvertTo-Json -Depth 10 | Out-File -FilePath (Join-Path $LogDirectory "InternalConfig.json") -Encoding UTF8
    }
    If (@($CurrentConfig.DeclineRules).Count -eq 0){
        Write-Warning "Not found decline rules"
    }

    # Load assemblies
    Import-Module UpdateServices
    If (@(Get-Module –Name UpdateServices).Count -eq 0){
        Write-Error "このスクリプトの動作に必要な UpdateServices が見つかりませんでした"
    }

    # Connect to WSUS server
    $WsusServer = Get-WsusServer -Name $CurrentConfig.Wsus.Server -PortNumber $CurrentConfig.Wsus.Port
    If ($WsusServer -eq $Null){
        Write-Error "WSUS サーバーに接続できませんでした" -ErrorAction Stop
        Break
    }
    If ($CurrentConfig.Wsus.PreferredCulture -ne $Null){
        $WsusServer.PreferredCulture = $CurrentConfig.Wsus.PreferredCulture
    }
    If ($CurrentConfig.Wsus.InvokeWsusSynchronization){
        Start-WsusServerSynchronization -WsusServer $WsusServer
    }
    Stop-WsusServerSynchronization -WsusServer $WsusServer

    $ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::unknown
    If ($FistLaunch){
        @($CurrentConfig.FistLaunch.ApprovedStates) | ForEach-Object {
            $ApprovedStates = $ApprovedStates -bor [Microsoft.UpdateServices.Administration.ApprovedStates]::$_
        }
    }
    Else{
        @($CurrentConfig.UpdatesFindMode.ApprovedStates) | ForEach-Object {
            $ApprovedStates = $ApprovedStates -bor [Microsoft.UpdateServices.Administration.ApprovedStates]::$_
        }
    }
    Write-Verbose "$(Get-Date -Format F): ApprovedStates is $ApprovedStates"
    If ($ApprovedStates -eq [Microsoft.UpdateServices.Administration.ApprovedStates]::unknown){
        Write-Verbose "$(Get-Date -Format F): ApprovedStates is NotApproved"
        $ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::NotApproved
    }


    $CurrentDate = [datetime]$CurrentConfig.UpdatesFindMode.MinimumDate
    $Months = @()
    Do{
        $Months += $CurrentDate
        $CurrentDate = $CurrentDate.AddMonths(1)
    }
    While($CurrentDate -lt (Get-Date -Day 1))

    $UpdateClassification = $WsusServer.GetUpdateClassifications()

    @($CurrentConfig.DeclineRules) | ForEach-Object {
        $DeclineRule = $_
        Write-Verbose "`n[DeclineRule] Mode: $($_.Mode), Product: $($_.Product), Version: $($_.Version), Type: $($_.Type), Architecture: $($_.Architecture)`n"
        # generate logfilename
        $Global:LogFileName = "$($_.Product)-$($_.Version)-$($_.Architecture)-$($_.Type)-$($_.Mode)"
        [System.IO.Path]::GetInvalidFileNameChars() | ForEach-Object {$Global:LogFileName = $LogFileName.Replace("$_","")}

        $UpdateScope = New-Object Microsoft.UpdateServices.Administration.UpdateScope
        Try{
            $UpdateScope.ApprovedStates = $ApprovedStates

            @($DeclineRule.TargetProductId -split "`n") | ForEach-Object {
                $Category = $WsusServer.GetUpdateCategory($_)
                If ($Category -ne $Null){
                    $TargetProductTitle = $Category.Title
                    Write-Verbose "[DeclineRule] TargetProductTitle: $TargetProductTitle, TargetProductId: $_"
                    $UpdateScope.Categories.Add($Category) | Out-Null
                }Else{
                    Write-Warning "[DeclineRule] Not found TargetProductId: $_"
                }
            }
            @($DeclineRule.TargetClassifications -split "`n") | ForEach-Object {
                $UpdateClassification | Where-Object Id -eq $_ | ForEach-Object {
                    $UpdateScope.Classifications.Add($_) | Out-Null
                    Write-Verbose "[DeclineRule] TargetClassifications: $($_.Title), TargetProductId: $($_.Id)"
                }
            }
            Try{
                If (-not $_.Mode -and $CurrentConfig.ChooseProducts.($_.Product).FilterType -eq "Title"){
                    Write-Verbose "[DeclineRule] Set to TextIncludes: $($_.Filter)"
                    $UpdateScope.TextIncludes = $_.Filter
                }
                $Script:LatestDeclineUpdate = $Null
                Deny-WsusFilteredUpdates -DeclineRule $DeclineRule -UpdateScope $UpdateScope -RetryCount 0
            }
            Catch{
                Write-Error "$(Get-Date -Format F): $LogFileName `n$($_.Exception)"
            }
        }
        Catch{
            Write-Error "$(Get-Date -Format F): UpdateScope error $LogFileName `n$($_.Exception)"
        }
    }

    If ($CurrentConfig.DeclineOptions.CleanupWizard.CompressUpdates){
        #CleanupWizard: 不要な更新および更新のリビジョン
        Write-Verbose "[CleanupWizard] CompressUpdates: $($WsusServer | Invoke-WsusServerCleanup -CompressUpdates)"
    }
    If ($CurrentConfig.DeclineOptions.CleanupWizard.CleanupObsoleteUpdates){
        #CleanupWizard: 期限の切れた更新
        Write-Verbose "[CleanupWizard] CleanupObsoleteUpdates: $($WsusServer | Invoke-WsusServerCleanup -CleanupObsoleteUpdates)"
    }
    If ($CurrentConfig.DeclineOptions.CleanupWizard.CleanupObsoleteComputers){
        #CleanupWizard: サーバーにアクセスしていないコンピューター
        Write-Verbose "[CleanupWizard] CleanupObsoleteComputers: $($WsusServer | Invoke-WsusServerCleanup -CleanupObsoleteComputers)"
    }
    If ($CurrentConfig.DeclineOptions.CleanupWizard.CleanupUnneededContentFiles){
        #CleanupWizard: 不要な更新ファイル
        Write-Verbose "[CleanupWizard] CleanupUnneededContentFiles: $($WsusServer | Invoke-WsusServerCleanup -CleanupUnneededContentFiles)"
    }
    If ($CurrentConfig.DeclineOptions.CleanupWizard.DeclineExpiredUpdates){
        #CleanupWizard: 期限の切れた更新
        Write-Verbose "[CleanupWizard] DeclineExpiredUpdates: $($WsusServer | Invoke-WsusServerCleanup -DeclineExpiredUpdates)"
    }
    If ($CurrentConfig.DeclineOptions.CleanupWizard.DeclineSupersededUpdates){
        #CleanupWizard: 置き換えられた更新
        Write-Verbose "[CleanupWizard] DeclineSupersededUpdates: $($WsusServer | Invoke-WsusServerCleanup -DeclineSupersededUpdates)"
    }
    If ($CurrentConfig.DeclineOptions.ForceDeclineSupersededUpdates){
        #置き換えられた更新 (強制)
        $UpdateScope = New-Object Microsoft.UpdateServices.Administration.UpdateScope
        $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::NotApproved
        $ActionDetails = ""
        $WsusServer.GetUpdates($UpdateScope) | Where-Object IsSuperseded | ForEach-Object {
            Try{
                $_.Decline()
                If ($CurrentConfig.Log.IsLogging){
                    If ($CurrentConfig.Log.Verbose){
                        $Update | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "ForceDeclineSupersededUpdates.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                    }
                    Else{
                        If ($IsMatch){
                            $Update | Select-Object Title, LegacyName, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "DeclineSupersededUpdates.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
                        }
                    }
                }
            }
            Catch{
                Write-Error "$(Get-Date -Format F): 更新プログラムを取得できませんでした`n$($_.Exception)"
            }
        }
    }

    If ($CurrentConfig.MaintenanceSql.SqlCmdMode -eq "psmodule"){
        If ($CurrentConfig.MaintenanceSql.UpdateStatisticsAndDbccDbReIndex){
            Write-Verbose "[psmodule] UpdateStatisticsAndDbccDbReIndex: $(Invoke-Sqlcmd -ServerInstance (Get-WsusSqlServerName) -InputFile (Join-Path $PSScriptRoot '..\Assets\UpdateStatisticsAndDbccDbReIndex.sql') -Encrypt Optional)"
        }
        If ($CurrentConfig.MaintenanceSql.WsusDBMaintenance){
            Write-Verbose "[psmodule] WsusDBMaintenance: $(Invoke-Sqlcmd -ServerInstance (Get-WsusSqlServerName) -InputFile (Join-Path $PSScriptRoot '..\Assets\WsusDBMaintenance.sql') -Encrypt Optional)"
        }
    }
    Else{
        If ($CurrentConfig.MaintenanceSql.UpdateStatisticsAndDbccDbReIndex){
            Write-Verbose "[$($CurrentConfig.MaintenanceSql.SqlCmdMode)] UpdateStatisticsAndDbccDbReIndex: $((Start-Process (Get-SqlCmdPath) -ArgumentList "-S $(Get-WsusSqlServerName) -i ""$(Join-Path $PSScriptRoot '..\Assets\UpdateStatisticsAndDbccDbReIndex.sql')"" -Wait -NoNewWindow -PassThru").ExitCode)"
        }
        If ($CurrentConfig.MaintenanceSql.WsusDBMaintenance){
            Write-Verbose "[$($CurrentConfig.MaintenanceSql.SqlCmdMode)] WsusDBMaintenance: $((Start-Process (Get-SqlCmdPath) -ArgumentList "-S $(Get-WsusSqlServerName) -i ""$(Join-Path $PSScriptRoot '..\Assets\WsusDBMaintenance.sql')"" -Wait -NoNewWindow -PassThru").ExitCode)"
        }
    }

    If ($CurrentConfig.Log.IsLogging){
        If ($CurrentConfig.Log.Verbose){
            $UpdateScope = New-Object Microsoft.UpdateServices.Administration.UpdateScope
            $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::NotApproved
            $ActionDetails = ""
            Try{
                $WsusServer.GetUpdates($UpdateScope) | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "NotApproved.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
            }
            Catch{
                Write-Error "$(Get-Date -Format F): [NotApproved] 更新プログラムを取得できませんでした`n$($_.Exception)"
            }

            $UpdateScope = New-Object Microsoft.UpdateServices.Administration.UpdateScope
            $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::HasStaleUpdateApprovals
            $ActionDetails = ""
            Try{
                $WsusServer.GetUpdates($UpdateScope) | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "HasStaleUpdateApprovals.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
            }
            Catch{
                Write-Error "$(Get-Date -Format F): [HasStaleUpdateApprovals] 更新プログラムを取得できませんでした`n$($_.Exception)"
            }

            $UpdateScope = New-Object Microsoft.UpdateServices.Administration.UpdateScope
            $UpdateScope.ApprovedStates = [Microsoft.UpdateServices.Administration.ApprovedStates]::LatestRevisionApproved
            $ActionDetails = ""
            Try{
                $WsusServer.GetUpdates($UpdateScope) | Select-Object Title, @{Name="ProductTitles";Expression={($_.GetUpdateCategories().Title -Join "`n")}}, @{Name="ProductIds";Expression={($_.GetUpdateCategories().Id -Join "`n")}}, CreationDate, LegacyName, @{Name="Id.RevisionNumber";Expression={($_.Id.RevisionNumber -Join "`n")}}, @{Name="Id.UpdateId";Expression={($_.Id.UpdateId -Join "`n")}}, @{Name="KnowledgebaseArticles";Expression={($_.KnowledgebaseArticles -Join "`n")}}, @{Name="SecurityBulletins";Expression={($_.SecurityBulletins -Join "`n")}}, UpdateClassificationTitle, @{Name="UpdateClassificationId";Expression={($_.GetUpdateClassification().Id -Join "`n")}}, @{Name="ProductFamilyTitles";Expression={($_.ProductFamilyTitles -Join "`n")}}, UpdateType, @{Name="Action";Expression={$ActionDetails}} | Export-Csv -Path (Join-Path $LogDirectory "HasStaleUpdateApprovals.csv") -Encoding UTF8 -NoTypeInformation -Append | Out-Null
            }
            Catch{
                Write-Error "$(Get-Date -Format F): [LatestRevisionApproved] 更新プログラムを取得できませんでした`n$($_.Exception)"
            }
        }
    }

    Write-Verbose "$(Get-Date -Format F): Finish Optimize-WsusContents"
    Stop-Logging
}
Export-ModuleMember -Function Optimize-WsusContents