# Import required modules
try {
Import-Module ActiveDirectory -ErrorAction Stop
} catch {
Write-Host "Active Directory module not available. Please install RSAT tools." -ForegroundColor Red
exit
}
# Output file setup
$outputFile = "ADHealthCheck_$(Get-Date -Format yyyyMMdd_HHmmss).txt"
$timestamp = Get-Date -Format "yyyy-MM-dd HH:mm:ss"
# Function to write output to console and file
function Write-OutputBoth {
param([string]$message)
Write-Host $message
Add-Content -Path $outputFile -Value $message
}
Write-OutputBoth "============================================="
Write-OutputBoth " Active Directory Health Check Report"
Write-OutputBoth " Generated on: $timestamp"
Write-OutputBoth "=============================================`n"
# 1. Domain Controller Availability
Write-OutputBoth "1. DOMAIN CONTROLLER AVAILABILITY"
Write-OutputBoth "---------------------------------"
try {
$domainControllers = Get-ADDomainController -Filter *
foreach ($dc in $domainControllers) {
$pingResult = Test-Connection -ComputerName $dc.HostName -Count 2 -Quiet
Write-OutputBoth "$($dc.HostName): $(if ($pingResult) {'Online'} else {'Offline'})"
}
} catch {
Write-OutputBoth "ERROR: Failed to check domain controllers - $_"
}
Write-OutputBoth ""
# 2. Replication Status
Write-OutputBoth "2. REPLICATION STATUS"
Write-OutputBoth "----------------------"
try {
$replStatus = repadmin /replsummary
Write-OutputBoth $replStatus
# More detailed replication check
Write-OutputBoth "`nDetailed Replication Status:"
$replErrors = repadmin /showrepl * /errorsonly
if ($replErrors) {
Write-OutputBoth "Replication errors detected:`n$replErrors" -ForegroundColor Red
} else {
Write-OutputBoth "No replication errors found."
}
} catch {
Write-OutputBoth "ERROR: Failed to check replication status - $_"
}
Write-OutputBoth ""
# 3. DNS Health Check
Write-OutputBoth "3. DNS HEALTH CHECK"
Write-OutputBoth "-------------------"
try {
Write-OutputBoth "DNS Test with DCDiag:"
$dnsTest = dcdiag /test:dns /v
Write-OutputBoth $dnsTest
# Check DNS forwarders
Write-OutputBoth "`nDNS Forwarders:"
$dnsServers = Get-DnsServerForwarder
if ($dnsServers) {
foreach ($server in $dnsServers.IPAddress) {
Write-OutputBoth "Forwarder: $server"
}
} else {
Write-OutputBoth "No DNS forwarders configured."
}
} catch {
Write-OutputBoth "ERROR: Failed to check DNS health - $_"
}
Write-OutputBoth ""
# 4. Service Status
Write-OutputBoth "4. CRITICAL SERVICES STATUS"
Write-OutputBoth "---------------------------"
$services = @("NTDS", "DNS", "Netlogon", "KDC", "DFS", "ADWS")
foreach ($service in $services) {
try {
$svc = Get-Service -Name $service -ErrorAction SilentlyContinue
if ($svc) {
Write-OutputBoth "$($service): $($svc.Status)"
}
} catch {
Write-OutputBoth "ERROR: Failed to check $service service - $_"
}
}
Write-OutputBoth ""
# 5. FSMO Roles Check
Write-OutputBoth "5. FSMO ROLES STATUS"
Write-OutputBoth "---------------------"
try {
$forest = Get-ADForest
$domain = Get-ADDomain
Write-OutputBoth "Forest-wide roles:"
Write-OutputBoth " Schema Master: $($forest.SchemaMaster)"
Write-OutputBoth " Domain Naming Master: $($forest.DomainNamingMaster)"
Write-OutputBoth "`nDomain-wide roles:"
Write-OutputBoth " PDC Emulator: $($domain.PDCEmulator)"
Write-OutputBoth " RID Master: $($domain.RIDMaster)"
Write-OutputBoth " Infrastructure Master: $($domain.InfrastructureMaster)"
# Check if roles are online
$allRoles = @($forest.SchemaMaster, $forest.DomainNamingMaster, $domain.PDCEmulator, $domain.RIDMaster, $domain.InfrastructureMaster) | Select-Object -Unique
foreach ($role in $allRoles) {
$online = Test-Connection -ComputerName $role -Count 1 -Quiet
Write-OutputBoth " $role is $(if ($online) {'online'} else {'offline'})"
}
} catch {
Write-OutputBoth "ERROR: Failed to check FSMO roles - $_"
}
Write-OutputBoth ""
# 6. Time Synchronization
Write-OutputBoth "6. TIME SYNCHRONIZATION"
Write-OutputBoth "-----------------------"
try {
$timeSource = w32tm /query /source
$timeConfig = w32tm /query /configuration
$timeStatus = w32tm /query /status
Write-OutputBoth "Time source: $timeSource"
Write-OutputBoth "`nTime configuration:`n$timeConfig"
Write-OutputBoth "`nTime status:`n$timeStatus"
} catch {
Write-OutputBoth "ERROR: Failed to check time synchronization - $_"
}
Write-OutputBoth ""
# 7. Recent Critical Events
Write-OutputBoth "7. RECENT CRITICAL EVENTS"
Write-OutputBoth "-------------------------"
try {
$events = Get-EventLog -LogName "Directory Service" -EntryType Error,Warning -After (Get-Date).AddDays(-1) -ErrorAction SilentlyContinue
if ($events) {
$events | Select-Object -First 10 -Property TimeGenerated, EntryType, EventID, Message | Format-Table -AutoSize | Out-String -Width 4096 | Write-OutputBoth
} else {
Write-OutputBoth "No critical Directory Service events in the last 24 hours."
}
$systemEvents = Get-EventLog -LogName "System" -EntryType Error,Warning -After (Get-Date).AddDays(-1) -ErrorAction SilentlyContinue | Where-Object {
$_.Source -like "*Active Directory*" -or $_.Source -like "*DNS*" -or $_.Source -like "*KDC*"
}
if ($systemEvents) {
Write-OutputBoth "`nRelevant System events:"
$systemEvents | Select-Object -First 10 -Property TimeGenerated, EntryType, EventID, Message | Format-Table -AutoSize | Out-String -Width 4096 | Write-OutputBoth
}
} catch {
Write-OutputBoth "ERROR: Failed to check event logs - $_"
}
# 8. Database and Disk Space
Write-OutputBoth "`n8. DATABASE AND DISK SPACE"
Write-OutputBoth "--------------------------"
try {
$ntdsPath = (Get-ItemProperty "HKLM:\SYSTEM\CurrentControlSet\Services\NTDS\Parameters")."DSA Database file"
$ntdsDrive = $ntdsPath.Split('\')[0]
$diskInfo = Get-WmiObject Win32_LogicalDisk -Filter "DeviceID='$ntdsDrive'"
$freeSpaceGB = [math]::Round($diskInfo.FreeSpace / 1GB, 2)
$totalSpaceGB = [math]::Round($diskInfo.Size / 1GB, 2)
$percentFree = [math]::Round(($freeSpaceGB / $totalSpaceGB) * 100, 2)
Write-OutputBoth "NTDS Database location: $ntdsPath"
Write-OutputBoth "Disk $ntdsDrive free space: $freeSpaceGB GB of $totalSpaceGB GB ($percentFree%)"
if ($percentFree -lt 15) {
Write-OutputBoth "WARNING: Low disk space on $ntdsDrive!" -ForegroundColor Red
}
} catch {
Write-OutputBoth "ERROR: Failed to check database and disk space - $_"
}
Write-OutputBoth "`n============================================="
Write-OutputBoth " Active Directory Health Check Completed"
Write-OutputBoth " Report saved to: $outputFile"
Write-OutputBoth "============================================="
# Open the report file
Invoke-Item $outputFile