Types/OpenAIClient.ps1
class OpenAIClient { [System.Management.Automation.HiddenAttribute()] [string]$apikey [System.Management.Automation.HiddenAttribute()] [string]$baseUri [System.Management.Automation.HiddenAttribute()] [string]$model [System.Management.Automation.HiddenAttribute()] [hashtable]$headers [System.Management.Automation.HiddenAttribute()] [string]$apiVersion [Assistant]$assistants [Vector_store]$vector_stores [File]$files [Thread]$threads OpenAIClient([string]$apiKey, [string]$baseUri, [string]$model, [string]$apiVersion) { $this.apikey = $apiKey $this.baseUri = $baseUri $this.model = $model $this.apiVersion = $apiVersion $this.init() } [System.Management.Automation.HiddenAttribute()] [void]init() { # check the apikey, endpoint and model, if empty, then return error $this.headers = @{ "OpenAI-Beta" = "assistants=v2" } if ($this.baseUri -match "azure") { $this.headers.Add("api-key", $this.apikey) $this.baseUri = $this.baseUri + "openai/" } else { $this.headers.Add("Authorization", "Bearer $($this.apikey)") $this.apiVersion = "" } $this.assistants = [Assistant]::new($this) $this.vector_stores = [Vector_store]::new($this) $this.files = [File]::new($this) $this.threads = [Thread]::new($this) } [psobject]web( [string]$urifragment, [string]$method = "GET", [psobject]$body = $null) { $url = "{0}{1}" -f $this.baseUri, $urifragment if ($this.apiVersion -ne "") { if ($url -match "\?") { $url = "{0}&api-version={1}" -f $url, $this.apiVersion } else { $url = "{0}?api-version={1}" -f $url, $this.apiVersion } } if ($method -eq "GET" -or $null -eq $body) { $params = @{ Method = $method Uri = $url Headers = $this.headers } return $this.unicodeiwr($params) } else { $params = @{ Method = $method Uri = $url Headers = $this.headers Body = ($body | ConvertTo-Json -Depth 10) } return $this.unicodeiwr($params) } } [System.Management.Automation.HiddenAttribute()] [psobject]unicodeiwr([hashtable]$params) { $oldProgressPreference = Get-Variable -Name ProgressPreference -ValueOnly Set-Variable -Name ProgressPreference -Value "SilentlyContinue" -Scope Script -Force $response = Invoke-WebRequest @params -ContentType "application/json;charset=utf-8" Set-Variable -Name ProgressPreference -Value $oldProgressPreference -Scope Script -Force $contentType = $response.Headers["Content-Type"] $version = Get-Variable -Name PSVersionTable -ValueOnly if ($version.PSVersion.Major -gt 5 -or $contentType -match 'charset=utf-8') { return $response.Content | ConvertFrom-Json } else { $response = $response.Content $charset = if ($contentType -match "charset=([^;]+)") { $matches[1] } else { "ISO-8859-1" } $dstEncoding = [System.Text.Encoding]::GetEncoding($charset) $srcEncoding = [System.Text.Encoding]::UTF8 $result = $srcEncoding.GetString([System.Text.Encoding]::Convert($srcEncoding, $dstEncoding, $srcEncoding.GetBytes($response))) return $result | ConvertFrom-Json } } [psobject]web($urifragment) { return $this.web($urifragment, "GET", @{}) } } class AssistantResource { [System.Management.Automation.HiddenAttribute()] [OpenAIClient]$client [System.Management.Automation.HiddenAttribute()] [string]$urifragment [System.Management.Automation.HiddenAttribute()] [string]$objTypeName AssistantResource([OpenAIClient]$client, [string]$urifragment, [string]$objTypeName) { $this.client = $client $this.urifragment = $urifragment $this.objTypeName = $objTypeName } [psobject[]]list() { if ($this.objTypeName) { return $this.client.web($this.urifragment).data | ForEach-Object { $temp = "{0}/{1}" -f $this.urifragment, $_.id $result = New-Object -TypeName $this.objTypeName -ArgumentList $_ $result | Add-Member -MemberType NoteProperty -Name client -Value $this.client $result | Add-Member -MemberType NoteProperty -Name urifragment -Value $temp $result } } return $this.client.web($this.urifragment).data } [psobject]get([string]$id) { if ($this.objTypeName) { $temp = "{0}/{1}" -f $this.urifragment, $id $result = New-Object -TypeName $this.objTypeName -ArgumentList $this.client.web($temp) $result | Add-Member -MemberType NoteProperty -Name client -Value $this.client $result | Add-Member -MemberType NoteProperty -Name urifragment -Value $temp return $result } return $this.client.web("$($this.urifragment)/$id") } [psobject]delete([string]$id) { return $this.client.web("$($this.urifragment)/$id", "DELETE", @{}) } [psobject]create([hashtable]$body) { if ($this.objTypeName) { $result = New-Object -TypeName $this.objTypeName -ArgumentList $this.client.web("$($this.urifragment)", "POST", $body) $result | Add-Member -MemberType NoteProperty -Name client -Value $this.client $result | Add-Member -MemberType NoteProperty -Name urifragment -Value "$($this.urifragment)/$($result.id)" return $result } return $this.client.web("$($this.urifragment)", "POST", $body) } [psobject]create() { return $this.create(@{}) } [void]clear() { # warn user this is very dangerous action, it will remove all the instance, and ask for confirmation $confirm = Read-Host "Are you sure you want to remove all the instances? (yes/no)" if ($confirm -ne "yes" -and $confirm -ne "y") { return } # get all the instances and remove it $this.list() | ForEach-Object { $this.delete($_.id) Write-Host "remove the instance: $($_.id)" } } } class AssistantResourceObject { AssistantResourceObject([psobject]$data) { # check all the properties and assign it to the object $data.PSObject.Properties | ForEach-Object { $this | Add-Member -MemberType NoteProperty -Name $_.Name -Value $_.Value } } [AssistantResourceObject]update([hashtable]$data) { $result = $this.client.web($this.urifragment, "POST", $data) return New-Object -TypeName $this.GetType().Name -ArgumentList $result } } class FileObject:AssistantResourceObject { FileObject([psobject]$data):base($data) {} [AssistantResourceObject]update([hashtable]$data) { Write-Host "You can't update the file object." return $this } } class File:AssistantResource { File([OpenAIClient]$client): base($client, "files", "FileObject") {} [psobject]create([hashtable]$body) { if ($body.files) { $files = $body.files return $this.upload($files) } throw "The body must contain 'files' key." } [System.Management.Automation.HiddenAttribute()] [FileObject[]]upload([string[]]$fullname) { $PSVersion = Get-Variable -Name PSVersionTable -ValueOnly if ($PSVersion.PSVersion.Major -lt 6) { throw "The upload file feature is only supported in PowerShell 6 or later." } # process the input, if it is a wildcard or a folder, then get all the files based on this pattern $fullname = $fullname | Get-ChildItem | Select-Object -ExpandProperty FullName # read all the files and check the filename, compute $existing_files = $this.list() | Select-Object id, @{l = "hash"; e = { $_.filename.split("-")[0] } } $localfiles = $fullname | Select-Object @{l = "fullname"; e = { $_ } }, @{l = "hash"; e = { (Get-FileHash $_).Hash } } $result = @( $existing_files | Where-Object { $_.hash -in $localfiles.hash } | ForEach-Object { [FileObject]::new($_) } ) $fullname = $localfiles | Where-Object { $_.hash -notin $existing_files.hash } | Select-Object -ExpandProperty fullname if ($fullname.Count -gt 0) { # confirm if user want to upload those files to openai $confirm = Read-Host "Are you sure you want to upload the $($fullname.Count) files? (yes/no)" if ($confirm -ne "yes" -and $confirm -ne "y") { throw "The user canceled the operation." } $url = "{0}{1}" -f $this.client.baseUri, $this.urifragment if ($this.client.baseUri -match "azure") { $url = "{0}?api-version=2024-05-01-preview" -f $url } foreach ($file in $fullname) { Write-Host "process file: $file" $name = "{0}-{1}" -f (Get-FileHash $file).Hash, (Split-Path $file -Leaf) # rename the file to the new name Rename-Item -Path $file -NewName $name $temppath = Join-Path -Path (Split-Path $file) -ChildPath $name try{ $form = @{ file = Get-Item -Path $temppath purpose = "assistants" } $response = Invoke-RestMethod -Uri $url -Method Post -Headers $this.client.headers -Form $form $result += [FileObject]::new($response) } finally{ # rename the file back to the original name Rename-Item -Path $temppath -NewName (Split-Path $file -Leaf) } } } return $result } } class Assistant:AssistantResource { Assistant([OpenAIClient]$client): base($client, "assistants", "AssistantObject") {} <# .SYNOPSIS Create a new assistant .DESCRIPTION Create a new assistant with the given name, model, and instructions. .PARAMETER body The body must contain 'name', 'model', and 'instructions' keys. But it can also contain 'config', 'vector_store_ids', 'functions', and 'files' keys. #> [AssistantObject]create([hashtable]$body) { if ($body.name -and $body.model -and $body.instructions) { $vector_store_ids = $body.vector_store_ids $functions = $body.functions $files = $body.files $config = $body.config if ($files) { # upload the files and create new vector store $file_ids = $this.client.files.create(@{ "files" = $files }) | Select-Object -ExpandProperty id $body.Add("tools", @( @{ "type" = "file_search" })) $body.Add("tool_resources", @{ "file_search" = @{ "vector_stores" = @( @{ file_ids = @($file_ids) }) } }) } if ($vector_store_ids -and $vector_store_ids.Count -gt 0) { $body.Add("tool_resources", @{ "file_search" = @{ "vector_store_ids" = @($vector_store_ids) } }) $body.Add("tools", @( @{ "type" = "file_search" })) } if ($functions -and $functions.Count -gt 0) { if ($null -eq $body.tools) { $body.Add("tools", @()) } $functions | ForEach-Object { $func = Get-FunctionJson -functionName $_ $body.tools += $func } } if ($config) { #if config is not hashtable, then convert it to hashtable if ($config -isnot [hashtable]) { $config = ConvertTo-Hashtable $config } Merge-Hashtable -table1 $body -table2 $config } # remove files, vector_store_ids, functions, and config from the body $body.Remove("files") $body.Remove("vector_store_ids") $body.Remove("functions") $body.Remove("config") $result = [AssistantObject]::new($this.client.web("$($this.urifragment)", "POST", $body)) $result | Add-Member -MemberType NoteProperty -Name client -Value $this.client $result | Add-Member -MemberType NoteProperty -Name urifragment -Value "$($this.urifragment)/$($result.id)" return $result } throw "The body must contain 'name' and 'model', 'instructions' keys." } } class AssistantObject:AssistantResourceObject { [ThreadObject]$thread AssistantObject([psobject]$data):base($data) {} [void]chat([bool]$clean = $false) { if (-not $this.thread) { # create a thread, and associate the assistant id $this.thread = $this.client.threads.create($this.id) } try { while ($true) { # ask use to input, until the user type 'q' or 'bye' $prompt = Read-Host ">" if ($prompt -eq "q" -or $prompt -eq "bye") { break } # send the message to the thread $response = $this.thread.send($prompt).run().get_last_message() if ($response) { Write-Host $response -ForegroundColor Green } } } finally { $this.client.threads.delete($this.thread.id) if ($clean) { Write-Host "clean up the thread, assistant, and vector_store..." -ForegroundColor Yellow # clean up the thread, assistant, and vector_store $vc_id = $this.tool_resources.file_search.vector_store_ids[0] $this.client.vector_stores.delete($vc_id) $this.client.assistants.delete($this.id) } } } } class ThreadObject:AssistantResourceObject { ThreadObject([psobject]$data):base($data) {} [ThreadObject]send([string]$message) { # send a message [AssistantResource]::new($this.client, ("threads/{0}/messages" -f $this.id), $null ).create(@{ role = "user" content = $message }) | Out-Null return $this } [ThreadObject]run([string]$assistantId) { $obj = [AssistantResource]::new($this.client, ("threads/{0}/runs" -f $this.id), $null ).create(@{assistant_id = $assistantId }) if ($null -eq $this.last_run_id) { $this | Add-Member -MemberType NoteProperty -Name last_run_id -Value $obj.id } else { $this.last_run_id = $obj.id } return $this } [ThreadObject]run() { return $this.run($this.assistant_id) } [string]get_last_message() { # check if the last_run is set, if not, then return null if ($this.last_run_id) { $run = [AssistantResource]::new($this.client, ("threads/{0}/runs" -f $this.id), $null ).get($this.last_run_id) while ($run.status -ne "completed") { Write-Verbose ("Run status: {0}" -f $run.status) if ($run.status -eq "failed") { Write-Host ("Run failed: {0}" -f $run.last_error.message) -ForegroundColor Red break } # The status of the run, which can be either queued, in_progress, requires_action, cancelling, cancelled, failed, completed, incomplete, or expired. if ($run.status -eq "requires_action") { $tool_calls = $run.required_action.submit_tool_outputs.tool_calls $tool_output = @() if ($tool_calls -and $tool_calls.Count -gt 0) { foreach ($tool_call in $tool_calls) { $call_id = $tool_call.id $function = $tool_call.function $function_args = $function.arguments | ConvertFrom-Json $exp = "{0} {1}" -f $function.name, (($function_args.PSObject.Properties | ForEach-Object { "-{0} '{1}'" -f $_.Name, $_.Value }) -join " ") Write-Verbose "calling function with arguments: $exp" $call_response = Invoke-Expression $exp $tool_output += @{ tool_call_id = $call_id output = $call_response } } } [AssistantResource]::new($this.client, ("threads/{0}/runs/{1}/submit_tool_outputs" -f $this.id, $this.last_run_id), $null ).create(@{tool_outputs = $tool_output }) } Start-Sleep -Milliseconds 500 $run = [AssistantResource]::new($this.client, ("threads/{0}/runs" -f $this.id), $null ).get($this.last_run_id) } $message = [AssistantResource]::new($this.client, ("threads/{0}/messages?limit=1" -f $this.id), $null).list() | Select-Object id, role, content -First 1 return $message.content.text.value } return $null } } class Thread:AssistantResource { Thread([OpenAIClient]$client): base($client, "threads", "ThreadObject") {} [psobject[]]list() { return @{ error = "It is not implemented yet, you can't get all the thread information." } } [ThreadObject]create([string]$assistantId) { $result = $this.create() $result | Add-Member -MemberType NoteProperty -Name assistant_id -Value $assistantId return $result } } class Vector_storeObject:AssistantResourceObject { Vector_storeObject([psobject]$data):base($data) {} [string[]]file_ids() { return $this.client.web("vector_stores/$($this.id)/files").data | Select-Object -ExpandProperty id } } class Vector_store:AssistantResource { Vector_store([OpenAIClient]$client): base($client, "vector_stores", "Vector_storeObject") {} [psobject]create([hashtable]$body) { <# .SYNOPSIS Create a new vector store .DESCRIPTION Create a new vector store with the given name, file_ids, and days_to_expire. .PARAMETER body The body must contain 'name', 'file_ids', and 'days_to_expire' keys. #> # check if the body contains name, file_ids, and days_to_expire if ($body.name -and $body.file_ids -and $body.days_to_expire) { #replace the days_to_expire with expires_after $body.expires_after = @{ "days" = $body.days_to_expire "anchor" = "last_active_at" } $body.Remove("days_to_expire") return $this.client.web("$($this.urifragment)", "POST", $body) } throw "The body must contain 'name', 'file_ids', and 'days_to_expire' keys." } } # SIG # Begin signature block # MIIdNQYJKoZIhvcNAQcCoIIdJjCCHSICAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCAaQabEpB04K8ZQ # slRxR16FlfEuvJV/xIqqKHkm2HOVy6CCAyowggMmMIICDqADAgECAhBcsg5m3zM9 # kUZxmeNzIQNjMA0GCSqGSIb3DQEBCwUAMCoxKDAmBgNVBAMMH0NIRU5YSVpIQU5H # IC0gQ29kZSBTaWduaW5nIENlcnQwIBcNMjQwMTA4MTMwMjA0WhgPMjA5OTEyMzEx # NjAwMDBaMCoxKDAmBgNVBAMMH0NIRU5YSVpIQU5HIC0gQ29kZSBTaWduaW5nIENl # cnQwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQDKDY3QG81JOKZG9jTb # QriDMDhq6gy93Pmoqgav9wErj+CgVvXKk+lGpUu74MWVyLUrJx8/ACb4b287wsXx # mQj8zQ3SqGn5CCjPKoAPsSbry0LOSl8bsFpwBr3YBJVL6cibhus2KLCbNu/u7sND # wyivKXYA1Iy1uTQPNVPcBx36krZTZyyE4CmngO75YbTMEzvHEjM3BIXdKtEt673t # iNOVSP6doh0zRwWEh2Y/eoOpv+FUokORwhKonxMtmIIET+ZPx7Ex+9aqHrliEabx # FsN4ETnuVT3rST++7Q2fquWFnl5scDnisFhU8JL8k+OGUzpLlo/nOpiRZkbKCEkZ # FCLhAgMBAAGjRjBEMA4GA1UdDwEB/wQEAwIHgDATBgNVHSUEDDAKBggrBgEFBQcD # AzAdBgNVHQ4EFgQUwcR3UUOZ6TxpBp9MxnBygyIMhUQwDQYJKoZIhvcNAQELBQAD # ggEBADwiE9nowKxUNN84BTk9an1ZkdU95ouj+q6MRbafH08u4XV7CxXpkPR8Za/c # BJWTOqCuz9pMPo0TylqWPm+++Tqy1OJ7Qewvy1+DXPuFGkTqY721uZ+YsHY3CueC # VSRZRNsWSYE9UxXXFRsjDu/M3+EvyaNDE4xQkwrP8obFJoHq7WaOCCD2wMbKjLb5 # bS/VgtOK7Yn9pU/ghrW+Em+zHOX87wNRh/I5jd+LsnY8bR6REzgdmogIyvD4dsJD # /IZLxRtbm2BHOn/aGBdu+GpEaYEEb6VkWcJhrQnpiNjjlu43CbRz5Bw14XPWGUDH # +EkUqkWS4h8zsRiyvR9Pnwklg6UxghlhMIIZXQIBATA+MCoxKDAmBgNVBAMMH0NI # RU5YSVpIQU5HIC0gQ29kZSBTaWduaW5nIENlcnQCEFyyDmbfMz2RRnGZ43MhA2Mw # DQYJYIZIAWUDBAIBBQCgfDAQBgorBgEEAYI3AgEMMQIwADAZBgkqhkiG9w0BCQMx # DAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAvBgkq # hkiG9w0BCQQxIgQgUzEYAeURw8nPzBIIklAzPVCgrB/HF87fHvLwsNuWObswDQYJ # KoZIhvcNAQEBBQAEggEAQ8jH/Mrkae4gKX370ZH8Bd0dzStOfcgDxpxQYX1N5Trs # uxyty8ecwT89zFh3FNV+UVdCTsledz/HebIF1hDVjWgyvE6A3BNCCJgbVskTZxwy # HzaKdesBB29tZbMhCKbQ9StyON4gPQkLUTqhLhjDka6HN3HB6VnuJf6qpiEftQdG # uryHvfjrKIOVKb8WwEYp57GSr24nfMglzTPzJqfx6JuBfgmWgl6pVZd0fvhg4zUP # qMPEaz+hRAQD6LcfIJmmj2uHrlqp0Yu5NIHcUzFw10yvbfixpPf4+2h8E4xdWzWc # gI6IKTP4R1ZhjOKZ+FXOdaV/Z4WlDpTa1ypjAeufOKGCF3YwghdyBgorBgEEAYI3 # AwMBMYIXYjCCF14GCSqGSIb3DQEHAqCCF08wghdLAgEDMQ8wDQYJYIZIAWUDBAIB # BQAwdwYLKoZIhvcNAQkQAQSgaARmMGQCAQEGCWCGSAGG/WwHATAxMA0GCWCGSAFl # AwQCAQUABCDZebEY+x1oCECyHXY68HedFaKfbVWIAqdxY42ENQ6v5QIQQeK6K4uY # UpaVpTmaLiDoCxgPMjAyNTA4MTAxMTI4MDVaoIITOjCCBu0wggTVoAMCAQICEAqA # 7xhLjfEFgtHEdqeVdGgwDQYJKoZIhvcNAQELBQAwaTELMAkGA1UEBhMCVVMxFzAV # BgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMUEwPwYDVQQDEzhEaWdpQ2VydCBUcnVzdGVk # IEc0IFRpbWVTdGFtcGluZyBSU0E0MDk2IFNIQTI1NiAyMDI1IENBMTAeFw0yNTA2 # MDQwMDAwMDBaFw0zNjA5MDMyMzU5NTlaMGMxCzAJBgNVBAYTAlVTMRcwFQYDVQQK # Ew5EaWdpQ2VydCwgSW5jLjE7MDkGA1UEAxMyRGlnaUNlcnQgU0hBMjU2IFJTQTQw # OTYgVGltZXN0YW1wIFJlc3BvbmRlciAyMDI1IDEwggIiMA0GCSqGSIb3DQEBAQUA # A4ICDwAwggIKAoICAQDQRqwtEsae0OquYFazK1e6b1H/hnAKAd/KN8wZQjBjMqiZ # 3xTWcfsLwOvRxUwXcGx8AUjni6bz52fGTfr6PHRNv6T7zsf1Y/E3IU8kgNkeECqV # Q+3bzWYesFtkepErvUSbf+EIYLkrLKd6qJnuzK8Vcn0DvbDMemQFoxQ2Dsw4vEjo # T1FpS54dNApZfKY61HAldytxNM89PZXUP/5wWWURK+IfxiOg8W9lKMqzdIo7VA1R # 0V3Zp3DjjANwqAf4lEkTlCDQ0/fKJLKLkzGBTpx6EYevvOi7XOc4zyh1uSqgr6Un # bksIcFJqLbkIXIPbcNmA98Oskkkrvt6lPAw/p4oDSRZreiwB7x9ykrjS6GS3NR39 # iTTFS+ENTqW8m6THuOmHHjQNC3zbJ6nJ6SXiLSvw4Smz8U07hqF+8CTXaETkVWz0 # dVVZw7knh1WZXOLHgDvundrAtuvz0D3T+dYaNcwafsVCGZKUhQPL1naFKBy1p6ll # N3QgshRta6Eq4B40h5avMcpi54wm0i2ePZD5pPIssoszQyF4//3DoK2O65Uck5Wg # gn8O2klETsJ7u8xEehGifgJYi+6I03UuT1j7FnrqVrOzaQoVJOeeStPeldYRNMmS # F3voIgMFtNGh86w3ISHNm0IaadCKCkUe2LnwJKa8TIlwCUNVwppwn4D3/Pt5pwID # AQABo4IBlTCCAZEwDAYDVR0TAQH/BAIwADAdBgNVHQ4EFgQU5Dv88jHt/f3X85Fx # YxlQQ89hjOgwHwYDVR0jBBgwFoAU729TSunkBnx6yuKQVvYv1Ensy04wDgYDVR0P # AQH/BAQDAgeAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMIGVBggrBgEFBQcBAQSB # iDCBhTAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMF0GCCsG # AQUFBzAChlFodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVz # dGVkRzRUaW1lU3RhbXBpbmdSU0E0MDk2U0hBMjU2MjAyNUNBMS5jcnQwXwYDVR0f # BFgwVjBUoFKgUIZOaHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1 # c3RlZEc0VGltZVN0YW1waW5nUlNBNDA5NlNIQTI1NjIwMjVDQTEuY3JsMCAGA1Ud # IAQZMBcwCAYGZ4EMAQQCMAsGCWCGSAGG/WwHATANBgkqhkiG9w0BAQsFAAOCAgEA # ZSqt8RwnBLmuYEHs0QhEnmNAciH45PYiT9s1i6UKtW+FERp8FgXRGQ/YAavXzWjZ # hY+hIfP2JkQ38U+wtJPBVBajYfrbIYG+Dui4I4PCvHpQuPqFgqp1PzC/ZRX4pvP/ # ciZmUnthfAEP1HShTrY+2DE5qjzvZs7JIIgt0GCFD9ktx0LxxtRQ7vllKluHWiKk # 6FxRPyUPxAAYH2Vy1lNM4kzekd8oEARzFAWgeW3az2xejEWLNN4eKGxDJ8WDl/FQ # USntbjZ80FU3i54tpx5F/0Kr15zW/mJAxZMVBrTE2oi0fcI8VMbtoRAmaaslNXdC # G1+lqvP4FbrQ6IwSBXkZagHLhFU9HCrG/syTRLLhAezu/3Lr00GrJzPQFnCEH1Y5 # 8678IgmfORBPC1JKkYaEt2OdDh4GmO0/5cHelAK2/gTlQJINqDr6JfwyYHXSd+V0 # 8X1JUPvB4ILfJdmL+66Gp3CSBXG6IwXMZUXBhtCyIaehr0XkBoDIGMUG1dUtwq1q # mcwbdUfcSYCn+OwncVUXf53VJUNOaMWMts0VlRYxe5nK+At+DI96HAlXHAL5SlfY # xJ7La54i71McVWRP66bW+yERNpbJCjyCYG2j+bdpxo/1Cy4uPcU3AWVPGrbn5PhD # Bf3Froguzzhk++ami+r3Qrx5bIbY3TVzgiFI7Gq3zWcwgga0MIIEnKADAgECAhAN # x6xXBf8hmS5AQyIMOkmGMA0GCSqGSIb3DQEBCwUAMGIxCzAJBgNVBAYTAlVTMRUw # EwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20x # ITAfBgNVBAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBHNDAeFw0yNTA1MDcwMDAw # MDBaFw0zODAxMTQyMzU5NTlaMGkxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdp # Q2VydCwgSW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3RlZCBHNCBUaW1lU3Rh # bXBpbmcgUlNBNDA5NiBTSEEyNTYgMjAyNSBDQTEwggIiMA0GCSqGSIb3DQEBAQUA # A4ICDwAwggIKAoICAQC0eDHTCphBcr48RsAcrHXbo0ZodLRRF51NrY0NlLWZloMs # VO1DahGPNRcybEKq+RuwOnPhof6pvF4uGjwjqNjfEvUi6wuim5bap+0lgloM2zX4 # kftn5B1IpYzTqpyFQ/4Bt0mAxAHeHYNnQxqXmRinvuNgxVBdJkf77S2uPoCj7GH8 # BLuxBG5AvftBdsOECS1UkxBvMgEdgkFiDNYiOTx4OtiFcMSkqTtF2hfQz3zQSku2 # Ws3IfDReb6e3mmdglTcaarps0wjUjsZvkgFkriK9tUKJm/s80FiocSk1VYLZlDwF # t+cVFBURJg6zMUjZa/zbCclF83bRVFLeGkuAhHiGPMvSGmhgaTzVyhYn4p0+8y9o # HRaQT/aofEnS5xLrfxnGpTXiUOeSLsJygoLPp66bkDX1ZlAeSpQl92QOMeRxykvq # 6gbylsXQskBBBnGy3tW/AMOMCZIVNSaz7BX8VtYGqLt9MmeOreGPRdtBx3yGOP+r # x3rKWDEJlIqLXvJWnY0v5ydPpOjL6s36czwzsucuoKs7Yk/ehb//Wx+5kMqIMRvU # BDx6z1ev+7psNOdgJMoiwOrUG2ZdSoQbU2rMkpLiQ6bGRinZbI4OLu9BMIFm1UUl # 9VnePs6BaaeEWvjJSjNm2qA+sdFUeEY0qVjPKOWug/G6X5uAiynM7Bu2ayBjUwID # AQABo4IBXTCCAVkwEgYDVR0TAQH/BAgwBgEB/wIBADAdBgNVHQ4EFgQU729TSunk # Bnx6yuKQVvYv1Ensy04wHwYDVR0jBBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4cD08w # DgYDVR0PAQH/BAQDAgGGMBMGA1UdJQQMMAoGCCsGAQUFBwMIMHcGCCsGAQUFBwEB # BGswaTAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEEGCCsG # AQUFBzAChjVodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVz # dGVkUm9vdEc0LmNydDBDBgNVHR8EPDA6MDigNqA0hjJodHRwOi8vY3JsMy5kaWdp # Y2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNybDAgBgNVHSAEGTAXMAgG # BmeBDAEEAjALBglghkgBhv1sBwEwDQYJKoZIhvcNAQELBQADggIBABfO+xaAHP4H # PRF2cTC9vgvItTSmf83Qh8WIGjB/T8ObXAZz8OjuhUxjaaFdleMM0lBryPTQM2qE # JPe36zwbSI/mS83afsl3YTj+IQhQE7jU/kXjjytJgnn0hvrV6hqWGd3rLAUt6vJy # 9lMDPjTLxLgXf9r5nWMQwr8Myb9rEVKChHyfpzee5kH0F8HABBgr0UdqirZ7bowe # 9Vj2AIMD8liyrukZ2iA/wdG2th9y1IsA0QF8dTXqvcnTmpfeQh35k5zOCPmSNq1U # H410ANVko43+Cdmu4y81hjajV/gxdEkMx1NKU4uHQcKfZxAvBAKqMVuqte69M9J6 # A47OvgRaPs+2ykgcGV00TYr2Lr3ty9qIijanrUR3anzEwlvzZiiyfTPjLbnFRsjs # Yg39OlV8cipDoq7+qNNjqFzeGxcytL5TTLL4ZaoBdqbhOhZ3ZRDUphPvSRmMThi0 # vw9vODRzW6AxnJll38F0cuJG7uEBYTptMSbhdhGQDpOXgpIUsWTjd6xpR6oaQf/D # Jbg3s6KCLPAlZ66RzIg9sC+NJpud/v4+7RWsWCiKi9EOLLHfMR2ZyJ/+xhCx9yHb # xtl5TPau1j/1MIDpMPx0LckTetiSuEtQvLsNz3Qbp7wGWqbIiOWCnb5WqxL3/BAP # vIXKUjPSxyZsq8WhbaM2tszWkPZPubdcMIIFjTCCBHWgAwIBAgIQDpsYjvnQLefv # 21DiCEAYWjANBgkqhkiG9w0BAQwFADBlMQswCQYDVQQGEwJVUzEVMBMGA1UEChMM # RGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQD # ExtEaWdpQ2VydCBBc3N1cmVkIElEIFJvb3QgQ0EwHhcNMjIwODAxMDAwMDAwWhcN # MzExMTA5MjM1OTU5WjBiMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQg # SW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSEwHwYDVQQDExhEaWdpQ2Vy # dCBUcnVzdGVkIFJvb3QgRzQwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoIC # AQC/5pBzaN675F1KPDAiMGkz7MKnJS7JIT3yithZwuEppz1Yq3aaza57G4QNxDAf # 8xukOBbrVsaXbR2rsnnyyhHS5F/WBTxSD1Ifxp4VpX6+n6lXFllVcq9ok3DCsrp1 # mWpzMpTREEQQLt+C8weE5nQ7bXHiLQwb7iDVySAdYyktzuxeTsiT+CFhmzTrBcZe # 7FsavOvJz82sNEBfsXpm7nfISKhmV1efVFiODCu3T6cw2Vbuyntd463JT17lNecx # y9qTXtyOj4DatpGYQJB5w3jHtrHEtWoYOAMQjdjUN6QuBX2I9YI+EJFwq1WCQTLX # 2wRzKm6RAXwhTNS8rhsDdV14Ztk6MUSaM0C/CNdaSaTC5qmgZ92kJ7yhTzm1EVgX # 9yRcRo9k98FpiHaYdj1ZXUJ2h4mXaXpI8OCiEhtmmnTK3kse5w5jrubU75KSOp49 # 3ADkRSWJtppEGSt+wJS00mFt6zPZxd9LBADMfRyVw4/3IbKyEbe7f/LVjHAsQWCq # sWMYRJUadmJ+9oCw++hkpjPRiQfhvbfmQ6QYuKZ3AeEPlAwhHbJUKSWJbOUOUlFH # dL4mrLZBdd56rF+NP8m800ERElvlEFDrMcXKchYiCd98THU/Y+whX8QgUWtvsauG # i0/C1kVfnSD8oR7FwI+isX4KJpn15GkvmB0t9dmpsh3lGwIDAQABo4IBOjCCATYw # DwYDVR0TAQH/BAUwAwEB/zAdBgNVHQ4EFgQU7NfjgtJxXWRM3y5nP+e6mK4cD08w # HwYDVR0jBBgwFoAUReuir/SSy4IxLVGLp6chnfNtyA8wDgYDVR0PAQH/BAQDAgGG # MHkGCCsGAQUFBwEBBG0wazAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGlnaWNl # cnQuY29tMEMGCCsGAQUFBzAChjdodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5jb20v # RGlnaUNlcnRBc3N1cmVkSURSb290Q0EuY3J0MEUGA1UdHwQ+MDwwOqA4oDaGNGh0 # dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5j # cmwwEQYDVR0gBAowCDAGBgRVHSAAMA0GCSqGSIb3DQEBDAUAA4IBAQBwoL9DXFXn # OF+go3QbPbYW1/e/Vwe9mqyhhyzshV6pGrsi+IcaaVQi7aSId229GhT0E0p6Ly23 # OO/0/4C5+KH38nLeJLxSA8hO0Cre+i1Wz/n096wwepqLsl7Uz9FDRJtDIeuWcqFI # tJnLnU+nBgMTdydE1Od/6Fmo8L8vC6bp8jQ87PcDx4eo0kxAGTVGamlUsLihVo7s # pNU96LHc/RzY9HdaXFSMb++hUD38dglohJ9vytsgjTVgHAIDyyCwrFigDkBjxZgi # wbJZ9VVrzyerbHbObyMt9H5xaiNrIv8SuFQtJ37YOtnwtoeW/VvRXKwYw02fc7cB # qZ9Xql4o4rmUMYIDfDCCA3gCAQEwfTBpMQswCQYDVQQGEwJVUzEXMBUGA1UEChMO # RGlnaUNlcnQsIEluYy4xQTA/BgNVBAMTOERpZ2lDZXJ0IFRydXN0ZWQgRzQgVGlt # ZVN0YW1waW5nIFJTQTQwOTYgU0hBMjU2IDIwMjUgQ0ExAhAKgO8YS43xBYLRxHan # lXRoMA0GCWCGSAFlAwQCAQUAoIHRMBoGCSqGSIb3DQEJAzENBgsqhkiG9w0BCRAB # BDAcBgkqhkiG9w0BCQUxDxcNMjUwODEwMTEyODA1WjArBgsqhkiG9w0BCRACDDEc # MBowGDAWBBTdYjCshgotMGvaOLFoeVIwB/tBfjAvBgkqhkiG9w0BCQQxIgQgiJPt # m03Cl79YlaX/fnZWxghP7+5C9QJHSvSphJpyIcwwNwYLKoZIhvcNAQkQAi8xKDAm # MCQwIgQgSqA/oizXXITFXJOPgo5na5yuyrM/420mmqM08UYRCjMwDQYJKoZIhvcN # AQEBBQAEggIAmeHPEYoOlGutayGyohq8o5uwgcUxuhqu/JdFN7+9AOUW9ZEdlBM5 # Aya8a1yOcijsk70yr3yCWLoTmhXOmLE3+gfzfzNn7bNzZt2E7MJqWP/hZnlHbJO5 # YZ/cjg8MBvK5FGGogYME1hOJPLsWbMsoLqapDjoMPwq7OWh2EQ290dug3aQTXty0 # NEvKwG3uUz/0cOFMK99p7qlsGKhIkNIiWIv6xp+6lf87vdmQznyfh22Ql079hTVa # s7aCZcXOthXthnHxymrKOdaLFhgNvNKcKCLPQnfDeWA+ONtAeHU39xDSFi3M8c+N # kmmfR8USDuNuef4VC41y49KwYwl22yTZnzvBmaxVtgxCv+oZ8cXRujKfPNceU0uT # SYa6Xg3UjbuLNeY04it9FTGE4WLQussVuxQI1USdx9uBeGYpJOf+7ZoVtqM9fhGZ # 7r2FELKb6P4kTOlmVEch9ZAjSSNlS40eKFGJILhjG1hvz5lJ7VJB2K0C5J2pOmn2 # emE0ebzYGwSgyCkPFbTgt3vkmT7FpXO3a1CdnDaxzU/IzT947Hxb54hVMxSR7hNk # AEn3IYalKmbsZTbkXoAaxqNNv4NhTFVk4obzsuWWGrPByjsK2drsoqgws4a80nnA # +xOovag1tYH8rdk2iu8x5+u5azAMKNs0vR+5GyYgtGIAQIxZj75rhis= # SIG # End signature block |