Private/networking-functions.ps1

function Show-ServicesWithoutEndpoints {
    param(
        [object]$KubeData,
        [int]$PageSize = 10,
        [switch]$Html,
        [switch]$Json,
        [switch]$ExcludeNamespaces
    )

    if (-not $Global:MakeReport -and -not $Html -and -not $json) { Clear-Host }
    Write-Host "`n[🔍 Services Without Endpoints]" -ForegroundColor Cyan
    Write-Host -NoNewline "`n🤖 Fetching Service Data..." -ForegroundColor Yellow

    try {
        if ($null -ne $KubeData) {
            $services = $KubeData.Services.items | Where-Object { $_.spec.type -ne "ExternalName" }
            $endpointsRaw = $KubeData.Endpoints.items
        } else {
            $services = kubectl get services --all-namespaces -o json | ConvertFrom-Json |
                Select-Object -ExpandProperty items |
                Where-Object { $_.spec.type -ne "ExternalName" }

            $endpointsRaw = kubectl get endpoints --all-namespaces -o json | ConvertFrom-Json |
                Select-Object -ExpandProperty items
        }
    }
    catch {
        Write-Host "`r🤖 ❌ Failed to fetch service or endpoint data: $_" -ForegroundColor Red
        if ($Global:MakeReport -and -not $Html) {
            Write-ToReport "`n[🔍 Services Without Endpoints]`n❌ Error: $_"
        }
        if (-not $Global:MakeReport -and -not $Html) {
            Read-Host "🤖 Press Enter to return to the menu"
        }
        return
    }

    if ($ExcludeNamespaces) {
        $services = Exclude-Namespaces -items $services
        $endpointsRaw = Exclude-Namespaces -items $endpointsRaw
    }

    Write-Host "`r🤖 ✅ Services fetched. (Total: $($services.Count))" -ForegroundColor Green
    Write-Host -NoNewline "`n🤖 Analyzing Endpoints..." -ForegroundColor Yellow

    # Build endpoint lookup
    $endpoints = $endpointsRaw | Group-Object { $_.metadata.namespace + "/" + $_.metadata.name }
    $endpointsLookup = @{}
    foreach ($ep in $endpoints) {
        $endpointsLookup[$ep.Name] = $true
    }

    $servicesWithoutEndpoints = $services | Where-Object {
        $key = $_.metadata.namespace + "/" + $_.metadata.name
        $ep = $endpointsRaw | Where-Object { $_.metadata.namespace + "/" + $_.metadata.name -eq $key }
    
        # If there's no endpoints object or it's empty
        -not $ep -or -not $ep.subsets -or $ep.subsets.Count -eq 0
    }
    
    $totalServices = $servicesWithoutEndpoints.Count
    Write-Host "`r🤖 ✅ Endpoint analysis complete. ($totalServices services without endpoints)" -ForegroundColor Green

    if ($totalServices -eq 0) {
        Write-Host "`r🤖 ✅ All services have endpoints." -ForegroundColor Green
        if ($Global:MakeReport -and -not $Html) {
            Write-ToReport "`n[🔍 Services Without Endpoints]`n✅ All services have endpoints."
        }
        if (-not $Global:MakeReport -and -not $Html) {
            Read-Host "🤖 Press Enter to return to the menu"
        }
        if ($Html) {
            return "<p><strong>✅ All services have endpoints.</strong></p>"
        }
        return
    }

    # Table content
    $tableData = $servicesWithoutEndpoints | ForEach-Object {
        [PSCustomObject]@{
            Namespace = $_.metadata.namespace
            Service   = $_.metadata.name
            Type      = $_.spec.type
            Status    = "⚠️ No Endpoints"
        }
    }

    if ($Json) {
        return @{ Total = $tableData.Count; Items = $tableData }
    }    

    if ($Html) {
        $htmlTable = $tableData |
            ConvertTo-Html -Fragment -Property Namespace, Service, Type, Status |
            Out-String
        return "<p><strong>⚠️ Total Services Without Endpoints:</strong> $totalServices</p>" + $htmlTable
    }

    if ($Global:MakeReport) {
        Write-ToReport "`n[🔍 Services Without Endpoints]`n⚠️ Total: $totalServices"
        $tableString = $tableData |
            Format-Table Namespace, Service, Type, Status -AutoSize |
            Out-String
        Write-ToReport $tableString
        return
    }

    # Console pagination
    $currentPage = 0
    $totalPages = [math]::Ceiling($totalServices / $PageSize)

    do {
        Clear-Host
        Write-Host "`n[🔍 Services Without Endpoints - Page $($currentPage + 1) of $totalPages]" -ForegroundColor Cyan

        if ($currentPage -eq 0) {
            Write-SpeechBubble -msg @(
                "🤖 These services lack endpoints.",
                "",
                "📌 Endpoints are needed to route traffic to pods.",
                " - Check if matching pods exist.",
                " - Validate service selectors.",
                "",
                "⚠️ Total: $totalServices"
            ) -color "Cyan" -icon "🤖" -lastColor "Red" -delay 50
        }

        $start = $currentPage * $PageSize
        $slice = $tableData | Select-Object -Skip $start -First $PageSize

        if ($slice) {
            $slice | Format-Table Namespace, Service, Type, Status -AutoSize | Out-Host
        }

        $newPage = Show-Pagination -currentPage $currentPage -totalPages $totalPages
        if ($newPage -eq -1) { break }
        $currentPage = $newPage
    } while ($true)
}

function Check-PubliclyAccessibleServices {
    param(
        [object]$KubeData,
        [int]$PageSize = 10,
        [switch]$Html,
        [switch]$Json,
        [switch]$ExcludeNamespaces
    )

    if (-not $Global:MakeReport -and -not $Html -and -not $json) { Clear-Host }
    Write-Host "`n[🌐 Publicly Accessible Services]" -ForegroundColor Cyan
    Write-Host -NoNewline "`n🤖 Fetching Services..." -ForegroundColor Yellow

    try {
        $services = if ($null -ne $KubeData) {
            $KubeData.Services.items
        } else {
            kubectl get services --all-namespaces -o json | ConvertFrom-Json | Select-Object -ExpandProperty items
        }
    } catch {
        Write-Host "`r🤖 ❌ Failed to fetch service data: $_" -ForegroundColor Red
        if ($Html) { return "<p>❌ Failed to fetch service data.</p>" }
        return
    }

    if (-not $services) {
        Write-Host "`r🤖 ❌ No services found." -ForegroundColor Red
        if ($Html) { return "<p>❌ No services found.</p>" }
        return
    }

    if ($ExcludeNamespaces) {
        $services = Exclude-Namespaces -items $services
    }

    Write-Host "`r🤖 ✅ Services fetched. ($($services.Count) total)" -ForegroundColor Green
    Write-Host -NoNewline "`n🤖 Analyzing for external exposure..." -ForegroundColor Yellow

    $internalIpPatterns = @(
        '^10\.',              # 10.0.0.0/8
        '^172\.(1[6-9]|2[0-9]|3[0-1])\.',  # 172.16.0.0/12
        '^192\.168\.',        # 192.168.0.0/16
        '^127\.',             # Loopback
        '^169\.254\.',        # APIPA
        '^100\.64\.',         # CGNAT
        '^0\.'                # Invalid
    )

    $isInternalIp = {
        param($ip)
        foreach ($pattern in $internalIpPatterns) {
            if ($ip -match $pattern) { return $true }
        }
        return $false
    }

    $publicServices = $services | Where-Object {
        $_.spec.type -in @("LoadBalancer", "NodePort")
    }

    $tableData = @()

    foreach ($svc in $publicServices) {
        $externalEntries = @()
        if ($svc.status.loadBalancer.ingress) {
            foreach ($entry in $svc.status.loadBalancer.ingress) {
                if ($entry.ip -and -not (&$isInternalIp $entry.ip)) {
                    $externalEntries += $entry.ip
                }
                elseif ($entry.hostname) {
                    $externalEntries += $entry.hostname
                }
            }
        }

        $hasNodePort = ($svc.spec.type -eq "NodePort")
        $hasExternalIp = $externalEntries.Count -gt 0

        if ($hasExternalIp -or $hasNodePort) {
            $tableData += [PSCustomObject]@{
                Namespace  = $svc.metadata.namespace
                Service    = $svc.metadata.name
                Type       = $svc.spec.type
                Ports      = if ($svc.spec.ports) {
                    ($svc.spec.ports | ForEach-Object { "$($_.port)/$($_.protocol)" }) -join ", "
                } else { "N/A" }
                ExternalIP = if ($externalEntries.Count -gt 0) { $externalEntries -join ", " } else { "None" }
            }
        }
    }

    $totalPublic = $tableData.Count
    Write-Host "`r🤖 ✅ Analysis complete. ($totalPublic public services)" -ForegroundColor Green

    if ($totalPublic -eq 0) {
        Write-Host "✅ No publicly accessible services found." -ForegroundColor Green
        if ($Html) { return "<p><strong>✅ No publicly accessible services found.</strong></p>" }
        if ($Global:MakeReport -and -not $Html) {
            Write-ToReport "`n[🌐 Publicly Accessible Services]`n✅ No publicly accessible services found."
        }
        if (-not $Global:MakeReport -and -not $Html) {
            Read-Host "🤖 Press Enter to return to the menu"
        }
        return
    }

    if ($Json) {
        return @{ Total = $tableData.Count; Items = $tableData }
    }    

    if ($Html) {
        $htmlTable = $tableData |
            ConvertTo-Html -Fragment -Property Namespace, Service, Type, Ports, ExternalIP |
            Out-String
        return "<p><strong>⚠️ Total Public Services Found:</strong> $totalPublic</p>" + $htmlTable
    }

    if ($Global:MakeReport) {
        Write-ToReport "`n[🌐 Publicly Accessible Services]`n⚠️ Total Public Services Found: $totalPublic"
        $tableString = $tableData | Format-Table Namespace, Service, Type, Ports, ExternalIP -AutoSize | Out-String 
        Write-ToReport $tableString
        return
    }

    # Console output with pagination
    $currentPage = 0
    $totalPages = [math]::Ceiling($totalPublic / $PageSize)

    do {
        Clear-Host
        Write-Host "`n[🌐 Publicly Accessible Services - Page $($currentPage + 1) of $totalPages]" -ForegroundColor Cyan

        if ($currentPage -eq 0) {
            Write-SpeechBubble -msg @(
                "🤖 Services of type LoadBalancer or NodePort may be publicly reachable.",
                "",
                "📌 This check detects services exposed via:",
                " - External IPs (if not private)",
                " - NodePort access across all cluster nodes",
                "",
                "⚠️ Total Public Services Found: $totalPublic"
            ) -color "Cyan" -icon "🤖" -lastColor "Red" -delay 50
        }

        $paged = $tableData | Select-Object -Skip ($currentPage * $PageSize) -First $PageSize
        $paged | Format-Table Namespace, Service, Type, Ports, ExternalIP -AutoSize | Out-Host

        $newPage = Show-Pagination -currentPage $currentPage -totalPages $totalPages
        if ($newPage -eq -1) { break }
        $currentPage = $newPage
    } while ($true)
}