Private/New-CAIQBreakGlassExclusionDashboard.ps1

Function New-CAIQBreakGlassExclusionDashboard {
    <#
        .SYNOPSIS
        Generates an HTML report for the Break Glass Assessment.
 
        .DESCRIPTION
        This function generates an HTML report displaying break glass accounts and the Conditional Access policies from which they are not excluded.
        The report uses existing HTML, CSS, and JavaScript templates.
 
        .PARAMETER AssessmentResults
        The results of the Break Glass Assessment from Invoke-CAIQBreakGlassAssessment.
 
        .PARAMETER OutputPath
        The path where the HTML report will be saved.
 
        .PARAMETER Title
        The title of the HTML report.
 
        .PARAMETER FileName
        The name of the HTML report file.
 
        .EXAMPLE
        New-CAIQBreakGlassExclusionDashboard -DataSet $results -OutputPath "C:\Reports"
 
        .INPUTS
        System.Object
 
        .OUTPUTS
        System.Object
     
    #>

    [CmdletBinding()]
    [OutputType([System.Object])]
    param (
        [Parameter(Mandatory=$true)]
        [object[]]$DataSet,
        [Parameter(Mandatory=$true)]
        [string]$OutputPath,
        [Parameter(Mandatory=$false)]
        [string]$Title = "Break Glass Account CA Policy Exclusion Assessment",
        [Parameter(Mandatory=$false)]
        [string]$FileName = "BreakGlass_ConditionalAccessIQ_Exclusion_Assessment.html"
    
    )
    Begin {
        # Initialize the output object
        $output_obj = [ordered]@{}
        
        # Get tenant info
        $tenant_info = Get-CAIQTenantInfo
        
        # Get templates
        Try {
            # Load the templates directly from the template path
            $html_template = $template_manager.GetTemplate("breakglass.html")
            $breakglass_styles = $template_manager.GetTemplate("breakglass_styles.css")
            $breakglass_script = $template_manager.GetTemplate("breakglass_script.js")

        } Catch {
            Write-Error "Error with templates: $_" -ErrorAction Stop
        
        }
        
        # Group assessment results by break glass account
        $grouped_results = $dataSet | Group-Object -Property BreakGlassAccount

        
        # Build tab buttons
        $tab_buttons = [System.Text.StringBuilder]::new()
        foreach ($group in $grouped_results) {
            $account_name = $group.Name
            $account_id = $account_name.Replace("@", "_").Replace(".", "_")
            [void]$tab_buttons.AppendLine("<button class=`"tablinks`" onclick=`"openTab(event, '$account_id')`">$account_name</button>")
        
        }
        
        # Build tab content
        $tab_content = [System.Text.StringBuilder]::new()

        # Initialize the total issues counter
        $total_issues = 0

        $policies_count = @($grouped_results[0].Group).Count
        
        foreach ($group in $grouped_results) {
            # Get the account name and ID
            $account_name = $group.Name

            # Get the account ID
            $account_id = $account_name.Replace("@", "_").Replace(".", "_")

            # Get the policies missing exclusions
            $policies_missing_exclusions = $group.Group | Where-Object {!$_.ExcludedFromPolicy}

            # Get the count of policies missing exclusions
            $missing_exclusions_count = @($policies_missing_exclusions).Count

            # Increment the total issues counter
            $total_issues += $missing_exclusions_count
            
            # Build the tab content
            [void]$tab_content.AppendLine("<div id=`"$account_id`" class=`"tabcontent`">")
            [void]$tab_content.AppendLine("<div class=`"account-info`">")
            [void]$tab_content.AppendLine("<h3>Account: $account_name</h3>")
            [void]$tab_content.AppendLine("<p>Number of policies without proper exclusion: $missing_exclusions_count</p>")
            [void]$tab_content.AppendLine("</div>")
            
            # If there are policies missing exclusions, build the tab content
            if ($policies_missing_exclusions) {
                [void]$tab_content.AppendLine("<h3>Policies Missing Exclusions</h3>")
                [void]$tab_content.AppendLine("<p>The following Conditional Access policies do not exclude this break glass account:</p>")
                [void]$tab_content.AppendLine("<div class=`"policy-cards`">")
                
                # Build the policy cards
                foreach ($policy in $policies_missing_exclusions) {
                    $policy_url = "https://entra.microsoft.com/#view/Microsoft_AAD_ConditionalAccess/PolicyBlade/policyId/$($policy.Id)"
                    [void]$tab_content.AppendLine("<div class=`"policy-card`">")
                    [void]$tab_content.AppendLine("<h3>$($policy.DisplayName)</h3>")
                    [void]$tab_content.AppendLine("<div class=`"policy-id`">ID: $($policy.Id)</div>")
                    
                    # Add status
                    If ($policy.State -eq "enabled") {
                        [void]$tab_content.AppendLine("<div class=`"policy-status`">Enabled</div>")
                    
                    } Else {
                        [void]$tab_content.AppendLine("<div class=`"policy-status`" style=`"background-color: #888;`">$($policy.State)</div>")
                    
                    }

                    # Add link to policy
                    [void]$tab_content.AppendLine("<a href=`"$policy_url`" target=`"_blank`">View Policy in Entra Portal</a>")
                    [void]$tab_content.AppendLine("</div>")
                
                }
                # Close policy cards
                [void]$tab_content.AppendLine("</div>")
            } Else {
                [void]$tab_content.AppendLine("<p>This account is properly excluded from all Conditional Access policies.</p>")
            
            }
            # Close tab content
            [void]$tab_content.AppendLine("</div>")
        
        }
        
        # Determine overall status
        $overall_status = "Good"
        $overall_status_class = "status-good"
        $summary_text = "All break glass accounts are properly excluded from Conditional Access policies."
        
        if ($total_issues) {
            $overall_status = "Critical"
            $overall_status_class = "status-critical"
            $summary_text = "One or more break glass accounts are not excluded from all Conditional Access policies."
        }
        
        $issues_class = if ($total_issues) { 
            "status-critical" 
        } Else { 
            "status-good" 
        
        }
    
    } Process {
        # Create tokens for replacement
        $tokens = @{}
        $tokens["TITLE"] = $title
        $tokens["TENANT_NAME"] = "$($tenant_info.DisplayName) ($($tenant_info.Id))"
        $tokens["STYLES"] = "<style>$breakglass_styles</style>"
        $tokens["SCRIPTS"] = "<script>$breakglass_script</script>"
        $tokens["OVERALL_STATUS_CLASS"] = $overall_status_class
        $tokens["OVERALL_STATUS"] = $overall_status
        $tokens["SUMMARY_TEXT"] = $summary_text
        $tokens["TOTAL_POLICIES"] = $policies_count
        $tokens["TOTAL_ACCOUNTS"] = @($grouped_results).Count
        $tokens["TOTAL_ISSUES"] = $total_issues
        $tokens["ISSUES_CLASS"] = $issues_class
        $tokens["TAB_BUTTONS"] = $tab_buttons.ToString()
        $tokens["TAB_CONTENT"] = $tab_content.ToString()
        $tokens["FOOTER"] = $footer_template
                
        # Replace all tokens in the HTML
        $html_content = $html_template

        # Replace all tokens in the HTML
        foreach ($key in $tokens.Keys) {
            $html_content = $html_content -replace "{{$key}}", $tokens[$key]
        }
        
        # Create the full path for the HTML file
        $report_path = Join-Path -Path $outputPath -ChildPath $FileName
        
        # Write the HTML file
        $html_content | Out-File -FilePath $report_path -Encoding utf8
        
        # Add the HTML file path to the output object
        $output_obj["Path"] = $report_path
        $output_obj["Html"] = $html_content
    
    } End {
        # Return the output object
        [PSCustomObject]$output_obj
    
    }
}