2021-06-08 21:46:20 +00:00
<#
. SYNOPSIS
Script to bootstrap a Haskell environment
. DESCRIPTION
This is the windows GHCup installer , installing :
* ghcup - The Haskell toolchain installer "
* ghc - The Glasgow Haskell Compiler "
* msys2 - Unix-style toolchain needed for dependencies and tools
* cabal - The Cabal build tool for managing Haskell software "
* stack - ( optional ) A cross-platform program for developing Haskell projects "
* hls - ( optional ) A language server for developers to integrate with their editor / IDE "
#>
param (
# Run a non-interactive installation
2021-06-10 08:58:07 +00:00
[ switch ] $Silent ,
2021-06-08 21:46:20 +00:00
# Specify the install root (default: 'C:\')
[ string ] $InstallDir ,
# Instead of installing a new MSys2, use an existing installation
[ string ] $ExistingMsys2Dir ,
2021-06-10 08:58:24 +00:00
# Specify the cabal root directory (default: '$InstallDir\cabal')
[ string ] $CabalDir ,
2021-06-08 21:46:20 +00:00
# Perform a quick installation, omitting some expensive operations (you may have to install dependencies yourself later)
[ bool ] $Quick ,
# Overwrite (or rather backup) a previous install
[ bool ] $Overwrite
)
2021-05-14 21:09:45 +00:00
function Print-Msg {
2021-06-07 15:06:44 +00:00
param ( [ Parameter ( Mandatory = $true , HelpMessage = 'String to output' ) ] [ string ] $msg , [ string ] $color = " Green " )
Write-Host ( '{0}' -f $msg ) -ForegroundColor $color
2021-05-14 21:09:45 +00:00
}
function Create-Shortcut {
param ( [ Parameter ( Mandatory = $true , HelpMessage = 'Target path' ) ] [ string ] $SourceExe , [ Parameter ( Mandatory = $true , HelpMessage = 'Arguments to the path/exe' ) ] [ AllowEmptyString ( ) ] $ArgumentsToSourceExe , [ Parameter ( Mandatory = $true , HelpMessage = 'The destination of the desktop link' ) ] [ string ] $DestinationPath )
$WshShell = New-Object -comObject WScript . Shell
$Shortcut = $WshShell . CreateShortcut ( $DestinationPath )
$Shortcut . TargetPath = $SourceExe
if ( $ArgumentsToSourceExe ) {
$Shortcut . Arguments = $ArgumentsToSourceExe
}
$Shortcut . Save ( )
}
function Add-EnvPath {
2021-06-08 21:46:20 +00:00
param (
[ Parameter ( Mandatory = $true , HelpMessage = 'The Path to add to Users environment' ) ]
[ string ] $Path ,
2021-05-14 21:09:45 +00:00
2021-06-08 21:46:20 +00:00
[ ValidateSet ( 'Machine' , 'User' , 'Session' ) ]
[ string ] $Container = 'Session'
)
2021-05-14 21:09:45 +00:00
2021-06-08 21:46:20 +00:00
if ( $Container -eq 'Session' ) {
$envPaths = [ Collections.Generic.List[String] ] ( $env:Path -split ( [ IO.Path ] :: PathSeparator ) )
if ( $envPaths -notcontains $Path ) {
$envPaths . Add ( $Path )
$env:PATH = $envPaths -join ( [ IO.Path ] :: PathSeparator )
}
}
else {
[ Microsoft.Win32.RegistryHive ] $hive , $keyPath = switch ( $Container ) {
'Machine' { 'LocalMachine' , 'SYSTEM\CurrentControlSet\Control\Session Manager\Environment' }
'User' { 'CurrentUser' , 'Environment' }
2021-05-14 21:09:45 +00:00
}
2021-06-08 21:46:20 +00:00
$hiveKey = $envKey = $null
try {
2021-06-11 12:36:52 +00:00
$hiveKey = [ Microsoft.Win32.RegistryKey ] :: OpenRemoteBaseKey ( $hive , '' )
2021-06-08 21:46:20 +00:00
$envKey = $hiveKey . OpenSubKey ( $keyPath , $true )
$rawPath = $envKey . GetValue ( 'PATH' , '' , 'DoNotExpandEnvironmentNames' )
$envPaths = [ Collections.Generic.List[String] ] ( $rawPath -split ( [ IO.Path ] :: PathSeparator ) )
if ( $envPaths -notcontains $Path ) {
$envPaths . Add ( $Path )
$envKey . SetValue ( 'PATH' , ( $envPaths -join ( [ IO.Path ] :: PathSeparator ) ) , 'ExpandString' )
}
}
finally {
2021-06-11 12:36:52 +00:00
if ( $envKey ) { $envKey . Close ( ) }
if ( $hiveKey ) { $hiveKey . Close ( ) }
2021-06-08 21:46:20 +00:00
}
}
2021-05-14 21:09:45 +00:00
}
2021-06-08 21:46:20 +00:00
2021-05-14 21:09:45 +00:00
filter Get-FileSize {
'{0:N2} {1}' -f $ (
if ( $_ -lt 1 kb ) { $_ , 'Bytes' }
elseif ( $_ -lt 1 mb ) { ( $_ / 1 kb ) , 'KB' }
elseif ( $_ -lt 1 gb ) { ( $_ / 1 mb ) , 'MB' }
elseif ( $_ -lt 1 tb ) { ( $_ / 1 gb ) , 'GB' }
elseif ( $_ -lt 1 pb ) { ( $_ / 1 tb ) , 'TB' }
else { ( $_ / 1 pb ) , 'PB' }
)
}
function Get-FileWCSynchronous {
param (
[ Parameter ( Mandatory = $true ) ]
[ string ] $url ,
[ string ] $destinationFolder = " $env:USERPROFILE \Downloads " ,
[ switch ] $includeStats
)
$wc = New-Object -TypeName Net . WebClient
$wc . UseDefaultCredentials = $true
$destination = Join-Path -Path $destinationFolder -ChildPath ( $url | Split-Path -Leaf )
$start = Get-Date
$wc . DownloadFile ( $url , $destination )
$elapsed = ( ( Get-Date ) - $start ) . ToString ( 'hh\:mm\:ss' )
$totalSize = ( Get-Item -Path $destination ) . Length | Get-FileSize
if ( $includeStats . IsPresent ) {
[ PSCustomObject ] @ { Name = $MyInvocation . MyCommand ; TotalSize = $totalSize ; Time = $elapsed }
}
Get-Item -Path $destination | Unblock-File
}
2021-06-08 11:57:01 +00:00
function Test-AbsolutePath {
Param (
[ Parameter ( Mandatory = $True ) ]
[ ValidateScript ( { [ System.IO.Path ] :: IsPathRooted ( $_ ) } ) ]
[ String ] $Path
)
}
2021-06-09 12:43:36 +00:00
function Exec
{
[ CmdletBinding ( ) ]
param (
[ Parameter ( Position = 0 , Mandatory = 1 ) ] [ string ] $cmd ,
[ Parameter ( ) ] [ string ] $errorMessage ,
[ parameter ( ValueFromRemainingArguments = $true ) ]
[ string[] ] $Passthrough
)
& $cmd @Passthrough
if ( $lastexitcode -ne 0 ) {
if ( ! ( $errorMessage ) ) {
throw ( 'Exec: Error executing command {0} with arguments ''{1}''' -f $cmd , " $Passthrough " )
} else {
throw ( 'Exec: ' + $errorMessage )
}
}
}
2021-05-14 21:09:45 +00:00
$ErrorActionPreference = 'Stop'
2021-06-08 21:46:20 +00:00
$GhcupBasePrefixEnv = [ System.Environment ] :: GetEnvironmentVariable ( 'GHCUP_INSTALL_BASE_PREFIX' , 'user' )
2021-06-08 11:57:01 +00:00
2021-06-08 21:46:20 +00:00
if ( $GhcupBasePrefixEnv ) {
$defaultGhcupBasePrefix = $GhcupBasePrefixEnv
2021-06-08 11:57:01 +00:00
} else {
$defaultGhcupBasePrefix = 'C:\'
}
2021-06-08 21:46:20 +00:00
if ( $Silent -and ! ( $InstallDir ) ) {
$GhcupBasePrefix = $defaultGhcupBasePrefix
} elseif ( $InstallDir ) {
if ( ! ( Test-Path -LiteralPath ( '{0}' -f $InstallDir ) -IsValid ) ) {
Print-Msg -color Red -msg " Not a valid directory! "
Exit 1
} elseif ( ! ( Split-Path -IsAbsolute -Path " $InstallDir " ) ) {
Print-Msg -color Red -msg " Non-absolute Path specified! "
Exit 1
2021-06-08 11:57:01 +00:00
} else {
2021-06-08 21:46:20 +00:00
$GhcupBasePrefix = $InstallDir
}
} else {
while ( $true ) {
Print-Msg -color Magenta -msg ( 'Where to install to (this should be a short Path, preferably a Drive like ''C:\''){1}Press enter to accept the default [{0}]:' -f $defaultGhcupBasePrefix , " `n " )
$basePrefixPrompt = Read-Host
$GhcupBasePrefix = ( $defaultGhcupBasePrefix , $basePrefixPrompt ) [ [bool ] $basePrefixPrompt ]
if ( ! ( $GhcupBasePrefix . EndsWith ( '\' ) ) ) {
$GhcupBasePrefix = ( '{0}\' -f $GhcupBasePrefix )
}
if ( ! ( $GhcupBasePrefix ) ) {
Print-Msg -color Red -msg " No directory specified! "
} elseif ( ! ( Test-Path -LiteralPath ( '{0}' -f $GhcupBasePrefix ) ) ) {
Print-Msg -color Red -msg " Directory does not exist, need to specify an existing Drive/Directory "
} elseif ( ! ( Split-Path -IsAbsolute -Path " $GhcupBasePrefix " ) ) {
Print-Msg -color Red -msg " Invalid/Non-absolute Path specified "
} else {
Break
}
2021-06-08 11:57:01 +00:00
}
}
2021-06-08 21:46:20 +00:00
2021-06-08 11:57:01 +00:00
Print-Msg -msg ( 'Setting env variable GHCUP_INSTALL_BASE_PREFIX to ''{0}''' -f $GhcupBasePrefix )
$null = [ Environment ] :: SetEnvironmentVariable ( " GHCUP_INSTALL_BASE_PREFIX " , $GhcupBasePrefix , [ System.EnvironmentVariableTarget ] :: User )
$GhcupDir = ( '{0}\ghcup' -f $GhcupBasePrefix )
2021-05-14 21:09:45 +00:00
$MsysDir = ( '{0}\msys64' -f $GhcupDir )
$Bash = ( '{0}\usr\bin\bash' -f $MsysDir )
2021-06-07 12:48:06 +00:00
$BootstrapUrl = 'https://www.haskell.org/ghcup/sh/bootstrap-haskell-windows'
2021-06-07 15:06:44 +00:00
$GhcupMsys2 = [ System.Environment ] :: GetEnvironmentVariable ( 'GHCUP_MSYS2' , 'user' )
2021-05-14 21:09:45 +00:00
Print-Msg -msg 'Preparing for GHCup installation...'
2021-06-08 11:57:01 +00:00
if ( Test-Path -LiteralPath ( '{0}' -f $GhcupDir ) ) {
2021-06-08 21:46:20 +00:00
Print-Msg -msg ( 'GHCup already installed at ''{0}''...' -f $GhcupDir )
if ( $Overwrite ) {
$decision = 0
} elseif ( ! ( $Silent ) ) {
$decision = $Host . UI . PromptForChoice ( 'Install GHCup'
, 'GHCup is already installed, what do you want to do?'
2021-06-09 12:43:48 +00:00
, [ System.Management.Automation.Host.ChoiceDescription[] ] @ ( '&Reinstall'
2021-06-08 21:46:20 +00:00
'&Continue'
'&Abort' ) , 1 )
} else {
$decision = 1
}
2021-05-14 21:09:45 +00:00
if ( $decision -eq 0 ) {
$suffix = [ IO.Path ] :: GetRandomFileName ( )
Print-Msg -msg ( 'Backing up {0} to {0}-{1} ...' -f $GhcupDir , $suffix )
Rename-Item -Path ( '{0}' -f $GhcupDir ) -NewName ( '{0}-{1}' -f $GhcupDir , $suffix )
} elseif ( $decision -eq 1 ) {
Print-Msg -msg 'Continuing installation...'
} elseif ( $decision -eq 2 ) {
2021-06-08 21:46:20 +00:00
Exit 0
2021-05-14 21:09:45 +00:00
}
}
$null = New-Item -Path ( '{0}' -f $GhcupDir ) -ItemType 'directory' -ErrorAction SilentlyContinue
$null = New-Item -Path ( '{0}' -f $GhcupDir ) -Name 'bin' -ItemType 'directory' -ErrorAction SilentlyContinue
Print-Msg -msg 'First checking for Msys2...'
2021-06-08 11:57:01 +00:00
if ( ! ( Test-Path -Path ( '{0}' -f $MsysDir ) ) ) {
2021-06-08 21:46:20 +00:00
if ( $Silent ) {
$msys2Decision = 0
} else {
$msys2Decision = $Host . UI . PromptForChoice ( 'Install MSys2'
, 'Do you want GHCup to install a default MSys2 toolchain (recommended)?'
2021-06-09 12:43:48 +00:00
, [ System.Management.Automation.Host.ChoiceDescription[] ] @ ( '&Yes'
2021-06-08 21:46:20 +00:00
'&No' ) , 0 )
}
2021-06-07 15:06:44 +00:00
if ( $msys2Decision -eq 0 ) {
2021-05-14 21:09:45 +00:00
Print-Msg -msg ( '...Msys2 doesn''t exist, installing into {0} ...this may take a while' -f $MsysDir )
# Download the archive
Print-Msg -msg 'Downloading Msys2 archive...'
$archive = 'msys2-x86_64-latest.sfx.exe'
if ( Get-Command -Name 'curl.exe' -ErrorAction SilentlyContinue ) {
2021-06-09 12:43:36 +00:00
Exec " curl.exe " '-o' ( '{0}\{1}' -f $env:TEMP , $archive ) ( 'https://repo.msys2.org/distrib/{0}' -f $archive )
2021-05-14 21:09:45 +00:00
} else {
Get-FileWCSynchronous -url ( 'https://repo.msys2.org/distrib/{0}' -f $archive ) -destinationFolder " $env:TEMP " -includeStats
}
Print-Msg -msg 'Extracting Msys2 archive...'
$null = & " $env:TEMP \ $archive " '-y' ( '-o{0}' -f $GhcupDir ) # Extract
Remove-Item -Path ( '{0}/{1}' -f $env:TEMP , $archive )
Print-Msg -msg 'Processing MSYS2 bash for first time use...'
2021-06-09 12:43:36 +00:00
Exec " $Bash " '-lc' 'exit'
2021-05-14 21:09:45 +00:00
2021-06-09 12:43:36 +00:00
Exec " $env:windir \system32\taskkill.exe " / F / FI `" MODULES eq msys - 2.0 . dll `"
2021-05-14 21:09:45 +00:00
Print-Msg -msg 'Upgrading full system...'
2021-06-09 12:43:36 +00:00
Exec " $Bash " '-lc' 'pacman --noconfirm -Syuu'
2021-05-14 21:09:45 +00:00
Print-Msg -msg 'Upgrading full system twice...'
2021-06-09 12:43:36 +00:00
Exec " $Bash " '-lc' 'pacman --noconfirm -Syuu'
2021-05-14 21:09:45 +00:00
2021-06-08 21:46:20 +00:00
if ( $Quick ) {
$ghcBuildDeps = $Quick
} elseif ( ! ( $Silent ) ) {
$ghcBuildDeps = $Host . UI . PromptForChoice ( 'Install Dependencies'
2021-06-11 20:12:23 +00:00
, 'Install a standard set of mingw64 packages to be able to build various haskell packages requiring unix libraries? (recommended, however this might take a while... if you skip this, you might have to do it manually later)'
2021-06-09 12:43:48 +00:00
, [ System.Management.Automation.Host.ChoiceDescription[] ] @ ( '&Yes'
2021-06-08 21:46:20 +00:00
'&No' ) , 0 )
} else {
$ghcBuildDeps = 0
}
if ( $ghcBuildDeps -eq 0 ) {
2021-06-11 20:12:23 +00:00
Print-Msg -msg 'Installing Dependencies...'
Exec " $Bash " '-lc' 'pacman --noconfirm -S --needed git tar curl wget base-devel gettext binutils autoconf make libtool automake pkgconf python p7zip patch unzip'
2021-06-08 21:46:20 +00:00
}
2021-05-14 21:09:45 +00:00
Print-Msg -msg 'Updating SSL root certificate authorities...'
2021-06-09 12:43:36 +00:00
Exec " $Bash " '-lc' 'pacman --noconfirm -S ca-certificates'
2021-05-14 21:09:45 +00:00
Print-Msg -msg 'Setting default home directory...'
2021-06-09 12:43:36 +00:00
Exec " $Bash " '-lc' " sed -i -e 's/db_home:.* $ /db_home: windows/' /etc/nsswitch.conf "
2021-06-07 15:06:44 +00:00
} elseif ( $msys2Decision -eq 1 ) {
2021-06-08 11:57:01 +00:00
Print-Msg -color Yellow -msg 'Skipping MSys2 installation.'
while ( $true ) {
if ( $GhcupMsys2 ) {
$defaultMsys2Dir = $GhcupMsys2
Print-Msg -color Magenta -msg ( 'Input existing MSys2 toolchain directory. Press enter to accept the default [{0}]:' -f $defaultMsys2Dir )
$MsysDirPrompt = Read-Host
$MsysDir = ( $defaultMsys2Dir , $MsysDirPrompt ) [ [bool ] $MsysDirPrompt ]
} else {
Print-Msg -color Magenta -msg 'Input existing MSys2 toolchain directory:'
$MsysDir = Read-Host
}
if ( ! ( $MsysDir ) ) {
Print-Msg -color Red -msg " No directory specified! "
} elseif ( ! ( Test-Path -LiteralPath ( '{0}' -f $MsysDir ) ) ) {
Print-Msg -color Red -msg ( 'MSys2 installation at ''{0}'' could not be found!' -f $MsysDir )
} elseif ( ! ( Split-Path -IsAbsolute -Path " $MsysDir " ) ) {
Print-Msg -color Red -msg " Invalid/Non-absolute Path specified "
} else {
Break
}
2021-06-07 15:06:44 +00:00
}
2021-06-08 11:57:01 +00:00
Print-Msg -msg ( 'Setting GHCUP_MSYS2 env var to ''{0}''' -f $MsysDir )
2021-06-07 15:06:44 +00:00
$null = [ Environment ] :: SetEnvironmentVariable ( " GHCUP_MSYS2 " , $MsysDir , [ System.EnvironmentVariableTarget ] :: User )
$Bash = ( '{0}\usr\bin\bash' -f $MsysDir )
}
2021-05-14 21:09:45 +00:00
} else {
Print-Msg -msg ( '...Msys2 found in {0} ...skipping Msys2 installation.' -f $MsysDir )
}
Print-Msg -msg 'Creating shortcuts...'
Create-Shortcut -SourceExe ( '{0}\msys2_shell.cmd' -f $MsysDir ) -ArgumentsToSourceExe '-mingw64' -DestinationPath ( '{0}\Desktop\Mingw haskell shell.lnk' -f $HOME )
Create-Shortcut -SourceExe 'https://www.msys2.org/docs/package-management' -ArgumentsToSourceExe '' -DestinationPath ( '{0}\Desktop\Mingw package management docs.url' -f $HOME )
Print-Msg -msg ( 'Adding {0}\bin to Users Path...' -f $GhcupDir )
2021-06-08 21:46:20 +00:00
Add-EnvPath -Path ( '{0}\bin' -f ( [ System.IO.Path ] :: GetFullPath ( " $GhcupDir " ) ) ) -Container 'User'
2021-05-14 21:09:45 +00:00
2021-06-10 08:58:24 +00:00
if ( $CabalDir ) {
$CabDirEnv = $CabalDir
if ( ! ( $CabDirEnv ) ) {
Print-Msg -color Red -msg " No directory specified! "
Exit 1
} elseif ( ! ( Split-Path -IsAbsolute -Path " $CabDirEnv " ) ) {
Print-Msg -color Red -msg " Invalid/Non-absolute Path specified "
Exit 1
}
} elseif ( ! ( $Silent ) ) {
while ( $true ) {
$defaultCabalDir = ( '{0}\cabal' -f $GhcupBasePrefix )
Print-Msg -color Magenta -msg ( 'Specify Cabal directory (this is where haskell packages end up). Press enter to accept the default [{0}]:' -f $defaultCabalDir )
$CabalDirPrompt = Read-Host
$CabDirEnv = ( $defaultCabalDir , $CabalDirPrompt ) [ [bool ] $CabalDirPrompt ]
if ( ! ( $CabDirEnv ) ) {
Print-Msg -color Red -msg " No directory specified! "
} elseif ( ! ( Split-Path -IsAbsolute -Path " $CabDirEnv " ) ) {
Print-Msg -color Red -msg " Invalid/Non-absolute Path specified "
} else {
Break
}
}
} else {
$CabDirEnv = ( '{0}\cabal' -f $GhcupBasePrefix )
}
$CabalDirFull = [ System.IO.Path ] :: GetFullPath ( " $CabDirEnv " )
Print-Msg -msg ( 'Setting CABAL_DIR to ''{0}''' -f $CabalDirFull )
$null = [ Environment ] :: SetEnvironmentVariable ( " CABAL_DIR " , $CabalDirFull , [ System.EnvironmentVariableTarget ] :: User )
2021-05-14 21:09:45 +00:00
Print-Msg -msg 'Starting GHCup installer...'
2021-06-07 12:48:06 +00:00
2021-06-07 15:06:44 +00:00
$Msys2Shell = ( '{0}\msys2_shell.cmd' -f $MsysDir )
2021-06-08 21:46:20 +00:00
if ( $Silent ) {
$SilentExport = 'export BOOTSTRAP_HASKELL_NONINTERACTIVE=1 ;'
} else {
$SilentExport = ''
}
2021-06-07 12:48:06 +00:00
if ( ( Get-Process -ID $PID ) . ProcessName . StartsWith ( " bootstrap-haskell " ) ) {
2021-06-10 08:58:24 +00:00
Exec " $Bash " '-lc' ( '{4} [ -n ''{1}'' ] && export GHCUP_MSYS2=$(cygpath -m ''{1}'') ; [ -n ''{2}'' ] && export GHCUP_INSTALL_BASE_PREFIX=$(cygpath -m ''{2}/'') ; export PATH=$(cygpath -u ''{3}/bin''):$PATH ; export CABAL_DIR=''{5}'' ; curl --proto ''=https'' --tlsv1.2 -sSf {0} | bash' -f $BootstrapUrl , $MsysDir , $GhcupBasePrefix , $GhcupDir , $SilentExport , $CabalDirFull )
2021-06-07 12:48:06 +00:00
} else {
2021-06-10 08:58:24 +00:00
Exec " $Msys2Shell " '-mingw64' '-mintty' '-c' ( '{4} [ -n ''{1}'' ] && export GHCUP_MSYS2=$(cygpath -m ''{1}'') ; [ -n ''{2}'' ] && export GHCUP_INSTALL_BASE_PREFIX=$(cygpath -m ''{2}/'') ; export PATH=$(cygpath -u ''{3}/bin''):$PATH ; export CABAL_DIR=''{5}'' ; trap ''echo Press any key to exit && read -n 1 && exit'' 2 ; curl --proto =https --tlsv1.2 -sSf -k {0} | bash ; echo ''Press any key to exit'' && read -n 1' -f $BootstrapUrl , $MsysDir , $GhcupBasePrefix , $GhcupDir , $SilentExport , $CabalDirFull )
2021-06-07 12:48:06 +00:00
}
2021-05-14 21:09:45 +00:00
# SIG # Begin signature block
# MIID4QYJKoZIhvcNAQcCoIID0jCCA84CAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB
# gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR
# AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQUVqKek181kF/Jx/P7z176herc
# ZyCgggH/MIIB+zCCAWSgAwIBAgIQGOezhGS1A5tHh9VubW0liDANBgkqhkiG9w0B
# AQUFADAYMRYwFAYDVQQDDA1KdWxpYW4gT3NwYWxkMB4XDTIxMDUzMDE4Mzk1OVoX
# DTI1MDUzMDAwMDAwMFowGDEWMBQGA1UEAwwNSnVsaWFuIE9zcGFsZDCBnzANBgkq
# hkiG9w0BAQEFAAOBjQAwgYkCgYEAs76XCXYPM14buR1RkVKhOB8pyM4Df6kPaz75
# nkbA0nq1VmMhBfCYFWyYHd7jniqTH0LoAKGGquN1bniREaCP9j2pFWpMIgLpQH3H
# +jpsfmxV2BTG8q+Jok88gTXS1FlAk72E85zO/Jhr6Fja1aFYAdibBRsRxcVMTVh7
# 4AGLNGUCAwEAAaNGMEQwEwYDVR0lBAwwCgYIKwYBBQUHAwMwHQYDVR0OBBYEFC+R
# hdhPo0Ty5HnzHyo1pN35IfZQMA4GA1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQUF
# AAOBgQAl3IdBVIwbJJDp7BksMYPeM4ivB3UyNvlw8aVxGwAzNgdSaezYIdMFtKXV
# CSv5bd4VnFRAPDJW9dhW0h3SkeJUoklUxMjKXhR3qygQhSxPDjIatAuOCffGACba
# ZZ7Om40b+pKXc6i/HnlApk9DGbXJ59bFcLGGcZ9QjoUae6Ex1DGCAUwwggFIAgEB
# MCwwGDEWMBQGA1UEAwwNSnVsaWFuIE9zcGFsZAIQGOezhGS1A5tHh9VubW0liDAJ
# BgUrDgMCGgUAoHgwGAYKKwYBBAGCNwIBDDEKMAigAoAAoQKAADAZBgkqhkiG9w0B
# CQMxDAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAj
# BgkqhkiG9w0BCQQxFgQUosm9nN1JgajqSBa1cUwxxhLrAsYwDQYJKoZIhvcNAQEB
# BQAEgYCnKzfsH1aDjS6xkC/uymjaBowHSnh6nFu2AkjcKu8RgcBZzP5SLBXgU9wm
# aED5Ujwyq3Qre+TGVRUqwkEauDhQiX2A008G00fRO6+di6yJRCRn5eaRAbdU3Xww
# E5VhEwLBnwzWrvLKtdEclhgUCo5Tq87QMXVdgX4aRmunl4ZE+Q==
# SIG # End signature block
2021-06-11 20:12:23 +00:00