Experimental/HelpParsers/Convert-NetshHelp.ps1
[CmdletBinding(DefaultParameterSetName="Default")] param ( [Parameter(Mandatory=$true,ParameterSetName="file")]$file, [Parameter(ParameterSetName="file")][switch]$Generate, [Parameter(ParameterSetName="file")][switch]$force ) if ( ! $IsWindows ) { throw "this can only be run on Windows" } $exe = "netsh.exe" $helpChar = "-?" $commandPattern = "Commands in this context:$" $optionPattern = "options are available:" $usagePattern = "^Usage: (?<usage>.*)" $argumentPattern = "arguments are available:" $headerLength = 0 $linkPattern = "^More help can be found at: (?<link>.*)" $parmPattern = "--(?<pname>\w+)\s+(?<phelp>.*)" class ParsedCommand { [string]$exe [string[]]$commandElements [string]$Verb [string]$Noun [cParameter[]]$Parameters [string]$Usage [string[]]$Help [string]$Link [string[]]$OriginalHelptext [object]GetCrescendoCommand() { $c = New-CrescendoCommand -Verb $this.Verb -Noun $this.Noun -originalname $this.exe if ( $this.Usage ) { $c.Usage = New-UsageInfo -usage $this.Usage } else { Write-Verbose ("skipping usage for " + ($this.commandElements -join " ")) } if ( $this.CommandElements ) { $c.OriginalCommandElements = $this.commandElements | foreach-object {$_} } $c.Platform = "Windows" $c.OriginalText = $this.OriginalHelptext -join ([char]5) $c.Description = $this.Help -join "`n" $c.HelpLinks = $this.Link foreach ( $p in $this.Parameters) { $parm = New-ParameterInfo -name $p.Name -originalName $p.OriginalName $parm.Description = $p.Help if ( $p.Position -ne [int]::MaxValue ) { $parm.Position = $p.Position } $c.Parameters.Add($parm) } return $c } [string]GetCrescendoJson() { return $this.GetCrescendoCommand().GetCrescendoConfiguration() } } class cParameter { [string]$Name [string]$OriginalName [string]$Help [int]$Position = [int]::MaxValue cParameter([string]$name, [string]$originalName, [string]$help) { $this.Name = $name $this.OriginalName = $originalName $this.Help = $help } } function Get-Parm ( $parameterString ) { write-verbose $parameterString } function Test-ParameterContinue { param ( [string[]]$text, [int]$offset ) [bool]$continues = $false if ($offset -ge $text.Count ) { break } elseif ( $text[$i+1] -match "^\s{9}(?<parm>[^\s].*)" ) { $continues = $true } elseif ( $text[$i+1] -match "^\s{13}[^\s]" ) { $continues = $true } elseif ( $text[$i].Trim() -match "\|$" ) { $continues = $true } elseif ( $text[$i+1].Trim() -match "^\|" ) { $continues = $true } return $continues } function Test-IsParameter { param ( [string[]]$text, [int]$offset ) if ( $offset -ge $text.Count ) { return $false } [string]$possibleParameter = $text[$offset].Trim() if ( $possibleParameter -eq "Not Supported." ) { return $false } elseif ( $possibleParameter -eq "Please go to the Network Connections folder to install." ){ return $false } elseif ( $text[$offset] -match "^\s{6}(?<parm>[^\s].*)" ) { return $true } return $false } function Get-ParameterFromUsage ( [string[]]$prolog, [string]$usage ) { #wait-debugger $pnum = $prolog.count - 1 if ( $pnum -ge 0 ) { foreach ( $i in 0..$pnum ) { $plog = $prolog[${i}..($pnum)] -join " " if ( $usage -match "$plog" ) { $parm = $usage -replace ".*${plog}" $parm = "$parm".Trim() # if ( $parm -match " " ) { wait-debugger } if ( "$parm" -eq "" ){ Write-Verbose ("trimmed everything from $usage with " + ($prolog -join ":"))} return "$parm" } } } if ( $usage -match ".*${exe} (?<parms>.*)" ) { $parm = $matches['parms'].Trim() # if ( $parm -match " " ) { wait-debugger } return $parm } # wait-debugger Write-verbose "no parameter in $usage" return "no parameter" } function parseHelp([string]$exe, [string[]]$commandProlog) { write-progress ("parsing help for '$exe " + ($commandProlog -join " ") + "'") if ( $commandProlog ) { $helpText = & $exe $commandProlog $helpChar } else { $helpText = & $exe $helpChar } $offset = $headerLength $cmdhelp = @() while ( $helpText[$offset] -ne "" -and $offset -lt $helpText.Count) { $cmdhelp += $helpText[$offset++] } #$cmdHelpString = $cmdhelp -join " " $parameters = @() $usage = "" for($i = $offset; $i -lt $helpText.Count; $i++) { if ($helpText[$i] -match $usagePattern) { $usage = $matches['usage'] try { $pp = get-parameterfromusage -prolog $commandprolog -usage $usage if ( $pp ) { $parameters += $pp } } catch { wait-debugger } continue # this mess is about finding parameters - skip for now while ( Test-ParameterContinue -text $helpText -offset $i ) { $i++ $usage += $helpText[$i].Trim() #wait-debugger } # manage parameters here $i++ while($i -lt $helpText.Count -and $helpText[$i][0] -eq " ") { if (Test-IsParameter -text $helpText -offset $i) { $helptext[$offset] -match "\s{6}(?<parm>[^\s].*)" try { $p = $matches['parm'].Trim() } catch { wait-debugger } while ( Test-ParameterContinue -text $helpText -offset $i ) { $i++ $p += $helpText[$i].Trim() #wait-debugger } if ( $p -match "not supported" ) { wait-debugger } if ( ! "$p".Trim() ) { wait-debugger } Get-Parm -parameterString $p $parameters += $p $p = "" if( $matches ) { $matches.Clear() } } $i++ } } elseif ($helpText[$i] -match $linkPattern ) { $link = $matches['link'] } elseif ($helpText[$i] -match $optionPattern) { $i++ while($helpText[$i] -ne "") { if ($helpText[$i] -match $parmPattern) { $originalName = "--" + $matches['pname'] $pHelp = $matches['phelp'] $pName = $originalName -replace "[- ]" $p = [cParameter]::new($pName, $originalName, $pHelp) $parameters += $p } $i++ } } elseif ($helpText[$i] -match $argumentPattern) { $i++ $position = 0 while($helpText[$i] -ne "") { if ($helpText[$i] -match $parmPattern -and $i -lt $helpText.Count) { $originalName = "--" + $matches['pname'] $pHelp = $matches['phelp'] $pName = $originalName -replace "[- ]" $p = [cParameter]::new($pName, $originalName, $pHelp) $p.Position = $position++ $parameters += $p } $i++ } } elseif ($helpText[$i] -match $commandPattern) { $i++ $subCommands = @() while($helpText[$i] -ne "" -and $i -le $helpText.Count) { try { $t = $helpText[$i].Trim() } catch { break } $subCommands, $subHelp = $t.split("-", 2, [System.StringSplitOptions]::RemoveEmptyEntries) $subcommand = $subCommands.Trim().Split(" ",[System.StringSplitOptions]::RemoveEmptyEntries)[-1] if ( $helpText[$i+1] -and $helpText[$i+1][0] -eq " " ) { $i++ while($helpText[$i+1] -and $helpText[$i][0] -eq " " -and $i -lt $helpText.Count) { $subHelp += " " + $helpText[$i].Trim() $i++ } } #$subCommand, $dash, $subHelp = $t.split(" ",3, [System.StringSplitOptions]::RemoveEmptyEntries) # skip '?' and 'help' if ( $subCommand -eq "?" -or $subCommand -eq "help") { $i++ continue } $cPro = $commandProlog $cPro += $subCommand #if ( $cPro[0] -eq "firewall" -and $cPro[1] -eq "add") { # wait-debugger #} #write-host (">>> " + ($cPro -join " ")) parseHelp -exe $exe -commandProlog $cPro $i++ } } } $c = [Parsedcommand]::new() $c.exe = $exe $c.commandElements = $commandProlog $c.Verb = "Invoke" $c.Noun = $("$exe" -replace ".exe";$commandProlog).Foreach({"$_".split("-")}).Foreach({[char]::ToUpper("$_"[0]) + "$_".SubString(1).toLower()}) -join "" # $c.Parameters = $parameters $c.Usage = $usage $c.Help = $cmdhelp $c.Link = $link $c.OriginalHelptext = $helpText $c } $commands = parseHelp -exe $exe -commandProlog @() $trimmedCommands = $commands.Where({$_.Usage -or ($_.commandElements -contains "show" -and $_.commandElements[-1] -ne "show")}) $convertedCommands = $trimmedCommands | ForEach-Object { $_.GetCrescendoCommand()} $h = [ordered]@{ '$schema' = 'https://aka.ms/PowerShell/Crescendo/Schemas/2021-11' 'Commands' = $convertedCommands } if ( ! $Generate ) { $h return } $sOptions = [System.Text.Json.JsonSerializerOptions]::new() $sOptions.WriteIndented = $true $sOptions.MaxDepth = 20 $sOptions.IgnoreNullValues = $true $ParsedConfig = [System.Text.Json.JsonSerializer]::Serialize($h, $sOptions) if ( $file ) { if (test-path $file) { if ($force) { $parsedConfig > $file } else { Write-Error "'$file' exists, use '-force' to overwrite" } } else { $parsedConfig > $file } } else { $parsedConfig } # SIG # Begin signature block # MIIoKgYJKoZIhvcNAQcCoIIoGzCCKBcCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCAxNHxcf2ondI6B # sA7fktHJzUzZ41Os4yyB2Aor/p1UHqCCDXYwggX0MIID3KADAgECAhMzAAADTrU8 # esGEb+srAAAAAANOMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p # bmcgUENBIDIwMTEwHhcNMjMwMzE2MTg0MzI5WhcNMjQwMzE0MTg0MzI5WjB0MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB # AQDdCKiNI6IBFWuvJUmf6WdOJqZmIwYs5G7AJD5UbcL6tsC+EBPDbr36pFGo1bsU # p53nRyFYnncoMg8FK0d8jLlw0lgexDDr7gicf2zOBFWqfv/nSLwzJFNP5W03DF/1 # 1oZ12rSFqGlm+O46cRjTDFBpMRCZZGddZlRBjivby0eI1VgTD1TvAdfBYQe82fhm # WQkYR/lWmAK+vW/1+bO7jHaxXTNCxLIBW07F8PBjUcwFxxyfbe2mHB4h1L4U0Ofa # +HX/aREQ7SqYZz59sXM2ySOfvYyIjnqSO80NGBaz5DvzIG88J0+BNhOu2jl6Dfcq # jYQs1H/PMSQIK6E7lXDXSpXzAgMBAAGjggFzMIIBbzAfBgNVHSUEGDAWBgorBgEE # AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUnMc7Zn/ukKBsBiWkwdNfsN5pdwAw # RQYDVR0RBD4wPKQ6MDgxHjAcBgNVBAsTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEW # MBQGA1UEBRMNMjMwMDEyKzUwMDUxNjAfBgNVHSMEGDAWgBRIbmTlUAXTgqoXNzci # tW2oynUClTBUBgNVHR8ETTBLMEmgR6BFhkNodHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpb3BzL2NybC9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3JsMGEG # CCsGAQUFBwEBBFUwUzBRBggrBgEFBQcwAoZFaHR0cDovL3d3dy5taWNyb3NvZnQu # Y29tL3BraW9wcy9jZXJ0cy9NaWNDb2RTaWdQQ0EyMDExXzIwMTEtMDctMDguY3J0 # MAwGA1UdEwEB/wQCMAAwDQYJKoZIhvcNAQELBQADggIBAD21v9pHoLdBSNlFAjmk # mx4XxOZAPsVxxXbDyQv1+kGDe9XpgBnT1lXnx7JDpFMKBwAyIwdInmvhK9pGBa31 # TyeL3p7R2s0L8SABPPRJHAEk4NHpBXxHjm4TKjezAbSqqbgsy10Y7KApy+9UrKa2 # kGmsuASsk95PVm5vem7OmTs42vm0BJUU+JPQLg8Y/sdj3TtSfLYYZAaJwTAIgi7d # hzn5hatLo7Dhz+4T+MrFd+6LUa2U3zr97QwzDthx+RP9/RZnur4inzSQsG5DCVIM # pA1l2NWEA3KAca0tI2l6hQNYsaKL1kefdfHCrPxEry8onJjyGGv9YKoLv6AOO7Oh # JEmbQlz/xksYG2N/JSOJ+QqYpGTEuYFYVWain7He6jgb41JbpOGKDdE/b+V2q/gX # UgFe2gdwTpCDsvh8SMRoq1/BNXcr7iTAU38Vgr83iVtPYmFhZOVM0ULp/kKTVoir # IpP2KCxT4OekOctt8grYnhJ16QMjmMv5o53hjNFXOxigkQWYzUO+6w50g0FAeFa8 # 5ugCCB6lXEk21FFB1FdIHpjSQf+LP/W2OV/HfhC3uTPgKbRtXo83TZYEudooyZ/A # Vu08sibZ3MkGOJORLERNwKm2G7oqdOv4Qj8Z0JrGgMzj46NFKAxkLSpE5oHQYP1H # tPx1lPfD7iNSbJsP6LiUHXH1MIIHejCCBWKgAwIBAgIKYQ6Q0gAAAAAAAzANBgkq # 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 # /Xmfwb1tbWrJUnMTDXpQzTGCGgowghoGAgEBMIGVMH4xCzAJBgNVBAYTAlVTMRMw # EQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN # aWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNp # Z25pbmcgUENBIDIwMTECEzMAAANOtTx6wYRv6ysAAAAAA04wDQYJYIZIAWUDBAIB # BQCgga4wGQYJKoZIhvcNAQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEO # MAwGCisGAQQBgjcCARUwLwYJKoZIhvcNAQkEMSIEIKcifsBOr8H4WW9fr+4nboIl # yDi0PINMUeb2jebv6/Z+MEIGCisGAQQBgjcCAQwxNDAyoBSAEgBNAGkAYwByAG8A # cwBvAGYAdKEagBhodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20wDQYJKoZIhvcNAQEB # BQAEggEA09LtMgW3/Dco208LNYH9V+xGk9gVZ7Z/+a9sKw0v6EqibAzGbO9ojggV # ipGzFT1P6ddrFYmRsNRTgOljEFAzn12WtVWQsoZq2G7NmrAfY/R7mkCtqjlqyNXa # NS78gA5j0FGCY7gCYYs/LzzsKIXM8wP2CTwrcPnqp80qHAz9DjfoAScCzi76qBz1 # 9FilvQBSuf97tASZVTvKcq9Owo7NK66p84jEiOYJIXQJFXSSduIepZhJw7vSwk83 # r2ztu/UX2FibOinKpshxWZHcMth5zxMz/VFK3t7/joh4uulDvgvVGlLFnYeEmeZh # xQnfwerZ1rAmZxXdPmYWKayomcaipKGCF5QwgheQBgorBgEEAYI3AwMBMYIXgDCC # F3wGCSqGSIb3DQEHAqCCF20wghdpAgEDMQ8wDQYJYIZIAWUDBAIBBQAwggFSBgsq # hkiG9w0BCRABBKCCAUEEggE9MIIBOQIBAQYKKwYBBAGEWQoDATAxMA0GCWCGSAFl # AwQCAQUABCAAKQaHk3yZTvJLW7IDJ7gK7pQzQcKF5i3eZwHrrww+6QIGZMvstFA9 # GBMyMDIzMDgwNDE3MTU0Ni4yNDRaMASAAgH0oIHRpIHOMIHLMQswCQYDVQQGEwJV # UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE # ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1l # cmljYSBPcGVyYXRpb25zMScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046OTYwMC0w # NUUwLUQ5NDcxJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2Wg # ghHqMIIHIDCCBQigAwIBAgITMwAAAdj8SzOlHdiFFQABAAAB2DANBgkqhkiG9w0B # AQsFADB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UE # BxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYD # VQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDAeFw0yMzA1MjUxOTEy # NDBaFw0yNDAyMDExOTEyNDBaMIHLMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2Fz # aGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENv # cnBvcmF0aW9uMSUwIwYDVQQLExxNaWNyb3NvZnQgQW1lcmljYSBPcGVyYXRpb25z # MScwJQYDVQQLEx5uU2hpZWxkIFRTUyBFU046OTYwMC0wNUUwLUQ5NDcxJTAjBgNV # BAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2UwggIiMA0GCSqGSIb3DQEB # AQUAA4ICDwAwggIKAoICAQDNeOsp0fXgAz7GUF0N+/0EHcQFri6wliTbmQNmFm8D # i0CeQ8n4bd2td5tbtzTsEk7dY2/nmWY9kqEvavbdYRbNc+Esv8Nfv6MMImH9tCr5 # Kxs254MQ0jmpRucrm3uHW421Cfva0hNQEKN1NS0rad1U/ZOme+V/QeSdWKofCThx # f/fsTeR41WbqUNAJN/ml3sbOH8aLhXyTHG7sVt/WUSLpT0fLlNXYGRXzavJ1qUOe # Pzyj86hiKyzQJLTjKr7GpTGFySiIcMW/nyK6NK7Rjfy1ofLdRvvtHIdJvpmPSze3 # CH/PYFU21TqhIhZ1+AS7RlDo18MSDGPHpTCWwo7lgtY1pY6RvPIguF3rbdtvhoyj # n5mPbs5pgjGO83odBNP7IlKAj4BbHUXeHit3Da2g7A4jicKrLMjo6sGeetJoeKoo # j5iNTXbDwLKM9HlUdXZSz62ftCZVuK9FBgkAO9MRN2pqBnptBGfllm+21FLk6E3v # VXMGHB5eOgFfAy84XlIieycQArIDsEm92KHIFOGOgZlWxe69leXvMHjYJlpo2VVM # tLwXLd3tjS/173ouGMRaiLInLm4oIgqDtjUIqvwYQUh3RN6wwdF75nOmrpr8wRw1 # n/BKWQ5mhQxaMBqqvkbuu1sLeSMPv2PMZIddXPbiOvAxadqPkBcMPUBmrySYoLTx # wwIDAQABo4IBSTCCAUUwHQYDVR0OBBYEFPbTj0x8PZBLYn0MZBI6nGh5qIlWMB8G # A1UdIwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8GA1UdHwRYMFYwVKBSoFCG # Tmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01pY3Jvc29mdCUy # MFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBsBggrBgEFBQcBAQRgMF4w # XAYIKwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2Vy # dHMvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3J0MAwG # A1UdEwEB/wQCMAAwFgYDVR0lAQH/BAwwCgYIKwYBBQUHAwgwDgYDVR0PAQH/BAQD # AgeAMA0GCSqGSIb3DQEBCwUAA4ICAQCunA6aSP48oJ1VD+SMF1/7SFiTGD6zyLC3 # Ju9HtLjqYYq1FJWUx10I5XqU0alcXTUFUoUIUPSvfeX/dX0MgofUG+cOXdokaHHS # lo6PZIDXnUClpkRix9xCN37yFBpcwGLzEZlDKJb2gDq/FBGC8snTlBSEOBjV0eE8 # ICVUkOJzIAttExaeQWJ5SerUr63nq6X7PmQvk1OLFl3FJoW4+5zKqriY/PKGssOa # A5ZjBZEyU+o7+P3icL/wZ0G3ymlT+Ea4h9f3q5aVdGVBdshYa/SehGmnUvGMA8j5 # Ct24inx+bVOuF/E/2LjIp+mEary5mOTrANVKLym2kW3eQxF/I9cj87xndiYH55Xf # rWMk9bsRToxOpRb9EpbCB5cSyKNvxQ8D00qd2TndVEJFpgyBHQJS/XEK5poeJZ5q # gmCFAj4VUPB/dPXHdTm1QXJI3cO7DRyPUZAYMwQ3KhPlM2hP2OfBJIr/VsDsh3sz # LL2ZJuerjshhxYGVboMud9aNoRjlz1Mcn4iEota4tam24FxDyHrqFm6EUQu/pDYE # DquuvQFGb5glIck4rKqBnRlrRoiRj0qdhO3nootVg/1SP0zTLC1RrxjuTEVe3PKr # ETbtvcODoGh912Xrtf4wbMwpra8jYszzr3pf0905zzL8b8n8kuMBChBYfFds916K # Tjc4TGNU9TCCB3EwggVZoAMCAQICEzMAAAAVxedrngKbSZkAAAAAABUwDQYJKoZI # hvcNAQELBQAwgYgxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAw # DgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24x # MjAwBgNVBAMTKU1pY3Jvc29mdCBSb290IENlcnRpZmljYXRlIEF1dGhvcml0eSAy # MDEwMB4XDTIxMDkzMDE4MjIyNVoXDTMwMDkzMDE4MzIyNVowfDELMAkGA1UEBhMC # VVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNV # BAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRp # bWUtU3RhbXAgUENBIDIwMTAwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoIC # AQDk4aZM57RyIQt5osvXJHm9DtWC0/3unAcH0qlsTnXIyjVX9gF/bErg4r25Phdg # M/9cT8dm95VTcVrifkpa/rg2Z4VGIwy1jRPPdzLAEBjoYH1qUoNEt6aORmsHFPPF # dvWGUNzBRMhxXFExN6AKOG6N7dcP2CZTfDlhAnrEqv1yaa8dq6z2Nr41JmTamDu6 # GnszrYBbfowQHJ1S/rboYiXcag/PXfT+jlPP1uyFVk3v3byNpOORj7I5LFGc6XBp # Dco2LXCOMcg1KL3jtIckw+DJj361VI/c+gVVmG1oO5pGve2krnopN6zL64NF50Zu # yjLVwIYwXE8s4mKyzbnijYjklqwBSru+cakXW2dg3viSkR4dPf0gz3N9QZpGdc3E # XzTdEonW/aUgfX782Z5F37ZyL9t9X4C626p+Nuw2TPYrbqgSUei/BQOj0XOmTTd0 # lBw0gg/wEPK3Rxjtp+iZfD9M269ewvPV2HM9Q07BMzlMjgK8QmguEOqEUUbi0b1q # GFphAXPKZ6Je1yh2AuIzGHLXpyDwwvoSCtdjbwzJNmSLW6CmgyFdXzB0kZSU2LlQ # +QuJYfM2BjUYhEfb3BvR/bLUHMVr9lxSUV0S2yW6r1AFemzFER1y7435UsSFF5PA # PBXbGjfHCBUYP3irRbb1Hode2o+eFnJpxq57t7c+auIurQIDAQABo4IB3TCCAdkw # EgYJKwYBBAGCNxUBBAUCAwEAATAjBgkrBgEEAYI3FQIEFgQUKqdS/mTEmr6CkTxG # NSnPEP8vBO4wHQYDVR0OBBYEFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMFwGA1UdIARV # MFMwUQYMKwYBBAGCN0yDfQEBMEEwPwYIKwYBBQUHAgEWM2h0dHA6Ly93d3cubWlj # cm9zb2Z0LmNvbS9wa2lvcHMvRG9jcy9SZXBvc2l0b3J5Lmh0bTATBgNVHSUEDDAK # BggrBgEFBQcDCDAZBgkrBgEEAYI3FAIEDB4KAFMAdQBiAEMAQTALBgNVHQ8EBAMC # AYYwDwYDVR0TAQH/BAUwAwEB/zAfBgNVHSMEGDAWgBTV9lbLj+iiXGJo0T2UkFvX # zpoYxDBWBgNVHR8ETzBNMEugSaBHhkVodHRwOi8vY3JsLm1pY3Jvc29mdC5jb20v # cGtpL2NybC9wcm9kdWN0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcmwwWgYI # KwYBBQUHAQEETjBMMEoGCCsGAQUFBzAChj5odHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpL2NlcnRzL01pY1Jvb0NlckF1dF8yMDEwLTA2LTIzLmNydDANBgkqhkiG # 9w0BAQsFAAOCAgEAnVV9/Cqt4SwfZwExJFvhnnJL/Klv6lwUtj5OR2R4sQaTlz0x # M7U518JxNj/aZGx80HU5bbsPMeTCj/ts0aGUGCLu6WZnOlNN3Zi6th542DYunKmC # VgADsAW+iehp4LoJ7nvfam++Kctu2D9IdQHZGN5tggz1bSNU5HhTdSRXud2f8449 # xvNo32X2pFaq95W2KFUn0CS9QKC/GbYSEhFdPSfgQJY4rPf5KYnDvBewVIVCs/wM # nosZiefwC2qBwoEZQhlSdYo2wh3DYXMuLGt7bj8sCXgU6ZGyqVvfSaN0DLzskYDS # PeZKPmY7T7uG+jIa2Zb0j/aRAfbOxnT99kxybxCrdTDFNLB62FD+CljdQDzHVG2d # Y3RILLFORy3BFARxv2T5JL5zbcqOCb2zAVdJVGTZc9d/HltEAY5aGZFrDZ+kKNxn # GSgkujhLmm77IVRrakURR6nxt67I6IleT53S0Ex2tVdUCbFpAUR+fKFhbHP+Crvs # QWY9af3LwUFJfn6Tvsv4O+S3Fb+0zj6lMVGEvL8CwYKiexcdFYmNcP7ntdAoGokL # jzbaukz5m/8K6TT4JDVnK+ANuOaMmdbhIurwJ0I9JZTmdHRbatGePu1+oDEzfbzL # 6Xu/OHBE0ZDxyKs6ijoIYn/ZcGNTTY3ugm2lBRDBcQZqELQdVTNYs6FwZvKhggNN # MIICNQIBATCB+aGB0aSBzjCByzELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hp # bmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jw # b3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2EgT3BlcmF0aW9uczEn # MCUGA1UECxMeblNoaWVsZCBUU1MgRVNOOjk2MDAtMDVFMC1EOTQ3MSUwIwYDVQQD # ExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNloiMKAQEwBwYFKw4DAhoDFQBI # p++xUJ+f85VrnbzdkRMSpBmvL6CBgzCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1w # IFBDQSAyMDEwMA0GCSqGSIb3DQEBCwUAAgUA6HcT8jAiGA8yMDIzMDgwNDA2MDY0 # MloYDzIwMjMwODA1MDYwNjQyWjB0MDoGCisGAQQBhFkKBAExLDAqMAoCBQDodxPy # AgEAMAcCAQACAjpCMAcCAQACAhLUMAoCBQDoeGVyAgEAMDYGCisGAQQBhFkKBAIx # KDAmMAwGCisGAQQBhFkKAwKgCjAIAgEAAgMHoSChCjAIAgEAAgMBhqAwDQYJKoZI # hvcNAQELBQADggEBAJSbgEMtxHgAIREsoO8twWVEWf/JWQ9f0cvwP22/IIWfF43C # aKRQ6Orx9m7PWbVAip/viTIfv/YGqHi9MeonV4yDGNHTqkUpTQYlRPaHz1A2RWkR # 4DDU+DEGoZs7NFT2zoEqKRqYxiK7HIIEkeGcheIc58CTMrGwIPzDGFqFb/MUEVEx # OC0xe3DM6Xnpfhdy9UfpkeGGpSQpduvxK6BlNz1RhJHG3Hle9nfZTcCGAaPT3Plg # f1WDAwHsOw+KDNxlosxSKVhR0x16a4AqLU8hqF4/xVp8J9TAsgjjaoj+cFKYaQ0M # ET4D9zATYKEy0lW1HWIVFX2wMRw1Sh0hpNfJ3hQxggQNMIIECQIBATCBkzB8MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNy # b3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMAITMwAAAdj8SzOlHdiFFQABAAAB2DAN # BglghkgBZQMEAgEFAKCCAUowGgYJKoZIhvcNAQkDMQ0GCyqGSIb3DQEJEAEEMC8G # CSqGSIb3DQEJBDEiBCABs0N/bg0r6oEHj7ug2lPbMK2wsyzb8k8/W/9t4wpg3DCB # +gYLKoZIhvcNAQkQAi8xgeowgecwgeQwgb0EIDrjIX/8CZN3RTABMNt5u73Mi3o3 # fmvq2j8Sik+2s75UMIGYMIGApH4wfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldh # c2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBD # b3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIw # MTACEzMAAAHY/EszpR3YhRUAAQAAAdgwIgQgj/Q7+E2Hwgkb7AXGUbdsfH7+kUg2 # m6NjaVViP2SxCjgwDQYJKoZIhvcNAQELBQAEggIAETANUmasYaFr30NLIryUxday # REZmOqOPYn7gKSH9wyPFojIWLcEakMZhuUL5W4OM0mcGJ/abZkuVC+gVi03+ddYi # LCAfBY7DIle3/yBcA3tN8YxqcrybMR5khp6B/h685GDrN56PA8gPUgdxV3ZpjtS3 # Q3tjndKb6SFSS0eF9HVUZeqj/mozWcKhfPkwoeUr3MJv+V9pFUKmYVNaeOJaJUUF # kpn4CXRvp8eAcNI+6djgvc+lN6cqaPnsCsH7VSmOTv+Aub7nx91wlFalc/8Ee9jR # 10G0sKtmqnsZIm/2pwtExarJuEBf/bV/kJpSvNUPMgmiz11IlF2sMgDG+HfVsuBf # /eY3rt6gSZmYh/mlriGtKASUur9S3oITUpwLzbzM2q/D69WhO22hNuuLudUc6sP5 # z1jOuMEaOz8ftTvTU29XpFWiYl+e3dZMJXGKhh56lBKIG6RbWA4TwTMgqntOtVmW # VFE0Z6lW1QqqkbvFdI/vtDSV+6DeYhXIvvRBGjHHu6UQ7TOmpgh4CIOswdNEzOoZ # 11Ep9m4GOgUDLCVW3/XGz8PikflQ8dGI5Kjg8ZZYmRP7Xdx14ow2Z7omodKSCeHV # cyzokKmvwqPtdFQUM+N2BXPyLzBDLaF3y/TlbnwtkZyMi3k/myyCp+AdTApypXKH # 9akcosmKJBrmy7y27wc= # SIG # End signature block |