UNPKG

@microsoft/windows-admin-center-sdk

Version:

Microsoft - Windows Admin Center Shell

821 lines (818 loc) 37.8 kB
import { from, of, Subject, zip } from 'rxjs'; import { filter, map, mergeMap, take, tap } from 'rxjs/operators'; import { LogLevel } from '../../diagnostics/log-level'; import { Logging } from '../../diagnostics/logging'; import { RpcExtensionBrokerResponseKey } from '../../rpc/extension-broker/rpc-extension-broker-model'; import { RpcExtensionBrokerModuleSubjectServer } from '../../rpc/extension-broker/rpc-extension-broker-module-request-subject-server'; import { RpcExtensionBrokerRequestClient } from '../../rpc/extension-broker/rpc-extension-broker-request-client'; export class ExtensionBrokerQuery { rpc; constructor(rpc) { this.rpc = rpc; } watcher; resultEmitter; /** * Handles rpc response messages from the extension host. * @param data the result of the extension host request */ onRpcResponse(data) { this.watcher.next(data); return Promise.resolve(); } /** * Calls a method on a extension instance and returns its result. * @param instanceId The instance id of the extension to call * @param method The method to call * @param version The version of the method to call * @param args The arguments to pass to the method * @returns an observable for the result of the call (may be null if method call is void) */ call(instanceId, method, version, ...args) { const payload = { requestType: 'call', instanceId: instanceId, version: version, method: method, arguments: args, callType: 'worker' }; return this.request(payload).pipe(map((response) => response.return)); } /** * Destroys a extension instance * @param instanceId The instance id of the extension to destroy * @returns an observable for when the extension is destroyed */ destroy(instanceId) { const payload = { requestType: 'destroy', instanceId: instanceId }; return this.request(payload); } /** * Moves a extension instance * @param instanceId The instance id of the extension to destroy * @param rect The rectangular position to move the extension to (relative to the iframe) * @param zIndex the z index to move to, default is 1 * @returns an observable for when the extension is destroyed */ move(instanceId, rect, zIndex = 1) { const payload = { requestType: 'move', instanceId: instanceId, rect: rect, zIndex: zIndex }; return this.request(payload); } } /** * Extension broker listener class * Register a listener without needed full extension broker service */ export class ExtensionBrokerListener { static extensionBrokerModuleSubjectServer; static requestReceivedSubject; static get requestReceived() { if (!ExtensionBrokerListener.requestReceivedSubject) { ExtensionBrokerListener.requestReceivedSubject = new Subject(); } return ExtensionBrokerListener.requestReceivedSubject.asObservable(); } static initialize(rpc) { // set up extension broker request channel if (!ExtensionBrokerListener.extensionBrokerModuleSubjectServer) { ExtensionBrokerListener.extensionBrokerModuleSubjectServer = new RpcExtensionBrokerModuleSubjectServer(rpc); ExtensionBrokerListener.extensionBrokerModuleSubjectServer.subject .subscribe(request => ExtensionBrokerListener.handleRpcRequest(request)); } } static handleRpcRequest(request) { Logging.log({ level: LogLevel.Debug, message: `Extension broker request received: ${request}`, source: 'ExtensionBroker', params: request }); request.deferred.resolve(); if (ExtensionBrokerListener.requestReceivedSubject) { ExtensionBrokerListener.requestReceivedSubject.next(request.data); } } } /** * Extension Broker Module Side Service. * Manages requests designated for other extensions through the shell using RPC */ export class ExtensionBroker extends ExtensionBrokerQuery { /** * The source name to use for logging */ get logSourceName() { return 'ExtensionBroker'; } /** * Initializes a new instance of the Extension Broker class * @param rpc The rpc to use to communicate with the shell */ constructor(rpc) { super(rpc); // set up extension broker response channel this.watcher = new Subject(); this.rpc.register(RpcExtensionBrokerResponseKey.command, this.onRpcResponse.bind(this)); // set up extension broker request channel ExtensionBrokerListener.initialize(rpc); // hook up emit event pipeline ExtensionBrokerListener.requestReceived.pipe(filter(request => request.payload && request.payload.requestType === 'emit')).subscribe(request => this.onEmitReceived(request)); this.resultEmitter = new Subject(); } /** * Occurs when an emit request is received from shell */ onEmitReceived(request) { Logging.log({ level: LogLevel.Debug, message: `Emit request received`, source: this.logSourceName, params: request }); const payload = request.payload; this.resultEmitter.next(payload); } /** * Gets the entry point ids for all the extensions that fulfill an extension targets contract * In some cases, as specified by the extension target, the shell may ask the user to select a preferred * extension that fulfills the target contract. Returning only the 1 entry point id. * @param extensionTargetId The id of the extension target * @return An observable for the entry point ids of the extensions that fulfill an extension targets contract. */ getTargetExtensions(extensionTargetId) { const payload = { requestType: 'fulfill', extensionTargetId: extensionTargetId }; return this.request(payload).pipe(map((response) => { return response.fulfillment; })); } /** * Creates a SnapIn instance given its entry point id. * This instance is tied to the calling extension and will close when destroy is called or the calling extension is destroyed. * @param entryPointId The id of the worker to create */ createSnapIn(entryPointId) { const payload = { requestType: 'create', entryPointId: entryPointId }; return this.request(payload).pipe(map((response) => { const instance = { instanceId: response.instanceId, call: (method, version, ...args) => this.call(response.instanceId, method, version, ...args), destroy: () => this.destroy(response.instanceId).pipe(map(MsftSme.noop)), move: (rect, zIndex) => this.move(response.instanceId, rect, zIndex).pipe(map(MsftSme.noop)), listen: (eventType) => this.listen(eventType, response.instanceId) }; return instance; })); } /** * Finds a worker given either its entryPointId or instanceId * @param searchOptions The search options for finding a worker instance */ findWorker(searchOptions) { // if search will always result in no worker, just return no worker. if (MsftSme.isNullOrUndefined(searchOptions) || (MsftSme.isNullOrUndefined(searchOptions.entryPointId) && MsftSme.isNullOrUndefined(searchOptions.instanceId))) { return of(null); } const payload = { requestType: 'find', searchOptions: searchOptions }; return this.request(payload).pipe(map((response) => { if (!response.found) { return null; } const instance = { instanceId: response.instanceId, extenderDefinition: response.extenderDefinition, contract: response.contract, call: (method, version, ...args) => this.call(response.instanceId, method, version, ...args), destroy: () => this.destroy(response.instanceId).pipe(map(MsftSme.noop)), listen: (eventType) => this.listen(eventType, response.instanceId) }; return instance; })); } /** * Finds extensions by search condition. * @param searchOptions The search options for finding manifest entry points that meet the condition. */ findEntriesByCondition(searchOptions) { // set of search options must meet. if (MsftSme.isNullOrUndefined(searchOptions) || !MsftSme.isNullOrUndefined(searchOptions.entryPointId) || !MsftSme.isNullOrUndefined(searchOptions.instanceId) || searchOptions.createIfNotFound || (searchOptions.extensionTypes == null || searchOptions.extensionTypes.length === 0) || MsftSme.isNullOrUndefined(searchOptions.condition)) { throw new Error('Argument error for \"searchOptions\" on findEntriesByCondition() function'); } const payload = { requestType: 'find', searchOptions: searchOptions }; return this.request(payload).pipe(map((response) => { const conditionValidated = searchOptions.condition.validationRequired; let entryPoints = []; if (response.found) { entryPoints = response.entryPoints; } return { entryPoints, conditionValidated }; })); } /** * return all registered entry points by type * @param types entry point types to return */ getExtensionEntryPointsByType(types) { if (MsftSme.isNullOrUndefined(types)) { return of(null); } const payload = { requestType: 'find', searchOptions: { extensionTypes: types } }; return this.request(payload).pipe(map((response) => { if (!response.found) { return null; } return response.entryPoints; })); } /** * Creates a worker instance given its entry point id. * This instance is tied to the calling extension and will close when destroy is called or the calling extension is destroyed. * @param entryPointId The id of the worker to create */ createWorker(entryPointId, extensionTarget) { const payload = { requestType: 'create', entryPointId: entryPointId, extensionTarget: extensionTarget }; return this.request(payload).pipe(map((response) => { const instance = { instanceId: response.instanceId, extenderDefinition: response.extenderDefinition, contract: response.contract, call: (method, version, ...args) => this.call(response.instanceId, method, version, ...args), destroy: () => this.destroy(response.instanceId).pipe(map(MsftSme.noop)), listen: (eventType) => this.listen(eventType, response.instanceId) }; return instance; })); } /** * subscribe to results emission for specific action */ listen(eventType, instanceId) { // send request to shell to add to database // return result emitter filtered by type and instance const payload = { requestType: 'listen', instanceId, eventType }; return this.request(payload).pipe(mergeMap(() => this.resultEmitter.asObservable()), filter(result => result.instanceId === instanceId && result.eventType === eventType)); } /** * Runs a worker instance and returns the result of one method. After which the worker is destroyed. * @param entryPointId The id of the worker to create * @param method The method to call * @param version The version of the method to call * @param args The arguments to pass to the method * @returns an observable for the result of the worker. */ runWorker(entryPointId, method, version, ...args) { const payload = { requestType: 'run', entryPointId: entryPointId, version: version, method: method, arguments: args }; return this.request(payload).pipe(map((response) => { return response.return; })); } /** * Runs a worker instance with no association to the calling extension. * The worker will run its own workflow until it is finished * @param entryPointId The id of the worker to create * @param method The method to call * @param version The version of the method to call * @param args The arguments to pass to the method * @returns an observable fro the creation of the worker. */ startWorker(entryPointId, method, version, ...args) { const payload = { requestType: 'run', expectReturn: false, entryPointId: entryPointId, version: version, method: method, arguments: args }; return this.request(payload).pipe(map(MsftSme.noop)); } /** * Runs a dialog instance and returns its result. After which the dialog is destroyed. * @param entryPointId The id of the dialog to create * @param version The version of the dialog to call * @param args The arguments to pass to the dialog * @returns an observable for the result of the dialog. */ createDialog(entryPointId) { const payload = { requestType: 'create', entryPointId: entryPointId }; return this.request(payload).pipe(map((response) => { const instance = { instanceId: response.instanceId, extenderDefinition: response.extenderDefinition, contract: response.contract, show: (version, ...args) => this.show(response.instanceId, version, ...args), hide: () => this.destroy(response.instanceId).pipe(map(MsftSme.noop)), listen: (eventType) => this.listen(eventType, response.instanceId) }; return instance; })); } /** * Runs a dialog instance and returns its result. After which the dialog is destroyed. * @param entryPointId The id of the dialog to create * @param version The version of the dialog to call * @param args The arguments to pass to the dialog * @returns an observable for the result of the dialog. */ showDialog(entryPointId, version, ...args) { const payload = { requestType: 'run', entryPointId: entryPointId, version: version, method: 'show', arguments: args }; return this.request(payload).pipe(map((response) => { return response.return; })); } /** * Runs a dialog instance and returns its result. After which the dialog is destroyed. * @param instanceId The id of the extension to show * @param version The version of the extension tp show * @param args The arguments to pass to the extension * @returns an observable for the result of the dialog. */ show(instanceId, version, ...args) { const payload = { requestType: 'call', instanceId: instanceId, version: version, method: 'show', arguments: args, callType: 'component' }; return this.request(payload).pipe(map((response) => { return response.return; })); } /** * Emit a result to listeners subscribed to the given eventType and instanceId * @param instanceId The instance Id of the component emitting the result * @param eventType the type of event being emitted, can be any string like "output", "validation", "changes" * so listeners can subscribe to specific event streams. listeners must be aware of chosen eventType * @param result the result to be sent to event listeners */ emitResult(instanceId, eventType, result) { const payload = { requestType: 'emit', instanceId: instanceId, eventType: eventType, data: result }; return this.request(payload); } /** * Calls a method on a service extension * @param entryPointId The id of the service to call * @param method The method to call * @param version The version of the method to call * @param args The arguments to pass to the method * @returns an observable for the result of the service method. */ callService(entryPointId, method, version, ...args) { const payload = { requestType: 'call', instanceId: entryPointId, version: version, method: method, arguments: args, callType: 'service' }; return this.request(payload).pipe(map((response) => { return response.return; })); } destroyRequested(entryPointId, version, args) { const payload = { requestType: 'call', instanceId: entryPointId, version: version, method: 'destroyRequested', arguments: args, callType: 'component' }; return this.request(payload).pipe(map((response) => { return response.return; })); } /** * Makes an extension broker request via the RPC * @param requestType The @see RpcExtensionBrokerRequestType. * @param payload The payload for this message. Depends on requestType. * @returns An observable for the response message from the shell */ request(payload) { const request = { requestId: MsftSme.newGuid(), payload: payload }; Logging.log({ level: LogLevel.Debug, message: `Sending request to Extension Broker Service: ${request.requestId}`, source: 'ExtensionBroker', params: { request: request } }); return zip( // wait for our request promise to be fulfilled from(RpcExtensionBrokerRequestClient.extensionBrokerRequest(this.rpc, request)).pipe(tap(() => { Logging.log({ level: LogLevel.Debug, message: `Request verified received from Extension Broker Shell Service: ${request.requestId}`, source: 'ExtensionBroker', params: { request: request } }); })), // wait for our response to come in via RPC, this may happen anytime after the request is sent, // but not necessarily after its promise is resolved this.watcher.pipe(filter((result) => { const isMatch = result.requestId === request.requestId; Logging.log({ level: LogLevel.Debug, message: `Checking potential response from Extension Broker Shell Service: ${result.requestId} === ${request.requestId} => ${isMatch}`, source: 'ExtensionBroker', params: { request: request, potentialResponse: result } }); return isMatch; }))).pipe(take(1), map(([_, result]) => { if (!MsftSme.isNullOrUndefined(result)) { if (!MsftSme.isNullOrUndefined(result.payload)) { return result.payload; } else if (!MsftSme.isNullOrUndefined(result.error)) { throw result.error; } } throw new Error('Unexpected Response from Extension Broker RPC Request'); })); } } //# sourceMappingURL=extension-broker.js.map // SIG // Begin signature block // SIG // MIIoKAYJKoZIhvcNAQcCoIIoGTCCKBUCAQExDzANBglg // SIG // hkgBZQMEAgEFADB3BgorBgEEAYI3AgEEoGkwZzAyBgor // SIG // BgEEAYI3AgEeMCQCAQEEEBDgyQbOONQRoqMAEEvTUJAC // SIG // AQACAQACAQACAQACAQAwMTANBglghkgBZQMEAgEFAAQg // SIG // uhBFGXX14b/OHCl5fQuSVqXB4L6GWOlLmz5QM4DGlxWg // SIG // gg12MIIF9DCCA9ygAwIBAgITMwAABARsdAb/VysncgAA // SIG // AAAEBDANBgkqhkiG9w0BAQsFADB+MQswCQYDVQQGEwJV // SIG // UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMH // SIG // UmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv // SIG // cmF0aW9uMSgwJgYDVQQDEx9NaWNyb3NvZnQgQ29kZSBT // SIG // aWduaW5nIFBDQSAyMDExMB4XDTI0MDkxMjIwMTExNFoX // SIG // DTI1MDkxMTIwMTExNFowdDELMAkGA1UEBhMCVVMxEzAR // SIG // BgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v // SIG // bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv // SIG // bjEeMBwGA1UEAxMVTWljcm9zb2Z0IENvcnBvcmF0aW9u // SIG // MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA // SIG // tCg32mOdDA6rBBnZSMwxwXegqiDEUFlvQH9Sxww07hY3 // SIG // w7L52tJxLg0mCZjcszQddI6W4NJYb5E9QM319kyyE0l8 // SIG // EvA/pgcxgljDP8E6XIlgVf6W40ms286Cr0azaA1f7vaJ // SIG // jjNhGsMqOSSSXTZDNnfKs5ENG0bkXeB2q5hrp0qLsm/T // SIG // WO3oFjeROZVHN2tgETswHR3WKTm6QjnXgGNj+V6rSZJO // SIG // /WkTqc8NesAo3Up/KjMwgc0e67x9llZLxRyyMWUBE9co // SIG // T2+pUZqYAUDZ84nR1djnMY3PMDYiA84Gw5JpceeED38O // SIG // 0cEIvKdX8uG8oQa047+evMfDRr94MG9EWwIDAQABo4IB // SIG // czCCAW8wHwYDVR0lBBgwFgYKKwYBBAGCN0wIAQYIKwYB // SIG // BQUHAwMwHQYDVR0OBBYEFPIboTWxEw1PmVpZS+AzTDwo // SIG // oxFOMEUGA1UdEQQ+MDykOjA4MR4wHAYDVQQLExVNaWNy // SIG // b3NvZnQgQ29ycG9yYXRpb24xFjAUBgNVBAUTDTIzMDAx // SIG // Mis1MDI5MjMwHwYDVR0jBBgwFoAUSG5k5VAF04KqFzc3 // SIG // IrVtqMp1ApUwVAYDVR0fBE0wSzBJoEegRYZDaHR0cDov // SIG // L3d3dy5taWNyb3NvZnQuY29tL3BraW9wcy9jcmwvTWlj // SIG // Q29kU2lnUENBMjAxMV8yMDExLTA3LTA4LmNybDBhBggr // SIG // BgEFBQcBAQRVMFMwUQYIKwYBBQUHMAKGRWh0dHA6Ly93 // SIG // d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY2VydHMvTWlj // SIG // Q29kU2lnUENBMjAxMV8yMDExLTA3LTA4LmNydDAMBgNV // SIG // HRMBAf8EAjAAMA0GCSqGSIb3DQEBCwUAA4ICAQCI5g/S // SIG // KUFb3wdUHob6Qhnu0Hk0JCkO4925gzI8EqhS+K4umnvS // SIG // BU3acsJ+bJprUiMimA59/5x7WhJ9F9TQYy+aD9AYwMtb // SIG // KsQ/rst+QflfML+Rq8YTAyT/JdkIy7R/1IJUkyIS6srf // SIG // G1AKlX8n6YeAjjEb8MI07wobQp1F1wArgl2B1mpTqHND // SIG // lNqBjfpjySCScWjUHNbIwbDGxiFr93JoEh5AhJqzL+8m // SIG // onaXj7elfsjzIpPnl8NyH2eXjTojYC9a2c4EiX0571Ko // SIG // mhENF3RtR25A7/X7+gk6upuE8tyMy4sBkl2MUSF08U+E // SIG // 2LOVcR8trhYxV1lUi9CdgEU2CxODspdcFwxdT1+G8YNc // SIG // gzHyjx3BNSI4nOZcdSnStUpGhCXbaOIXfvtOSfQX/UwJ // SIG // oruhCugvTnub0Wna6CQiturglCOMyIy/6hu5rMFvqk9A // SIG // ltIJ0fSR5FwljW6PHHDJNbCWrZkaEgIn24M2mG1M/Ppb // SIG // /iF8uRhbgJi5zWxo2nAdyDBqWvpWxYIoee/3yIWpquVY // SIG // cYGhJp/1I1sq/nD4gBVrk1SKX7Do2xAMMO+cFETTNSJq // SIG // fTSSsntTtuBLKRB5mw5qglHKuzapDiiBuD1Zt4QwxA/1 // SIG // kKcyQ5L7uBayG78kxlVNNbyrIOFH3HYmdH0Pv1dIX/Mq // SIG // 7avQpAfIiLpOWwcbjzCCB3owggVioAMCAQICCmEOkNIA // SIG // AAAAAAMwDQYJKoZIhvcNAQELBQAwgYgxCzAJBgNVBAYT // SIG // AlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQH // SIG // EwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29y // SIG // cG9yYXRpb24xMjAwBgNVBAMTKU1pY3Jvc29mdCBSb290 // SIG // IENlcnRpZmljYXRlIEF1dGhvcml0eSAyMDExMB4XDTEx // SIG // MDcwODIwNTkwOVoXDTI2MDcwODIxMDkwOVowfjELMAkG // SIG // A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAO // SIG // BgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m // SIG // dCBDb3Jwb3JhdGlvbjEoMCYGA1UEAxMfTWljcm9zb2Z0 // SIG // IENvZGUgU2lnbmluZyBQQ0EgMjAxMTCCAiIwDQYJKoZI // SIG // hvcNAQEBBQADggIPADCCAgoCggIBAKvw+nIQHC6t2G6q // SIG // ghBNNLrytlghn0IbKmvpWlCquAY4GgRJun/DDB7dN2vG // SIG // EtgL8DjCmQawyDnVARQxQtOJDXlkh36UYCRsr55JnOlo // SIG // XtLfm1OyCizDr9mpK656Ca/XllnKYBoF6WZ26DJSJhIv // SIG // 56sIUM+zRLdd2MQuA3WraPPLbfM6XKEW9Ea64DhkrG5k // SIG // NXimoGMPLdNAk/jj3gcN1Vx5pUkp5w2+oBN3vpQ97/vj // SIG // K1oQH01WKKJ6cuASOrdJXtjt7UORg9l7snuGG9k+sYxd // SIG // 6IlPhBryoS9Z5JA7La4zWMW3Pv4y07MDPbGyr5I4ftKd // SIG // gCz1TlaRITUlwzluZH9TupwPrRkjhMv0ugOGjfdf8NBS // SIG // v4yUh7zAIXQlXxgotswnKDglmDlKNs98sZKuHCOnqWbs // SIG // YR9q4ShJnV+I4iVd0yFLPlLEtVc/JAPw0XpbL9Uj43Bd // SIG // D1FGd7P4AOG8rAKCX9vAFbO9G9RVS+c5oQ/pI0m8GLhE // SIG // fEXkwcNyeuBy5yTfv0aZxe/CHFfbg43sTUkwp6uO3+xb // SIG // n6/83bBm4sGXgXvt1u1L50kppxMopqd9Z4DmimJ4X7Iv // SIG // hNdXnFy/dygo8e1twyiPLI9AN0/B4YVEicQJTMXUpUMv // SIG // dJX3bvh4IFgsE11glZo+TzOE2rCIF96eTvSWsLxGoGyY // SIG // 0uDWiIwLAgMBAAGjggHtMIIB6TAQBgkrBgEEAYI3FQEE // SIG // AwIBADAdBgNVHQ4EFgQUSG5k5VAF04KqFzc3IrVtqMp1 // SIG // ApUwGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBDAEEwCwYD // SIG // VR0PBAQDAgGGMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0j // SIG // BBgwFoAUci06AjGQQ7kUBU7h6qfHMdEjiTQwWgYDVR0f // SIG // BFMwUTBPoE2gS4ZJaHR0cDovL2NybC5taWNyb3NvZnQu // SIG // Y29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0 // SIG // MjAxMV8yMDExXzAzXzIyLmNybDBeBggrBgEFBQcBAQRS // SIG // MFAwTgYIKwYBBQUHMAKGQmh0dHA6Ly93d3cubWljcm9z // SIG // b2Z0LmNvbS9wa2kvY2VydHMvTWljUm9vQ2VyQXV0MjAx // SIG // MV8yMDExXzAzXzIyLmNydDCBnwYDVR0gBIGXMIGUMIGR // SIG // BgkrBgEEAYI3LgMwgYMwPwYIKwYBBQUHAgEWM2h0dHA6 // SIG // Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvZG9jcy9w // SIG // cmltYXJ5Y3BzLmh0bTBABggrBgEFBQcCAjA0HjIgHQBM // SIG // AGUAZwBhAGwAXwBwAG8AbABpAGMAeQBfAHMAdABhAHQA // SIG // ZQBtAGUAbgB0AC4gHTANBgkqhkiG9w0BAQsFAAOCAgEA // SIG // Z/KGpZjgVHkaLtPYdGcimwuWEeFjkplCln3SeQyQwWVf // SIG // Liw++MNy0W2D/r4/6ArKO79HqaPzadtjvyI1pZddZYSQ // SIG // fYtGUFXYDJJ80hpLHPM8QotS0LD9a+M+By4pm+Y9G6XU // SIG // tR13lDni6WTJRD14eiPzE32mkHSDjfTLJgJGKsKKELuk // SIG // qQUMm+1o+mgulaAqPyprWEljHwlpblqYluSD9MCP80Yr // SIG // 3vw70L01724lruWvJ+3Q3fMOr5kol5hNDj0L8giJ1h/D // SIG // Mhji8MUtzluetEk5CsYKwsatruWy2dsViFFFWDgycSca // SIG // f7H0J/jeLDogaZiyWYlobm+nt3TDQAUGpgEqKD6CPxNN // SIG // ZgvAs0314Y9/HG8VfUWnduVAKmWjw11SYobDHWM2l4bf // SIG // 2vP48hahmifhzaWX0O5dY0HjWwechz4GdwbRBrF1HxS+ // SIG // YWG18NzGGwS+30HHDiju3mUv7Jf2oVyW2ADWoUa9WfOX // SIG // pQlLSBCZgB/QACnFsZulP0V3HjXG0qKin3p6IvpIlR+r // SIG // +0cjgPWe+L9rt0uX4ut1eBrs6jeZeRhL/9azI2h15q/6 // SIG // /IvrC4DqaTuv/DDtBEyO3991bWORPdGdVk5Pv4BXIqF4 // SIG // ETIheu9BCrE/+6jMpF3BoYibV3FWTkhFwELJm3ZbCoBI // SIG // a/15n8G9bW1qyVJzEw16UM0xghoKMIIaBgIBATCBlTB+ // SIG // MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3Rv // SIG // bjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWlj // SIG // cm9zb2Z0IENvcnBvcmF0aW9uMSgwJgYDVQQDEx9NaWNy // SIG // b3NvZnQgQ29kZSBTaWduaW5nIFBDQSAyMDExAhMzAAAE // SIG // BGx0Bv9XKydyAAAAAAQEMA0GCWCGSAFlAwQCAQUAoIGu // SIG // MBkGCSqGSIb3DQEJAzEMBgorBgEEAYI3AgEEMBwGCisG // SIG // AQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMC8GCSqGSIb3 // SIG // DQEJBDEiBCCv7szbEoMlPC0DK7FYO57ehHhWxYlHTEkT // SIG // IWAQOPcZDjBCBgorBgEEAYI3AgEMMTQwMqAUgBIATQBp // SIG // AGMAcgBvAHMAbwBmAHShGoAYaHR0cDovL3d3dy5taWNy // SIG // b3NvZnQuY29tMA0GCSqGSIb3DQEBAQUABIIBAF3qM7sH // SIG // S0sZEBM9cM0cdwL5SsoCLbiXXpjw2QRNdt/tilKMeSni // SIG // /sJwvCaIRHUVaUJmpNExrb8m0aeDfgfzonlEPAtyqO4E // SIG // 0HNG6f23xMe5G1eLSg15kzqRBbaMIj2lOsa5/AnPu02U // SIG // 5BVZ64wTtqpYRcNMF+XyygYr8tVSZS2omYlYJ63HQO0Q // SIG // aerh9p4gHP7o8YZYtBnTjoaDl0CjyQ4d3iDY1sM3Oame // SIG // 27tAYV3KLXJ8/B86Prgg5i/UT5c2oLGGkAoFxdpqLk78 // SIG // 9B1tJl70ah8vsy4LY9cYYBf+7wmWEsphLOE8bGK7YQP0 // SIG // ZfT+C3W2HzdJyATfdZvufTsVxDGhgheUMIIXkAYKKwYB // SIG // BAGCNwMDATGCF4Awghd8BgkqhkiG9w0BBwKgghdtMIIX // SIG // aQIBAzEPMA0GCWCGSAFlAwQCAQUAMIIBUgYLKoZIhvcN // SIG // AQkQAQSgggFBBIIBPTCCATkCAQEGCisGAQQBhFkKAwEw // SIG // MTANBglghkgBZQMEAgEFAAQgjeU8z9bsVio3jqRi8xD1 // SIG // 6kgrMFhbmUtajdVIYMzv6bYCBmeuC226yRgTMjAyNTAy // SIG // MjAxNTI4MzUuNzAzWjAEgAIB9KCB0aSBzjCByzELMAkG // SIG // A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAO // SIG // BgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m // SIG // dCBDb3Jwb3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0 // SIG // IEFtZXJpY2EgT3BlcmF0aW9uczEnMCUGA1UECxMeblNo // SIG // aWVsZCBUU1MgRVNOOjdGMDAtMDVFMC1EOTQ3MSUwIwYD // SIG // VQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNl // SIG // oIIR6jCCByAwggUIoAMCAQICEzMAAAHwKnwdWTvmH60A // SIG // AQAAAfAwDQYJKoZIhvcNAQELBQAwfDELMAkGA1UEBhMC // SIG // VVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcT // SIG // B1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jw // SIG // b3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUt // SIG // U3RhbXAgUENBIDIwMTAwHhcNMjMxMjA2MTg0NTUxWhcN // SIG // MjUwMzA1MTg0NTUxWjCByzELMAkGA1UEBhMCVVMxEzAR // SIG // BgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v // SIG // bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv // SIG // bjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2EgT3Bl // SIG // cmF0aW9uczEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNO // SIG // OjdGMDAtMDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3Nv // SIG // ZnQgVGltZS1TdGFtcCBTZXJ2aWNlMIICIjANBgkqhkiG // SIG // 9w0BAQEFAAOCAg8AMIICCgKCAgEAtR4tU6M4dztHMxPM // SIG // X0Z68ppSTQmhlqJpj8tHiDX/uBCa+63/LUs5YBPCJeGY // SIG // +PE+X00zgIepyE0X9pSu/rqXJ2f8YNACqA2KQUDlqy/T // SIG // mxUocpSB36/w0OD7EV/BkbSJ7ibDJMEoOt23weENlIpD // SIG // BD/wrWR4vMVuV7QCqfeXLN5r1AjLyAAPNya/1MgAwAiV // SIG // 1VOJmmIrHM1M+ydddXg9SqxvZkPiE4J0Uf19sUwzAs/o // SIG // qPGWRRxsBGYPnN75j6fO5uDqjilsoXKjFLqT73jv4EAv // SIG // Ub+LMyzRg2qHj3iuiFNCanBo16sW3BKEv7NYQoD3e1Me // SIG // mFnlq1F2sW2/iaLIDms1IFBrNWqqZy489GCn1Kp/IuU2 // SIG // 5kdXahJUeEAPjmX3lYaU6J6zOLBPzJSdSS6UdhcACB1H // SIG // jH6LVzUIsrWH0QVDiRxXiWBH5WYFZNF8f+JGQXc4BUDz // SIG // ln1XdjaM15QtnqRrVI2pbgNqLDr0B2cxjqCk71lD1/fT // SIG // LLBjmztMSR3dA8oD/yQFafm0RiamtruwKi5tvrQE9usi // SIG // Ob3nHA5jWbIN7w4dR3KQiWvUKUVvzA92vgBdsoFdbEZx // SIG // lVFRt1Tg19wYRk/EzFPxolMt4ewpVzsZmfsHgriDswpE // SIG // w2yuTpQEvKkj6YKflxpTixD2jN9TIH+mE6aQGnj9KKYc // SIG // d+OaIbcCAwEAAaOCAUkwggFFMB0GA1UdDgQWBBTQgrKn // SIG // V0cvzJo2RlUwL8e2BVqbJTAfBgNVHSMEGDAWgBSfpxVd // SIG // AF5iXYP05dJlpxtTNRnpcjBfBgNVHR8EWDBWMFSgUqBQ // SIG // hk5odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3Bz // SIG // L2NybC9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENB // SIG // JTIwMjAxMCgxKS5jcmwwbAYIKwYBBQUHAQEEYDBeMFwG // SIG // CCsGAQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29mdC5j // SIG // b20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUyMFRpbWUt // SIG // U3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNydDAMBgNVHRMB // SIG // Af8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMA4G // SIG // A1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQsFAAOCAgEA // SIG // OCL0m56+IxD4KOVuMY9//tHQhsLM/Ot0BmdpgfWeEsLR // SIG // hizL8H7EVLNzSJTw/y7FgMXVWB5JQM8C08EOTPj0Xcvd // SIG // gxn7chDhjB37rexqvC90VHL6769AC/zTbXxKWwBJAhc7 // SIG // HnsbWObN4c49619sL6AWZtsrYcHC3mZjIB0Apo2af9tH // SIG // x1iYK4z2I7HukQybVE5b1LI6/vO/P7fr60BCKpZnmwnh // SIG // IvlUFcXO8BdC7jE8P4AlnXKh6Ki+diaLcSs2PI2UkO3H // SIG // DR4QuHhxhUaWinokkNBl7ZWxiGz+JFtTtyc5So38Butj // SIG // Qkr35jNYjw/dF2IhZu//JoEdegJqnnw7H4wQlsH96oXi // SIG // DH4Gc1qnhM/JWhZjPA3ZF47jgkBP9i9E8Ya41LXTJAE3 // SIG // 13WY2EZbHAQ8q/MxjJaaxQuy3Magl5YcYbXdgjPpqXE3 // SIG // PEQdg9xKK9FHaD9+kPa+F1glVf9ip9AF7b1sbyH8jhZu // SIG // Wi5dHhM5IX7/15lJQXjghJUu43XXVqQZUIybhx1B4zlR // SIG // l5ayU+1+IYnBdaNt8eVPo+6jygXHq8j9v9aMX5h3OrgV // SIG // 5VwSrpFf0AmQVQIgTGCYZ5LWpFh6aPbiHkp2E+kMe8H9 // SIG // kWrmByBEfEi0Zm5TMzzrPiR0M674e8Urcd9dCzqftA2j // SIG // l7PMY2b4aZ/lrmYo+UZmYowwggdxMIIFWaADAgECAhMz // SIG // AAAAFcXna54Cm0mZAAAAAAAVMA0GCSqGSIb3DQEBCwUA // SIG // MIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGlu // SIG // Z3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMV // SIG // TWljcm9zb2Z0IENvcnBvcmF0aW9uMTIwMAYDVQQDEylN // SIG // aWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0ZSBBdXRob3Jp // SIG // dHkgMjAxMDAeFw0yMTA5MzAxODIyMjVaFw0zMDA5MzAx // SIG // ODMyMjVaMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpX // SIG // YXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYD // SIG // VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNV // SIG // BAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEw // SIG // MIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA // SIG // 5OGmTOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51yMo1 // SIG // V/YBf2xK4OK9uT4XYDP/XE/HZveVU3Fa4n5KWv64NmeF // SIG // RiMMtY0Tz3cywBAY6GB9alKDRLemjkZrBxTzxXb1hlDc // SIG // wUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9cmmvHaus // SIG // 9ja+NSZk2pg7uhp7M62AW36MEBydUv626GIl3GoPz130 // SIG // /o5Tz9bshVZN7928jaTjkY+yOSxRnOlwaQ3KNi1wjjHI // SIG // NSi947SHJMPgyY9+tVSP3PoFVZhtaDuaRr3tpK56KTes // SIG // y+uDRedGbsoy1cCGMFxPLOJiss254o2I5JasAUq7vnGp // SIG // F1tnYN74kpEeHT39IM9zfUGaRnXNxF803RKJ1v2lIH1+ // SIG // /NmeRd+2ci/bfV+AutuqfjbsNkz2K26oElHovwUDo9Fz // SIG // pk03dJQcNIIP8BDyt0cY7afomXw/TNuvXsLz1dhzPUNO // SIG // wTM5TI4CvEJoLhDqhFFG4tG9ahhaYQFzymeiXtcodgLi // SIG // Mxhy16cg8ML6EgrXY28MyTZki1ugpoMhXV8wdJGUlNi5 // SIG // UPkLiWHzNgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9Q // SIG // BXpsxREdcu+N+VLEhReTwDwV2xo3xwgVGD94q0W29R6H // SIG // XtqPnhZyacaue7e3PmriLq0CAwEAAaOCAd0wggHZMBIG // SIG // CSsGAQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUCBBYE // SIG // FCqnUv5kxJq+gpE8RjUpzxD/LwTuMB0GA1UdDgQWBBSf // SIG // pxVdAF5iXYP05dJlpxtTNRnpcjBcBgNVHSAEVTBTMFEG // SIG // DCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIBFjNodHRw // SIG // Oi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL0RvY3Mv // SIG // UmVwb3NpdG9yeS5odG0wEwYDVR0lBAwwCgYIKwYBBQUH // SIG // AwgwGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBDAEEwCwYD // SIG // VR0PBAQDAgGGMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0j // SIG // BBgwFoAU1fZWy4/oolxiaNE9lJBb186aGMQwVgYDVR0f // SIG // BE8wTTBLoEmgR4ZFaHR0cDovL2NybC5taWNyb3NvZnQu // SIG // Y29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0 // SIG // XzIwMTAtMDYtMjMuY3JsMFoGCCsGAQUFBwEBBE4wTDBK // SIG // BggrBgEFBQcwAoY+aHR0cDovL3d3dy5taWNyb3NvZnQu // SIG // Y29tL3BraS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0w // SIG // Ni0yMy5jcnQwDQYJKoZIhvcNAQELBQADggIBAJ1Vffwq // SIG // reEsH2cBMSRb4Z5yS/ypb+pcFLY+TkdkeLEGk5c9MTO1 // SIG // OdfCcTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulmZzpT // SIG // Td2YurYeeNg2LpypglYAA7AFvonoaeC6Ce5732pvvinL // SIG // btg/SHUB2RjebYIM9W0jVOR4U3UkV7ndn/OOPcbzaN9l // SIG // 9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECWOKz3+SmJ // SIG // w7wXsFSFQrP8DJ6LGYnn8AtqgcKBGUIZUnWKNsIdw2Fz // SIG // Lixre24/LAl4FOmRsqlb30mjdAy87JGA0j3mSj5mO0+7 // SIG // hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3UwxTSwethQ/gpY // SIG // 3UA8x1RtnWN0SCyxTkctwRQEcb9k+SS+c23Kjgm9swFX // SIG // SVRk2XPXfx5bRAGOWhmRaw2fpCjcZxkoJLo4S5pu+yFU // SIG // a2pFEUep8beuyOiJXk+d0tBMdrVXVAmxaQFEfnyhYWxz // SIG // /gq77EFmPWn9y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/ // SIG // AsGConsXHRWJjXD+57XQKBqJC4822rpM+Zv/Cuk0+CQ1 // SIG // ZyvgDbjmjJnW4SLq8CdCPSWU5nR0W2rRnj7tfqAxM328 // SIG // y+l7vzhwRNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEG // SIG // ahC0HVUzWLOhcGbyoYIDTTCCAjUCAQEwgfmhgdGkgc4w // SIG // gcsxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n // SIG // dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN // SIG // aWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNVBAsTHE1p // SIG // Y3Jvc29mdCBBbWVyaWNhIE9wZXJhdGlvbnMxJzAlBgNV // SIG // BAsTHm5TaGllbGQgVFNTIEVTTjo3RjAwLTA1RTAtRDk0 // SIG // NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAg // SIG // U2VydmljZaIjCgEBMAcGBSsOAwIaAxUAwigGSir+ZbHQ // SIG // tsqDBimIkMu/af+ggYMwgYCkfjB8MQswCQYDVQQGEwJV // SIG // UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMH // SIG // UmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv // SIG // cmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1T // SIG // dGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsFAAIFAOth // SIG // w94wIhgPMjAyNTAyMjAxNTA4MTRaGA8yMDI1MDIyMTE1 // SIG // MDgxNFowdDA6BgorBgEEAYRZCgQBMSwwKjAKAgUA62HD // SIG // 3gIBADAHAgEAAgIX3zAHAgEAAgISPjAKAgUA62MVXgIB // SIG // ADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMC // SIG // oAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3 // SIG // DQEBCwUAA4IBAQBTCwRZN+RQnPzfRg+SwFhXnr5AWd1c // SIG // VHxADo35cG3C1mb197EFGcSfOb1+Nny36CQTIVq4R0YW // SIG // md4BkwMvlAvgW5iHWo09TiVM65vL1PQsp0ik6DssIb5q // SIG // np7urfjkYs4G9qYK5Q+5xbetXsdGNdERGXo3FSRO8cy0 // SIG // TjRanuVlVpxD35g7Was96kwEQche0jFhUKwYaGL+1a6m // SIG // g/vjg5BrGRQC0dS1PdfXjYWn+jhCWJdjLikdGFZxrVdc // SIG // uX1ZLKwpUzxG+iRa/7ZnV/XDd1BxUhLzPNJyfwmNwNRA // SIG // fKGWaPJHHj8cjvy7ThUBeEidXvKfm+hG93fQ9bYu1lv4 // SIG // ECNiMYIEDTCCBAkCAQEwgZMwfDELMAkGA1UEBhMCVVMx // SIG // EzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl // SIG // ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3Jh // SIG // dGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3Rh // SIG // bXAgUENBIDIwMTACEzMAAAHwKnwdWTvmH60AAQAAAfAw // SIG // DQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzEN // SIG // BgsqhkiG9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQguXkh // SIG // pk6Tw+vFOMo5MCu2oSZDFdAtsZnA/FZcDWve9YMwgfoG // SIG // CyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCBcAZqjpXL9 // SIG // 5FO5gAebFke+EBQIkupm/gjOmrJxQGVLizCBmDCBgKR+ // SIG // MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n // SIG // dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN // SIG // aWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1p // SIG // Y3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMzAAAB // SIG // 8Cp8HVk75h+tAAEAAAHwMCIEICfu7D/aEYVtoD42f8pa // SIG // W2rfdnAyYN5qaKGMxqC6AkyoMA0GCSqGSIb3DQEBCwUA // SIG // BIICAGAi0q9GUJHMZzVg0Jzhl0UANpyNN4vGirMxcKw0 // SIG // uHBXek198KbJPIGC1I9NHC6C19A+u/nNzg7IXwsKD6eF // SIG // On4yGybZTspgh8CAWRENbyQg+c/FW/I3XNQxn4wHcwtE // SIG // 7JrUPf4THn7VCO1ZphwOAlc6AskbjoDCR5oEfuheV3N9 // SIG // EOuzQWg4MPbF2sHObYg3c3ZglFqZp2ouu4nVAhaPoc+K // SIG // STlT26JD/rFu1i1/imBB03S7asiRVx0ro5u3UTaV6AAd // SIG // W+RqK3TwKBhGQaORAeN7Oe3gpEiJDRKeUtXV12Lt1MKb // SIG // jrp6SILSz16mkgRNUYyTVxX7u8jlrRo4O2FUqeOvm+nd // SIG // dR/eRTSPptgF5XCSlgFOwBpLic+RxRScH+Q95XtJoR9l // SIG // Cz5bArXLpsZvjTy5Z5BpRbE2wpDDOpxf93bhq14GHunZ // SIG // HghBljf7w4tRjwZNn/JGzCKFKrduW3Yt2FAK+Yq4tvx/ // SIG // CZtjnMUHYprBP3wjwI1751qDf6J0PMuEMY0LoYgSk16p // SIG // DGDAENczuyF+GFfcUejAVPHMSeDaLugP49oTKbdmMi2Y // SIG // 9VhrBn7pbLofuIM1LABGaMUG+0/KjYHE2t4F4ndgmZjB // SIG // BaGBQg+uN/TX0tPY8rClIw7QDx0/iIpIz/y3TmZhDBeN // SIG // UhqWXxBxuJ6sef9QXP+aBw/8pFb8 // SIG // End signature block