tests/Test-HostServerFeatures.ps1
function Test-HostServerFeatures { [CmdletBinding()] param ( [parameter()][string] $TestName = "Installed Windows Features", [parameter()][string] $TestGroup = "configuration", [parameter()][string] $Description = "Validate Windows Server roles and features for CM site systems", [parameter()][hashtable] $ScriptParams ) try { $startTime = (Get-Date) [System.Collections.Generic.List[PSObject]]$tempdata = @() # for detailed test output to return if needed $stat = "PASS" $except = "FAIL" $msg = "No issues found" Import-Module ServerManager if ($ScriptParams.ComputerName -ne $env:COMPUTERNAME) { if ($ScriptParams.Credential) { $features = Get-WindowsFeature -ComputerName $ScriptParams.ComputerName -Credential $ScriptParams.Credential -ErrorAction Stop | Sort-Object Name } else { $features = Get-WindowsFeature -ComputerName $ScriptParams.ComputerName -ErrorAction Stop | Sort-Object Name } } else { $features = Get-WindowsFeature -ErrorAction Stop | Sort-Object Name } $LogFile = Join-Path $env:TEMP "serverfeatures.log" if ($ScriptParams.Remediate -eq $True -and ([string]::IsNullOrEmpty($ScriptParams.Source))) { throw "Source parameter is required for -Remediate but was not specified" } $flist = @($CmHealthConfig.windowsfeatures.Feature) if ($flist.Count -lt 1) { throw "failed to read features list from cmhealth settings file" } $exceptions = 0 [System.Collections.Generic.List[PSObject]]$missing = @() foreach ($feature in $features) { if ($feature.Name -in $flist) { if ($feature.Installed -ne $True) { Write-Log -Message "feature not installed: $($feature.Name)" if ($ScriptParams.Remediate -eq $True) { try { Write-Log -Message "installing: $($Feature.Name)" if ($ScriptParams.ComputerName -ne $env:COMPUTERNAME) { if ($ScriptParams.Credential) { Install-WindowsFeature -Name "$($Feature.Name)" -ComputerName $ScriptParams.ComputerName -Credential $ScriptParams.Credential -Source $ScriptParams.Source -LogPath $LogFile -WarningAction SilentlyContinue -ErrorAction Stop } else { Install-WindowsFeature -Name "$($Feature.Name)" -ComputerName $ScriptParams.ComputerName -Source $ScriptParams.Source -LogPath $LogFile -WarningAction SilentlyContinue -ErrorAction Stop } } else { Install-WindowsFeature -Name "$($Feature.Name)" -Source $ScriptParams.Source -LogPath $LogFile -WarningAction SilentlyContinue -ErrorAction Stop } $tempdata.Add( [pscustomobject]@{ Feature = $feature.Name Status = "Remediated" Message = "Success" } ) } catch { $tempdata.Add( [pscustomobject]@{ Feature = $feature.Name Status = "ERROR" Message = $_.Exception.Message -join ';' } ) } } else { $exceptions++ $tempdata.Add( [pscustomobject]@{ Feature = $feature.Name Status = $except Message = "Not installed" } ) $missing.Add($feature.Name) } } else { Write-Log -Message "feature is installed: $($feature.Name)" $tempdata.Add( [pscustomobject]@{ Feature = $feature.Name Status = "PASS" Message = "Already installed" } ) } } } if ($exceptions -gt 0) { $stat = $except $msg = "$exceptions features are missing: $($missing -join ',')" } } catch { $stat = 'ERROR' $msg = $_.Exception.Message -join ';' } finally { Write-Output $([pscustomobject]@{ TestName = $TestName TestGroup = $TestGroup TestData = $tempdata Description = $Description Status = $stat Message = $msg RunTime = $(Get-RunTime -BaseTime $startTime) Credential = $(if($ScriptParams.Credential){$($ScriptParams.Credential).UserName} else { $env:USERNAME }) }) } } |