Obs/bin/ObsDep/content/Powershell/Roles/Common/Servicing/Scripts/Modules/FolderHash.psm1

#region C# code
$FolderHashSource = @"
using System;
using System.Collections.Generic;
using System.IO;
using System.Runtime.Serialization.Formatters.Binary;
 
namespace Microsoft.Solutions.IO
{
    [Serializable]
    public class FolderHashEntry : IEquatable<FolderHashEntry>
    {
        public string Path { get; private set; }
        public long LastWriteTimeUTCTicks { get; private set; }
        public int Size { get; private set; }
        public string Hash { get; private set; }
 
        public FolderHashEntry(string path, long lastWriteTimeUTCTicks, int size, string hash)
        {
            Path = path;
            LastWriteTimeUTCTicks = lastWriteTimeUTCTicks;
            Size = size;
            Hash = hash;
        }
 
        public void UpdateHash(string hash)
        {
            Hash = hash;
        }
 
        public override int GetHashCode() { return Path.GetHashCode(); }
 
        public override bool Equals(object obj)
        {
            if (obj == null) return false;
            FolderHashEntry fhe = obj as FolderHashEntry;
            if (fhe == null) return false;
            else return Equals(fhe);
        }
 
        public bool Equals(FolderHashEntry other)
        {
            if (other == null) { return false; }
            return (this.Path.Equals(other.Path));
        }
    }
 
    [Serializable]
    public class FolderHash
    {
        public string Algorithm { get; private set; }
        public List<FolderHashEntry> Entries { get; private set; }
 
        public FolderHash(string algorithm)
        {
            Algorithm = algorithm;
            Entries = new List<FolderHashEntry>();
        }
    }
}
"@


Add-Type -TypeDefinition $FolderHashSource -Language CSharp
#endregion C# code


<#
.Synopsis
    Perform a deep copy on an object
#>

function Invoke-DeepCopy ($object)
{
    $ms = $null
    $newObject = $null

    try
    {
        # Serialize and Deserialize data using BinaryFormatter
        $ms = New-Object System.IO.MemoryStream
        $bf = New-Object System.Runtime.Serialization.Formatters.Binary.BinaryFormatter
        $bf.Serialize($ms, $object)
        $ms.Position = 0
        $newObject = $bf.Deserialize($ms)
    }
    finally
    {
        if ($ms)
        {
            $ms.Dispose()
            $ms = $null
        }
    }

    $newObject
}


function Get-HashProviderFromString
{
    [CmdletBinding()]
    param (
        [ValidateSet('SHA1','SHA256', 'SHA384', 'SHA512')]
        [string]
        $Algorithm = 'SHA256'
    )

    $sha = $null

    switch ($Algorithm)
    {
        'SHA1'
        {
            $sha = New-Object -TypeName System.Security.Cryptography.SHA1CryptoServiceProvider
        }

        'SHA256'
        {
            $sha = New-Object -TypeName System.Security.Cryptography.SHA256CryptoServiceProvider
        }

        'SHA384'
        {
            $sha = New-Object -TypeName System.Security.Cryptography.SHA384CryptoServiceProvider
        }

        'SHA512'
        {
            $sha = New-Object -TypeName System.Security.Cryptography.SHA512CryptoServiceProvider
        }
    }
    
    return $sha
}

<#
.Synopsis
    For a given Path and Algorithm, returns a hash for the file.
    We write our own function because build machines are v3.0 and Get-FileHash in PowerShell is v4.0.
#>

function Get-FileHash
{
    [CmdletBinding()]
    param (
        [Parameter(Mandatory=$true)]
        [string[]]
        $Path,

        [ValidateSet('SHA1', 'SHA256', 'SHA384', 'SHA512')]
        [string]
        $Algorithm = 'SHA256'
    )

    $sha = Get-HashProviderFromString -Algorithm $Algorithm

    $file = [System.IO.File]::Open($Path,[System.IO.Filemode]::Open, [System.IO.FileAccess]::Read)
    [System.BitConverter]::ToString($sha.ComputeHash($file)) -replace "-",""
    $file.Close()
}

<#
.Synopsis
    For a given path, returns a Microsoft.Solutions.IO.FolderHash object with an entry for each file found.
#>

function Get-FolderHash
{
    [CmdletBinding()]
    param (
        [Parameter(Mandatory=$true)]
        [string[]]
        $Path,

        [string]
        $Filter,

        [string[]]
        $Exclude,

        [string[]]
        $Include,

        [switch]
        $Recurse,

        [switch]
        $NoHash,

        [switch]
        $AllowSkippedFiles,

        [ValidateSet('SHA1', 'SHA256', 'SHA384', 'SHA512')]
        [string]
        $Algorithm = 'SHA256'
    )

    $folderHash = New-Object Microsoft.Solutions.IO.FolderHash -ArgumentList $Algorithm

    if (-not (Test-Path $Path))
    {
        throw "Path $Path does not exist or access denied."
    }

    $args = @{
        'Path' = $Path
        'Filter' = $Filter
        'Exclude' = $Exclude
        'Include' = $Include
        'Recurse' = $Recurse
        'ErrorAction' = 'SilentlyContinue'
    }

    $files = Get-ChildItem @args -File

    foreach ($file in $files)
    {
        $fullName = $file.FullName
        Write-Verbose "Processing: $fullName."

        try 
        {
            # remove $Path from name so we store only the relative path
            $relativePath = $fullName -replace [regex]::Escape($($Path))
            $lastWriteTimeUTCTicks = $file.LastWriteTimeUtc.Ticks
            $size = $file.Length
            if ($NoHash)
            {
                $hash = [Guid]::Empty
            }
            else
            {
                $hash = Get-FileHash -Path $fullName -Algorithm $Algorithm -ErrorAction Stop
            }
            $entry = New-Object Microsoft.Solutions.IO.FolderHashEntry -ArgumentList $relativePath.TrimStart("\"), $lastWriteTimeUTCTicks, $size, $hash.Hash

            $folderHash.Entries.Add($entry)
        }
        catch [Microsoft.PowerShell.Commands.WriteErrorException]
        {
            if (-not $AllowSkippedFiles)
            {
                Write-Warning "WriteErrorException: $fullName."
                $ex = $_ | Select-Object *
                Write-Warning $ex
                throw
            }

            Write-Verbose "Skipping $fullName."
        }
        catch
        {
            Write-Warning "Error processing: $fullName."
            $ex = $_ | Select-Object *
            Write-Warning $ex
            throw
        }
    }

    $folderHash
}


<#
.Synopsis
    For a given path and Microsoft.Solutions.IO.FolderHash.Entries object or JSON file containing Entries,
    test each file and report files that are same, different, missing, extra files from the Entries list.
    Errors will be reported as skipped unless critical which will throw.
#>

function Test-FolderHash
{
    [CmdletBinding()]
    param (
        [Parameter(Mandatory=$true)]
        [string]
        $Path,

        [string]
        $Filter,

        [string[]]
        $Exclude,

        [string[]]
        $Include,

        [switch]
        $Recurse,

        [switch]
        $NoHash,

        [switch]
        $AllowSkippedFiles,

        [Parameter(ParameterSetName='Entry', Mandatory=$true)]
        [Microsoft.Solutions.IO.FolderHash]
        $FolderHash,

        [Parameter(ParameterSetName='JSON', Mandatory=$true)]
        [string]
        $JsonFile
    )

    if ($JsonFile)
    {
        Write-Verbose "Processing JSON: $JsonFile."
        $fh = Get-Content $JsonFile -Raw | ConvertFrom-Json
        $folderHashSrc = New-Object Microsoft.Solutions.IO.FolderHash -ArgumentList $fh.Algorithm
        $entries = $fh.Entries

        foreach ($entry in $entries) {
            $entry = New-Object Microsoft.Solutions.IO.FolderHashEntry -ArgumentList $entry.Path, $entry.LastWriteTimeUTCTicks, $entry.Size, $entry.Hash
            $folderHashSrc.Entries.Add($entry)
        }
    }
    else
    {
        Write-Verbose "Processing Microsoft.Solutions.IO.FolderHash object."
        # use a clone so we do not change the original object
        $folderHashSrc = Invoke-DeepCopy $FolderHash
    }

    Write-Verbose "Found $($folderHashSrc.Entries.Count) entries."

    Write-Verbose "Reading files from: $Path"
    # always call with -NoHash first for performance reasons, so we don't hash files with different sizes or LastWriteTimeUTCTicks
    $args = @{
        'Path' = $Path
        'Filter' = $Filter
        'Exclude' = $Exclude
        'Include' = $Include
        'Recurse' = $Recurse
        'NoHash' = $true
        'Algorithm' = $folderHashSrc.Algorithm
    }
    $folderHashDst = Get-FolderHash @args

    Write-Verbose "Found $($folderHashDst.Entries.Count) entries."
    
    Write-Verbose "Starting test against $Path."
    # create new lists for each category we are tracking by moving entries from the original lists
    $same = New-Object System.Collections.Generic.List[Object]
    $different = New-Object System.Collections.Generic.List[Object]
    # could not process
    $skipped = New-Object System.Collections.Generic.List[Object]
    # in src, not in dst
    $missing = New-Object System.Collections.Generic.List[Object]
    # not in src, in dst
    $extra = New-Object System.Collections.Generic.List[Object]

    while ($folderHashSrc.Entries.Count)
    {
        $srcEntry = $folderHashSrc.Entries[0]
        Write-Verbose "Finding: $($srcEntry.Path)."
        $dstEntry = $folderHashDst.Entries | Where-Object {$_.Path -eq $srcEntry.Path}

        if ($dstEntry)
        {
            Write-Verbose "Found: $($dstEntry.Path)."

            # we save ticks which are in 100-ns units. we are not guaranteed this granularity when storing files
            # on unc paths, locally, etc. Use granularity to the second to optimize not calculating the hash.
            $srcSeconds = [long]($srcEntry.LastWriteTimeUTCTicks / 10000000) * 10000000
            $dstSeconds = [long]($dstEntry.LastWriteTimeUTCTicks / 10000000) * 10000000
            if ($srcSeconds -ne $dstSeconds)
            {
                # ok to report LastWriteTimeUTCTicks here
                Write-Verbose "LastWriteTimeUTCTicks mismatch: $($srcEntry.LastWriteTimeUTCTicks) -ne $($dstEntry.LastWriteTimeUTCTicks)."
                # different
                $different.Add($srcEntry)
            }
            elseif ($srcEntry.Size -ne $dstEntry.Size)
            {
                Write-Verbose "Size mismatch: $($srcEntry.Size) -ne $($dstEntry.Size)."
                # different
                $different.Add($srcEntry)
            }
            else {
                # try to get hash now
                $fullName = Join-Path $Path $dstEntry.Path
                try
                {
                    $hash = (Get-FileHash -Path $fullName -Algorithm $folderHashSrc.Algorithm -ErrorAction Stop).Hash
                    $dstEntry.UpdateHash($hash)

                    if ($srcEntry.Hash -ne $dstEntry.Hash)
                    {
                        Write-Verbose "Hash mismatch: $($srcEntry.Hash) -ne $($dstEntry.Hash)."
                        # different
                        $different.Add($srcEntry)
                    }
                    else
                    {
                        Write-Verbose "Same: $($srcEntry.Path)."
                        # same
                        $same.Add($srcEntry)
                    }
                }
                catch [Microsoft.PowerShell.Commands.WriteErrorException]
                {
                    if (-not $AllowSkippedFiles)
                    {
                        Write-Warning "WriteErrorException: $fullName."
                        $ex = $_ | Select-Object *
                        Write-Warning $ex
                        throw
                    }

                    Write-Verbose "Skipping $fullName."
                    # skipped
                    $skipped.Add($srcEntry)
                }
                catch
                {
                    Write-Warning "Error processing: $fullName."
                    $ex = $_ | Select-Object *
                    Write-Warning $ex
                    throw
                }
            }

            if (-not ($folderHashDst.Entries.Remove($dstEntry)))
            {
                throw "Error removing dstEntry from folderHashDst.Entries."
            }
        }
        else
        {
            Write-Verbose "Missing $($srcEntry.Path)."
            # missing
            $missing.Add($srcEntry)
        }

        if (-not ($folderHashSrc.Entries.Remove($srcEntry)))
        {
            throw "Error removing srcEntry from folderHashSrc.Entries."
        }
    }

    # anything left in the dst list is extra
    $extra.AddRange($folderHashDst.Entries)

    $results = [ordered]@{}
    $results.Add('NumFailures', ($different.Count + $skipped.Count + $missing.Count + $extra.Count))
    $results.Add('Same', $same)
    $results.Add('Different', $different)
    $results.Add('Skipped', $skipped)
    $results.Add('Missing', $missing)
    $results.Add('Extra', $extra)

    Write-Verbose "Test-FolderHash results:"
    Write-Verbose " Same: $($same.Count)"
    Write-Verbose " Different: $($different.Count)"
    Write-Verbose " Skipped: $($skipped.Count)"
    Write-Verbose " Missing: $($missing.Count)"
    Write-Verbose " Extra: $($extra.Count)"

    $results
}

function Get-FolderContentSummaryHash
{
    [CmdletBinding()]
    param (
        [Parameter(Mandatory=$true)]
        [string]
        $Path,

        [ValidateSet('SHA1', 'SHA256', 'SHA384', 'SHA512')]
        [string]
        $Algorithm = 'SHA256'
    )

    $folderHash = New-Object Microsoft.Solutions.IO.FolderHash -ArgumentList $Algorithm

    if (-not (Test-Path $Path))
    {
        throw "Path $Path does not exist or access denied."
    }

    $args = @{
        'Path' = $Path
        'Recurse' = $true
        'ErrorAction' = 'SilentlyContinue'
    }

    $files = Get-ChildItem @args -File

    foreach ($file in $files)
    {
        $fullName = $file.FullName
        Write-Verbose "Processing: $fullName."

        try 
        {
            # Remove $Path from name so we store only the relative path
            $relativePath = $fullName -replace [regex]::Escape($($Path))
            $lastWriteTimeUTCTicks = $file.LastWriteTimeUtc.Ticks
            $size = $file.Length
            $hash = Get-FileHash -Path $fullName -Algorithm $Algorithm -ErrorAction Stop
            $entry = New-Object Microsoft.Solutions.IO.FolderHashEntry -ArgumentList $relativePath.TrimStart("\"), $lastWriteTimeUTCTicks, $size, $hash.Hash

            $folderHash.Entries.Add($entry)
        }
        catch [Microsoft.PowerShell.Commands.WriteErrorException]
        {
            Write-Warning "WriteErrorException: $fullName."
            $ex = $_ | Select-Object *
            Write-Warning $ex
            throw
        }
        catch
        {
            Write-Warning "Error processing: $fullName."
            $ex = $_ | Select-Object *
            Write-Warning $ex
            throw
        }
    }

    # Sort the file list by name and then create a byte stream for hashing based on the names and hashes
    $fileNameAndHashBytes = new-object System.Collections.Generic.List[byte]
    foreach ($entry in $folderHash.Entries)
    {
        $fileNameAndHashBytes.AddRange([System.Text.Encoding]::UTF8.GetBytes($entry.Path))
        $fileNameAndHashBytes.AddRange([System.Text.Encoding]::UTF8.GetBytes($entry.Hash))
    }

    $sha = Get-HashProviderFromString -Algorithm $Algorithm

    return [System.BitConverter]::ToString($sha.ComputeHash($fileNameAndHashBytes.ToArray())) -replace "-",""
}

#region Exports
Export-ModuleMember Get-FolderContentSummaryHash
Export-ModuleMember Get-FolderHash
Export-ModuleMember Test-FolderHash
#endregion Exports

# SIG # Begin signature block
# MIIoRgYJKoZIhvcNAQcCoIIoNzCCKDMCAQExDzANBglghkgBZQMEAgEFADB5Bgor
# BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG
# KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCBzh2j+pjx1Jn5O
# 2VuDyNN5fl8LCWpODT0gGlEJPNFZyqCCDXYwggX0MIID3KADAgECAhMzAAADrzBA
# DkyjTQVBAAAAAAOvMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p
# bmcgUENBIDIwMTEwHhcNMjMxMTE2MTkwOTAwWhcNMjQxMTE0MTkwOTAwWjB0MQsw
# CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u
# ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB
# AQDOS8s1ra6f0YGtg0OhEaQa/t3Q+q1MEHhWJhqQVuO5amYXQpy8MDPNoJYk+FWA
# hePP5LxwcSge5aen+f5Q6WNPd6EDxGzotvVpNi5ve0H97S3F7C/axDfKxyNh21MG
# 0W8Sb0vxi/vorcLHOL9i+t2D6yvvDzLlEefUCbQV/zGCBjXGlYJcUj6RAzXyeNAN
# xSpKXAGd7Fh+ocGHPPphcD9LQTOJgG7Y7aYztHqBLJiQQ4eAgZNU4ac6+8LnEGAL
# go1ydC5BJEuJQjYKbNTy959HrKSu7LO3Ws0w8jw6pYdC1IMpdTkk2puTgY2PDNzB
# tLM4evG7FYer3WX+8t1UMYNTAgMBAAGjggFzMIIBbzAfBgNVHSUEGDAWBgorBgEE
# AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQURxxxNPIEPGSO8kqz+bgCAQWGXsEw
# RQYDVR0RBD4wPKQ6MDgxHjAcBgNVBAsTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEW
# MBQGA1UEBRMNMjMwMDEyKzUwMTgyNjAfBgNVHSMEGDAWgBRIbmTlUAXTgqoXNzci
# tW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j
# b20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3JsMGEG
# CCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDovL3d3dy5taWNyb3NvZnQu
# Y29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3J0
# MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIBAISxFt/zR2frTFPB45Yd
# mhZpB2nNJoOoi+qlgcTlnO4QwlYN1w/vYwbDy/oFJolD5r6FMJd0RGcgEM8q9TgQ
# 2OC7gQEmhweVJ7yuKJlQBH7P7Pg5RiqgV3cSonJ+OM4kFHbP3gPLiyzssSQdRuPY
# 1mIWoGg9i7Y4ZC8ST7WhpSyc0pns2XsUe1XsIjaUcGu7zd7gg97eCUiLRdVklPmp
# XobH9CEAWakRUGNICYN2AgjhRTC4j3KJfqMkU04R6Toyh4/Toswm1uoDcGr5laYn
# TfcX3u5WnJqJLhuPe8Uj9kGAOcyo0O1mNwDa+LhFEzB6CB32+wfJMumfr6degvLT
# e8x55urQLeTjimBQgS49BSUkhFN7ois3cZyNpnrMca5AZaC7pLI72vuqSsSlLalG
# OcZmPHZGYJqZ0BacN274OZ80Q8B11iNokns9Od348bMb5Z4fihxaBWebl8kWEi2O
# PvQImOAeq3nt7UWJBzJYLAGEpfasaA3ZQgIcEXdD+uwo6ymMzDY6UamFOfYqYWXk
# ntxDGu7ngD2ugKUuccYKJJRiiz+LAUcj90BVcSHRLQop9N8zoALr/1sJuwPrVAtx
# HNEgSW+AKBqIxYWM4Ev32l6agSUAezLMbq5f3d8x9qzT031jMDT+sUAoCw0M5wVt
# CUQcqINPuYjbS1WgJyZIiEkBMIIHejCCBWKgAwIBAgIKYQ6Q0gAAAAAAAzANBgkq
# hkiG9w0BAQsFADCBiDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24x
# EDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv
# bjEyMDAGA1UEAxMpTWljcm9zb2Z0IFJvb3QgQ2VydGlmaWNhdGUgQXV0aG9yaXR5
# IDIwMTEwHhcNMTEwNzA4MjA1OTA5WhcNMjYwNzA4MjEwOTA5WjB+MQswCQYDVQQG
# EwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwG
# A1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSgwJgYDVQQDEx9NaWNyb3NvZnQg
# Q29kZSBTaWduaW5nIFBDQSAyMDExMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIIC
# CgKCAgEAq/D6chAcLq3YbqqCEE00uvK2WCGfQhsqa+laUKq4BjgaBEm6f8MMHt03
# a8YS2AvwOMKZBrDIOdUBFDFC04kNeWSHfpRgJGyvnkmc6Whe0t+bU7IKLMOv2akr
# rnoJr9eWWcpgGgXpZnboMlImEi/nqwhQz7NEt13YxC4Ddato88tt8zpcoRb0Rrrg
# OGSsbmQ1eKagYw8t00CT+OPeBw3VXHmlSSnnDb6gE3e+lD3v++MrWhAfTVYoonpy
# 4BI6t0le2O3tQ5GD2Xuye4Yb2T6xjF3oiU+EGvKhL1nkkDstrjNYxbc+/jLTswM9
# sbKvkjh+0p2ALPVOVpEhNSXDOW5kf1O6nA+tGSOEy/S6A4aN91/w0FK/jJSHvMAh
# dCVfGCi2zCcoOCWYOUo2z3yxkq4cI6epZuxhH2rhKEmdX4jiJV3TIUs+UsS1Vz8k
# A/DRelsv1SPjcF0PUUZ3s/gA4bysAoJf28AVs70b1FVL5zmhD+kjSbwYuER8ReTB
# w3J64HLnJN+/RpnF78IcV9uDjexNSTCnq47f7Fufr/zdsGbiwZeBe+3W7UvnSSmn
# Eyimp31ngOaKYnhfsi+E11ecXL93KCjx7W3DKI8sj0A3T8HhhUSJxAlMxdSlQy90
# lfdu+HggWCwTXWCVmj5PM4TasIgX3p5O9JawvEagbJjS4NaIjAsCAwEAAaOCAe0w
# ggHpMBAGCSsGAQQBgjcVAQQDAgEAMB0GA1UdDgQWBBRIbmTlUAXTgqoXNzcitW2o
# ynUClTAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMCAYYwDwYD
# VR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBRyLToCMZBDuRQFTuHqp8cx0SOJNDBa
# BgNVHR8EUzBRME+gTaBLhklodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20vcGtpL2Ny
# bC9wcm9kdWN0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3JsMF4GCCsG
# AQUFBwEBBFIwUDBOBggrBgEFBQcwAoZCaHR0cDovL3d3dy5taWNyb3NvZnQuY29t
# L3BraS9jZXJ0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFfMDNfMjIuY3J0MIGfBgNV
# HSAEgZcwgZQwgZEGCSsGAQQBgjcuAzCBgzA/BggrBgEFBQcCARYzaHR0cDovL3d3
# dy5taWNyb3NvZnQuY29tL3BraW9wcy9kb2NzL3ByaW1hcnljcHMuaHRtMEAGCCsG
# AQUFBwICMDQeMiAdAEwAZQBnAGEAbABfAHAAbwBsAGkAYwB5AF8AcwB0AGEAdABl
# AG0AZQBuAHQALiAdMA0GCSqGSIb3DQEBCwUAA4ICAQBn8oalmOBUeRou09h0ZyKb
# C5YR4WOSmUKWfdJ5DJDBZV8uLD74w3LRbYP+vj/oCso7v0epo/Np22O/IjWll11l
# hJB9i0ZQVdgMknzSGksc8zxCi1LQsP1r4z4HLimb5j0bpdS1HXeUOeLpZMlEPXh6
# I/MTfaaQdION9MsmAkYqwooQu6SpBQyb7Wj6aC6VoCo/KmtYSWMfCWluWpiW5IP0
# wI/zRive/DvQvTXvbiWu5a8n7dDd8w6vmSiXmE0OPQvyCInWH8MyGOLwxS3OW560
# STkKxgrCxq2u5bLZ2xWIUUVYODJxJxp/sfQn+N4sOiBpmLJZiWhub6e3dMNABQam
# ASooPoI/E01mC8CzTfXhj38cbxV9Rad25UAqZaPDXVJihsMdYzaXht/a8/jyFqGa
# J+HNpZfQ7l1jQeNbB5yHPgZ3BtEGsXUfFL5hYbXw3MYbBL7fQccOKO7eZS/sl/ah
# XJbYANahRr1Z85elCUtIEJmAH9AAKcWxm6U/RXceNcbSoqKfenoi+kiVH6v7RyOA
# 9Z74v2u3S5fi63V4GuzqN5l5GEv/1rMjaHXmr/r8i+sLgOppO6/8MO0ETI7f33Vt
# Y5E90Z1WTk+/gFcioXgRMiF670EKsT/7qMykXcGhiJtXcVZOSEXAQsmbdlsKgEhr
# /Xmfwb1tbWrJUnMTDXpQzTGCGiYwghoiAgEBMIGVMH4xCzAJBgNVBAYTAlVTMRMw
# EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN
# aWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNp
# Z25pbmcgUENBIDIwMTECEzMAAAOvMEAOTKNNBUEAAAAAA68wDQYJYIZIAWUDBAIB
# BQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEO
# MAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEICOU4zc8snhm8wfpMSot7Zq3
# zUgclC+dSX4dQPrpBkjNMEIGCisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8A
# cwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEB
# BQAEggEAbx0cWnQr8ILejMQXAUeGB0MKGVct4OzPIOwfpU7slXQWyl+6+dXX8zlZ
# 4Q/jKGJ1NdzfhuZvyMnhfx24qvhXkIE44G0EPe7dd5c1VhC3atYrlbwCgaCL/0Oh
# VytrHMPboTBJsnd7SuUoMI+OTXo6FxAiR7BCJXLkefiA9ez0ZeJHPZ3FGaLMB0mc
# 239COHG5sMw5Y+JhfenZ/4nch1QHNhhZO89dI8FCu4tOow3BqEjRO7e957G+XtJN
# Zo2UBOajZxI9y+6763DxUXwXmNXW2KNforiO0N+tdjZIdsujDOiPBta1ymgbFszS
# zN3LT5QkvKv5SaiJHTZ/yhJJ9rxXKqGCF7AwghesBgorBgEEAYI3AwMBMYIXnDCC
# F5gGCSqGSIb3DQEHAqCCF4kwgheFAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFaBgsq
# hkiG9w0BCRABBKCCAUkEggFFMIIBQQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFl
# AwQCAQUABCCAUi/OXIb3dDQuVm0OXPrRQoVrbYcWdRzXgH2mk0EisgIGZuswzYkH
# GBMyMDI0MTAwOTAxMTMzNy45MzlaMASAAgH0oIHZpIHWMIHTMQswCQYDVQQGEwJV
# UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE
# ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJl
# bGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVT
# TjoyRDFBLTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAg
# U2VydmljZaCCEf4wggcoMIIFEKADAgECAhMzAAAB/XP5aFrNDGHtAAEAAAH9MA0G
# CSqGSIb3DQEBCwUAMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9u
# MRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRp
# b24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMB4XDTI0
# MDcyNTE4MzExNloXDTI1MTAyMjE4MzExNlowgdMxCzAJBgNVBAYTAlVTMRMwEQYD
# VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy
# b3NvZnQgQ29ycG9yYXRpb24xLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9w
# ZXJhdGlvbnMgTGltaXRlZDEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOjJEMUEt
# MDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNl
# MIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEAoWWs+D+Ou4JjYnRHRedu
# 0MTFYzNJEVPnILzc02R3qbnujvhZgkhp+p/lymYLzkQyG2zpxYceTjIF7HiQWbt6
# FW3ARkBrthJUz05ZnKpcF31lpUEb8gUXiD2xIpo8YM+SD0S+hTP1TCA/we38yZ3B
# EtmZtcVnaLRp/Avsqg+5KI0Kw6TDJpKwTLl0VW0/23sKikeWDSnHQeTprO0zIm/b
# tagSYm3V/8zXlfxy7s/EVFdSglHGsUq8EZupUO8XbHzz7tURyiD3kOxNnw5ox1eZ
# X/c/XmW4H6b4yNmZF0wTZuw37yA1PJKOySSrXrWEh+H6++Wb6+1ltMCPoMJHUtPP
# 3Cn0CNcNvrPyJtDacqjnITrLzrsHdOLqjsH229Zkvndk0IqxBDZgMoY+Ef7ffFRP
# 2pPkrF1F9IcBkYz8hL+QjX+u4y4Uqq4UtT7VRnsqvR/x/+QLE0pcSEh/XE1w1fcp
# 6Jmq8RnHEXikycMLN/a/KYxpSP3FfFbLZuf+qIryFL0gEDytapGn1ONjVkiKpVP2
# uqVIYj4ViCjy5pLUceMeqiKgYqhpmUHCE2WssLLhdQBHdpl28+k+ZY6m4dPFnEoG
# cJHuMcIZnw4cOwixojROr+Nq71cJj7Q4L0XwPvuTHQt0oH7RKMQgmsy7CVD7v55d
# OhdHXdYsyO69dAdK+nWlyYcCAwEAAaOCAUkwggFFMB0GA1UdDgQWBBTpDMXA4ZW8
# +yL2+3vA6RmU7oEKpDAfBgNVHSMEGDAWgBSfpxVdAF5iXYP05dJlpxtTNRnpcjBf
# BgNVHR8EWDBWMFSgUqBQhk5odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3Bz
# L2NybC9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENBJTIwMjAxMCgxKS5jcmww
# bAYIKwYBBQUHAQEEYDBeMFwGCCsGAQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29m
# dC5jb20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0El
# MjAyMDEwKDEpLmNydDAMBgNVHRMBAf8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUF
# BwMIMA4GA1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQsFAAOCAgEAY9hYX+T5AmCr
# YGaH96TdR5T52/PNOG7ySYeopv4flnDWQLhBlravAg+pjlNv5XSXZrKGv8e4s5dJ
# 5WdhfC9ywFQq4TmXnUevPXtlubZk+02BXK6/23hM0TSKs2KlhYiqzbRe8QbMfKXE
# DtvMoHSZT7r+wI2IgjYQwka+3P9VXgERwu46/czz8IR/Zq+vO5523Jld6ssVuzs9
# uwIrJhfcYBj50mXWRBcMhzajLjWDgcih0DuykPcBpoTLlOL8LpXooqnr+QLYE4Bp
# Uep3JySMYfPz2hfOL3g02WEfsOxp8ANbcdiqM31dm3vSheEkmjHA2zuM+Tgn4j5n
# +Any7IODYQkIrNVhLdML09eu1dIPhp24lFtnWTYNaFTOfMqFa3Ab8KDKicmp0Ath
# RNZVg0BPAL58+B0UcoBGKzS9jscwOTu1JmNlisOKkVUVkSJ5Fo/ctfDSPdCTVaIX
# XF7l40k1cM/X2O0JdAS97T78lYjtw/PybuzX5shxBh/RqTPvCyAhIxBVKfN/hfs4
# CIoFaqWJ0r/8SB1CGsyyIcPfEgMo8ceq1w5Zo0JfnyFi6Guo+z3LPFl/exQaRubE
# rsAUTfyBY5/5liyvjAgyDYnEB8vHO7c7Fg2tGd5hGgYs+AOoWx24+XcyxpUkAajD
# hky9Dl+8JZTjts6BcT9sYTmOodk/SgIwggdxMIIFWaADAgECAhMzAAAAFcXna54C
# m0mZAAAAAAAVMA0GCSqGSIb3DQEBCwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UE
# CBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9z
# b2Z0IENvcnBvcmF0aW9uMTIwMAYDVQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZp
# Y2F0ZSBBdXRob3JpdHkgMjAxMDAeFw0yMTA5MzAxODIyMjVaFw0zMDA5MzAxODMy
# MjVaMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQH
# EwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNV
# BAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMIICIjANBgkqhkiG9w0B
# AQEFAAOCAg8AMIICCgKCAgEA5OGmTOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51
# yMo1V/YBf2xK4OK9uT4XYDP/XE/HZveVU3Fa4n5KWv64NmeFRiMMtY0Tz3cywBAY
# 6GB9alKDRLemjkZrBxTzxXb1hlDcwUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9
# cmmvHaus9ja+NSZk2pg7uhp7M62AW36MEBydUv626GIl3GoPz130/o5Tz9bshVZN
# 7928jaTjkY+yOSxRnOlwaQ3KNi1wjjHINSi947SHJMPgyY9+tVSP3PoFVZhtaDua
# Rr3tpK56KTesy+uDRedGbsoy1cCGMFxPLOJiss254o2I5JasAUq7vnGpF1tnYN74
# kpEeHT39IM9zfUGaRnXNxF803RKJ1v2lIH1+/NmeRd+2ci/bfV+AutuqfjbsNkz2
# K26oElHovwUDo9Fzpk03dJQcNIIP8BDyt0cY7afomXw/TNuvXsLz1dhzPUNOwTM5
# TI4CvEJoLhDqhFFG4tG9ahhaYQFzymeiXtcodgLiMxhy16cg8ML6EgrXY28MyTZk
# i1ugpoMhXV8wdJGUlNi5UPkLiWHzNgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9Q
# BXpsxREdcu+N+VLEhReTwDwV2xo3xwgVGD94q0W29R6HXtqPnhZyacaue7e3Pmri
# Lq0CAwEAAaOCAd0wggHZMBIGCSsGAQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUC
# BBYEFCqnUv5kxJq+gpE8RjUpzxD/LwTuMB0GA1UdDgQWBBSfpxVdAF5iXYP05dJl
# pxtTNRnpcjBcBgNVHSAEVTBTMFEGDCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIB
# FjNodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL0RvY3MvUmVwb3NpdG9y
# eS5odG0wEwYDVR0lBAwwCgYIKwYBBQUHAwgwGQYJKwYBBAGCNxQCBAweCgBTAHUA
# YgBDAEEwCwYDVR0PBAQDAgGGMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU
# 1fZWy4/oolxiaNE9lJBb186aGMQwVgYDVR0fBE8wTTBLoEmgR4ZFaHR0cDovL2Ny
# bC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0XzIw
# MTAtMDYtMjMuY3JsMFoGCCsGAQUFBwEBBE4wTDBKBggrBgEFBQcwAoY+aHR0cDov
# L3d3dy5taWNyb3NvZnQuY29tL3BraS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0w
# Ni0yMy5jcnQwDQYJKoZIhvcNAQELBQADggIBAJ1VffwqreEsH2cBMSRb4Z5yS/yp
# b+pcFLY+TkdkeLEGk5c9MTO1OdfCcTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulm
# ZzpTTd2YurYeeNg2LpypglYAA7AFvonoaeC6Ce5732pvvinLbtg/SHUB2RjebYIM
# 9W0jVOR4U3UkV7ndn/OOPcbzaN9l9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECW
# OKz3+SmJw7wXsFSFQrP8DJ6LGYnn8AtqgcKBGUIZUnWKNsIdw2FzLixre24/LAl4
# FOmRsqlb30mjdAy87JGA0j3mSj5mO0+7hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3Uw
# xTSwethQ/gpY3UA8x1RtnWN0SCyxTkctwRQEcb9k+SS+c23Kjgm9swFXSVRk2XPX
# fx5bRAGOWhmRaw2fpCjcZxkoJLo4S5pu+yFUa2pFEUep8beuyOiJXk+d0tBMdrVX
# VAmxaQFEfnyhYWxz/gq77EFmPWn9y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/AsGC
# onsXHRWJjXD+57XQKBqJC4822rpM+Zv/Cuk0+CQ1ZyvgDbjmjJnW4SLq8CdCPSWU
# 5nR0W2rRnj7tfqAxM328y+l7vzhwRNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEG
# ahC0HVUzWLOhcGbyoYIDWTCCAkECAQEwggEBoYHZpIHWMIHTMQswCQYDVQQGEwJV
# UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE
# ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQgSXJl
# bGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxJzAlBgNVBAsTHm5TaGllbGQgVFNTIEVT
# TjoyRDFBLTA1RTAtRDk0NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAg
# U2VydmljZaIjCgEBMAcGBSsOAwIaAxUAoj0WtVVQUNSKoqtrjinRAsBUdoOggYMw
# gYCkfjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE
# BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYD
# VQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsF
# AAIFAOqwDB8wIhgPMjAyNDEwMDgxOTUzMDNaGA8yMDI0MTAwOTE5NTMwM1owdzA9
# BgorBgEEAYRZCgQBMS8wLTAKAgUA6rAMHwIBADAKAgEAAgIKEwIB/zAHAgEAAgIU
# SjAKAgUA6rFdnwIBADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMCoAow
# CAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3DQEBCwUAA4IBAQAuhrfpECOd
# nolMLnYDeA+YAltDHr2+4D53GrSSkgpWHVyrf15LzJX6pgqbI/IZJI4yHzdU1pnv
# x0tCX0jXgfnZzo58ATwyypAmYhICdQNuhIWoMJdLr9VERAFbdR1QSe1qo//hzU3A
# eZJSYIHrAqBVh2PnjjBsnx5xsSVRLYCYYu6Pp2hj5FoIHSMQ6kSQVTny1AJxh2/A
# XLr6G+mUhiC+YluXBbjpf+ZuWgOcPioTHn9lht4jvLUpiJ8d5rETBKj5wbQDwbLZ
# ddHoYtuqWRJe6dF07OlymYOsLuksaPGXnZqHSQVgctLJIm4zC7g/boActaDcxF4l
# oXpc+Ayqud2BMYIEDTCCBAkCAQEwgZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgT
# Cldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m
# dCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENB
# IDIwMTACEzMAAAH9c/loWs0MYe0AAQAAAf0wDQYJYIZIAWUDBAIBBQCgggFKMBoG
# CSqGSIb3DQEJAzENBgsqhkiG9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQgQgAv8hr+
# k5ttvpyVVsrVyzm9CmhJN56yGhsIDh1hu74wgfoGCyqGSIb3DQEJEAIvMYHqMIHn
# MIHkMIG9BCCAKEgNyUowvIfx/eDfYSupHkeF1p6GFwjKBs8lRB4NRzCBmDCBgKR+
# MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdS
# ZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMT
# HU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMzAAAB/XP5aFrNDGHtAAEA
# AAH9MCIEIBsy2Zap+BoYTAXS6nPV0emrd7W9fWnVFoYx5Z5QfVFYMA0GCSqGSIb3
# DQEBCwUABIICACzt5cmQBFxsaVBRJSe402HMjKXBd+ulQZg7YorC9K9Qq9iW6NZC
# Ohi3WyutqZTaoHVDTknCuzbTySNfTT1Qgmg/ogNg7iB5IDdHB7pYyMq8iGBgPARJ
# 80KWf8boJj1nQ9fKXo+6WiEY5Rc+kfqYGipLVe6NpWI6ydum92Q9AZgwCEXn6+bS
# aa/xTJcBDg9GmYdgvC6h42NLWAK5f5ImXKdi5xWNuVWRHUdVEUjIgjsg81qC09NS
# XF29rCeEMXSrAn8vFnNDtMsTAUNS6IXLDKtD5GQQDEut9uCObijEMFhAS5csV+Pp
# rA4QV9eCJvXtN3BMGj9bHdiMdjGSxXVtO1HXbybfs6orue1Ulq2dI5KJY8rqz6NO
# ULc4KNwApTSr2W91j4k/+SaRSSZJvJyIsEb75LoCnW+Rkpz1/nQbl/A1Rq52z1c4
# QLi0OoGhhkknlcL2r6gaaU8BdOgRH0xPj/e/eF40IcHtoSW0gXQ1aBnf0eH2LA66
# 92jTKhXtDX5gVzTanc/H/YBMikPufusj+rRP75b7hS+/Ia0KsEUlq6Sv9PpILbjR
# PVknSAp9nbH3z5+ntltuepPN0vBA/1XZWnvSlFGdbcwMak6QBKgZj0GKLpEUJVih
# InBWxlwC0eYlYuHBAcXrTZfZP6q8B+2zWb190+9ynMsMfydImJIy08vt
# SIG # End signature block