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 # MIInzgYJKoZIhvcNAQcCoIInvzCCJ7sCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCC/AJYVnB8kXTzN # D/TgdHk3/VlKZy4201atVRZEaL/flqCCDYUwggYDMIID66ADAgECAhMzAAADri01 # UchTj1UdAAAAAAOuMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p # bmcgUENBIDIwMTEwHhcNMjMxMTE2MTkwODU5WhcNMjQxMTE0MTkwODU5WjB0MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB # AQD0IPymNjfDEKg+YyE6SjDvJwKW1+pieqTjAY0CnOHZ1Nj5irGjNZPMlQ4HfxXG # yAVCZcEWE4x2sZgam872R1s0+TAelOtbqFmoW4suJHAYoTHhkznNVKpscm5fZ899 # QnReZv5WtWwbD8HAFXbPPStW2JKCqPcZ54Y6wbuWV9bKtKPImqbkMcTejTgEAj82 # 6GQc6/Th66Koka8cUIvz59e/IP04DGrh9wkq2jIFvQ8EDegw1B4KyJTIs76+hmpV # M5SwBZjRs3liOQrierkNVo11WuujB3kBf2CbPoP9MlOyyezqkMIbTRj4OHeKlamd # WaSFhwHLJRIQpfc8sLwOSIBBAgMBAAGjggGCMIIBfjAfBgNVHSUEGDAWBgorBgEE # AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUhx/vdKmXhwc4WiWXbsf0I53h8T8w # VAYDVR0RBE0wS6RJMEcxLTArBgNVBAsTJE1pY3Jvc29mdCBJcmVsYW5kIE9wZXJh # dGlvbnMgTGltaXRlZDEWMBQGA1UEBRMNMjMwMDEyKzUwMTgzNjAfBgNVHSMEGDAW # gBRIbmTlUAXTgqoXNzcitW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8v # d3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIw # MTEtMDctMDguY3JsMGEGCCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDov # L3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDEx # XzIwMTEtMDctMDguY3J0MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIB # AGrJYDUS7s8o0yNprGXRXuAnRcHKxSjFmW4wclcUTYsQZkhnbMwthWM6cAYb/h2W # 5GNKtlmj/y/CThe3y/o0EH2h+jwfU/9eJ0fK1ZO/2WD0xi777qU+a7l8KjMPdwjY # 0tk9bYEGEZfYPRHy1AGPQVuZlG4i5ymJDsMrcIcqV8pxzsw/yk/O4y/nlOjHz4oV # APU0br5t9tgD8E08GSDi3I6H57Ftod9w26h0MlQiOr10Xqhr5iPLS7SlQwj8HW37 # ybqsmjQpKhmWul6xiXSNGGm36GarHy4Q1egYlxhlUnk3ZKSr3QtWIo1GGL03hT57 # xzjL25fKiZQX/q+II8nuG5M0Qmjvl6Egltr4hZ3e3FQRzRHfLoNPq3ELpxbWdH8t # Nuj0j/x9Crnfwbki8n57mJKI5JVWRWTSLmbTcDDLkTZlJLg9V1BIJwXGY3i2kR9i # 5HsADL8YlW0gMWVSlKB1eiSlK6LmFi0rVH16dde+j5T/EaQtFz6qngN7d1lvO7uk # 6rtX+MLKG4LDRsQgBTi6sIYiKntMjoYFHMPvI/OMUip5ljtLitVbkFGfagSqmbxK # 7rJMhC8wiTzHanBg1Rrbff1niBbnFbbV4UDmYumjs1FIpFCazk6AADXxoKCo5TsO # zSHqr9gHgGYQC2hMyX9MGLIpowYCURx3L7kUiGbOiMwaMIIHejCCBWKgAwIBAgIK # YQ6Q0gAAAAAAAzANBgkqhkiG9w0BAQsFADCBiDELMAkGA1UEBhMCVVMxEzARBgNV # BAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jv # c29mdCBDb3Jwb3JhdGlvbjEyMDAGA1UEAxMpTWljcm9zb2Z0IFJvb3QgQ2VydGlm # aWNhdGUgQXV0aG9yaXR5IDIwMTEwHhcNMTEwNzA4MjA1OTA5WhcNMjYwNzA4MjEw # OTA5WjB+MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSgwJgYD # VQQDEx9NaWNyb3NvZnQgQ29kZSBTaWduaW5nIFBDQSAyMDExMIICIjANBgkqhkiG # 9w0BAQEFAAOCAg8AMIICCgKCAgEAq/D6chAcLq3YbqqCEE00uvK2WCGfQhsqa+la # UKq4BjgaBEm6f8MMHt03a8YS2AvwOMKZBrDIOdUBFDFC04kNeWSHfpRgJGyvnkmc # 6Whe0t+bU7IKLMOv2akrrnoJr9eWWcpgGgXpZnboMlImEi/nqwhQz7NEt13YxC4D # dato88tt8zpcoRb0RrrgOGSsbmQ1eKagYw8t00CT+OPeBw3VXHmlSSnnDb6gE3e+ # lD3v++MrWhAfTVYoonpy4BI6t0le2O3tQ5GD2Xuye4Yb2T6xjF3oiU+EGvKhL1nk # kDstrjNYxbc+/jLTswM9sbKvkjh+0p2ALPVOVpEhNSXDOW5kf1O6nA+tGSOEy/S6 # A4aN91/w0FK/jJSHvMAhdCVfGCi2zCcoOCWYOUo2z3yxkq4cI6epZuxhH2rhKEmd # X4jiJV3TIUs+UsS1Vz8kA/DRelsv1SPjcF0PUUZ3s/gA4bysAoJf28AVs70b1FVL # 5zmhD+kjSbwYuER8ReTBw3J64HLnJN+/RpnF78IcV9uDjexNSTCnq47f7Fufr/zd # sGbiwZeBe+3W7UvnSSmnEyimp31ngOaKYnhfsi+E11ecXL93KCjx7W3DKI8sj0A3 # T8HhhUSJxAlMxdSlQy90lfdu+HggWCwTXWCVmj5PM4TasIgX3p5O9JawvEagbJjS # 4NaIjAsCAwEAAaOCAe0wggHpMBAGCSsGAQQBgjcVAQQDAgEAMB0GA1UdDgQWBBRI # bmTlUAXTgqoXNzcitW2oynUClTAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTAL # BgNVHQ8EBAMCAYYwDwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBRyLToCMZBD # uRQFTuHqp8cx0SOJNDBaBgNVHR8EUzBRME+gTaBLhklodHRwOi8vY3JsLm1pY3Jv # c29mdC5jb20vcGtpL2NybC9wcm9kdWN0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFf # MDNfMjIuY3JsMF4GCCsGAQUFBwEBBFIwUDBOBggrBgEFBQcwAoZCaHR0cDovL3d3 # dy5taWNyb3NvZnQuY29tL3BraS9jZXJ0cy9NaWNSb29DZXJBdXQyMDExXzIwMTFf # MDNfMjIuY3J0MIGfBgNVHSAEgZcwgZQwgZEGCSsGAQQBgjcuAzCBgzA/BggrBgEF # BQcCARYzaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9kb2NzL3ByaW1h # cnljcHMuaHRtMEAGCCsGAQUFBwICMDQeMiAdAEwAZQBnAGEAbABfAHAAbwBsAGkA # YwB5AF8AcwB0AGEAdABlAG0AZQBuAHQALiAdMA0GCSqGSIb3DQEBCwUAA4ICAQBn # 8oalmOBUeRou09h0ZyKbC5YR4WOSmUKWfdJ5DJDBZV8uLD74w3LRbYP+vj/oCso7 # v0epo/Np22O/IjWll11lhJB9i0ZQVdgMknzSGksc8zxCi1LQsP1r4z4HLimb5j0b # pdS1HXeUOeLpZMlEPXh6I/MTfaaQdION9MsmAkYqwooQu6SpBQyb7Wj6aC6VoCo/ # KmtYSWMfCWluWpiW5IP0wI/zRive/DvQvTXvbiWu5a8n7dDd8w6vmSiXmE0OPQvy # CInWH8MyGOLwxS3OW560STkKxgrCxq2u5bLZ2xWIUUVYODJxJxp/sfQn+N4sOiBp # mLJZiWhub6e3dMNABQamASooPoI/E01mC8CzTfXhj38cbxV9Rad25UAqZaPDXVJi # hsMdYzaXht/a8/jyFqGaJ+HNpZfQ7l1jQeNbB5yHPgZ3BtEGsXUfFL5hYbXw3MYb # BL7fQccOKO7eZS/sl/ahXJbYANahRr1Z85elCUtIEJmAH9AAKcWxm6U/RXceNcbS # oqKfenoi+kiVH6v7RyOA9Z74v2u3S5fi63V4GuzqN5l5GEv/1rMjaHXmr/r8i+sL # gOppO6/8MO0ETI7f33VtY5E90Z1WTk+/gFcioXgRMiF670EKsT/7qMykXcGhiJtX # cVZOSEXAQsmbdlsKgEhr/Xmfwb1tbWrJUnMTDXpQzTGCGZ8wghmbAgEBMIGVMH4x # CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRt # b25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01p # Y3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMTECEzMAAAOuLTVRyFOPVR0AAAAA # A64wDQYJYIZIAWUDBAIBBQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQw # HAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIIRx # 6w5PtwX73ytgFLY1JXGhHSttU3hF6kZGNbm29kT2MEIGCisGAQQBgjcCAQwxNDAy # oBSAEgBNAGkAYwByAG8AcwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20wDQYJKoZIhvcNAQEBBQAEggEAmahWEuslUNPfrbrM2REgPOoG8TaE6pvIvgfo # MtiKFeX//O78bCh+TJbsyVi95v/QvCYjPqr+7LLpolZM/ROFx7OviuWWGNwZ/ohF # h8SPr2f28ULppixu4/N9ZzrMm/q0NJGW0L/s7cWMSSbw5+YrXeTem8uoQPsEDODf # IeHtuvVfeXbdleUyLHkEXUAjCJhtE32pAD8I//+O+Kcr0kLgBmN0ay4R76O1uC7u # E3ucr4GaNZr6rNEBqLFBLhuW0cAUXiAYBffyn3t/N+WETp3QuC4hrF1p7rsRXdGw # 9TsW+O1Dkb6ieecil4S3wBR12HLAESbqEKBMYOZnk88g0E49aKGCFykwghclBgor # BgEEAYI3AwMBMYIXFTCCFxEGCSqGSIb3DQEHAqCCFwIwghb+AgEDMQ8wDQYJYIZI # AWUDBAIBBQAwggFZBgsqhkiG9w0BCRABBKCCAUgEggFEMIIBQAIBAQYKKwYBBAGE # WQoDATAxMA0GCWCGSAFlAwQCAQUABCDRtG0qkFZOqaVV4MqJam1k7f9iW/i0j7rZ # 0N49UQsF1AIGZbqhWn4FGBMyMDI0MDIwODAwMTczMy43MjNaMASAAgH0oIHYpIHV # MIHSMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMH # UmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMS0wKwYDVQQL # EyRNaWNyb3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxJjAkBgNVBAsT # HVRoYWxlcyBUU1MgRVNOOjg2REYtNEJCQy05MzM1MSUwIwYDVQQDExxNaWNyb3Nv # ZnQgVGltZS1TdGFtcCBTZXJ2aWNloIIReDCCBycwggUPoAMCAQICEzMAAAHdXVcd # ldStqhsAAQAAAd0wDQYJKoZIhvcNAQELBQAwfDELMAkGA1UEBhMCVVMxEzARBgNV # BAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jv # c29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAg # UENBIDIwMTAwHhcNMjMxMDEyMTkwNzA5WhcNMjUwMTEwMTkwNzA5WjCB0jELMAkG # A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQx # HjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEtMCsGA1UECxMkTWljcm9z # b2Z0IElyZWxhbmQgT3BlcmF0aW9ucyBMaW1pdGVkMSYwJAYDVQQLEx1UaGFsZXMg # VFNTIEVTTjo4NkRGLTRCQkMtOTMzNTElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUt # U3RhbXAgU2VydmljZTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAKhO # A5RE6i53nHURH4lnfKLp+9JvipuTtctairCxMUSrPSy5CWK2DtriQP+T52HXbN2g # 7AktQ1pQZbTDGFzK6d03vYYNrCPuJK+PRsP2FPVDjBXy5mrLRFzIHHLaiAaobE5v # FJuoxZ0ZWdKMCs8acjhHUmfaY+79/CR7uN+B4+xjJqwvdpU/mp0mAq3earyH+AKm # v6lkrQN8zgrcbCgHwsqvvqT6lEFqYpi7uKn7MAYbSeLe0pMdatV5EW6NVnXMYOTR # KuGPfyfBKdShualLo88kG7qa2mbA5l77+X06JAesMkoyYr4/9CgDFjHUpcHSODuj # lFBKMi168zRdLerdpW0bBX9EDux2zBMMaEK8NyxawCEuAq7++7ktFAbl3hUKtuzY # C1FUZuUl2Bq6U17S4CKsqR3itLT9qNcb2pAJ4jrIDdll5Tgoqef5gpv+YcvBM834 # bXFNwytd3ujDD24P9Dd8xfVJvumjsBQQkK5T/qy3HrQJ8ud1nHSvtFVi5Sa/ubGu # YEpS8gF6GDWN5/KbveFkdsoTVIPo8pkWhjPs0Q7nA5+uBxQB4zljEjKz5WW7BA4w # pmFm24fhBmRjV4Nbp+n78cgAjvDSfTlA6DYBcv2kx1JH2dIhaRnSeOXePT6hMF0I # l598LMu0rw35ViUWcAQkUNUTxRnqGFxz5w+ZusMDAgMBAAGjggFJMIIBRTAdBgNV # HQ4EFgQUbqL1toyPUdpFyyHSDKWj0I4lw/EwHwYDVR0jBBgwFoAUn6cVXQBeYl2D # 9OXSZacbUzUZ6XIwXwYDVR0fBFgwVjBUoFKgUIZOaHR0cDovL3d3dy5taWNyb3Nv # ZnQuY29tL3BraW9wcy9jcmwvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUy # MDIwMTAoMSkuY3JsMGwGCCsGAQUFBwEBBGAwXjBcBggrBgEFBQcwAoZQaHR0cDov # L3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9jZXJ0cy9NaWNyb3NvZnQlMjBUaW1l # LVN0YW1wJTIwUENBJTIwMjAxMCgxKS5jcnQwDAYDVR0TAQH/BAIwADAWBgNVHSUB # Af8EDDAKBggrBgEFBQcDCDAOBgNVHQ8BAf8EBAMCB4AwDQYJKoZIhvcNAQELBQAD # ggIBAC5U2bINLgXIHWbMcqVuf9jkUT/K8zyLBvu5h8JrqYR2z/eaO2yo1Ooc9Shy # vxbe9GZDu7kkUzxSyJ1IZksZZw6FDq6yZNT3PEjAEnREpRBL8S+mbXg+O4VLS0LS # mb8XIZiLsaqZ0fDEcv3HeA+/y/qKnCQWkXghpaEMwGMQzRkhGwcGdXr1zGpQ7HTx # vfu57xFxZX1MkKnWFENJ6urd+4teUgXj0ngIOx//l3XMK3Ht8T2+zvGJNAF+5/5q # Bk7nr079zICbFXvxtidNN5eoXdW+9rAIkS+UGD19AZdBrtt6dZ+OdAquBiDkYQ5k # VfUMKS31yHQOGgmFxuCOzTpWHalrqpdIllsy8KNsj5U9sONiWAd9PNlyEHHbQZDm # i9/BNlOYyTt0YehLbDovmZUNazk79Od/A917mqCdTqrExwBGUPbMP+/vdYUqaJsp # upBnUtjOf/76DAhVy8e/e6zR98PkplmliO2brL3Q3rD6+ZCVdrGM9Rm6hUDBBkvY # h+YjmGdcQ5HB6WT9Rec8+qDHmbhLhX4Zdaard5/OXeLbgx2f7L4QQQj3KgqjqDOW # InVhNE1gYtTWLHe4882d/k7Lui0K1g8EZrKD7maOrsJLKPKlegceJ9FCqY1sDUKU # hRa0EHUW+ZkKLlohKrS7FwjdrINWkPBgbQznCjdE2m47QjTbMIIHcTCCBVmgAwIB # AgITMwAAABXF52ueAptJmQAAAAAAFTANBgkqhkiG9w0BAQsFADCBiDELMAkGA1UE # BhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAc # BgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEyMDAGA1UEAxMpTWljcm9zb2Z0 # IFJvb3QgQ2VydGlmaWNhdGUgQXV0aG9yaXR5IDIwMTAwHhcNMjEwOTMwMTgyMjI1 # WhcNMzAwOTMwMTgzMjI1WjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGlu # Z3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv # cmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDCC # AiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAOThpkzntHIhC3miy9ckeb0O # 1YLT/e6cBwfSqWxOdcjKNVf2AX9sSuDivbk+F2Az/1xPx2b3lVNxWuJ+Slr+uDZn # hUYjDLWNE893MsAQGOhgfWpSg0S3po5GawcU88V29YZQ3MFEyHFcUTE3oAo4bo3t # 1w/YJlN8OWECesSq/XJprx2rrPY2vjUmZNqYO7oaezOtgFt+jBAcnVL+tuhiJdxq # D89d9P6OU8/W7IVWTe/dvI2k45GPsjksUZzpcGkNyjYtcI4xyDUoveO0hyTD4MmP # frVUj9z6BVWYbWg7mka97aSueik3rMvrg0XnRm7KMtXAhjBcTyziYrLNueKNiOSW # rAFKu75xqRdbZ2De+JKRHh09/SDPc31BmkZ1zcRfNN0Sidb9pSB9fvzZnkXftnIv # 231fgLrbqn427DZM9ituqBJR6L8FA6PRc6ZNN3SUHDSCD/AQ8rdHGO2n6Jl8P0zb # r17C89XYcz1DTsEzOUyOArxCaC4Q6oRRRuLRvWoYWmEBc8pnol7XKHYC4jMYcten # IPDC+hIK12NvDMk2ZItboKaDIV1fMHSRlJTYuVD5C4lh8zYGNRiER9vcG9H9stQc # xWv2XFJRXRLbJbqvUAV6bMURHXLvjflSxIUXk8A8FdsaN8cIFRg/eKtFtvUeh17a # j54WcmnGrnu3tz5q4i6tAgMBAAGjggHdMIIB2TASBgkrBgEEAYI3FQEEBQIDAQAB # MCMGCSsGAQQBgjcVAgQWBBQqp1L+ZMSavoKRPEY1Kc8Q/y8E7jAdBgNVHQ4EFgQU # n6cVXQBeYl2D9OXSZacbUzUZ6XIwXAYDVR0gBFUwUzBRBgwrBgEEAYI3TIN9AQEw # QTA/BggrBgEFBQcCARYzaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9E # b2NzL1JlcG9zaXRvcnkuaHRtMBMGA1UdJQQMMAoGCCsGAQUFBwMIMBkGCSsGAQQB # gjcUAgQMHgoAUwB1AGIAQwBBMAsGA1UdDwQEAwIBhjAPBgNVHRMBAf8EBTADAQH/ # MB8GA1UdIwQYMBaAFNX2VsuP6KJcYmjRPZSQW9fOmhjEMFYGA1UdHwRPME0wS6BJ # oEeGRWh0dHA6Ly9jcmwubWljcm9zb2Z0LmNvbS9wa2kvY3JsL3Byb2R1Y3RzL01p # Y1Jvb0NlckF1dF8yMDEwLTA2LTIzLmNybDBaBggrBgEFBQcBAQROMEwwSgYIKwYB # BQUHMAKGPmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2VydHMvTWljUm9v # Q2VyQXV0XzIwMTAtMDYtMjMuY3J0MA0GCSqGSIb3DQEBCwUAA4ICAQCdVX38Kq3h # LB9nATEkW+Geckv8qW/qXBS2Pk5HZHixBpOXPTEztTnXwnE2P9pkbHzQdTltuw8x # 5MKP+2zRoZQYIu7pZmc6U03dmLq2HnjYNi6cqYJWAAOwBb6J6Gngugnue99qb74p # y27YP0h1AdkY3m2CDPVtI1TkeFN1JFe53Z/zjj3G82jfZfakVqr3lbYoVSfQJL1A # oL8ZthISEV09J+BAljis9/kpicO8F7BUhUKz/AyeixmJ5/ALaoHCgRlCGVJ1ijbC # HcNhcy4sa3tuPywJeBTpkbKpW99Jo3QMvOyRgNI95ko+ZjtPu4b6MhrZlvSP9pEB # 9s7GdP32THJvEKt1MMU0sHrYUP4KWN1APMdUbZ1jdEgssU5HLcEUBHG/ZPkkvnNt # yo4JvbMBV0lUZNlz138eW0QBjloZkWsNn6Qo3GcZKCS6OEuabvshVGtqRRFHqfG3 # rsjoiV5PndLQTHa1V1QJsWkBRH58oWFsc/4Ku+xBZj1p/cvBQUl+fpO+y/g75LcV # v7TOPqUxUYS8vwLBgqJ7Fx0ViY1w/ue10CgaiQuPNtq6TPmb/wrpNPgkNWcr4A24 # 5oyZ1uEi6vAnQj0llOZ0dFtq0Z4+7X6gMTN9vMvpe784cETRkPHIqzqKOghif9lw # Y1NNje6CbaUFEMFxBmoQtB1VM1izoXBm8qGCAtQwggI9AgEBMIIBAKGB2KSB1TCB # 0jELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEtMCsGA1UECxMk # TWljcm9zb2Z0IElyZWxhbmQgT3BlcmF0aW9ucyBMaW1pdGVkMSYwJAYDVQQLEx1U # aGFsZXMgVFNTIEVTTjo4NkRGLTRCQkMtOTMzNTElMCMGA1UEAxMcTWljcm9zb2Z0 # IFRpbWUtU3RhbXAgU2VydmljZaIjCgEBMAcGBSsOAwIaAxUANiNHGWXbNaDPxnyi # DbEOciSjFhCggYMwgYCkfjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGlu # Z3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv # cmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDAN # BgkqhkiG9w0BAQUFAAIFAOluWg0wIhgPMjAyNDAyMDgwMzM1NDFaGA8yMDI0MDIw # OTAzMzU0MVowdDA6BgorBgEEAYRZCgQBMSwwKjAKAgUA6W5aDQIBADAHAgEAAgIQ # MjAHAgEAAgIRdTAKAgUA6W+rjQIBADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEE # AYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3DQEBBQUAA4GB # AGDRq2bvFZFqai06uZpqfzzZlEFk+HAQqAwbOojwLH2qfZGD5dTDxjfa7pHOFEqj # UyJeTXfSrPWyF5b45lxYdx5yEqELGI0AHQrIJBqj9x/AX5Q/SgFiba3v13PLiZal # a0K5BqupXZip+hbXa/3AphTE4SdleXuWowhxeSpZosDBMYIEDTCCBAkCAQEwgZMw # fDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMd # TWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAHdXVcdldStqhsAAQAA # Ad0wDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzENBgsqhkiG9w0BCRAB # BDAvBgkqhkiG9w0BCQQxIgQgLxSE18i8z0NKtA48xLINhP5f3EtXPvGZuN2Cs61t # ENswgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCBh/w4tmmWsT3iZnHtH0Vk3 # 7UCN02lRxY+RiON6wDFjZjCBmDCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQI # EwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3Nv # ZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBD # QSAyMDEwAhMzAAAB3V1XHZXUraobAAEAAAHdMCIEIM7tl0fcU1uXID8B5PtSnecK # 2FnZi6Fm6hJlYwhsLSjYMA0GCSqGSIb3DQEBCwUABIICABDJxjZmneYMHaJj82XL # qQstLs+HlsfVOtCnwDCyeRTD1ejVJnIfz0qpOADEyQc+o7blu5EHVGRcpnT5gbrg # +g9g6dtjFFfT3P147C/AGG0FVdRBanQyK9eN+tYl+EJF70d+kSQ826j11Tk2i+VM # Zy13qG2JDbZd8rIcJSOeE6VQ0wLS5rcWY3M76JRjaIkyF4/iiDpDJEwfN6UUouNK # omJqyv+6mWWfGTiI2PqhkHV0iyt+NiSpzAsBh4tSL+cQIP2QdHJuBIUOAeYuVugC # s5Z9+i5ncSdGjYRGEQ63ChhFzjLBhNmZJfffRuXPEUplsT3kUMJTphHZqX8T26vB # TiE/spsFRcwEtCMrrdPoT20FTuGZcwl0r/SNBwhlmNpNvTnb7QBLqj6jRy/w2p2K # T31doK15ogDZNe8M41Lj/aHG9fhwoDHVzUTVkacUa1HWqjIpoZDlm9s+1rBFpfo4 # EvJA7Gn/PtuFh+eehwZhy3yvs1AWALuOFRTJ7m3qwDa4o4CDv4eAg8UmhVCt+Cpx # KjLCUxdykwCUHrHJMtAylP7fnxdEhSY5vhSpdiK9UjT2gI+lnpOC07S6ZMrmYSO0 # uLvtHBeQpmRDKBHHD7ot9yxDsU84oC1RkGHuJSuGY7W2QN9vJ0dJ4xCpel4b9YfE # c9AR/phRlfA6+ShON7pijrD5 # SIG # End signature block |