Modules/PSNativeCmdDevKit/0.1.0/PSNativeCmdDevKit.psm1
#Region './Public/Add-SudoPreferenceRule.ps1' 0 # Add-SudoPreferenceRule -Executable dpkg -ParameterFilterRule {$_.parameters -contains '-i' -or $_.parameters -contains '--install')} # Add-SudoPreferenceRule -Executable dpkg -ParameterFilterRule {$_.parameters -contains '-W' -or $_.parameters -contains '--show')} -SudoUser otheruser # Add-SudoPreferenceRule -EnableSudoForAllCommands # Add-SudoPreferenceRule -EnableSudoForAllCommands -SudoUser otheruser # Add-SudoPreferenceRule -DisableSudoForAllCommands function Add-SudoPreferenceRule { param ( [Parameter(ParameterSetName = 'Sudo', Mandatory = $true)] [Alias('Command')] # The binary or command the rule will affect. [string] $Executable, [Parameter(ParameterSetName = 'Sudo', Mandatory = $true)] # The Parameter filter to be evaluated for the command. # if you want to use sudo for an Executable, regardless of the parameters, use: # `-ParameterFilterRule *` or `-ParameterFilterRule {$true}` # Otherwise, you can evaluate the Parameters to be used, populated the $Args variable: # `-ParameterFilterRule {$args -contains '-i' -or $args -contains '--install'} [string] $ParameterFilterRule, [Parameter(ParameterSetName = 'SudoAll', Mandatory = $true)] # This will Enable sudo for any command, but won't destroy your # registered settings. You can set a $SudoUser to be used along. [switch] $EnableSudoForAllCommands, [Parameter(ParameterSetName = 'NoSudoAll', Mandatory = $true)] # This will ensure sudo is not automatically added to each command, # instead it will use the Sudo Preference rules registered with `Add-SudoPreferenceRule`. [switch] $DisableSudoForAllCommands, [Parameter(ParameterSetName = 'Sudo')] [Parameter(ParameterSetName = 'SudoAll')] # The executable that is invoked with sudo should be run as this user. # the resulting command invoked will be `sudo <sudo user> <executable> <parameters>`. [string] $SudoUser ) if ($script:SudoPreferenceRules -isnot [System.Collections.ArrayList]) { # There is no default rules store, let's create an array list $script:SudoPreferenceRules = [System.Collections.ArrayList]::new() } if ($EnableSudoForAllCommands.IsPresent -or $DisableSudoForAllCommands.IsPresent) { $Script:SudoAll = switch ($PSCmdlet.ParameterSetName) { NoSudoAll { $false } SudoAll { $true } } # If sudoUser is specified, set to SudoAllAs. Clean up if disabling SudoAll $script:SudoAllAs = $SudoUser return } elseif ($Executable -eq '*') { $Script:SudoAll = switch -regex ($ParameterFilterRule.Trim()) { '^\$true$' { $true } '^\$false$' { $false } Default { $true } } $script:SudoAllAs = $SudoUser } $index = $null if (Get-SudoPreferenceRule -Executable $Executable -ParameterFilterRule $ParameterFilterRule) { Write-Warning "Sudo Preference Rule found. Replacing" $index = [int](Remove-SudoPreferenceRule -Executable $Executable -ParameterFilterRule $ParameterFilterRule) } # copy hash with Executable, ParameterFilterRule, and SudoUser if present $newRule = @{ Executable = $Executable ParameterFilterRule = $ParameterFilterRule SudoUser = $SudoUser } if ($index) { Write-Debug "Replacing Sudo rule for '$Executable' with filter '$ParameterFilterRule' at index $index" $null = $script:SudoPreferenceRules.Insert($index, $newRule) } else { Write-Debug "Adding Sudo rule for '$Executable' with filter '$ParameterFilterRule'" $null = $script:SudoPreferenceRules.Add($newRule) } } #EndRegion './Public/Add-SudoPreferenceRule.ps1' 102 #Region './Public/Get-PropertyHashFromListOutput.ps1' 0 function Get-PropertyHashFromListOutput { [CmdletBinding(DefaultParameterSetName = 'AddExtraPropertiesUnderKey')] [OutputType([hashtable])] param ( [Parameter(ValueFromPipeline = $true, ValueFromPipelineByPropertyName = $true, Mandatory = $true)] [Object] # Output from a command, typically the result of Invoke-LinuxCommand. # Error records will be handled by the scriptblock in -ErrorHandling parameter. # The latter defaults to send the error record to Write-Error. $Output, [Parameter()] # Regex with 'property' & 'val' Named groups # of a string to extract an hashtable key/value pair from a string. [regex] $Regex = '^\s*(?<property>[\w-\s]*):\s*(?<val>.*)', [Parameter()] # List of property names allowed to be parsed. # Default to '*' for all properties, otherwise the parsed properties # not listed here will either be discarded if -DiscardExtraProperties is set # or will be added to a hashtable under the key named $AddExtraPropertiesAsKey. [string[]] $AllowedPropertyName = '*', [Parameter(ValueFromPipelineByPropertyName = $true, ParameterSetName = 'DiscardExtraProperties')] # When only a limited number of Property named is allowed using -AllowedPropertyName # parameter, the extra properties will be discarded. [switch] $DiscardExtraProperties, [Parameter(ParameterSetName = 'AddExtraPropertiesUnderKey')] # When only a limited number of Property named is allowed using -AllowedPropertyName # parameter, the extra properties will be added under the `$property[$AddExtraPropertiesAsKey]` # hash. For instance, `$property['ExtraProperties']['NotAllowedPropertyName'] = $ParsedValue` [string] $AddExtraPropertiesAsKey = 'ExtraProperties', [Parameter()] # When the output of a native command has had its `STDERR` redirected # using `2>&1`, we'll send the ErrorRecords (output from STDERR) to # this scriptblock. By default: `$errorRecord | &{ Write-Error $_}`. [scriptblock] $ErrorHandling = { Write-Error $_ } ) begin { $properties = @{} if (-not $DiscardExtraProperties.isPresent) { $properties[$AddExtraPropertiesAsKey] = @{} } } process { foreach ($line in $Output) { Write-Debug "Output Line: $line" if ($line -is [System.Management.Automation.ErrorRecord]) { $line | &$ErrorHandling } elseif ($line -match $Regex) { $propertyName = $Matches.property.replace('-','').replace(' ','') if ($AllowedPropertyName -contains '*' -or $AllowedPropertyName -contains $propertyName) { $properties.Add($propertyName, $Matches.val) } else { if (-not $DiscardExtraProperties.isPresent) { Write-Debug " Adding Property '$propertyName' to $AddExtraPropertiesAsKey" $properties[$AddExtraPropertiesAsKey].Add($propertyName, $Matches.val) } } $lastProperty = $propertyName } else { if (-not $lastProperty) { Write-Verbose $line } elseif ($AllowedPropertyName -contains '*' -or $AllowedPropertyName -contains $lastProperty) { Write-Debug " Adding second line to property $lastProperty" $properties[$lastProperty] += "`n" + $line.TrimEnd() } else { $properties[$AddExtraPropertiesAsKey][$lastProperty] += $line.Trim() } } } } end { if ($properties[$AddExtraPropertiesAsKey].Count -eq 0) { Write-Debug "No Extra properties where found, removing unnecessary key '$AddExtraPropertiesAsKey'" $properties.Remove($AddExtraPropertiesAsKey) } $properties } } #EndRegion './Public/Get-PropertyHashFromListOutput.ps1' 114 #Region './Public/Get-SudoPreference.ps1' 0 function Get-SudoPreference { [CmdletBinding()] [OutputType([hashtable])] param ( [Parameter(Mandatory = $true)] [Alias('Command')] # The binary or command to be executed. [string] $Executable, [Parameter()] # List of parameters to pass to the invocation that will be # evaluated against the registered Sudo Preference Rules. [String[]] $Parameters ) if ($script:SudoAll) { @{ $Sudo = $true $SudoAs = $script:SudoAllAs } } elseif ($script:SudoPreferenceRules) { $enumerator = $script:SudoPreferenceRules.GetEnumerator() $RuleMatchFound = $false while ($enumerator.MoveNext() -and -not $RuleMatchFound) { $RuleMatchFound = $script:SudoPreferenceRules | Where-Object -FilterScript { $Executable -eq $_.Executable -and ($_.ParameterFilterRule.ToString().Trim() -eq '*' -or [scriptblock]::create($_.ParameterFilterRule).Invoke($Parameters)) } | Select-Object -First 1 } if ($RuleMatchFound) { return [hashtable]$RuleMatchFound } else { Write-Debug "No matching rules for '$Executable' with params '$Parameters'" } } } #EndRegion './Public/Get-SudoPreference.ps1' 48 #Region './Public/Get-SudoPreferenceRule.ps1' 0 function Get-SudoPreferenceRule { [CmdletBinding(DefaultParameterSetName = 'all')] [OutputType([System.Object[]])] param ( [Parameter(ParameterSetName = 'byCommand', Mandatory = $true)] [Alias('Command')] # The binary or command to be executed. [string] $Executable, [Parameter(ParameterSetName = 'byCommand')] [string] $ParameterFilterRule, [Parameter(ParameterSetName = 'all')] [switch] $All ) if ($script:SudoPreferenceRules -isnot [System.Collections.ArrayList]) { # There is no default rules store, let's create an array list and return it $script:SudoPreferenceRules = [System.Collections.ArrayList]::new() } if ($PSCmdlet.ParameterSetName -eq 'All') { $script:SudoPreferenceRules } else { $script:SudoPreferenceRules.Where{ $_.Executable -eq $Executable -and $( if ($ParameterFilterRule -and $ParameterFilterRule.Trim() -ne '*') { $_.ParameterFilterRule -eq $ParameterFilterRule } else { $true } ) } } } #EndRegion './Public/Get-SudoPreferenceRule.ps1' 50 #Region './Public/Invoke-NativeCommand.ps1' 0 function Invoke-NativeCommand { [cmdletBinding()] param ( [Parameter(Mandatory = $true)] [Alias('Command')] # The binary or command you would like to execute. [string] $Executable, [Parameter()] # Whether you want to sudo the command invocation, on non-windows OSes. # If you want to sudo as a different user, use the parameter `-SudoAs`. [switch] $Sudo, [Parameter()] # Specify a user to sudo he command as. i.e.: `sudo otheruser ls -alh` [String] $SudoAs, [Parameter()] # list of Parameters to pass to the invocation. # For binaries and commands requiring a specific order # make sure it is respected as no further check is done. [String[]] $Parameters ) # If Sudo or SudoAs is not specified, lookup in the Module variable DefaultCommandToSudo if ( -not ($PSBoundParameters.ContainsKey('Sudo') -or $PSBoundParameters.ContainsKey('SudoAs')) ) { if ($DefaultSudo = Get-SudoPreference @PSBoundParameters) { $Sudo = $DefaultSudo.Sudo $SudoAs = $DefaultSudo.SudoAs } } [string[]]$CommandExpression = @() if ($SudoAs -and ($IsLinux -or $IsMacOS)) { $commandExpression += "sudo -u $SudoAs $Executable" } elseif ($Sudo -and ($IsLinux -or $IsMacOS)) { $commandExpression += "sudo $Executable" } else { $commandExpression += $Executable } $commandExpression += $Parameters # Mixes the Error stream and the success streams (redirect STDERR with STDOUT) # What was in STDERR will be of type [ErrorRecord] if you need to differentiate for parsing. $commandExpression += '2>&1' Write-Verbose -Message "Running #> $commandExpression" [scriptblock]$commandExpression = [scriptblock]::create($commandExpression) # Stream the output through the pipeline & $commandExpression } #EndRegion './Public/Invoke-NativeCommand.ps1' 67 #Region './Public/Remove-SudoPreferenceRule.ps1' 0 function Remove-SudoPreferenceRule { [cmdletBinding()] param ( [Parameter(ParameterSetName = 'ByValue', Mandatory = $true, ValueFromPipelineByPropertyName = $true)] [Alias('Command')] # The executable that has the rule applied to. [string] $Executable, [Parameter(ParameterSetName = 'ByValue', Mandatory = $true, ValueFromPipelineByPropertyName = $true)] # The parameter filter rule to match with the executable to remve. [string] $ParameterFilterRule, [Parameter(Dontshow = $true, ParameterSetName = 'ByIndex', Mandatory = $true, ValueFromPipelineByPropertyName = $true)] # Remove the Rule stored in the module's $script:SudoPreferenceRules by its index (advanced user only) [int] $index, [Parameter(ParameterSetName = 'All', Mandatory = $true)] # Remove all previously registered rules. [switch] $All ) begin { if ($script:SudoPreferenceRules -isnot [System.Collections.ArrayList]) { # There is no default rules store return } } process { if ($PSCmdlet.ParameterSetName -eq 'ByIndex') { $script:SudoPreferenceRules.RemoveAt($Index) return } elseif ($PSCmdlet.ParameterSetName -eq 'All') { $script:SudoPreferenceRules.Clear() return } $CurrentIndex = 0 $indexesToRemove = $script:SudoPreferenceRules.Foreach{ if ($_.Executable -eq $Executable -and ($_.ParameterFilterRule.ToString().Trim() -eq '*' -or $_.ParameterFilterRule -eq $ParameterFilterRule) ) { $CurrentIndex } $CurrentIndex++ } $indexesToRemove.Foreach{ $script:SudoPreferenceRules.RemoveAt($_) # return the Indexes where the rule has been removed $_ } } } #EndRegion './Public/Remove-SudoPreferenceRule.ps1' 68 # SIG # Begin signature block # MIInoQYJKoZIhvcNAQcCoIInkjCCJ44CAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCC/AJYVnB8kXTzN # D/TgdHk3/VlKZy4201atVRZEaL/flqCCDYEwggX/MIID56ADAgECAhMzAAACUosz # qviV8znbAAAAAAJSMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p # bmcgUENBIDIwMTEwHhcNMjEwOTAyMTgzMjU5WhcNMjIwOTAxMTgzMjU5WjB0MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB # AQDQ5M+Ps/X7BNuv5B/0I6uoDwj0NJOo1KrVQqO7ggRXccklyTrWL4xMShjIou2I # sbYnF67wXzVAq5Om4oe+LfzSDOzjcb6ms00gBo0OQaqwQ1BijyJ7NvDf80I1fW9O # L76Kt0Wpc2zrGhzcHdb7upPrvxvSNNUvxK3sgw7YTt31410vpEp8yfBEl/hd8ZzA # v47DCgJ5j1zm295s1RVZHNp6MoiQFVOECm4AwK2l28i+YER1JO4IplTH44uvzX9o # RnJHaMvWzZEpozPy4jNO2DDqbcNs4zh7AWMhE1PWFVA+CHI/En5nASvCvLmuR/t8 # q4bc8XR8QIZJQSp+2U6m2ldNAgMBAAGjggF+MIIBejAfBgNVHSUEGDAWBgorBgEE # AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUNZJaEUGL2Guwt7ZOAu4efEYXedEw # UAYDVR0RBEkwR6RFMEMxKTAnBgNVBAsTIE1pY3Jvc29mdCBPcGVyYXRpb25zIFB1 # ZXJ0byBSaWNvMRYwFAYDVQQFEw0yMzAwMTIrNDY3NTk3MB8GA1UdIwQYMBaAFEhu # ZOVQBdOCqhc3NyK1bajKdQKVMFQGA1UdHwRNMEswSaBHoEWGQ2h0dHA6Ly93d3cu # bWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01pY0NvZFNpZ1BDQTIwMTFfMjAxMS0w # Ny0wOC5jcmwwYQYIKwYBBQUHAQEEVTBTMFEGCCsGAQUFBzAChkVodHRwOi8vd3d3 # Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NlcnRzL01pY0NvZFNpZ1BDQTIwMTFfMjAx # MS0wNy0wOC5jcnQwDAYDVR0TAQH/BAIwADANBgkqhkiG9w0BAQsFAAOCAgEAFkk3 # uSxkTEBh1NtAl7BivIEsAWdgX1qZ+EdZMYbQKasY6IhSLXRMxF1B3OKdR9K/kccp # kvNcGl8D7YyYS4mhCUMBR+VLrg3f8PUj38A9V5aiY2/Jok7WZFOAmjPRNNGnyeg7 # l0lTiThFqE+2aOs6+heegqAdelGgNJKRHLWRuhGKuLIw5lkgx9Ky+QvZrn/Ddi8u # TIgWKp+MGG8xY6PBvvjgt9jQShlnPrZ3UY8Bvwy6rynhXBaV0V0TTL0gEx7eh/K1 # o8Miaru6s/7FyqOLeUS4vTHh9TgBL5DtxCYurXbSBVtL1Fj44+Od/6cmC9mmvrti # yG709Y3Rd3YdJj2f3GJq7Y7KdWq0QYhatKhBeg4fxjhg0yut2g6aM1mxjNPrE48z # 6HWCNGu9gMK5ZudldRw4a45Z06Aoktof0CqOyTErvq0YjoE4Xpa0+87T/PVUXNqf # 7Y+qSU7+9LtLQuMYR4w3cSPjuNusvLf9gBnch5RqM7kaDtYWDgLyB42EfsxeMqwK # WwA+TVi0HrWRqfSx2olbE56hJcEkMjOSKz3sRuupFCX3UroyYf52L+2iVTrda8XW # esPG62Mnn3T8AuLfzeJFuAbfOSERx7IFZO92UPoXE1uEjL5skl1yTZB3MubgOA4F # 8KoRNhviFAEST+nG8c8uIsbZeb08SeYQMqjVEmkwggd6MIIFYqADAgECAgphDpDS # AAAAAAADMA0GCSqGSIb3DQEBCwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMK # V2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0 # IENvcnBvcmF0aW9uMTIwMAYDVQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0 # ZSBBdXRob3JpdHkgMjAxMTAeFw0xMTA3MDgyMDU5MDlaFw0yNjA3MDgyMTA5MDla # MH4xCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdS # ZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMT # H01pY3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMTEwggIiMA0GCSqGSIb3DQEB # AQUAA4ICDwAwggIKAoICAQCr8PpyEBwurdhuqoIQTTS68rZYIZ9CGypr6VpQqrgG # OBoESbp/wwwe3TdrxhLYC/A4wpkGsMg51QEUMULTiQ15ZId+lGAkbK+eSZzpaF7S # 35tTsgosw6/ZqSuuegmv15ZZymAaBelmdugyUiYSL+erCFDPs0S3XdjELgN1q2jz # y23zOlyhFvRGuuA4ZKxuZDV4pqBjDy3TQJP4494HDdVceaVJKecNvqATd76UPe/7 # 4ytaEB9NViiienLgEjq3SV7Y7e1DkYPZe7J7hhvZPrGMXeiJT4Qa8qEvWeSQOy2u # M1jFtz7+MtOzAz2xsq+SOH7SnYAs9U5WkSE1JcM5bmR/U7qcD60ZI4TL9LoDho33 # X/DQUr+MlIe8wCF0JV8YKLbMJyg4JZg5SjbPfLGSrhwjp6lm7GEfauEoSZ1fiOIl # XdMhSz5SxLVXPyQD8NF6Wy/VI+NwXQ9RRnez+ADhvKwCgl/bwBWzvRvUVUvnOaEP # 6SNJvBi4RHxF5MHDcnrgcuck379GmcXvwhxX24ON7E1JMKerjt/sW5+v/N2wZuLB # l4F77dbtS+dJKacTKKanfWeA5opieF+yL4TXV5xcv3coKPHtbcMojyyPQDdPweGF # RInECUzF1KVDL3SV9274eCBYLBNdYJWaPk8zhNqwiBfenk70lrC8RqBsmNLg1oiM # CwIDAQABo4IB7TCCAekwEAYJKwYBBAGCNxUBBAMCAQAwHQYDVR0OBBYEFEhuZOVQ # BdOCqhc3NyK1bajKdQKVMBkGCSsGAQQBgjcUAgQMHgoAUwB1AGIAQwBBMAsGA1Ud # DwQEAwIBhjAPBgNVHRMBAf8EBTADAQH/MB8GA1UdIwQYMBaAFHItOgIxkEO5FAVO # 4eqnxzHRI4k0MFoGA1UdHwRTMFEwT6BNoEuGSWh0dHA6Ly9jcmwubWljcm9zb2Z0 # LmNvbS9wa2kvY3JsL3Byb2R1Y3RzL01pY1Jvb0NlckF1dDIwMTFfMjAxMV8wM18y # Mi5jcmwwXgYIKwYBBQUHAQEEUjBQME4GCCsGAQUFBzAChkJodHRwOi8vd3d3Lm1p # Y3Jvc29mdC5jb20vcGtpL2NlcnRzL01pY1Jvb0NlckF1dDIwMTFfMjAxMV8wM18y # Mi5jcnQwgZ8GA1UdIASBlzCBlDCBkQYJKwYBBAGCNy4DMIGDMD8GCCsGAQUFBwIB # FjNodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2RvY3MvcHJpbWFyeWNw # cy5odG0wQAYIKwYBBQUHAgIwNB4yIB0ATABlAGcAYQBsAF8AcABvAGwAaQBjAHkA # XwBzAHQAYQB0AGUAbQBlAG4AdAAuIB0wDQYJKoZIhvcNAQELBQADggIBAGfyhqWY # 4FR5Gi7T2HRnIpsLlhHhY5KZQpZ90nkMkMFlXy4sPvjDctFtg/6+P+gKyju/R6mj # 82nbY78iNaWXXWWEkH2LRlBV2AySfNIaSxzzPEKLUtCw/WvjPgcuKZvmPRul1LUd # d5Q54ulkyUQ9eHoj8xN9ppB0g430yyYCRirCihC7pKkFDJvtaPpoLpWgKj8qa1hJ # Yx8JaW5amJbkg/TAj/NGK978O9C9Ne9uJa7lryft0N3zDq+ZKJeYTQ49C/IIidYf # wzIY4vDFLc5bnrRJOQrGCsLGra7lstnbFYhRRVg4MnEnGn+x9Cf43iw6IGmYslmJ # aG5vp7d0w0AFBqYBKig+gj8TTWYLwLNN9eGPfxxvFX1Fp3blQCplo8NdUmKGwx1j # NpeG39rz+PIWoZon4c2ll9DuXWNB41sHnIc+BncG0QaxdR8UvmFhtfDcxhsEvt9B # xw4o7t5lL+yX9qFcltgA1qFGvVnzl6UJS0gQmYAf0AApxbGbpT9Fdx41xtKiop96 # eiL6SJUfq/tHI4D1nvi/a7dLl+LrdXga7Oo3mXkYS//WsyNodeav+vyL6wuA6mk7 # r/ww7QRMjt/fdW1jkT3RnVZOT7+AVyKheBEyIXrvQQqxP/uozKRdwaGIm1dxVk5I # RcBCyZt2WwqASGv9eZ/BvW1taslScxMNelDNMYIZdjCCGXICAQEwgZUwfjELMAkG # A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQx # HjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEoMCYGA1UEAxMfTWljcm9z # b2Z0IENvZGUgU2lnbmluZyBQQ0EgMjAxMQITMwAAAlKLM6r4lfM52wAAAAACUjAN # BglghkgBZQMEAgEFAKCBrjAZBgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgor # BgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQghHHrDk+3 # BfvfK2AUtjUlcaEdK21TeEXqRkY1ubb2RPYwQgYKKwYBBAGCNwIBDDE0MDKgFIAS # AE0AaQBjAHIAbwBzAG8AZgB0oRqAGGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbTAN # BgkqhkiG9w0BAQEFAASCAQAcq3vA0Fku6zFwkYkiUoQwHKtAXjlr10CkRnP9gF7F # 72s8rbjX1XsrqJ8vGaltb/HX8KhqGbQeupjoO2VC4n1nn8YU1mkCVfHzjPoZzQJj # MitcMwaTju+7l8TRipfPidg2oFVH6GMk2udIESVRTJNmnKiMmExlKfiiqKZw+Wml # 1cZASrpbJaRZ0PLAzeYjQgcSXv1IXTKqiPDZ5h2sFGO/LykqwMV7ZeA+il3F1puM # Bkb1HBXeQfZG+WDFns/NqgAprGQ8DKc8PiKKfUPowhx7xIMpFBxyFyGxmsIN9whT # BqrUaBCZ4C3qfp9VeEvIYckKbe/m6bh2urMsYLlsRSzloYIXADCCFvwGCisGAQQB # gjcDAwExghbsMIIW6AYJKoZIhvcNAQcCoIIW2TCCFtUCAQMxDzANBglghkgBZQME # AgEFADCCAVEGCyqGSIb3DQEJEAEEoIIBQASCATwwggE4AgEBBgorBgEEAYRZCgMB # MDEwDQYJYIZIAWUDBAIBBQAEIBtUHmvUPJ+YoSeC1qc7UssgGOTB2cU/HfsfL3LI # kNnbAgZiFmBi/7YYEzIwMjIwMzIyMjIyMzMwLjkxMVowBIACAfSggdCkgc0wgcox # CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRt # b25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNVBAsTHE1p # Y3Jvc29mdCBBbWVyaWNhIE9wZXJhdGlvbnMxJjAkBgNVBAsTHVRoYWxlcyBUU1Mg # RVNOOjhBODItRTM0Ri05RERBMSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFt # cCBTZXJ2aWNloIIRVzCCBwwwggT0oAMCAQICEzMAAAGZyI+vrbZ9vosAAQAAAZkw # DQYJKoZIhvcNAQELBQAwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0 # b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3Jh # dGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAwHhcN # MjExMjAyMTkwNTE2WhcNMjMwMjI4MTkwNTE2WjCByjELMAkGA1UEBhMCVVMxEzAR # BgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1p # Y3Jvc29mdCBDb3Jwb3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2Eg # T3BlcmF0aW9uczEmMCQGA1UECxMdVGhhbGVzIFRTUyBFU046OEE4Mi1FMzRGLTlE # REExJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2UwggIiMA0G # CSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQC4E/lXXKMsy9rVa2a8bRb0Ar/Pj4+b # KiAgMgKayvCMFn3ddGof8eWFgJWp5JdKjWjrnmW1r9tHpcP2kFpjXp2Udrj55jt5 # NYi1MERcoIo+E29XuCwFAMJftGdvsWea/OTQPIFsZEWqEteXdRncyVwct5xFzBIC # 1JWCdmfc7R59RMIyvgWjIz8356mweowkOstN1fe53KIJ8flrYILIQWsNRMOT3znA # GwIb9kyL54C6jZjFxOSusGYmVQ+Gr/qZQELw1ipx9s5jNP1LSpOpfTEBFu+y9KLN # BmMBARkSPpTFkGEyGSwGGgSdOi6BU6FPK+6urZ830jrRemK4JkIJ9tQhlGcIhAjh # cqZStn+38lRjVvrfbBI5EpI2NwlVIK2ibGW7sWeTAz/yNPNISUbQhGAJse/OgGj/ # 1qz/Ha9mqfYZ8BHchNxn08nWkqyrjrKicQyxuD8mCatTrVSbOJYfQyZdHR9a4vgy # GeZEXBYQNAlIuB37QCOAgs/VeDU8M4dc/IlrTyC0uV1SS4Gk8zV+5X5eRu+XORN8 # FWqzI6k/9y6cWwOWMK6aUN1XqLcaF/sm9rX84eKW2lhDc3C31WLjp8UOfOHZfPuy # y54xfilnhhCPy4QKJ9jggoqqeeEhCEfgDYjy+PByV/e5HDB2xHdtlL93wltAkI3a # Cxo84kVPBCa0OwIDAQABo4IBNjCCATIwHQYDVR0OBBYEFI26Vrg+nGWvrvIh0dQP # EonENR0QMB8GA1UdIwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8GA1UdHwRY # MFYwVKBSoFCGTmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01p # Y3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBsBggrBgEF # BQcBAQRgMF4wXAYIKwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9w # a2lvcHMvY2VydHMvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAo # MSkuY3J0MAwGA1UdEwEB/wQCMAAwEwYDVR0lBAwwCgYIKwYBBQUHAwgwDQYJKoZI # hvcNAQELBQADggIBAHGzWh29ibBNro3ns8E3EOHGsLB1Gzk90SFYUKBilIu4jDbR # 7qbvXNd8nnl/z5D9LKgw3T81jqy5tMiWp+p4jYBBk3PRx1ySqLUfhF5ZMWolRzW+ # cQZGXV38iSmdAUG0CpR5x1rMdPIrTczVUFsOYGqmkoUQ/dRiVL4iAXJLCNTj4x3Y # wIQcCPt0ijJVinPIMAYzA8f99BbeiskyI0BHGAd0kGUX2I2/puYnlyS8toBnANjh # 21xgvEuaZ2dvRqvWk/i1XIlO67au/XCeMTvXhPOIUmq80U32Tifw3SSiBKTyir7m # oWH1i7H2q5QAnrBxuyy//ZsDfARDV/Atmj5jr6ATfRHDdUanQpeoBS+iylNU6RAR # u8g+TMCu/ZndZmrs9w+8galUIGg+GmlNk07fXJ58Oc+qFqgNAsNkMi+dSzKkWGA4 # /klJFn0XichXL8+t7KOayXKGzQja6CdtCjisnyS8hbv4PKhaeMtf68wJWKKOs0tt # 2AJfYC5vSbH9ck8BGj2e/yQXEZEu88L5/fHK5XUk/IKXx3zaLkxXTSZ43Ea/WKXV # BzMasHZ3Pmny0moEekAXx1UhLNNYv4Vum33VirxSB6r/GKQxFSHu7yFfrWQpYyyD # H119TmhAedS8T1VabqdtO5ZP2E14TK82Vyxy3xEPelOo4dRIlhm7XY6k9B68MIIH # cTCCBVmgAwIBAgITMwAAABXF52ueAptJmQAAAAAAFTANBgkqhkiG9w0BAQsFADCB # iDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEyMDAGA1UEAxMp # TWljcm9zb2Z0IFJvb3QgQ2VydGlmaWNhdGUgQXV0aG9yaXR5IDIwMTAwHhcNMjEw # OTMwMTgyMjI1WhcNMzAwOTMwMTgzMjI1WjB8MQswCQYDVQQGEwJVUzETMBEGA1UE # CBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9z # b2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQ # Q0EgMjAxMDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAOThpkzntHIh # C3miy9ckeb0O1YLT/e6cBwfSqWxOdcjKNVf2AX9sSuDivbk+F2Az/1xPx2b3lVNx # WuJ+Slr+uDZnhUYjDLWNE893MsAQGOhgfWpSg0S3po5GawcU88V29YZQ3MFEyHFc # UTE3oAo4bo3t1w/YJlN8OWECesSq/XJprx2rrPY2vjUmZNqYO7oaezOtgFt+jBAc # nVL+tuhiJdxqD89d9P6OU8/W7IVWTe/dvI2k45GPsjksUZzpcGkNyjYtcI4xyDUo # veO0hyTD4MmPfrVUj9z6BVWYbWg7mka97aSueik3rMvrg0XnRm7KMtXAhjBcTyzi # YrLNueKNiOSWrAFKu75xqRdbZ2De+JKRHh09/SDPc31BmkZ1zcRfNN0Sidb9pSB9 # fvzZnkXftnIv231fgLrbqn427DZM9ituqBJR6L8FA6PRc6ZNN3SUHDSCD/AQ8rdH # GO2n6Jl8P0zbr17C89XYcz1DTsEzOUyOArxCaC4Q6oRRRuLRvWoYWmEBc8pnol7X # KHYC4jMYctenIPDC+hIK12NvDMk2ZItboKaDIV1fMHSRlJTYuVD5C4lh8zYGNRiE # R9vcG9H9stQcxWv2XFJRXRLbJbqvUAV6bMURHXLvjflSxIUXk8A8FdsaN8cIFRg/ # eKtFtvUeh17aj54WcmnGrnu3tz5q4i6tAgMBAAGjggHdMIIB2TASBgkrBgEEAYI3 # FQEEBQIDAQABMCMGCSsGAQQBgjcVAgQWBBQqp1L+ZMSavoKRPEY1Kc8Q/y8E7jAd # BgNVHQ4EFgQUn6cVXQBeYl2D9OXSZacbUzUZ6XIwXAYDVR0gBFUwUzBRBgwrBgEE # AYI3TIN9AQEwQTA/BggrBgEFBQcCARYzaHR0cDovL3d3dy5taWNyb3NvZnQuY29t # L3BraW9wcy9Eb2NzL1JlcG9zaXRvcnkuaHRtMBMGA1UdJQQMMAoGCCsGAQUFBwMI # MBkGCSsGAQQBgjcUAgQMHgoAUwB1AGIAQwBBMAsGA1UdDwQEAwIBhjAPBgNVHRMB # Af8EBTADAQH/MB8GA1UdIwQYMBaAFNX2VsuP6KJcYmjRPZSQW9fOmhjEMFYGA1Ud # HwRPME0wS6BJoEeGRWh0dHA6Ly9jcmwubWljcm9zb2Z0LmNvbS9wa2kvY3JsL3By # b2R1Y3RzL01pY1Jvb0NlckF1dF8yMDEwLTA2LTIzLmNybDBaBggrBgEFBQcBAQRO # MEwwSgYIKwYBBQUHMAKGPmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2Vy # dHMvTWljUm9vQ2VyQXV0XzIwMTAtMDYtMjMuY3J0MA0GCSqGSIb3DQEBCwUAA4IC # AQCdVX38Kq3hLB9nATEkW+Geckv8qW/qXBS2Pk5HZHixBpOXPTEztTnXwnE2P9pk # bHzQdTltuw8x5MKP+2zRoZQYIu7pZmc6U03dmLq2HnjYNi6cqYJWAAOwBb6J6Gng # ugnue99qb74py27YP0h1AdkY3m2CDPVtI1TkeFN1JFe53Z/zjj3G82jfZfakVqr3 # lbYoVSfQJL1AoL8ZthISEV09J+BAljis9/kpicO8F7BUhUKz/AyeixmJ5/ALaoHC # gRlCGVJ1ijbCHcNhcy4sa3tuPywJeBTpkbKpW99Jo3QMvOyRgNI95ko+ZjtPu4b6 # MhrZlvSP9pEB9s7GdP32THJvEKt1MMU0sHrYUP4KWN1APMdUbZ1jdEgssU5HLcEU # BHG/ZPkkvnNtyo4JvbMBV0lUZNlz138eW0QBjloZkWsNn6Qo3GcZKCS6OEuabvsh # VGtqRRFHqfG3rsjoiV5PndLQTHa1V1QJsWkBRH58oWFsc/4Ku+xBZj1p/cvBQUl+ # fpO+y/g75LcVv7TOPqUxUYS8vwLBgqJ7Fx0ViY1w/ue10CgaiQuPNtq6TPmb/wrp # NPgkNWcr4A245oyZ1uEi6vAnQj0llOZ0dFtq0Z4+7X6gMTN9vMvpe784cETRkPHI # qzqKOghif9lwY1NNje6CbaUFEMFxBmoQtB1VM1izoXBm8qGCAs4wggI3AgEBMIH4 # oYHQpIHNMIHKMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4G # A1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUw # IwYDVQQLExxNaWNyb3NvZnQgQW1lcmljYSBPcGVyYXRpb25zMSYwJAYDVQQLEx1U # aGFsZXMgVFNTIEVTTjo4QTgyLUUzNEYtOUREQTElMCMGA1UEAxMcTWljcm9zb2Z0 # IFRpbWUtU3RhbXAgU2VydmljZaIjCgEBMAcGBSsOAwIaAxUAku/zYujnqapN6BJ9 # MJ5jtgDrlOuggYMwgYCkfjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGlu # Z3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv # cmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDAN # BgkqhkiG9w0BAQUFAAIFAOXkdaswIhgPMjAyMjAzMjMwMDE5NTVaGA8yMDIyMDMy # NDAwMTk1NVowdzA9BgorBgEEAYRZCgQBMS8wLTAKAgUA5eR1qwIBADAKAgEAAgIR # cgIB/zAHAgEAAgIR/TAKAgUA5eXHKwIBADA2BgorBgEEAYRZCgQCMSgwJjAMBgor # BgEEAYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3DQEBBQUA # A4GBAAkO+NmTHZh23SxvKRXH7zDXRBEK/7Pbj0VF6Z1BrY9miFLXqvG6u+esGn7Q # xcVTdRSpmQiB3xMTV5JBeCiEpEm9HpXBkKVsUFgOAWH34eM98HljzqpgRfHoDT51 # N+KEG1tMSWSE2Rn6YZWVf5wm1R8EsH4YO8JBFW6PESKZbvfgMYIEDTCCBAkCAQEw # gZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcT # B1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UE # AxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAGZyI+vrbZ9vosA # AQAAAZkwDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzENBgsqhkiG9w0B # CRABBDAvBgkqhkiG9w0BCQQxIgQg+e5M2vkJSj1KNjKrh/oj/tkhoLicno5sFNJ7 # 0INvVpUwgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCBmfWn58qKN7WWpBTYO # rUO1BSCSnKPLC/G7wCOIc2JsfjCBmDCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1w # IFBDQSAyMDEwAhMzAAABmciPr622fb6LAAEAAAGZMCIEIIbKUSUHzKgZfk+CS57P # JmcF9Lkys7mfbuHMWcpDlZW9MA0GCSqGSIb3DQEBCwUABIICAAF2GFiT7D0nrxF9 # D0uscZi+YyA2A8zULoyoiDSrZjp5QPQjbPa8EttKsJjX0VfsSzvY9E9TOKvbZxG5 # 0MmhITNs2auPqGUDxvbROyiM6409/FwpwmTv1tGZFy6rHFBPiaXyfLKvHe86t1rI # fM9cX11WDXg8EoaBuwh89uZQqOIfgEqrxSfsmXQ/AWkEVEphJJ3Lz5CNh1N8CvBY # ZyGQ67KaSBhZtMK+h8ckVdPIl1d+R0FLYn88qVSBbgtagoZI0S1TTMz6/Ig2t6A9 # ewqglQOZzc665srhnZM3bf8MtiVudzAdSR3VjfzbQ2w+asVs0uqqk8owwsWONB/I # 6UDb9vmXgd7h/8IX3QQD4JCkIZKZMgnt9qRszbEIEXyCI+c+lJhjikRMVmVfqzrk # 6TUYxbhJXI/3QZ7OEdRe30gE7HpDucVdr/lBCRYKvY7JcmWztDNKi8H+Rev0uXNh # RvNpA/EIRk8PPEgCSdNg+SQQUyj0sK0Pb+XjNWLJdTWYjk5b11SQ6tMB9HY4w97g # bl2QsAxN2q/W/xY0tgPQY574HCnsNIg4u7TwdQpoOM53Zih3b2oqM2+GDXBKL9iI # byN9QidIwzriRMnP7jPbnv/G0oNKDu452BHnGx/AgzHeOeEBap0tO7bBmERy/yOJ # qbYOvslOIwGTtuSY2BaiD6KOmHw1 # SIG # End signature block |