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 # MIIdNgYJKoZIhvcNAQcCoIIdJzCCHSMCAQExDzANBglghkgBZQMEAgEFADB5Bgor # 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 # +EkUqkWS4h8zsRiyvR9Pnwklg6UxghliMIIZXgIBATA+MCoxKDAmBgNVBAMMH0NI # RU5YSVpIQU5HIC0gQ29kZSBTaWduaW5nIENlcnQCEFyyDmbfMz2RRnGZ43MhA2Mw # DQYJYIZIAWUDBAIBBQCgfDAQBgorBgEEAYI3AgEMMQIwADAZBgkqhkiG9w0BCQMx # DAYKKwYBBAGCNwIBBDAcBgorBgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAvBgkq # hkiG9w0BCQQxIgQgUzEYAeURw8nPzBIIklAzPVCgrB/HF87fHvLwsNuWObswDQYJ # KoZIhvcNAQEBBQAEggEAQ8jH/Mrkae4gKX370ZH8Bd0dzStOfcgDxpxQYX1N5Trs # uxyty8ecwT89zFh3FNV+UVdCTsledz/HebIF1hDVjWgyvE6A3BNCCJgbVskTZxwy # HzaKdesBB29tZbMhCKbQ9StyON4gPQkLUTqhLhjDka6HN3HB6VnuJf6qpiEftQdG # uryHvfjrKIOVKb8WwEYp57GSr24nfMglzTPzJqfx6JuBfgmWgl6pVZd0fvhg4zUP # qMPEaz+hRAQD6LcfIJmmj2uHrlqp0Yu5NIHcUzFw10yvbfixpPf4+2h8E4xdWzWc # gI6IKTP4R1ZhjOKZ+FXOdaV/Z4WlDpTa1ypjAeufOKGCF3cwghdzBgorBgEEAYI3 # AwMBMYIXYzCCF18GCSqGSIb3DQEHAqCCF1AwghdMAgEDMQ8wDQYJYIZIAWUDBAIB # BQAweAYLKoZIhvcNAQkQAQSgaQRnMGUCAQEGCWCGSAGG/WwHATAxMA0GCWCGSAFl # AwQCAQUABCDZebEY+x1oCECyHXY68HedFaKfbVWIAqdxY42ENQ6v5QIRAJM6YMyt # LLD+ZkoXEmhD6BYYDzIwMjUwODE5MDgxMDQxWqCCEzowggbtMIIE1aADAgECAhAK # gO8YS43xBYLRxHanlXRoMA0GCSqGSIb3DQEBCwUAMGkxCzAJBgNVBAYTAlVTMRcw # FQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1c3Rl # ZCBHNCBUaW1lU3RhbXBpbmcgUlNBNDA5NiBTSEEyNTYgMjAyNSBDQTEwHhcNMjUw # NjA0MDAwMDAwWhcNMzYwOTAzMjM1OTU5WjBjMQswCQYDVQQGEwJVUzEXMBUGA1UE # ChMORGlnaUNlcnQsIEluYy4xOzA5BgNVBAMTMkRpZ2lDZXJ0IFNIQTI1NiBSU0E0 # MDk2IFRpbWVzdGFtcCBSZXNwb25kZXIgMjAyNSAxMIICIjANBgkqhkiG9w0BAQEF # AAOCAg8AMIICCgKCAgEA0EasLRLGntDqrmBWsytXum9R/4ZwCgHfyjfMGUIwYzKo # md8U1nH7C8Dr0cVMF3BsfAFI54um8+dnxk36+jx0Tb+k+87H9WPxNyFPJIDZHhAq # lUPt281mHrBbZHqRK71Em3/hCGC5KyyneqiZ7syvFXJ9A72wzHpkBaMUNg7MOLxI # 6E9RaUueHTQKWXymOtRwJXcrcTTPPT2V1D/+cFllESviH8YjoPFvZSjKs3SKO1QN # UdFd2adw44wDcKgH+JRJE5Qg0NP3yiSyi5MxgU6cehGHr7zou1znOM8odbkqoK+l # J25LCHBSai25CFyD23DZgPfDrJJJK77epTwMP6eKA0kWa3osAe8fcpK40uhktzUd # /Yk0xUvhDU6lvJukx7jphx40DQt82yepyekl4i0r8OEps/FNO4ahfvAk12hE5FVs # 9HVVWcO5J4dVmVzix4A77p3awLbr89A90/nWGjXMGn7FQhmSlIUDy9Z2hSgctaep # ZTd0ILIUbWuhKuAeNIeWrzHKYueMJtItnj2Q+aTyLLKLM0MheP/9w6CtjuuVHJOV # oIJ/DtpJRE7Ce7vMRHoRon4CWIvuiNN1Lk9Y+xZ66lazs2kKFSTnnkrT3pXWETTJ # khd76CIDBbTRofOsNyEhzZtCGmnQigpFHti58CSmvEyJcAlDVcKacJ+A9/z7eacC # AwEAAaOCAZUwggGRMAwGA1UdEwEB/wQCMAAwHQYDVR0OBBYEFOQ7/PIx7f391/OR # cWMZUEPPYYzoMB8GA1UdIwQYMBaAFO9vU0rp5AZ8esrikFb2L9RJ7MtOMA4GA1Ud # DwEB/wQEAwIHgDAWBgNVHSUBAf8EDDAKBggrBgEFBQcDCDCBlQYIKwYBBQUHAQEE # gYgwgYUwJAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBdBggr # BgEFBQcwAoZRaHR0cDovL2NhY2VydHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1 # c3RlZEc0VGltZVN0YW1waW5nUlNBNDA5NlNIQTI1NjIwMjVDQTEuY3J0MF8GA1Ud # HwRYMFYwVKBSoFCGTmh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFRy # dXN0ZWRHNFRpbWVTdGFtcGluZ1JTQTQwOTZTSEEyNTYyMDI1Q0ExLmNybDAgBgNV # HSAEGTAXMAgGBmeBDAEEAjALBglghkgBhv1sBwEwDQYJKoZIhvcNAQELBQADggIB # AGUqrfEcJwS5rmBB7NEIRJ5jQHIh+OT2Ik/bNYulCrVvhREafBYF0RkP2AGr181o # 2YWPoSHz9iZEN/FPsLSTwVQWo2H62yGBvg7ouCODwrx6ULj6hYKqdT8wv2UV+Kbz # /3ImZlJ7YXwBD9R0oU62PtgxOao872bOySCILdBghQ/ZLcdC8cbUUO75ZSpbh1oi # pOhcUT8lD8QAGB9lctZTTOJM3pHfKBAEcxQFoHlt2s9sXoxFizTeHihsQyfFg5fx # UFEp7W42fNBVN4ueLaceRf9Cq9ec1v5iQMWTFQa0xNqItH3CPFTG7aEQJmmrJTV3 # Qhtfparz+BW60OiMEgV5GWoBy4RVPRwqxv7Mk0Sy4QHs7v9y69NBqycz0BZwhB9W # OfOu/CIJnzkQTwtSSpGGhLdjnQ4eBpjtP+XB3pQCtv4E5UCSDag6+iX8MmB10nfl # dPF9SVD7weCC3yXZi/uuhqdwkgVxuiMFzGVFwYbQsiGnoa9F5AaAyBjFBtXVLcKt # apnMG3VH3EmAp/jsJ3FVF3+d1SVDTmjFjLbNFZUWMXuZyvgLfgyPehwJVxwC+UpX # 2MSey2ueIu9THFVkT+um1vshETaWyQo8gmBto/m3acaP9QsuLj3FNwFlTxq25+T4 # QwX9xa6ILs84ZPvmpovq90K8eWyG2N01c4IhSOxqt81nMIIGtDCCBJygAwIBAgIQ # DcesVwX/IZkuQEMiDDpJhjANBgkqhkiG9w0BAQsFADBiMQswCQYDVQQGEwJVUzEV # MBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29t # MSEwHwYDVQQDExhEaWdpQ2VydCBUcnVzdGVkIFJvb3QgRzQwHhcNMjUwNTA3MDAw # MDAwWhcNMzgwMTE0MjM1OTU5WjBpMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGln # aUNlcnQsIEluYy4xQTA/BgNVBAMTOERpZ2lDZXJ0IFRydXN0ZWQgRzQgVGltZVN0 # YW1waW5nIFJTQTQwOTYgU0hBMjU2IDIwMjUgQ0ExMIICIjANBgkqhkiG9w0BAQEF # AAOCAg8AMIICCgKCAgEAtHgx0wqYQXK+PEbAHKx126NGaHS0URedTa2NDZS1mZaD # LFTtQ2oRjzUXMmxCqvkbsDpz4aH+qbxeLho8I6jY3xL1IusLopuW2qftJYJaDNs1 # +JH7Z+QdSKWM06qchUP+AbdJgMQB3h2DZ0Mal5kYp77jYMVQXSZH++0trj6Ao+xh # /AS7sQRuQL37QXbDhAktVJMQbzIBHYJBYgzWIjk8eDrYhXDEpKk7RdoX0M980EpL # tlrNyHw0Xm+nt5pnYJU3Gmq6bNMI1I7Gb5IBZK4ivbVCiZv7PNBYqHEpNVWC2ZQ8 # BbfnFRQVESYOszFI2Wv82wnJRfN20VRS3hpLgIR4hjzL0hpoYGk81coWJ+KdPvMv # aB0WkE/2qHxJ0ucS638ZxqU14lDnki7CcoKCz6eum5A19WZQHkqUJfdkDjHkccpL # 6uoG8pbF0LJAQQZxst7VvwDDjAmSFTUms+wV/FbWBqi7fTJnjq3hj0XbQcd8hjj/ # q8d6ylgxCZSKi17yVp2NL+cnT6Toy+rN+nM8M7LnLqCrO2JP3oW//1sfuZDKiDEb # 1AQ8es9Xr/u6bDTnYCTKIsDq1BtmXUqEG1NqzJKS4kOmxkYp2WyODi7vQTCBZtVF # JfVZ3j7OgWmnhFr4yUozZtqgPrHRVHhGNKlYzyjlroPxul+bgIspzOwbtmsgY1MC # AwEAAaOCAV0wggFZMBIGA1UdEwEB/wQIMAYBAf8CAQAwHQYDVR0OBBYEFO9vU0rp # 5AZ8esrikFb2L9RJ7MtOMB8GA1UdIwQYMBaAFOzX44LScV1kTN8uZz/nupiuHA9P # MA4GA1UdDwEB/wQEAwIBhjATBgNVHSUEDDAKBggrBgEFBQcDCDB3BggrBgEFBQcB # AQRrMGkwJAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBBBggr # BgEFBQcwAoY1aHR0cDovL2NhY2VydHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1 # c3RlZFJvb3RHNC5jcnQwQwYDVR0fBDwwOjA4oDagNIYyaHR0cDovL2NybDMuZGln # aWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZFJvb3RHNC5jcmwwIAYDVR0gBBkwFzAI # BgZngQwBBAIwCwYJYIZIAYb9bAcBMA0GCSqGSIb3DQEBCwUAA4ICAQAXzvsWgBz+ # Bz0RdnEwvb4LyLU0pn/N0IfFiBowf0/Dm1wGc/Do7oVMY2mhXZXjDNJQa8j00DNq # hCT3t+s8G0iP5kvN2n7Jd2E4/iEIUBO41P5F448rSYJ59Ib61eoalhnd6ywFLery # cvZTAz40y8S4F3/a+Z1jEMK/DMm/axFSgoR8n6c3nuZB9BfBwAQYK9FHaoq2e26M # HvVY9gCDA/JYsq7pGdogP8HRtrYfctSLANEBfHU16r3J05qX3kId+ZOczgj5kjat # VB+NdADVZKON/gnZruMvNYY2o1f4MXRJDMdTSlOLh0HCn2cQLwQCqjFbqrXuvTPS # egOOzr4EWj7PtspIHBldNE2K9i697cvaiIo2p61Ed2p8xMJb82Yosn0z4y25xUbI # 7GIN/TpVfHIqQ6Ku/qjTY6hc3hsXMrS+U0yy+GWqAXam4ToWd2UQ1KYT70kZjE4Y # tL8Pbzg0c1ugMZyZZd/BdHLiRu7hAWE6bTEm4XYRkA6Tl4KSFLFk43esaUeqGkH/ # wyW4N7OigizwJWeukcyIPbAvjSabnf7+Pu0VrFgoiovRDiyx3zEdmcif/sYQsfch # 28bZeUz2rtY/9TCA6TD8dC3JE3rYkrhLULy7Dc90G6e8BlqmyIjlgp2+VqsS9/wQ # D7yFylIz0scmbKvFoW2jNrbM1pD2T7m3XDCCBY0wggR1oAMCAQICEA6bGI750C3n # 79tQ4ghAGFowDQYJKoZIhvcNAQEMBQAwZTELMAkGA1UEBhMCVVMxFTATBgNVBAoT # DERpZ2lDZXJ0IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTEkMCIGA1UE # AxMbRGlnaUNlcnQgQXNzdXJlZCBJRCBSb290IENBMB4XDTIyMDgwMTAwMDAwMFoX # DTMxMTEwOTIzNTk1OVowYjELMAkGA1UEBhMCVVMxFTATBgNVBAoTDERpZ2lDZXJ0 # IEluYzEZMBcGA1UECxMQd3d3LmRpZ2ljZXJ0LmNvbTEhMB8GA1UEAxMYRGlnaUNl # cnQgVHJ1c3RlZCBSb290IEc0MIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKC # AgEAv+aQc2jeu+RdSjwwIjBpM+zCpyUuySE98orYWcLhKac9WKt2ms2uexuEDcQw # H/MbpDgW61bGl20dq7J58soR0uRf1gU8Ug9SH8aeFaV+vp+pVxZZVXKvaJNwwrK6 # dZlqczKU0RBEEC7fgvMHhOZ0O21x4i0MG+4g1ckgHWMpLc7sXk7Ik/ghYZs06wXG # XuxbGrzryc/NrDRAX7F6Zu53yEioZldXn1RYjgwrt0+nMNlW7sp7XeOtyU9e5TXn # Mcvak17cjo+A2raRmECQecN4x7axxLVqGDgDEI3Y1DekLgV9iPWCPhCRcKtVgkEy # 19sEcypukQF8IUzUvK4bA3VdeGbZOjFEmjNAvwjXWkmkwuapoGfdpCe8oU85tRFY # F/ckXEaPZPfBaYh2mHY9WV1CdoeJl2l6SPDgohIbZpp0yt5LHucOY67m1O+Skjqe # PdwA5EUlibaaRBkrfsCUtNJhbesz2cXfSwQAzH0clcOP9yGyshG3u3/y1YxwLEFg # qrFjGESVGnZifvaAsPvoZKYz0YkH4b235kOkGLimdwHhD5QMIR2yVCkliWzlDlJR # R3S+Jqy2QXXeeqxfjT/JvNNBERJb5RBQ6zHFynIWIgnffEx1P2PsIV/EIFFrb7Gr # hotPwtZFX50g/KEexcCPorF+CiaZ9eRpL5gdLfXZqbId5RsCAwEAAaOCATowggE2 # MA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYEFOzX44LScV1kTN8uZz/nupiuHA9P # MB8GA1UdIwQYMBaAFEXroq/0ksuCMS1Ri6enIZ3zbcgPMA4GA1UdDwEB/wQEAwIB # hjB5BggrBgEFBQcBAQRtMGswJAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2lj # ZXJ0LmNvbTBDBggrBgEFBQcwAoY3aHR0cDovL2NhY2VydHMuZGlnaWNlcnQuY29t # L0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENBLmNydDBFBgNVHR8EPjA8MDqgOKA2hjRo # dHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRBc3N1cmVkSURSb290Q0Eu # Y3JsMBEGA1UdIAQKMAgwBgYEVR0gADANBgkqhkiG9w0BAQwFAAOCAQEAcKC/Q1xV # 5zhfoKN0Gz22Ftf3v1cHvZqsoYcs7IVeqRq7IviHGmlUIu2kiHdtvRoU9BNKei8t # tzjv9P+Aufih9/Jy3iS8UgPITtAq3votVs/59PesMHqai7Je1M/RQ0SbQyHrlnKh # SLSZy51PpwYDE3cnRNTnf+hZqPC/Lwum6fI0POz3A8eHqNJMQBk1RmppVLC4oVaO # 7KTVPeix3P0c2PR3WlxUjG/voVA9/HYJaISfb8rbII01YBwCA8sgsKxYoA5AY8WY # IsGyWfVVa88nq2x2zm8jLfR+cWojayL/ErhULSd+2DrZ8LaHlv1b0VysGMNNn3O3 # AamfV6peKOK5lDGCA3wwggN4AgEBMH0waTELMAkGA1UEBhMCVVMxFzAVBgNVBAoT # DkRpZ2lDZXJ0LCBJbmMuMUEwPwYDVQQDEzhEaWdpQ2VydCBUcnVzdGVkIEc0IFRp # bWVTdGFtcGluZyBSU0E0MDk2IFNIQTI1NiAyMDI1IENBMQIQCoDvGEuN8QWC0cR2 # p5V0aDANBglghkgBZQMEAgEFAKCB0TAaBgkqhkiG9w0BCQMxDQYLKoZIhvcNAQkQ # AQQwHAYJKoZIhvcNAQkFMQ8XDTI1MDgxOTA4MTA0MVowKwYLKoZIhvcNAQkQAgwx # HDAaMBgwFgQU3WIwrIYKLTBr2jixaHlSMAf7QX4wLwYJKoZIhvcNAQkEMSIEIIZ8 # f88m6yKaMrDVc8p/4CklUKzVX9XLd7sq+znacX/AMDcGCyqGSIb3DQEJEAIvMSgw # JjAkMCIEIEqgP6Is11yExVyTj4KOZ2ucrsqzP+NtJpqjNPFGEQozMA0GCSqGSIb3 # DQEBAQUABIICADbc88B0U6yc8orSJZwXA2iyaMxJvvLLGMnFhX9oDziGLXExYIDz # DStK/9yccjKvg6ekOSUSTs1K+aex+rjt011adWdc0fRfIAnRX+2Vdgj5FoRcricW # YDIRa83P2UHEMK+8jdFqf7LFO93cEXjY/NZ/aSVezuX0/AkXcZK5i4Y8RE3pWuwN # i2s7w10BB/kCWCcsyNkoj5pUPKkui5/fWub2e1chckuLSID0b5d3IPc6uLBIYzhk # kfd7nQ0p0LrOD3YN3hnKssxwj3kXDzVd2U38bu7VIg/XUo4s9hEF7aNDnm1zoNUy # k3AZGV4Q/lTBgwI/qdoLp0CHU7FmjbkXaOIHcp6HvS6yjhB1ZxqocO9707dB2Fur # zV0A3+ZyyawbRmqLWSk0XvLaPrm5C50cRJp1FobcopWaM4vphpIHnFBv9woMRW52 # w/IqJP08YC/vzunddb3hHP2gj3YoKLY04/tUKoCOs0nVu5T54fbWZnLukkMhppAK # Wv4lJothzCBQxzkN5VR+iaiKmVA5WQsckgWuPU6uz8+0yGnm5KpaR3jQD3Trbtcr # 0Vr/Su29gpw91hrsk1SLChkFUtKYP3yu8AZctXwb2PodgnkQFVHNi4/DUcx2TF/t # HyuuluZzt2wL5YvYvOev8h6sFWoxOSNqpQfwYCqADYYLvsqKXs1Cf5h7 # SIG # End signature block |