Functions/Show-ADPrincipalCertificate.ps1

<#
 
.SYNOPSIS
    Enumerate Active Directory principals that have certificates attached.
 
.DESCRIPTION
    This function retrieves AD principals (users, computers, or service accounts) that have certificates attached. The user can filter by principal type or specify a security group to check. Only principals with certificates are returned.
 
.PARAMETER PrincipalType
    Specifies the type of principals to enumerate. Acceptable values are 'All', 'User', 'Computer', 'ServiceAccount', or 'Group'. Default is 'All'. The -GroupName parameter is required when PrincipalType is set to 'Group'.
 
.PARAMETER GroupName
    The name of the security group to check. Required when PrincipalType is set to 'Group'.
 
.PARAMETER ExcludeSelfSignedCertificate
    When specified, excludes principals that only have self-signed certificates (where Subject equals Issuer).
 
.INPUTS
    None.
 
.OUTPUTS
    Custom objects containing AccountName and AccountType properties for principals with certificates.
 
.EXAMPLE
    Show-ADPrincipalCertificate
 
    Returns all AD principals (users, computers, and service accounts) that have certificates attached.
 
.EXAMPLE
    Show-ADPrincipalCertificate -PrincipalType All
 
    Returns all AD principals (users, computers, and service accounts) that have certificates attached.
 
.EXAMPLE
    Show-ADPrincipalCertificate -PrincipalType User
 
    Returns only user accounts that have certificates attached.
 
.EXAMPLE
    Show-ADPrincipalCertificate -PrincipalType Computer
 
    Returns only computer accounts that have certificates attached.
 
.EXAMPLE
    Show-ADPrincipalCertificate -PrincipalType ServiceAccount
 
    Returns only service accounts that have certificates attached.
 
.EXAMPLE
    Show-ADPrincipalCertificate -PrincipalType Group -GroupName 'Management Workstations'
 
    Returns members of the 'Management Workstations' group that have certificates attached.
 
.EXAMPLE
    Show-ADPrincipalCertificate -ExcludeSelfSignedCertificate
 
    Returns all AD principals that have certificates attached, excluding those with only self-signed certificates.
 
.EXAMPLE
    Show-ADPrincipalCertificate -PrincipalType User -ExcludeSelfSignedCertificate
 
    Returns user accounts that have certificates attached, excluding those with only self-signed certificates.
 
.LINK
    https://github.com/richardhicks/adprincipalcertificate/blob/main/Functions/Show-ADPrincipalCertificate.ps1
 
.LINK
    https://www.richardhicks.com/
 
.NOTES
    Version: 1.0
    Creation Date: February 7, 2026
    Last Updated: February 7, 2026
    Author: Richard Hicks
    Organization: Richard M. Hicks Consulting, Inc.
    Contact: rich@richardhicks.com
    Website: https://www.richardhicks.com/
 
#>


Function Show-ADPrincipalCertificate {

    # Prerequisites
    #Requires -Module ActiveDirectory

    [CmdletBinding()]

    Param (

        [Parameter(HelpMessage = 'Specify the type of principals to enumerate')]
        [ValidateSet('All', 'User', 'Computer', 'ServiceAccount', 'Group')]
        [Alias('Type')]
        [string]$PrincipalType = 'All',
        [Parameter(HelpMessage = "Specify the security group name (required when PrincipalType is 'Group')")]
        [string]$GroupName,
        [Parameter(HelpMessage = 'Exclude principals with only self-signed certificates')]
        [switch]$ExcludeSelfSignedCertificate

    )

    # Validate that GroupName is provided when PrincipalType is Group
    If ($PrincipalType -eq 'Group' -and [string]::IsNullOrWhiteSpace($GroupName)) {

        Write-Warning "The -GroupName parameter is required when PrincipalType is set to 'Group'."
        Return

    }

    # Function to process and output principals with certificates
    Function Get-PrincipalWithCertificate {

        [CmdletBinding()]

        Param (

            [Parameter(Mandatory)]
            [array]$Principals,
            [switch]$ExcludeSelfSigned

        )

        $Counter = 0
        $Total = $Principals.Count

        ForEach ($Principal in $Principals) {

            $Counter++

            # Show progress for better user experience
            If ($Total -gt 10) {

                Write-Progress -Activity "Checking principals for certificates" `
                    -Status "Processing $Counter of $Total" `
                    -PercentComplete (($Counter / $Total) * 100)

            }

            # Check if userCertificate attribute exists and has values
            If ($Principal.userCertificate.Count -gt 0) {

                # If ExcludeSelfSigned is specified, check if the principal has any non-self-signed certificates
                If ($ExcludeSelfSigned) {

                    $HasNonSelfSignedCert = $False

                    ForEach ($CertBytes in $Principal.userCertificate) {

                        Try {

                            $Cert = [System.Security.Cryptography.X509Certificates.X509Certificate2]::New($CertBytes)

                            # A certificate is self-signed if Subject equals Issuer
                            If ($Cert.Subject -ne $Cert.Issuer) {

                                $HasNonSelfSignedCert = $True
                                Break

                            }

                        }

                        Catch {

                            Write-Verbose "Unable to parse certificate for $($Principal.SamAccountName): $_"

                        }

                    }

                    # Skip this principal if all certificates are self-signed
                    If (-not $HasNonSelfSignedCert) {

                        Continue

                    }

                }

                # Output directly to pipeline (no array building)
                [PSCustomObject]@{

                    AccountName = $Principal.SamAccountName
                    AccountType = $Principal.ObjectClass -replace '^msDS-', ''

                }

            }

        }

        If ($Total -gt 10) {

            Write-Progress -Activity "Checking principals for certificates" -Completed

        }

    }

    # Main script logic
    Try {

        Write-Verbose "Retrieving AD principals with certificate information..."

        # Properties to retrieve
        $Properties = @('SamAccountName', 'userCertificate', 'ObjectClass', 'DistinguishedName')

        Switch ($PrincipalType) {

            'All' {

                # Retrieve principals with certificate property in single query (more efficient)
                Write-Verbose "Enumerating all principals (users, computers, and service accounts)..."
                $Users = Get-ADUser -Filter * -Properties $Properties
                $Computers = Get-ADComputer -Filter * -Properties $Properties
                $ServiceAccounts = Get-ADServiceAccount -Filter * -Properties $Properties
                $Principals = $Users + $Computers + $ServiceAccounts

            }

            'User' {

                Write-Verbose "Enumerating user accounts..."
                $Principals = Get-ADUser -Filter * -Properties $Properties

            }

            'Computer' {

                Write-Verbose "Enumerating computer accounts..."
                $Principals = Get-ADComputer -Filter * -Properties $Properties

            }

            'ServiceAccount' {

                Write-Verbose "Enumerating service accounts..."
                $Principals = Get-ADServiceAccount -Filter * -Properties $Properties

            }

            'Group' {

                Write-Verbose "Enumerating members of group '$GroupName'..."
                Try {

                    # Verify group exists
                    $Group = Get-ADGroup -Identity $GroupName -ErrorAction Stop
                    $Members = Get-ADGroupMember -Identity $Group -Recursive -ErrorAction Stop

                    # Filter to users, computers, and service accounts, then get full objects with certificate property
                    $UserMembers = $Members | Where-Object { $_.objectClass -eq 'user' }
                    $ComputerMembers = $Members | Where-Object { $_.objectClass -eq 'computer' }
                    $ServiceAccountMembers = $Members | Where-Object { $_.objectClass -eq 'msDS-ManagedServiceAccount' -or $_.objectClass -eq 'msDS-GroupManagedServiceAccount' }

                    # Build principals array
                    $Principals = @(

                        # Get full AD objects with properties for users
                        If ($UserMembers.Count -gt 0) {

                            $UserMembers | ForEach-Object {

                                Get-ADUser -Identity $_.DistinguishedName -Properties $Properties

                            }

                        }

                        # Get full AD objects with properties for computers
                        If ($ComputerMembers.Count -gt 0) {

                            $ComputerMembers | ForEach-Object {

                                Get-ADComputer -Identity $_.DistinguishedName -Properties $Properties

                            }

                        }

                        # Get full AD objects with properties for service accounts
                        If ($ServiceAccountMembers.Count -gt 0) {

                            $ServiceAccountMembers | ForEach-Object {

                                Get-ADServiceAccount -Identity $_.DistinguishedName -Properties $Properties

                            }

                        }

                    )

                    If ($Principals.Count -eq 0) {

                        Write-Warning "No user, computer, or service account principals found in group '$GroupName'."
                        Return

                    }

                }

                Catch {

                    Write-Warning "Error retrieving group '$GroupName': $_"
                    Exit 1

                }

            }

        }

        If ($Principals.Count -eq 0) {

            Write-Warning 'No principals found matching the specified criteria.'
            Return

        }

        Write-Verbose "Found $($Principals.Count) principal(s). Checking for certificates..."

        # Process principals and output those with certificates directly to pipeline. Pipeline automatically sorts and formats results
        $Results = Get-PrincipalWithCertificate -Principals $Principals -ExcludeSelfSigned:$ExcludeSelfSignedCertificate | Sort-Object -Property AccountType, AccountName

        # Warn if no principals with certificates were found
        If (-not $Results) {

            Write-Warning 'No certificates found for any of the specified principals.'

        }

        Else {

            $Results

        }

    }

    Catch {

        Write-Warning "Operation failed: $($_.Exception.Message)"
        Throw $_

    }

}

# SIG # Begin signature block
# MIIf2gYJKoZIhvcNAQcCoIIfyzCCH8cCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCT+wtx3ruiyXwX
# JroC2Z83CGkt9azfTyP7rzAVFSQ8laCCGpkwggNZMIIC36ADAgECAhAPuKdAuRWN
# A1FDvFnZ8EApMAoGCCqGSM49BAMDMGExCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxE
# aWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xIDAeBgNVBAMT
# F0RpZ2lDZXJ0IEdsb2JhbCBSb290IEczMB4XDTIxMDQyOTAwMDAwMFoXDTM2MDQy
# ODIzNTk1OVowZDELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMu
# MTwwOgYDVQQDEzNEaWdpQ2VydCBHbG9iYWwgRzMgQ29kZSBTaWduaW5nIEVDQyBT
# SEEzODQgMjAyMSBDQTEwdjAQBgcqhkjOPQIBBgUrgQQAIgNiAAS7tKwnpUgNolNf
# jy6BPi9TdrgIlKKaqoqLmLWx8PwqFbu5s6UiL/1qwL3iVWhga5c0wWZTcSP8GtXK
# IA8CQKKjSlpGo5FTK5XyA+mrptOHdi/nZJ+eNVH8w2M1eHbk+HejggFXMIIBUzAS
# BgNVHRMBAf8ECDAGAQH/AgEAMB0GA1UdDgQWBBSbX7A2up0GrhknvcCgIsCLizh3
# 7TAfBgNVHSMEGDAWgBSz20ik+aHF2K42QcwRY2liKbxLxjAOBgNVHQ8BAf8EBAMC
# AYYwEwYDVR0lBAwwCgYIKwYBBQUHAwMwdgYIKwYBBQUHAQEEajBoMCQGCCsGAQUF
# BzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wQAYIKwYBBQUHMAKGNGh0dHA6
# Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEdsb2JhbFJvb3RHMy5jcnQw
# QgYDVR0fBDswOTA3oDWgM4YxaHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lD
# ZXJ0R2xvYmFsUm9vdEczLmNybDAcBgNVHSAEFTATMAcGBWeBDAEDMAgGBmeBDAEE
# ATAKBggqhkjOPQQDAwNoADBlAjB4vUmVZXEB0EZXaGUOaKncNgjB7v3UjttAZT8N
# /5Ovwq5jhqN+y7SRWnjsBwNnB3wCMQDnnx/xB1usNMY4vLWlUM7m6jh+PnmQ5KRb
# qwIN6Af8VqZait2zULLd8vpmdJ7QFmMwggP+MIIDhKADAgECAhANSjTahpCPwBMs
# vIE3k68kMAoGCCqGSM49BAMDMGQxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdp
# Q2VydCwgSW5jLjE8MDoGA1UEAxMzRGlnaUNlcnQgR2xvYmFsIEczIENvZGUgU2ln
# bmluZyBFQ0MgU0hBMzg0IDIwMjEgQ0ExMB4XDTI0MTIwNjAwMDAwMFoXDTI3MTIy
# NDIzNTk1OVowgYYxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpDYWxpZm9ybmlhMRYw
# FAYDVQQHEw1NaXNzaW9uIFZpZWpvMSQwIgYDVQQKExtSaWNoYXJkIE0uIEhpY2tz
# IENvbnN1bHRpbmcxJDAiBgNVBAMTG1JpY2hhcmQgTS4gSGlja3MgQ29uc3VsdGlu
# ZzBZMBMGByqGSM49AgEGCCqGSM49AwEHA0IABFCbtcqpc7vGGM4hVM79U+7f0tKz
# o8BAGMJ/0E7JUwKJfyMJj9jsCNpp61+mBNdTwirEm/K0Vz02vak0Ftcb/3yjggHz
# MIIB7zAfBgNVHSMEGDAWgBSbX7A2up0GrhknvcCgIsCLizh37TAdBgNVHQ4EFgQU
# KIMkVkfISNUyQJ7bwvLm9sCIkxgwPgYDVR0gBDcwNTAzBgZngQwBBAEwKTAnBggr
# BgEFBQcCARYbaHR0cDovL3d3dy5kaWdpY2VydC5jb20vQ1BTMA4GA1UdDwEB/wQE
# AwIHgDATBgNVHSUEDDAKBggrBgEFBQcDAzCBqwYDVR0fBIGjMIGgME6gTKBKhkho
# dHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRHbG9iYWxHM0NvZGVTaWdu
# aW5nRUNDU0hBMzg0MjAyMUNBMS5jcmwwTqBMoEqGSGh0dHA6Ly9jcmw0LmRpZ2lj
# ZXJ0LmNvbS9EaWdpQ2VydEdsb2JhbEczQ29kZVNpZ25pbmdFQ0NTSEEzODQyMDIx
# Q0ExLmNybDCBjgYIKwYBBQUHAQEEgYEwfzAkBggrBgEFBQcwAYYYaHR0cDovL29j
# c3AuZGlnaWNlcnQuY29tMFcGCCsGAQUFBzAChktodHRwOi8vY2FjZXJ0cy5kaWdp
# Y2VydC5jb20vRGlnaUNlcnRHbG9iYWxHM0NvZGVTaWduaW5nRUNDU0hBMzg0MjAy
# MUNBMS5jcnQwCQYDVR0TBAIwADAKBggqhkjOPQQDAwNoADBlAjBMOsBb80qx6E6S
# 2lnnHafuyY2paoDtPjcfddKaB1HKnAy7WLaEVc78xAC84iW3l6ECMQDhOPD5JHtw
# YxEH6DxVDle5pLKfuyQHiY1i0I9PrSn1plPUeZDTnYKmms1P66nBvCkwggWNMIIE
# daADAgECAhAOmxiO+dAt5+/bUOIIQBhaMA0GCSqGSIb3DQEBDAUAMGUxCzAJBgNV
# BAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdp
# Y2VydC5jb20xJDAiBgNVBAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9vdCBDQTAe
# Fw0yMjA4MDEwMDAwMDBaFw0zMTExMDkyMzU5NTlaMGIxCzAJBgNVBAYTAlVTMRUw
# EwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20x
# ITAfBgNVBAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDCCAiIwDQYJKoZIhvcN
# AQEBBQADggIPADCCAgoCggIBAL/mkHNo3rvkXUo8MCIwaTPswqclLskhPfKK2FnC
# 4SmnPVirdprNrnsbhA3EMB/zG6Q4FutWxpdtHauyefLKEdLkX9YFPFIPUh/GnhWl
# fr6fqVcWWVVyr2iTcMKyunWZanMylNEQRBAu34LzB4TmdDttceItDBvuINXJIB1j
# KS3O7F5OyJP4IWGbNOsFxl7sWxq868nPzaw0QF+xembud8hIqGZXV59UWI4MK7dP
# pzDZVu7Ke13jrclPXuU15zHL2pNe3I6PgNq2kZhAkHnDeMe2scS1ahg4AxCN2NQ3
# pC4FfYj1gj4QkXCrVYJBMtfbBHMqbpEBfCFM1LyuGwN1XXhm2ToxRJozQL8I11pJ
# pMLmqaBn3aQnvKFPObURWBf3JFxGj2T3wWmIdph2PVldQnaHiZdpekjw4KISG2aa
# dMreSx7nDmOu5tTvkpI6nj3cAORFJYm2mkQZK37AlLTSYW3rM9nF30sEAMx9HJXD
# j/chsrIRt7t/8tWMcCxBYKqxYxhElRp2Yn72gLD76GSmM9GJB+G9t+ZDpBi4pncB
# 4Q+UDCEdslQpJYls5Q5SUUd0viastkF13nqsX40/ybzTQRESW+UQUOsxxcpyFiIJ
# 33xMdT9j7CFfxCBRa2+xq4aLT8LWRV+dIPyhHsXAj6KxfgommfXkaS+YHS312amy
# HeUbAgMBAAGjggE6MIIBNjAPBgNVHRMBAf8EBTADAQH/MB0GA1UdDgQWBBTs1+OC
# 0nFdZEzfLmc/57qYrhwPTzAfBgNVHSMEGDAWgBRF66Kv9JLLgjEtUYunpyGd823I
# DzAOBgNVHQ8BAf8EBAMCAYYweQYIKwYBBQUHAQEEbTBrMCQGCCsGAQUFBzABhhho
# dHRwOi8vb2NzcC5kaWdpY2VydC5jb20wQwYIKwYBBQUHMAKGN2h0dHA6Ly9jYWNl
# cnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcnQwRQYD
# VR0fBD4wPDA6oDigNoY0aHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0
# QXNzdXJlZElEUm9vdENBLmNybDARBgNVHSAECjAIMAYGBFUdIAAwDQYJKoZIhvcN
# AQEMBQADggEBAHCgv0NcVec4X6CjdBs9thbX979XB72arKGHLOyFXqkauyL4hxpp
# VCLtpIh3bb0aFPQTSnovLbc47/T/gLn4offyct4kvFIDyE7QKt76LVbP+fT3rDB6
# mouyXtTP0UNEm0Mh65ZyoUi0mcudT6cGAxN3J0TU53/oWajwvy8LpunyNDzs9wPH
# h6jSTEAZNUZqaVSwuKFWjuyk1T3osdz9HNj0d1pcVIxv76FQPfx2CWiEn2/K2yCN
# NWAcAgPLILCsWKAOQGPFmCLBsln1VWvPJ6tsds5vIy30fnFqI2si/xK4VC0nftg6
# 2fC2h5b9W9FcrBjDTZ9ztwGpn1eqXijiuZQwgga0MIIEnKADAgECAhANx6xXBf8h
# mS5AQyIMOkmGMA0GCSqGSIb3DQEBCwUAMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQK
# EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNV
# BAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDAeFw0yNTA1MDcwMDAwMDBaFw0z
# ODAxMTQyMzU5NTlaMGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwg
# SW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBUaW1lU3RhbXBpbmcg
# UlNBNDA5NiBTSEEyNTYgMjAyNSBDQTEwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAw
# ggIKAoICAQC0eDHTCphBcr48RsAcrHXbo0ZodLRRF51NrY0NlLWZloMsVO1DahGP
# NRcybEKq+RuwOnPhof6pvF4uGjwjqNjfEvUi6wuim5bap+0lgloM2zX4kftn5B1I
# pYzTqpyFQ/4Bt0mAxAHeHYNnQxqXmRinvuNgxVBdJkf77S2uPoCj7GH8BLuxBG5A
# vftBdsOECS1UkxBvMgEdgkFiDNYiOTx4OtiFcMSkqTtF2hfQz3zQSku2Ws3IfDRe
# b6e3mmdglTcaarps0wjUjsZvkgFkriK9tUKJm/s80FiocSk1VYLZlDwFt+cVFBUR
# Jg6zMUjZa/zbCclF83bRVFLeGkuAhHiGPMvSGmhgaTzVyhYn4p0+8y9oHRaQT/ao
# fEnS5xLrfxnGpTXiUOeSLsJygoLPp66bkDX1ZlAeSpQl92QOMeRxykvq6gbylsXQ
# skBBBnGy3tW/AMOMCZIVNSaz7BX8VtYGqLt9MmeOreGPRdtBx3yGOP+rx3rKWDEJ
# lIqLXvJWnY0v5ydPpOjL6s36czwzsucuoKs7Yk/ehb//Wx+5kMqIMRvUBDx6z1ev
# +7psNOdgJMoiwOrUG2ZdSoQbU2rMkpLiQ6bGRinZbI4OLu9BMIFm1UUl9VnePs6B
# aaeEWvjJSjNm2qA+sdFUeEY0qVjPKOWug/G6X5uAiynM7Bu2ayBjUwIDAQABo4IB
# XTCCAVkwEgYDVR0TAQH/BAgwBgEB/wIBADAdBgNVHQ4EFgQU729TSunkBnx6yuKQ
# VvYv1Ensy04wHwYDVR0jBBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4cD08wDgYDVR0P
# AQH/BAQDAgGGMBMGA1UdJQQMMAoGCCsGAQUFBwMIMHcGCCsGAQUFBwEBBGswaTAk
# BggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEEGCCsGAQUFBzAC
# hjVodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9v
# dEc0LmNydDBDBgNVHR8EPDA6MDigNqA0hjJodHRwOi8vY3JsMy5kaWdpY2VydC5j
# b20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNybDAgBgNVHSAEGTAXMAgGBmeBDAEE
# AjALBglghkgBhv1sBwEwDQYJKoZIhvcNAQELBQADggIBABfO+xaAHP4HPRF2cTC9
# vgvItTSmf83Qh8WIGjB/T8ObXAZz8OjuhUxjaaFdleMM0lBryPTQM2qEJPe36zwb
# SI/mS83afsl3YTj+IQhQE7jU/kXjjytJgnn0hvrV6hqWGd3rLAUt6vJy9lMDPjTL
# xLgXf9r5nWMQwr8Myb9rEVKChHyfpzee5kH0F8HABBgr0UdqirZ7bowe9Vj2AIMD
# 8liyrukZ2iA/wdG2th9y1IsA0QF8dTXqvcnTmpfeQh35k5zOCPmSNq1UH410ANVk
# o43+Cdmu4y81hjajV/gxdEkMx1NKU4uHQcKfZxAvBAKqMVuqte69M9J6A47OvgRa
# Ps+2ykgcGV00TYr2Lr3ty9qIijanrUR3anzEwlvzZiiyfTPjLbnFRsjsYg39OlV8
# cipDoq7+qNNjqFzeGxcytL5TTLL4ZaoBdqbhOhZ3ZRDUphPvSRmMThi0vw9vODRz
# W6AxnJll38F0cuJG7uEBYTptMSbhdhGQDpOXgpIUsWTjd6xpR6oaQf/DJbg3s6KC
# LPAlZ66RzIg9sC+NJpud/v4+7RWsWCiKi9EOLLHfMR2ZyJ/+xhCx9yHbxtl5TPau
# 1j/1MIDpMPx0LckTetiSuEtQvLsNz3Qbp7wGWqbIiOWCnb5WqxL3/BAPvIXKUjPS
# xyZsq8WhbaM2tszWkPZPubdcMIIG7TCCBNWgAwIBAgIQCoDvGEuN8QWC0cR2p5V0
# aDANBgkqhkiG9w0BAQsFADBpMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGlnaUNl
# cnQsIEluYy4xQTA/BgNVBAMTOERpZ2lDZXJ0IFRydXN0ZWQgRzQgVGltZVN0YW1w
# aW5nIFJTQTQwOTYgU0hBMjU2IDIwMjUgQ0ExMB4XDTI1MDYwNDAwMDAwMFoXDTM2
# MDkwMzIzNTk1OVowYzELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJ
# bmMuMTswOQYDVQQDEzJEaWdpQ2VydCBTSEEyNTYgUlNBNDA5NiBUaW1lc3RhbXAg
# UmVzcG9uZGVyIDIwMjUgMTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIB
# ANBGrC0Sxp7Q6q5gVrMrV7pvUf+GcAoB38o3zBlCMGMyqJnfFNZx+wvA69HFTBdw
# bHwBSOeLpvPnZ8ZN+vo8dE2/pPvOx/Vj8TchTySA2R4QKpVD7dvNZh6wW2R6kSu9
# RJt/4QhguSssp3qome7MrxVyfQO9sMx6ZAWjFDYOzDi8SOhPUWlLnh00Cll8pjrU
# cCV3K3E0zz09ldQ//nBZZREr4h/GI6Dxb2UoyrN0ijtUDVHRXdmncOOMA3CoB/iU
# SROUINDT98oksouTMYFOnHoRh6+86Ltc5zjPKHW5KqCvpSduSwhwUmotuQhcg9tw
# 2YD3w6ySSSu+3qU8DD+nigNJFmt6LAHvH3KSuNLoZLc1Hf2JNMVL4Q1OpbybpMe4
# 6YceNA0LfNsnqcnpJeItK/DhKbPxTTuGoX7wJNdoRORVbPR1VVnDuSeHVZlc4seA
# O+6d2sC26/PQPdP51ho1zBp+xUIZkpSFA8vWdoUoHLWnqWU3dCCyFG1roSrgHjSH
# lq8xymLnjCbSLZ49kPmk8iyyizNDIXj//cOgrY7rlRyTlaCCfw7aSUROwnu7zER6
# EaJ+AliL7ojTdS5PWPsWeupWs7NpChUk555K096V1hE0yZIXe+giAwW00aHzrDch
# Ic2bQhpp0IoKRR7YufAkprxMiXAJQ1XCmnCfgPf8+3mnAgMBAAGjggGVMIIBkTAM
# BgNVHRMBAf8EAjAAMB0GA1UdDgQWBBTkO/zyMe39/dfzkXFjGVBDz2GM6DAfBgNV
# HSMEGDAWgBTvb1NK6eQGfHrK4pBW9i/USezLTjAOBgNVHQ8BAf8EBAMCB4AwFgYD
# VR0lAQH/BAwwCgYIKwYBBQUHAwgwgZUGCCsGAQUFBwEBBIGIMIGFMCQGCCsGAQUF
# BzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wXQYIKwYBBQUHMAKGUWh0dHA6
# Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRHNFRpbWVTdGFt
# cGluZ1JTQTQwOTZTSEEyNTYyMDI1Q0ExLmNydDBfBgNVHR8EWDBWMFSgUqBQhk5o
# dHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkRzRUaW1lU3Rh
# bXBpbmdSU0E0MDk2U0hBMjU2MjAyNUNBMS5jcmwwIAYDVR0gBBkwFzAIBgZngQwB
# BAIwCwYJYIZIAYb9bAcBMA0GCSqGSIb3DQEBCwUAA4ICAQBlKq3xHCcEua5gQezR
# CESeY0ByIfjk9iJP2zWLpQq1b4URGnwWBdEZD9gBq9fNaNmFj6Eh8/YmRDfxT7C0
# k8FUFqNh+tshgb4O6Lgjg8K8elC4+oWCqnU/ML9lFfim8/9yJmZSe2F8AQ/UdKFO
# tj7YMTmqPO9mzskgiC3QYIUP2S3HQvHG1FDu+WUqW4daIqToXFE/JQ/EABgfZXLW
# U0ziTN6R3ygQBHMUBaB5bdrPbF6MRYs03h4obEMnxYOX8VBRKe1uNnzQVTeLni2n
# HkX/QqvXnNb+YkDFkxUGtMTaiLR9wjxUxu2hECZpqyU1d0IbX6Wq8/gVutDojBIF
# eRlqAcuEVT0cKsb+zJNEsuEB7O7/cuvTQasnM9AWcIQfVjnzrvwiCZ85EE8LUkqR
# hoS3Y50OHgaY7T/lwd6UArb+BOVAkg2oOvol/DJgddJ35XTxfUlQ+8Hggt8l2Yv7
# roancJIFcbojBcxlRcGG0LIhp6GvReQGgMgYxQbV1S3CrWqZzBt1R9xJgKf47Cdx
# VRd/ndUlQ05oxYy2zRWVFjF7mcr4C34Mj3ocCVccAvlKV9jEnstrniLvUxxVZE/r
# ptb7IRE2lskKPIJgbaP5t2nGj/ULLi49xTcBZU8atufk+EMF/cWuiC7POGT75qaL
# 6vdCvHlshtjdNXOCIUjsarfNZzGCBJcwggSTAgEBMHgwZDELMAkGA1UEBhMCVVMx
# FzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMTwwOgYDVQQDEzNEaWdpQ2VydCBHbG9i
# YWwgRzMgQ29kZSBTaWduaW5nIEVDQyBTSEEzODQgMjAyMSBDQTECEA1KNNqGkI/A
# Eyy8gTeTryQwDQYJYIZIAWUDBAIBBQCggYQwGAYKKwYBBAGCNwIBDDEKMAigAoAA
# oQKAADAZBgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4w
# DAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQgV6ijwvZ4SQVFo6goNo8emLBK
# OyXTxi1xg4bkNr1cmLwwCwYHKoZIzj0CAQUABEcwRQIhAIicDrj67JrODSmq6E8s
# eIHFU52jZD3U1wr4W6TE+uBBAiBqwIHZiLTSCo+G6X4LFBt1Kw9kThF1m22AxbKd
# aryauqGCAyYwggMiBgkqhkiG9w0BCQYxggMTMIIDDwIBATB9MGkxCzAJBgNVBAYT
# AlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQg
# VHJ1c3RlZCBHNCBUaW1lU3RhbXBpbmcgUlNBNDA5NiBTSEEyNTYgMjAyNSBDQTEC
# EAqA7xhLjfEFgtHEdqeVdGgwDQYJYIZIAWUDBAIBBQCgaTAYBgkqhkiG9w0BCQMx
# CwYJKoZIhvcNAQcBMBwGCSqGSIb3DQEJBTEPFw0yNjAyMDcyMTM1NTFaMC8GCSqG
# SIb3DQEJBDEiBCDft3nv9fJBkduqUaRM9NxtgYpoIGftCx9enLbwEZ6u/jANBgkq
# hkiG9w0BAQEFAASCAgBN1QBLqZZGho07AlXAyKwwDRMo05WcldazAtLG1qGT8W/r
# sQzH0pL3IvBJGWlQFZMr/xaFh81TnFkDpQZVoaqT33BbV27le+TNwNgdCvZQbgGy
# mDlZRPxiIhQhLOrPsY06XIM5c05HB1FLvL/S7AhfI8bJbmX4tUcQY2kKUAlMCnWa
# CcYDqKXKnSuBbE2ihyOVAz2OKfrUVZ9sKkRoMc8h8Kki4u7yCy2oWYFi2f0qqj+x
# JBNGJZBGM+BBAGvVhD9u3wtaRdp9Vn8jDTCkUTY3SMxRm9Rtm9fxfSaGXZuZpb63
# gZVjdQ3BtW5zbsv4s9dzsEMKk9Lap0tACp6xl3+9LO+uOgoGr+Usyt/utdE7MOlH
# zNHsMZ5lN8prbVczJ/nYesG5ZfyxnnXfcy1M/xSmalh82lqKaFwvm3VYRt/KxEhm
# CvTib+lP+s+lKtASy3SfmNo1/oOwjCA1yHvjGzXwfuhFqaCuGajXeQFdn1Xamdf2
# w+2X1Dz+nAScRMimH3wFVt8bySzUI76pERnbvZ+mK/M7VaZJbGg60zr67s7fLX6A
# W565vf7OvfS0ENSZ7KbHZNnlDK8km+k+pqVp9BCf3Uyl2ob9WRB5MxU+fRPUbsEK
# KB5/BiGqHY/SuHUQ8CBOJr8PUvAjEwmlTaEjgOjzBjzoZGenlbVpsovSBw8Fpg==
# SIG # End signature block