helpers/aws/Get-AWSDashboard.ps1

<#
.SYNOPSIS
    Generates an interactive HTML dashboard report for AWS environments.
.DESCRIPTION
    Orchestration script that authenticates to AWS (via profile, access keys, or
    existing session), scans one or more regions for EC2 instances, RDS databases,
    and ELBv2 load balancers, and produces a searchable, sortable Bootstrap Table
    HTML dashboard. Output includes both a JSON data file and a self-contained HTML report.
.PARAMETER AccessKey
    AWS access key ID for authentication. Use with SecretKey. If omitted and no
    ProfileName is specified, uses the existing AWS session.
.PARAMETER SecretKey
    AWS secret access key for authentication. Use with AccessKey.
.PARAMETER ProfileName
    Name of a stored AWS credential profile to use for authentication.
.PARAMETER Regions
    One or more AWS region names to scan (e.g. us-east-1, eu-west-1). If omitted,
    prompts interactively. Defaults to us-east-1 if none provided.
.PARAMETER SkipRDS
    Exclude RDS instances from the dashboard results.
.PARAMETER SkipELB
    Exclude ELBv2 load balancers from the dashboard results.
.EXAMPLE
    .\Get-AWSDashboard.ps1
 
    Prompts for region(s) and uses existing AWS session credentials.
.EXAMPLE
    .\Get-AWSDashboard.ps1 -Regions "us-east-1"
 
    Generates a dashboard for a single AWS region.
.EXAMPLE
    .\Get-AWSDashboard.ps1 -ProfileName "MyProfile" -Regions "us-east-1","eu-west-1"
 
    Authenticates with a named profile and scans multiple regions.
.EXAMPLE
    .\Get-AWSDashboard.ps1 -Regions "us-east-1" -SkipRDS -SkipELB
 
    Generates a dashboard with only EC2 instances (no RDS or ELB).
.OUTPUTS
    System.Void
    Produces a JSON file (aws_dashboard.json) and an HTML dashboard
    (AWS-Dashboard.html) in the system temp directory, then opens the HTML in the default browser.
.NOTES
    Author : jason@wug.ninja
    Version : 1.0.0
    Date : 2025-07-15
    Requires: PowerShell 5.1+, AWS.Tools PowerShell modules, AWSHelpers.ps1 in the same directory.
.LINK
    https://github.com/jayyx2/WhatsUpGoldPS
#>


param (
    [Parameter(Mandatory = $false)]
    [string]$AccessKey,

    [Parameter(Mandatory = $false)]
    [string]$SecretKey,

    [Parameter(Mandatory = $false)]
    [string]$ProfileName,

    [Parameter(Mandatory = $false)]
    [string[]]$Regions,

    [Parameter(Mandatory = $false)]
    [switch]$SkipRDS,

    [Parameter(Mandatory = $false)]
    [switch]$SkipELB
)

# --- Configuration -----------------------------------------------------------
$helpersPath = Join-Path $PSScriptRoot "AWSHelpers.ps1"
if (Test-Path $helpersPath) {
    . $helpersPath
}
else {
    throw "AWSHelpers.ps1 not found at $helpersPath. Ensure it is in the same directory."
}

if (Get-Module -ListAvailable -Name WhatsUpGoldPS) {
    if (-not (Get-Module -Name WhatsUpGoldPS)) {
        Import-Module -Name WhatsUpGoldPS
    }
}

# --- Authenticate ------------------------------------------------------------
if ($ProfileName) {
    Connect-AWSProfile -ProfileName $ProfileName -Region ($Regions | Select-Object -First 1 -ErrorAction SilentlyContinue) 2>$null
}
elseif ($AccessKey -and $SecretKey) {
    $defaultRegion = if ($Regions) { $Regions[0] } else { "us-east-1" }
    Connect-AWSProfile -AccessKey $AccessKey -SecretKey $SecretKey -Region $defaultRegion
}
else {
    Write-Host "No credentials specified. Attempting to use existing AWS session/profile..." -ForegroundColor Yellow
}

# --- Input prompts -----------------------------------------------------------
if (-not $Regions -or $Regions.Count -eq 0) {
    $regionInput = Read-Host -Prompt "Enter AWS region(s) to scan (comma-separated, e.g. us-east-1)"
    $Regions = $regionInput -split ',' | ForEach-Object { $_.Trim() } | Where-Object { $_ -ne '' }
}

if (-not $Regions -or $Regions.Count -eq 0) {
    $Regions = @("us-east-1")
    Write-Host "Defaulting to us-east-1" -ForegroundColor Yellow
}

# Output paths
$outputDir = if ($env:TEMP) { $env:TEMP } else { "C:\temp" }
$jsonPath  = Join-Path $outputDir "aws_dashboard.json"
$htmlPath  = Join-Path $outputDir "AWS-Dashboard.html"
if (-not (Test-Path $outputDir)) { New-Item -ItemType Directory -Path $outputDir -Force | Out-Null }

# --- Collect data ------------------------------------------------------------
Write-Host "`nScanning AWS regions: $($Regions -join ', ')" -ForegroundColor Cyan
$dashboardData = Get-AWSDashboard -Regions $Regions -IncludeRDS (-not $SkipRDS) -IncludeELB (-not $SkipELB)

if (-not $dashboardData -or $dashboardData.Count -eq 0) {
    Write-Warning "No resources collected. Exiting."
    return
}

# --- Summary -----------------------------------------------------------------
$running = @($dashboardData | Where-Object { $_.State -in 'running','available','active' }).Count
$stopped = $dashboardData.Count - $running
$ec2     = @($dashboardData | Where-Object { $_.ResourceType -eq 'EC2' }).Count
$rds     = @($dashboardData | Where-Object { $_.ResourceType -eq 'RDS' }).Count
$elb     = @($dashboardData | Where-Object { $_.ResourceType -eq 'ELB' }).Count

Write-Host "`n--- AWS Summary ---" -ForegroundColor Yellow
Write-Host " Total: $($dashboardData.Count)"
Write-Host " Running: $running" -ForegroundColor Green
Write-Host " Stopped: $stopped" -ForegroundColor Red
Write-Host " EC2: $ec2 | RDS: $rds | ELB: $elb"

# --- Generate outputs --------------------------------------------------------
$dashboardData | ConvertTo-Json -Depth 5 | Out-File $jsonPath -Force -Encoding UTF8
Write-Host "`nJSON data written to $jsonPath" -ForegroundColor Yellow

$templatePath = Join-Path $PSScriptRoot "AWS-Dashboard-Template.html"
Export-AWSDashboardHtml -DashboardData $dashboardData -OutputPath $htmlPath -ReportTitle "AWS Dashboard" -TemplatePath $templatePath
Write-Host "HTML dashboard written to $htmlPath" -ForegroundColor Yellow

# --- Optional: Open in browser -----------------------------------------------
if ($env:OS -match 'Windows') {
    $openBrowser = Read-Host -Prompt "Open dashboard in browser? (Y/N)"
    if ($openBrowser -match '^[Yy]') {
        Start-Process $htmlPath
    }
}

Write-Host "Done." -ForegroundColor Green

# SIG # Begin signature block
# MIIVlwYJKoZIhvcNAQcCoIIViDCCFYQCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCqwkSybWl1LU/p
# wqhZ0Kndr4BVXrgm6ke4L4EY3oqb+qCCEdMwggVvMIIEV6ADAgECAhBI/JO0YFWU
# jTanyYqJ1pQWMA0GCSqGSIb3DQEBDAUAMHsxCzAJBgNVBAYTAkdCMRswGQYDVQQI
# DBJHcmVhdGVyIE1hbmNoZXN0ZXIxEDAOBgNVBAcMB1NhbGZvcmQxGjAYBgNVBAoM
# EUNvbW9kbyBDQSBMaW1pdGVkMSEwHwYDVQQDDBhBQUEgQ2VydGlmaWNhdGUgU2Vy
# dmljZXMwHhcNMjEwNTI1MDAwMDAwWhcNMjgxMjMxMjM1OTU5WjBWMQswCQYDVQQG
# EwJHQjEYMBYGA1UEChMPU2VjdGlnbyBMaW1pdGVkMS0wKwYDVQQDEyRTZWN0aWdv
# IFB1YmxpYyBDb2RlIFNpZ25pbmcgUm9vdCBSNDYwggIiMA0GCSqGSIb3DQEBAQUA
# A4ICDwAwggIKAoICAQCN55QSIgQkdC7/FiMCkoq2rjaFrEfUI5ErPtx94jGgUW+s
# hJHjUoq14pbe0IdjJImK/+8Skzt9u7aKvb0Ffyeba2XTpQxpsbxJOZrxbW6q5KCD
# J9qaDStQ6Utbs7hkNqR+Sj2pcaths3OzPAsM79szV+W+NDfjlxtd/R8SPYIDdub7
# P2bSlDFp+m2zNKzBenjcklDyZMeqLQSrw2rq4C+np9xu1+j/2iGrQL+57g2extme
# me/G3h+pDHazJyCh1rr9gOcB0u/rgimVcI3/uxXP/tEPNqIuTzKQdEZrRzUTdwUz
# T2MuuC3hv2WnBGsY2HH6zAjybYmZELGt2z4s5KoYsMYHAXVn3m3pY2MeNn9pib6q
# RT5uWl+PoVvLnTCGMOgDs0DGDQ84zWeoU4j6uDBl+m/H5x2xg3RpPqzEaDux5mcz
# mrYI4IAFSEDu9oJkRqj1c7AGlfJsZZ+/VVscnFcax3hGfHCqlBuCF6yH6bbJDoEc
# QNYWFyn8XJwYK+pF9e+91WdPKF4F7pBMeufG9ND8+s0+MkYTIDaKBOq3qgdGnA2T
# OglmmVhcKaO5DKYwODzQRjY1fJy67sPV+Qp2+n4FG0DKkjXp1XrRtX8ArqmQqsV/
# AZwQsRb8zG4Y3G9i/qZQp7h7uJ0VP/4gDHXIIloTlRmQAOka1cKG8eOO7F/05QID
# AQABo4IBEjCCAQ4wHwYDVR0jBBgwFoAUoBEKIz6W8Qfs4q8p74Klf9AwpLQwHQYD
# VR0OBBYEFDLrkpr/NZZILyhAQnAgNpFcF4XmMA4GA1UdDwEB/wQEAwIBhjAPBgNV
# HRMBAf8EBTADAQH/MBMGA1UdJQQMMAoGCCsGAQUFBwMDMBsGA1UdIAQUMBIwBgYE
# VR0gADAIBgZngQwBBAEwQwYDVR0fBDwwOjA4oDagNIYyaHR0cDovL2NybC5jb21v
# ZG9jYS5jb20vQUFBQ2VydGlmaWNhdGVTZXJ2aWNlcy5jcmwwNAYIKwYBBQUHAQEE
# KDAmMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5jb21vZG9jYS5jb20wDQYJKoZI
# hvcNAQEMBQADggEBABK/oe+LdJqYRLhpRrWrJAoMpIpnuDqBv0WKfVIHqI0fTiGF
# OaNrXi0ghr8QuK55O1PNtPvYRL4G2VxjZ9RAFodEhnIq1jIV9RKDwvnhXRFAZ/ZC
# J3LFI+ICOBpMIOLbAffNRk8monxmwFE2tokCVMf8WPtsAO7+mKYulaEMUykfb9gZ
# pk+e96wJ6l2CxouvgKe9gUhShDHaMuwV5KZMPWw5c9QLhTkg4IUaaOGnSDip0TYl
# d8GNGRbFiExmfS9jzpjoad+sPKhdnckcW67Y8y90z7h+9teDnRGWYpquRRPaf9xH
# +9/DUp/mBlXpnYzyOmJRvOwkDynUWICE5EV7WtgwggYaMIIEAqADAgECAhBiHW0M
# UgGeO5B5FSCJIRwKMA0GCSqGSIb3DQEBDAUAMFYxCzAJBgNVBAYTAkdCMRgwFgYD
# VQQKEw9TZWN0aWdvIExpbWl0ZWQxLTArBgNVBAMTJFNlY3RpZ28gUHVibGljIENv
# ZGUgU2lnbmluZyBSb290IFI0NjAeFw0yMTAzMjIwMDAwMDBaFw0zNjAzMjEyMzU5
# NTlaMFQxCzAJBgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0ZWQxKzAp
# BgNVBAMTIlNlY3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYwggGiMA0G
# CSqGSIb3DQEBAQUAA4IBjwAwggGKAoIBgQCbK51T+jU/jmAGQ2rAz/V/9shTUxjI
# ztNsfvxYB5UXeWUzCxEeAEZGbEN4QMgCsJLZUKhWThj/yPqy0iSZhXkZ6Pg2A2NV
# DgFigOMYzB2OKhdqfWGVoYW3haT29PSTahYkwmMv0b/83nbeECbiMXhSOtbam+/3
# 6F09fy1tsB8je/RV0mIk8XL/tfCK6cPuYHE215wzrK0h1SWHTxPbPuYkRdkP05Zw
# mRmTnAO5/arnY83jeNzhP06ShdnRqtZlV59+8yv+KIhE5ILMqgOZYAENHNX9SJDm
# +qxp4VqpB3MV/h53yl41aHU5pledi9lCBbH9JeIkNFICiVHNkRmq4TpxtwfvjsUe
# dyz8rNyfQJy/aOs5b4s+ac7IH60B+Ja7TVM+EKv1WuTGwcLmoU3FpOFMbmPj8pz4
# 4MPZ1f9+YEQIQty/NQd/2yGgW+ufflcZ/ZE9o1M7a5Jnqf2i2/uMSWymR8r2oQBM
# dlyh2n5HirY4jKnFH/9gRvd+QOfdRrJZb1sCAwEAAaOCAWQwggFgMB8GA1UdIwQY
# MBaAFDLrkpr/NZZILyhAQnAgNpFcF4XmMB0GA1UdDgQWBBQPKssghyi47G9IritU
# pimqF6TNDDAOBgNVHQ8BAf8EBAMCAYYwEgYDVR0TAQH/BAgwBgEB/wIBADATBgNV
# HSUEDDAKBggrBgEFBQcDAzAbBgNVHSAEFDASMAYGBFUdIAAwCAYGZ4EMAQQBMEsG
# A1UdHwREMEIwQKA+oDyGOmh0dHA6Ly9jcmwuc2VjdGlnby5jb20vU2VjdGlnb1B1
# YmxpY0NvZGVTaWduaW5nUm9vdFI0Ni5jcmwwewYIKwYBBQUHAQEEbzBtMEYGCCsG
# AQUFBzAChjpodHRwOi8vY3J0LnNlY3RpZ28uY29tL1NlY3RpZ29QdWJsaWNDb2Rl
# U2lnbmluZ1Jvb3RSNDYucDdjMCMGCCsGAQUFBzABhhdodHRwOi8vb2NzcC5zZWN0
# aWdvLmNvbTANBgkqhkiG9w0BAQwFAAOCAgEABv+C4XdjNm57oRUgmxP/BP6YdURh
# w1aVcdGRP4Wh60BAscjW4HL9hcpkOTz5jUug2oeunbYAowbFC2AKK+cMcXIBD0Zd
# OaWTsyNyBBsMLHqafvIhrCymlaS98+QpoBCyKppP0OcxYEdU0hpsaqBBIZOtBajj
# cw5+w/KeFvPYfLF/ldYpmlG+vd0xqlqd099iChnyIMvY5HexjO2AmtsbpVn0OhNc
# WbWDRF/3sBp6fWXhz7DcML4iTAWS+MVXeNLj1lJziVKEoroGs9Mlizg0bUMbOalO
# hOfCipnx8CaLZeVme5yELg09Jlo8BMe80jO37PU8ejfkP9/uPak7VLwELKxAMcJs
# zkyeiaerlphwoKx1uHRzNyE6bxuSKcutisqmKL5OTunAvtONEoteSiabkPVSZ2z7
# 6mKnzAfZxCl/3dq3dUNw4rg3sTCggkHSRqTqlLMS7gjrhTqBmzu1L90Y1KWN/Y5J
# KdGvspbOrTfOXyXvmPL6E52z1NZJ6ctuMFBQZH3pwWvqURR8AgQdULUvrxjUYbHH
# j95Ejza63zdrEcxWLDX6xWls/GDnVNueKjWUH3fTv1Y8Wdho698YADR7TNx8X8z2
# Bev6SivBBOHY+uqiirZtg0y9ShQoPzmCcn63Syatatvx157YK9hlcPmVoa1oDE5/
# L9Uo2bC5a4CH2RwwggY+MIIEpqADAgECAhAHnODk0RR/hc05c892LTfrMA0GCSqG
# SIb3DQEBDAUAMFQxCzAJBgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0
# ZWQxKzApBgNVBAMTIlNlY3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYw
# HhcNMjYwMjA5MDAwMDAwWhcNMjkwNDIxMjM1OTU5WjBVMQswCQYDVQQGEwJVUzEU
# MBIGA1UECAwLQ29ubmVjdGljdXQxFzAVBgNVBAoMDkphc29uIEFsYmVyaW5vMRcw
# FQYDVQQDDA5KYXNvbiBBbGJlcmlubzCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCC
# AgoCggIBAPN6aN4B1yYWkI5b5TBj3I0VV/peETrHb6EY4BHGxt8Ap+eT+WpEpJyE
# tRYPxEmNJL3A38Bkg7mwzPE3/1NK570ZBCuBjSAn4mSDIgIuXZnvyBO9W1OQs5d6
# 7MlJLUAEufl18tOr3ST1DeO9gSjQSAE5Nql0QDxPnm93OZBon+Fz3CmE+z3MwAe2
# h4KdtRAnCqwM+/V7iBdbw+JOxolpx+7RVjGyProTENIG3pe/hKvPb501lf8uBAAD
# LdjZr5ip8vIWbf857Yw1Bu10nVI7HW3eE8Cl5//d1ribHlzTzQLfttW+k+DaFsKZ
# BBL56l4YAlIVRsrOiE1kdHYYx6IGrEA809R7+TZA9DzGqyFiv9qmJAbL4fDwetDe
# yIq+Oztz1LvEdy8Rcd0JBY+J4S0eDEFIA3X0N8VcLeAwabKb9AjulKXwUeqCJLvN
# 79CJ90UTZb2+I+tamj0dn+IKMEsJ4v4Ggx72sxFr9+6XziodtTg5Luf2xd6+Phha
# mOxF2px9LObhBLLEMyRsCHZIzVZOFKu9BpHQH7ufGB+Sa80Tli0/6LEyn9+bMYWi
# 2ttn6lLOPThXMiQaooRUq6q2u3+F4SaPlxVFLI7OJVMhar6nW6joBvELTJPmANSM
# jDSRFDfHRCdGbZsL/keELJNy+jZctF6VvxQEjFM8/bazu6qYhrA7AgMBAAGjggGJ
# MIIBhTAfBgNVHSMEGDAWgBQPKssghyi47G9IritUpimqF6TNDDAdBgNVHQ4EFgQU
# 6YF0o0D5AVhKHbVocr8GaSIBibAwDgYDVR0PAQH/BAQDAgeAMAwGA1UdEwEB/wQC
# MAAwEwYDVR0lBAwwCgYIKwYBBQUHAwMwSgYDVR0gBEMwQTA1BgwrBgEEAbIxAQIB
# AwIwJTAjBggrBgEFBQcCARYXaHR0cHM6Ly9zZWN0aWdvLmNvbS9DUFMwCAYGZ4EM
# AQQBMEkGA1UdHwRCMEAwPqA8oDqGOGh0dHA6Ly9jcmwuc2VjdGlnby5jb20vU2Vj
# dGlnb1B1YmxpY0NvZGVTaWduaW5nQ0FSMzYuY3JsMHkGCCsGAQUFBwEBBG0wazBE
# BggrBgEFBQcwAoY4aHR0cDovL2NydC5zZWN0aWdvLmNvbS9TZWN0aWdvUHVibGlj
# Q29kZVNpZ25pbmdDQVIzNi5jcnQwIwYIKwYBBQUHMAGGF2h0dHA6Ly9vY3NwLnNl
# Y3RpZ28uY29tMA0GCSqGSIb3DQEBDAUAA4IBgQAEIsm4xnOd/tZMVrKwi3doAXvC
# wOA/RYQnFJD7R/bSQRu3wXEK4o9SIefye18B/q4fhBkhNAJuEvTQAGfqbbpxow03
# J5PrDTp1WPCWbXKX8Oz9vGWJFyJxRGftkdzZ57JE00synEMS8XCwLO9P32MyR9Z9
# URrpiLPJ9rQjfHMb1BUdvaNayomm7aWLAnD+X7jm6o8sNT5An1cwEAob7obWDM6s
# X93wphwJNBJAstH9Ozs6LwISOX6sKS7CKm9N3Kp8hOUue0ZHAtZdFl6o5u12wy+z
# zieGEI50fKnN77FfNKFOWKlS6OJwlArcbFegB5K89LcE5iNSmaM3VMB2ADV1FEcj
# GSHw4lTg1Wx+WMAMdl/7nbvfFxJ9uu5tNiT54B0s+lZO/HztwXYQUczdsFon3pjs
# Nrsk9ZlalBi5SHkIu+F6g7tWiEv3rtVApmJRnLkUr2Xq2a4nbslUCt4jKs5UX4V1
# nSX8OM++AXoyVGO+iTj7z+pl6XE9Gw/Td6WKKKsxggMaMIIDFgIBATBoMFQxCzAJ
# BgNVBAYTAkdCMRgwFgYDVQQKEw9TZWN0aWdvIExpbWl0ZWQxKzApBgNVBAMTIlNl
# Y3RpZ28gUHVibGljIENvZGUgU2lnbmluZyBDQSBSMzYCEAec4OTRFH+FzTlzz3Yt
# N+swDQYJYIZIAWUDBAIBBQCggYQwGAYKKwYBBAGCNwIBDDEKMAigAoAAoQKAADAZ
# BgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYB
# BAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQgVFaSIo/iRbl3rG+ZASE8eQDG7/iOeQoX
# lk+o3lWVrKQwDQYJKoZIhvcNAQEBBQAEggIA8TZ+0g2+ibU+ZgH3be1kK0XLKHEn
# 729H1C++LMun835GvzNJxhoq9U3OVyR7OqCDKysQPjKUzrvqs3YCKHF2o6qZQbwo
# bWWi47I3JkxicNKjui/C5qRlU1cDFnMcnGFEIKYvm7VJvQAFF59CtVYEHecP2Nks
# jACUm0dS30kLZOu/PBg/uOM1pim+02kmgr6rakTxfmNpde8NXk2d/Wly23T4v6Ry
# o3xvnQas7r4ZMZVuigV/HDiGrwwXhHXlKHS81HX8tlRiRnovqE5JhE/J8t5MafFL
# 5Kt+oD9JYw23vXsLUDDvazBrG5DyIcLZuX9gmq0ab5ji6/8+FkuiG4uKHTGztKMW
# eg/KFfLQtF6HxA95XFkd/9TtfC9xHBlejfPHbGrFlLeqt2xH0a/lIOQP3I3QsiG7
# 3AKgz3pXVhw6jRklKM3RH5bB8WDCTqcqZ0GAkM7D0CCnRnFj7eEGa30+jcqbGxN3
# F1lU3pG0ZjpvJ4VAU94CRlJWA3YJBqO4nX38ZwbZB8om5XICNXcZewL7EcgDnKpw
# eI7GviV/FR6qr/vAEMHs+epPv0BovFDs4MAy7u1/2jQhVd1HK0HPQy8iVx4K5roz
# Wlp7syuLl0LDjaryb4uJK/YVlJ36HRAPrzXItXdNa5HcMNz4CJ9GyqpNYQPY3reh
# h5bUr09klUZ+a2w=
# SIG # End signature block