PSGit.psm1
# Internal Functions #region Interal Functions function WriteMessage { [CmdletBinding()] param ( # The Type of message you'd like, its just a prefix to the message. [Parameter(Mandatory=$true,Position=0)] [string] $Type, # The message to display [Parameter(Mandatory=$true,Position=1)] [string] $Message, # Color of the message, it will default to the hosts' verbose color. [Parameter(Position=2)] [PoshCode.Pansies.RgbColor] $ForegroundColor = ($host.PrivateData.VerboseForegroundColor|ConvertColor), # Background color of the message, it will default to the hosts' verbose color. [Parameter(Position=3)] [PoshCode.Pansies.RgbColor] $BackgroundColor = ($host.PrivateData.VerboseBackgroundColor|ConvertColor -default "black") ) #todo check to see if the preference is on or off Write-Host "$($Type.ToUpper()): $Message" -ForegroundColor $ForegroundColor -BackgroundColor $BackgroundColor } function ConvertColor { #.Synopsis # A color converter that handles WPF colors #.Description # ConvertColor exists specifically to handle the WPF colors that ISE uses as it's defaults [CmdletBinding()] param ( [Parameter(ValueFromPipeline=$true)] $color, $default="yellow" ) if($color -is [string] -and $color.Length -gt 7) { if($color -match "#00[0-9A-F]{6}") { $color = if($host.PrivateData.ConsolePaneBackgroundColor) { $host.PrivateData.ConsolePaneBackgroundColor } else { $host.UI.RawUI.BackgroundColor } } if("System.Windows.Media.Color" -as [type]) { #if its a transparent color, just use the background color of the host $color = "#" + ([System.Windows.Media.Color]"$color").ToString().Substring(3) } elseif("$color"[0] -eq "#") { $color = "#" + "$color".SubString($color.Length - 6) } } if(($color -as [PoshCode.Pansies.RgbColor]) -ne $null) { ([PoshCode.Pansies.RgbColor]$color).ConsoleColor } else { ([PoshCode.Pansies.RgbColor]$default).ConsoleColor } } #endregion function Get-RootFolder { #.Synopsis # Search up the directory tree recursively for a git root (and corresponding .git folder) [CmdletBinding(DefaultParameterSetName="IndexAndWorkDir")] param ( # Where to start searching [Parameter()] [ValidateNotNullOrEmpty()] [String]$Root = $Pwd ) end { # Git Repositories are File System Based, and don't care aabout PSDrives [LibGit2Sharp.Repository]::Discover((Convert-Path $Root)) } } # TODO: DOCUMENT ME function Get-Change { [CmdletBinding(DefaultParameterSetName="IndexAndWorkDir")] param ( [Parameter()] [ValidateNotNullOrEmpty()] [String]$Root = $Pwd, [Parameter(Position = 0)] [String[]]$PathSpec, [Parameter(ParameterSetName="WorkDirOnly", Mandatory=$true)] [Switch]$UnStagedOnly, [Parameter(ParameterSetName="IndexOnly", Mandatory=$true)] [Switch]$StagedOnly, [Parameter()] [switch] $HideUntracked, [Parameter()] [switch] $HideSubmodules, [Parameter()] [switch] $ShowIgnored ) end { $Path = [LibGit2Sharp.Repository]::Discover((Convert-Path $Root)) if(!$Path) { Write-Warning "The path is not in a git repository!" return } $PathSpec = $PathSpec | Where { "$_".Trim().Length -gt 0 } try { $repo = New-Object LibGit2Sharp.Repository $Path $Path = $repo.Info.WorkingDirectory $Options = New-Object LibGit2Sharp.StatusOptions $Options.Show = $PSCmdlet.ParameterSetName # Don't touch PathSpec unless you're serious, it breaks the output if($PathSpec) { $Options.PathSpec = $PathSpec } $Options.DetectRenamesInWorkDir = $true if($HideSubmodules) { $Options.ExcludeSubmodules = $true } $status = $repo.RetrieveStatus($Options) } finally { $repo.Dispose() } # Unaltered, Added, Staged, Removed, RenamedInIndex, StagedTypeChange, # Untracked, Modified, Missing, TypeChanged, RenamedInWorkDir, # Unreadable, Ignored, Nonexistent # Output staged changes, if any foreach($file in $status.Added) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $true Change = "Added" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) } } foreach($file in $status.RenamedInIndex) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $true Change = "Renamed" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) OldPath = $File.HeadToIndexRenameDetails.OldFilePath + $(if(Test-Path (Join-Path $Path $File.HeadToIndexRenameDetails.OldFilePath) -Type Container){ "\" }) } } foreach($file in $status.Removed) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $true Change = "Removed" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) } } foreach($file in $status.Staged) { #BUGBUG: hides rename + edit, but avoids double-outputs (and behaves like git) if(($file.State -band [LibGit2Sharp.FileStatus]::RenamedInIndex) -eq 0) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $true Change = "Modified" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) } } } # Output unstaged changes, if any foreach($file in $status.RenamedInWorkDir) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $false Change = "Renamed" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) OldPath = $File.IndexToWorkDirRenameDetails.OldFilePath + $(if(Test-Path (Join-Path $Path $File.IndexToWorkDirRenameDetails.OldFilePath) -Type Container){ "\" }) } } foreach($file in $status.Modified) { #BUGBUG: hides rename + edit, but avoids double-outputs (and behaves like git) if(($file.State -band [LibGit2Sharp.FileStatus]::RenamedInWorkDir) -eq 0) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $false Change = "Modified" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) } } } foreach($file in $status.Missing) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $false Change = "Removed" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) } } # Optional output if(!$HideUntracked) { foreach($file in $status.Untracked) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $false Change = "Added" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) } } } if($ShowIgnored) { foreach($file in $status.Ignored) { New-Object PSCustomObject -Property @{ PSTypeName = "PSGit.FileStatus" Staged = $false Change = "Ignored" Path = $file.FilePath + $(if(Test-Path (Join-Path $Path $File.FilePath) -Type Container){ "\" }) } } } } } $BranchProperties = @{ Name="Branch"; Expr={$_.Name}}, @{ Name="IsHead"; Expr={ $_.IsCurrentRepositoryHead}}, "IsRemote", "IsTracking", @{ Name="Tip"; Expr={$_.Tip.Sha}}, @{ Name="Remote"; expr = { $_.Remote.Url } }, @{ Name="Ahead"; Expr= { $_.TrackingDetails.AheadBy }}, @{ Name="Behind"; Expr = { $_.TrackingDetails.BehindBy }}, @{ Name="CommonAncestor"; Expr={ $_.TrackingDetails.CommonAncestor.Sha }}, @{ Name="GitDir"; Expr= {$_.Repository.Info.WorkingDirectory}} # TODO: DOCUMENT ME function Get-Info { [CmdletBinding()] param ( [Parameter()] [ValidateNotNullOrEmpty()] [String]$Root = $Pwd ) end { $Path = [LibGit2Sharp.Repository]::Discover((Convert-Path $Root)) if(!$Path) { Write-Warning "The path is not in a git repository!" return } try { $repo = New-Object LibGit2Sharp.Repository $Path # We have to transform the object to keep the data around after .Dispose() $repo.Head | Select-Object $BranchProperties | ForEach-Object { $_.PSTypeNames.Insert(0,"PSGit.Branch"); $_ } } finally { if($null -ne $repo) { $repo.Dispose() } } } } # TODO: DOCUMENT ME function Get-Branch { [CmdletBinding()] param ( [Parameter()] [ValidateNotNullOrEmpty()] [String]$Root = $Pwd, [Switch]$Force ) end { $Path = [LibGit2Sharp.Repository]::Discover((Convert-Path $Root)) if(!$Path) { Write-Warning "The path is not in a git repository!" return } try { $repo = New-Object LibGit2Sharp.Repository $Path $( # In the initialized state, there are no "Branches" if([Linq.Enumerable]::Count($repo.Branches) -eq 0) { # But really, there is the master! $repo.Head } elseif($Force) { $repo.Branches } else { $repo.Branches | Where-Object { !$_.IsRemote } } # We have to transform the object to keep the data around after .Dispose() ) | Select-Object $BranchProperties | ForEach-Object { $_.PSTypeNames.Insert(0,"PSGit.Branch"); $_ } } finally { if($null -ne $repo) { $repo.Dispose() } } } } Set-Alias "Branch" "Get-Branch" function Get-Status { [CmdletBinding()] param ( [Parameter()] [ValidateNotNullOrEmpty()] [String]$Root = $Pwd ) Get-Info -Root $Root | Add-Member -Type NoteProperty -Name Changes -Value ( Get-Change -Root $Root ) -Passthru } # TODO: DOCUMENT ME function Show-Status { [CmdletBinding()] param ( [Parameter()] [ValidateNotNullOrEmpty()] [String]$Root = $Pwd ) $status = Get-Status -Root $Root $changes = $status.Changes $status | Out-Default if($info.BehindBy) { WriteMessage "Action" " (use `git pull` to merge the remote branch into yours)" } $staged = $changes | where { $_.Staged } $unstaged = $changes | where { !$_.Staged} $added = $unstaged | where { $_.Change -eq "Added" } $unstaged = $unstaged | where { $_.Change -ne "Added" } if($staged) { WriteMessage "Changes to be committed" "`n (use ``git reset HEAD `${file}`` to unstage)" -ForegroundColor "Green" # $fg, $Host.UI.RawUI.ForegroundColor = $Host.UI.RawUI.ForegroundColor, "Green" $staged | Out-Default # $Host.UI.RawUI.ForegroundColor = $fg } if($unstaged) { WriteMessage "Changes not staged for commit" "`n (use ``git add `${file}`` to update what will be committed)`n (use ``git checkout -- `${file}`` to discard changes in the working directory)" -ForegroundColor "DarkYellow" $unstaged | Out-Default } if($added) { WriteMessage "Untracked Files" "`n (use ``git add `${file}`` to include them in what will be committed)" -ForegroundColor "Red" $added | Out-Default } } Set-Alias "Status" "Show-Status" # Export-ModuleMember -Function *-* -Alias * # For PSTypes?? # Update-TypeData -TypeName LibGit2Sharp.StatusEntry -MemberType ScriptMethod -MemberName ToString -Value { $this.FilePath } # OR -Value { # switch($this.State){ # "Ignored" { } # "Untracked" { "?? " + $this.FilePath } # "Added" { "A " + $this.FilePath } # "Modified" { " M " + $this.FilePath} # "Added, Modified" { "AM " + $this.FilePath} # default { $this.State + " " + $this.FilePath} # } # } -Force # TODO: DOCUMENT ME function New-Repository { [CmdletBinding()] param ( [Parameter()] [ValidateNotNullOrEmpty()] [String]$Root = $Pwd ) end { $Path = Convert-Path $Root # Not sure why this is needed, but if you do a folder on the root it fails $Path = Join-Path $path "." $null = mkdir $Root -Force -ErrorAction SilentlyContinue try { $rtn = [LibGit2Sharp.Repository]::Init($Path) } finally {} } } . $PSScriptRoot\PSGitPrompt.ps1 . $PSScriptRoot\PSGitPowerline.ps1 # SIG # Begin signature block # MIIXzgYJKoZIhvcNAQcCoIIXvzCCF7sCAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB # gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR # AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQUvLRTLNtV06Ic2ISWR6ou8+HY # ASugghMBMIID7jCCA1egAwIBAgIQfpPr+3zGTlnqS5p31Ab8OzANBgkqhkiG9w0B # AQUFADCBizELMAkGA1UEBhMCWkExFTATBgNVBAgTDFdlc3Rlcm4gQ2FwZTEUMBIG # A1UEBxMLRHVyYmFudmlsbGUxDzANBgNVBAoTBlRoYXd0ZTEdMBsGA1UECxMUVGhh # d3RlIENlcnRpZmljYXRpb24xHzAdBgNVBAMTFlRoYXd0ZSBUaW1lc3RhbXBpbmcg # Q0EwHhcNMTIxMjIxMDAwMDAwWhcNMjAxMjMwMjM1OTU5WjBeMQswCQYDVQQGEwJV # UzEdMBsGA1UEChMUU3ltYW50ZWMgQ29ycG9yYXRpb24xMDAuBgNVBAMTJ1N5bWFu # dGVjIFRpbWUgU3RhbXBpbmcgU2VydmljZXMgQ0EgLSBHMjCCASIwDQYJKoZIhvcN # AQEBBQADggEPADCCAQoCggEBALGss0lUS5ccEgrYJXmRIlcqb9y4JsRDc2vCvy5Q # WvsUwnaOQwElQ7Sh4kX06Ld7w3TMIte0lAAC903tv7S3RCRrzV9FO9FEzkMScxeC # i2m0K8uZHqxyGyZNcR+xMd37UWECU6aq9UksBXhFpS+JzueZ5/6M4lc/PcaS3Er4 # ezPkeQr78HWIQZz/xQNRmarXbJ+TaYdlKYOFwmAUxMjJOxTawIHwHw103pIiq8r3 # +3R8J+b3Sht/p8OeLa6K6qbmqicWfWH3mHERvOJQoUvlXfrlDqcsn6plINPYlujI # fKVOSET/GeJEB5IL12iEgF1qeGRFzWBGflTBE3zFefHJwXECAwEAAaOB+jCB9zAd # BgNVHQ4EFgQUX5r1blzMzHSa1N197z/b7EyALt0wMgYIKwYBBQUHAQEEJjAkMCIG # CCsGAQUFBzABhhZodHRwOi8vb2NzcC50aGF3dGUuY29tMBIGA1UdEwEB/wQIMAYB # Af8CAQAwPwYDVR0fBDgwNjA0oDKgMIYuaHR0cDovL2NybC50aGF3dGUuY29tL1Ro # YXd0ZVRpbWVzdGFtcGluZ0NBLmNybDATBgNVHSUEDDAKBggrBgEFBQcDCDAOBgNV # HQ8BAf8EBAMCAQYwKAYDVR0RBCEwH6QdMBsxGTAXBgNVBAMTEFRpbWVTdGFtcC0y # MDQ4LTEwDQYJKoZIhvcNAQEFBQADgYEAAwmbj3nvf1kwqu9otfrjCR27T4IGXTdf # plKfFo3qHJIJRG71betYfDDo+WmNI3MLEm9Hqa45EfgqsZuwGsOO61mWAK3ODE2y # 0DGmCFwqevzieh1XTKhlGOl5QGIllm7HxzdqgyEIjkHq3dlXPx13SYcqFgZepjhq # IhKjURmDfrYwggSjMIIDi6ADAgECAhAOz/Q4yP6/NW4E2GqYGxpQMA0GCSqGSIb3 # DQEBBQUAMF4xCzAJBgNVBAYTAlVTMR0wGwYDVQQKExRTeW1hbnRlYyBDb3Jwb3Jh # dGlvbjEwMC4GA1UEAxMnU3ltYW50ZWMgVGltZSBTdGFtcGluZyBTZXJ2aWNlcyBD # QSAtIEcyMB4XDTEyMTAxODAwMDAwMFoXDTIwMTIyOTIzNTk1OVowYjELMAkGA1UE # BhMCVVMxHTAbBgNVBAoTFFN5bWFudGVjIENvcnBvcmF0aW9uMTQwMgYDVQQDEytT # eW1hbnRlYyBUaW1lIFN0YW1waW5nIFNlcnZpY2VzIFNpZ25lciAtIEc0MIIBIjAN # BgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAomMLOUS4uyOnREm7Dv+h8GEKU5Ow # mNutLA9KxW7/hjxTVQ8VzgQ/K/2plpbZvmF5C1vJTIZ25eBDSyKV7sIrQ8Gf2Gi0 # jkBP7oU4uRHFI/JkWPAVMm9OV6GuiKQC1yoezUvh3WPVF4kyW7BemVqonShQDhfu # ltthO0VRHc8SVguSR/yrrvZmPUescHLnkudfzRC5xINklBm9JYDh6NIipdC6Anqh # d5NbZcPuF3S8QYYq3AhMjJKMkS2ed0QfaNaodHfbDlsyi1aLM73ZY8hJnTrFxeoz # C9Lxoxv0i77Zs1eLO94Ep3oisiSuLsdwxb5OgyYI+wu9qU+ZCOEQKHKqzQIDAQAB # o4IBVzCCAVMwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEFBQcDCDAO # BgNVHQ8BAf8EBAMCB4AwcwYIKwYBBQUHAQEEZzBlMCoGCCsGAQUFBzABhh5odHRw # Oi8vdHMtb2NzcC53cy5zeW1hbnRlYy5jb20wNwYIKwYBBQUHMAKGK2h0dHA6Ly90 # cy1haWEud3Muc3ltYW50ZWMuY29tL3Rzcy1jYS1nMi5jZXIwPAYDVR0fBDUwMzAx # oC+gLYYraHR0cDovL3RzLWNybC53cy5zeW1hbnRlYy5jb20vdHNzLWNhLWcyLmNy # bDAoBgNVHREEITAfpB0wGzEZMBcGA1UEAxMQVGltZVN0YW1wLTIwNDgtMjAdBgNV # HQ4EFgQURsZpow5KFB7VTNpSYxc/Xja8DeYwHwYDVR0jBBgwFoAUX5r1blzMzHSa # 1N197z/b7EyALt0wDQYJKoZIhvcNAQEFBQADggEBAHg7tJEqAEzwj2IwN3ijhCcH # bxiy3iXcoNSUA6qGTiWfmkADHN3O43nLIWgG2rYytG2/9CwmYzPkSWRtDebDZw73 # BaQ1bHyJFsbpst+y6d0gxnEPzZV03LZc3r03H0N45ni1zSgEIKOq8UvEiCmRDoDR # EfzdXHZuT14ORUZBbg2w6jiasTraCXEQ/Bx5tIB7rGn0/Zy2DBYr8X9bCT2bW+IW # yhOBbQAuOA2oKY8s4bL0WqkBrxWcLC9JG9siu8P+eJRRw4axgohd8D20UaF5Mysu # e7ncIAkTcetqGVvP6KUwVyyJST+5z3/Jvz4iaGNTmr1pdKzFHTx/kuDDvBzYBHUw # ggUwMIIEGKADAgECAhAECRgbX9W7ZnVTQ7VvlVAIMA0GCSqGSIb3DQEBCwUAMGUx # CzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3 # dy5kaWdpY2VydC5jb20xJDAiBgNVBAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9v # dCBDQTAeFw0xMzEwMjIxMjAwMDBaFw0yODEwMjIxMjAwMDBaMHIxCzAJBgNVBAYT # AlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2Vy # dC5jb20xMTAvBgNVBAMTKERpZ2lDZXJ0IFNIQTIgQXNzdXJlZCBJRCBDb2RlIFNp # Z25pbmcgQ0EwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQD407Mcfw4R # r2d3B9MLMUkZz9D7RZmxOttE9X/lqJ3bMtdx6nadBS63j/qSQ8Cl+YnUNxnXtqrw # nIal2CWsDnkoOn7p0WfTxvspJ8fTeyOU5JEjlpB3gvmhhCNmElQzUHSxKCa7JGnC # wlLyFGeKiUXULaGj6YgsIJWuHEqHCN8M9eJNYBi+qsSyrnAxZjNxPqxwoqvOf+l8 # y5Kh5TsxHM/q8grkV7tKtel05iv+bMt+dDk2DZDv5LVOpKnqagqrhPOsZ061xPeM # 0SAlI+sIZD5SlsHyDxL0xY4PwaLoLFH3c7y9hbFig3NBggfkOItqcyDQD2RzPJ6f # pjOp/RnfJZPRAgMBAAGjggHNMIIByTASBgNVHRMBAf8ECDAGAQH/AgEAMA4GA1Ud # DwEB/wQEAwIBhjATBgNVHSUEDDAKBggrBgEFBQcDAzB5BggrBgEFBQcBAQRtMGsw # JAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBDBggrBgEFBQcw # AoY3aHR0cDovL2NhY2VydHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElE # Um9vdENBLmNydDCBgQYDVR0fBHoweDA6oDigNoY0aHR0cDovL2NybDQuZGlnaWNl # cnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENBLmNybDA6oDigNoY0aHR0cDov # L2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENBLmNybDBP # BgNVHSAESDBGMDgGCmCGSAGG/WwAAgQwKjAoBggrBgEFBQcCARYcaHR0cHM6Ly93 # d3cuZGlnaWNlcnQuY29tL0NQUzAKBghghkgBhv1sAzAdBgNVHQ4EFgQUWsS5eyoK # o6XqcQPAYPkt9mV1DlgwHwYDVR0jBBgwFoAUReuir/SSy4IxLVGLp6chnfNtyA8w # DQYJKoZIhvcNAQELBQADggEBAD7sDVoks/Mi0RXILHwlKXaoHV0cLToaxO8wYdd+ # C2D9wz0PxK+L/e8q3yBVN7Dh9tGSdQ9RtG6ljlriXiSBThCk7j9xjmMOE0ut119E # efM2FAaK95xGTlz/kLEbBw6RFfu6r7VRwo0kriTGxycqoSkoGjpxKAI8LpGjwCUR # 4pwUR6F6aGivm6dcIFzZcbEMj7uo+MUSaJ/PQMtARKUT8OZkDCUIQjKyNookAv4v # cn4c10lFluhZHen6dGRrsutmQ9qzsIzV6Q3d9gEgzpkxYz0IGhizgZtPxpMQBvwH # gfqL2vmCSfdibqFT+hKUGIUukpHqaGxEMrJmoecYpJpkUe8wggUwMIIEGKADAgEC # AhAFmB+6PJIk/oqP7b4FPfHsMA0GCSqGSIb3DQEBCwUAMHIxCzAJBgNVBAYTAlVT # MRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5j # b20xMTAvBgNVBAMTKERpZ2lDZXJ0IFNIQTIgQXNzdXJlZCBJRCBDb2RlIFNpZ25p # bmcgQ0EwHhcNMTcwNjE0MDAwMDAwWhcNMTgwNjAxMTIwMDAwWjBtMQswCQYDVQQG # EwJVUzERMA8GA1UECBMITmV3IFlvcmsxFzAVBgNVBAcTDldlc3QgSGVucmlldHRh # MRgwFgYDVQQKEw9Kb2VsIEguIEJlbm5ldHQxGDAWBgNVBAMTD0pvZWwgSC4gQmVu # bmV0dDCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBANALmLHevB7LvTmI # p2oVErnz915fP1JUKoC+/5BRWUtAGooxg95jxX8+qT1yc02ZnkK7u1UyM0Mfs3b8 # MzhSqe5OkkQeT2RHrGe52+0/0ZWD68pvUBZoMQxrAnWJETjFO6IoXPKmoXN3zzpF # +5s/UIbNGI5mdiN4v4F93Yaajzu2ymsJsXK6NgRh/AUbUzUlefpOas+o06wT0vqp # LniGWw26321zJo//2QEo5PBrJvDDDIBBN6Xn5A2ww6v6fH2KGk2qf4vpr58rhDIH # fLOHLg9s35effaktygUMQBCFmxOAbPLKWId8n5+O7zbMfKw3qxqCp2QeXhjkIh9v # ETIX9pECAwEAAaOCAcUwggHBMB8GA1UdIwQYMBaAFFrEuXsqCqOl6nEDwGD5LfZl # dQ5YMB0GA1UdDgQWBBQ8xh3xoTXbMfJUSyFBfPsrxoD8XzAOBgNVHQ8BAf8EBAMC # B4AwEwYDVR0lBAwwCgYIKwYBBQUHAwMwdwYDVR0fBHAwbjA1oDOgMYYvaHR0cDov # L2NybDMuZGlnaWNlcnQuY29tL3NoYTItYXNzdXJlZC1jcy1nMS5jcmwwNaAzoDGG # L2h0dHA6Ly9jcmw0LmRpZ2ljZXJ0LmNvbS9zaGEyLWFzc3VyZWQtY3MtZzEuY3Js # MEwGA1UdIARFMEMwNwYJYIZIAYb9bAMBMCowKAYIKwYBBQUHAgEWHGh0dHBzOi8v # d3d3LmRpZ2ljZXJ0LmNvbS9DUFMwCAYGZ4EMAQQBMIGEBggrBgEFBQcBAQR4MHYw # JAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBOBggrBgEFBQcw # AoZCaHR0cDovL2NhY2VydHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0U0hBMkFzc3Vy # ZWRJRENvZGVTaWduaW5nQ0EuY3J0MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQEL # BQADggEBAGvlfIiin9JAyL16oeCNApnAWLfZpUBob4D+XLzdRJXidPq/pvNkE9Rg # pRZFaWs30f2WPhWeqCpSCahoHzFsD5S9mOzsGTXsT+EdjAS0yEe1t9LfMvEC/pI3 # aBQJeJ/DdgpTMUEUJSvddc0P0NbDJ6TJC/niEMOJ8XvsfF75J4YVJ10yVNahbAuU # MrRrRLe30pW74MRv1s7SKxwPmLhcsMQuK0mWGERtGYMwDHwW0ZdRHKNDGHRsl0Wh # DS1P8+JRpE3eNFPcO17yiOfKDnVh+/1AOg7QopD6R6+P9rErorebsvW680s4WTlr # hDcMsTOX0js2KFF6uT4nSojS4GNlSxExggQ3MIIEMwIBATCBhjByMQswCQYDVQQG # EwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNl # cnQuY29tMTEwLwYDVQQDEyhEaWdpQ2VydCBTSEEyIEFzc3VyZWQgSUQgQ29kZSBT # aWduaW5nIENBAhAFmB+6PJIk/oqP7b4FPfHsMAkGBSsOAwIaBQCgeDAYBgorBgEE # AYI3AgEMMQowCKACgAChAoAAMBkGCSqGSIb3DQEJAzEMBgorBgEEAYI3AgEEMBwG # CisGAQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMCMGCSqGSIb3DQEJBDEWBBSPYdSo # NL1Re1N1/Q1rRGtApnCnTDANBgkqhkiG9w0BAQEFAASCAQDNKBR7lZBbdRWYtCST # 93fVH+tj0Yvq5AWUKeE4rjVcv9bZFp2i824113tW23k0CXdJX2hLaPo2VszffExE # 9ifyOtA7kHu5OIul0ZDOS9BiZyRZIAoLWPrQ/X1R+Bu4tdSSHbcUfiBmbyLJb/RN # KWou7it4nUaZxRy/daBMG9MXeB7DkVS2RJHW6sWwUxAhQjT/G3rhwMXyGcG9QiAA # MMMCXWvsW2Ti//SPCtqkeDXL/wYxu1Ip5FOgfiEYxLMohMOJNF3ydETcKwCqqOz2 # P6Tbqau9uBPbRJKFVoRJhPFEMa/CBZy1LBNlZQFdw6XwxR7PcMJ8H1wYok0zKr9J # uPbDoYICCzCCAgcGCSqGSIb3DQEJBjGCAfgwggH0AgEBMHIwXjELMAkGA1UEBhMC # VVMxHTAbBgNVBAoTFFN5bWFudGVjIENvcnBvcmF0aW9uMTAwLgYDVQQDEydTeW1h # bnRlYyBUaW1lIFN0YW1waW5nIFNlcnZpY2VzIENBIC0gRzICEA7P9DjI/r81bgTY # apgbGlAwCQYFKw4DAhoFAKBdMBgGCSqGSIb3DQEJAzELBgkqhkiG9w0BBwEwHAYJ # KoZIhvcNAQkFMQ8XDTE3MTExNzA3NTEzMFowIwYJKoZIhvcNAQkEMRYEFD1EXiSU # 84wgCflEXZ7wc8pD1Sk2MA0GCSqGSIb3DQEBAQUABIIBABKIfmM7PEJLqEVhN2n9 # ZmqNpZryMM9XMKNhW97ryI8EUNpNzrtrAFYKUXNPnUWXApRwr/3EJwNdAnYYfBcR # scgl5+2gQnDkcXGRkttL+q+VZtXAFhlPMJFu2xe475X2XlfCAPDi/wHb/Cu4YIXl # DL0NxJGKffci/U/P8V/e71qVmN5O0SrcZa4OK/gWPHPyDpr1GbaykPaNic3RdYNI # 0bSJSY7YhOYQNkK/TDM1sffxs14bR82oipIy4R/4qDp6qAi2H2Cab5TiRdBB0HEG # 6esOl1mdivxeIyXthuRcxa6pEMcJUtbsPxFuNqTp0jrsPRZFG6kChlPCkjmu0gaR # Mec= # SIG # End signature block |