UNPKG

@microsoft/windows-admin-center-sdk

Version:

Microsoft - Windows Admin Center Shell

934 lines (931 loc) 44.6 kB
import { EMPTY, of } from 'rxjs'; import { catchError, expand, filter, map, take } from 'rxjs/operators'; import { Net } from '../data/net'; import { PowerShell } from '../data/powershell'; import { LogLevel } from '../diagnostics/log-level'; import { Logging } from '../diagnostics/logging'; import { EnvironmentModule, EnvironmentModuleToolState } from '../manifest/environment-modules'; import { GatewayMode } from '../shared/gateway-inventory/gateway-inventory'; import { ToolInventoryCache } from '../shared/tool-inventory/tool-inventory-cache'; /** * The class handles conditions of tools to be presented on tools' menu. * @dynamic */ export class ToolConditionValidator { appContext; /** * Support the following condition name. * It can be a string, number, boolean and version string. */ static serverInventoryProperties = [ // string { conditionName: 'computerManufacturer', dataName: 'computerManufacturer' }, // string { conditionName: 'computerModel', dataName: 'computerModel' }, // number { conditionName: 'operatingSystemSKU', dataName: 'operatingSystemSKU' }, // version string { conditionName: 'operatingSystemVersion', dataName: 'operatingSystemVersion' }, // number { conditionName: 'windowsProductType', dataName: 'productType' }, // string { conditionName: 'clusterFqdn', dataName: 'clusterFqdn' }, // string { conditionName: 'systemLockdownPolicy', dataName: 'systemLockdownPolicy' }, // boolean { conditionName: 'isHyperVRoleInstalled', dataName: 'isHyperVRoleInstalled' }, // boolean { conditionName: 'isHyperVPowershellInstalled', dataName: 'isHyperVPowershellInstalled' }, // boolean { conditionName: 'isManagementToolsAvailable', dataName: 'isManagementToolsAvailable' }, // boolean { conditionName: 'isWmfInstalled', dataName: 'isWmfInstalled' }, // boolean { conditionName: 'isRemoteAppEnabled', dataName: 'isRemoteAppEnabled' }, // boolean { conditionName: 'isDomainController', dataName: 'isDomainController' }, // boolean { conditionName: 'isS2dEnabled', dataName: 'isS2dEnabled' }, // boolean { conditionName: 'isBritannicaEnabled', dataName: 'isBritannicaEnabled' }, // boolean { conditionName: 'isHciServer', dataName: 'isHciServer' } ]; /** * The following operators are supported. * String comparison uses caseinsesitive pattern. */ static operators = { /** * Greater than: number, string (caseinsesitive), version */ gt: 'gt', /** * Greater or equal: number, string (caseinsesitive), version */ ge: 'ge', /** * Less than: number, string (caseinsesitive), version */ lt: 'lt', /** * Less or equal: number, string (caseinsesitive), version */ le: 'le', /** * Equal: number, string (caseinsesitive), version (Accept '*' like "1.2.*") */ eq: 'eq', /** * Not equal: number, string (caseinsesitive), version (Accept '*' like "1.2.*") */ ne: 'ne', /** * Test true: number, string, boolean */ is: 'is', /** * Test false: number, string, boolean */ not: 'not', /** * Contains a string: string (caseinsesitive) */ contains: 'contains', /** * Not contains a string: string (caseinsesitive) */ notContains: 'notContains', /** * Any one of value (number or string) equal: numberArray, stringArray */ anyEq: 'anyEq', /** * Any one of value (number or string) not equal: numberArray, stringArray */ anyNe: 'anyNe', /** * Any one of string contains: stringArray */ anyContains: 'anyContains', /** * Any one of string not contains: stringArray */ anyNotContains: 'anyNotContains' }; static internalCurrent; caches; toolInventoryCache; errorStrings = MsftSme.getStrings().MsftSmeShell.Core.Error; /** * Gets the current object of the ToolConditionValidator class, and maintains as singleton. * * @param appContext the application context. * @param caches the instance of the inventory query caches to share the resource. */ static current(appContext, caches) { if (!ToolConditionValidator.internalCurrent) { ToolConditionValidator.internalCurrent = new ToolConditionValidator(appContext, caches); } return ToolConditionValidator.internalCurrent; } /** * Initializes a new instance of the ToolConditionValidator class. * * @param appContext the application context. * @param caches the instance of the inventory query caches to share the resource. */ constructor(appContext, caches) { this.appContext = appContext; const statusExpiration = 4 * 60 * 1000; // 4 min this.caches = caches; this.toolInventoryCache = new ToolInventoryCache(appContext, { expiration: statusExpiration }); } /** * Scan the tool condition to be present or not. * * @param connection the connection object. * @param solution The entry point object of solution. * @param tool The entry point object of tool. * @param scanMode The mode of scanning. * @return the result observable. */ scanToolCondition(connection, solution, tool) { if (!tool.requirements) { // tool is missing requirements, never show. return { weight: 0 /* ToolConditionWeight.NonObservable */, result: { ...tool, ...{ show: false, detail: EnvironmentModuleToolState.NotSupported } } }; } const solutionId = EnvironmentModule.createFormattedEntrypoint(solution); const toolId = EnvironmentModule.createFormattedEntrypoint(tool); const checkersCollection = []; let weight = 0 /* ToolConditionWeight.NonObservable */; for (const requirement of tool.requirements) { const checkers = []; // tool must specify the solutions it can show in if (!requirement.solutionIds || requirement.solutionIds.every(id => id !== solutionId)) { continue; } // if ths solution is a connections type, then the tool must specify the connection type if (solution.rootNavigationBehavior === 'connections' && (!requirement.connectionTypes || requirement.connectionTypes.every(type => type !== connection.type))) { continue; } // if there are no conditions to check, then this tool has been satisfied. if (!requirement.conditions || requirement.conditions.length === 0) { checkers.push(this.installationTypeValidate(connection)); weight = Math.max(weight, 2 /* ToolConditionWeight.Gateway */); } else { for (const condition of requirement.conditions) { checkers.push(this.installationTypeValidate(connection, condition.installationTypes)); weight = Math.max(weight, 2 /* ToolConditionWeight.Gateway */); if (condition.localhost !== undefined && !condition.localhost) { // if connection is localhost and not supported. checkers.push(this.localhostValidate(connection)); weight = Math.max(weight, 2 /* ToolConditionWeight.Gateway */); } if (condition.inventory) { checkers.push(this.inventoryValidate(connection, condition.inventory)); weight = Math.max(weight, 3 /* ToolConditionWeight.Server */); } if (condition.properties) { checkers.push(this.propertyValidate(connection, condition.properties)); weight = Math.max(weight, 1 /* ToolConditionWeight.Property */); } if (condition.powerShell) { // powerShell { command, script } checkers.push(this.toolInventoryValidate(connection, toolId, condition.powerShell)); weight = Math.max(weight, 4 /* ToolConditionWeight.Script */); } else if (condition.script) { // deprecated checkers.push(this.toolInventoryValidate(connection, toolId, condition.script)); weight = Math.max(weight, 4 /* ToolConditionWeight.Script */); } } } checkersCollection.push(checkers); } if (checkersCollection.length === 0) { return { weight: 0 /* ToolConditionWeight.NonObservable */, result: { ...tool, ...{ show: false, detail: EnvironmentModuleToolState.NotSupported, connectionId: connection ? connection.id : null } } }; } let collectionIndex = 0; let checkerIndex = 0; let lastDetail = null; let lastMessage = null; return { weight: weight, result: this.runChecker(checkersCollection[collectionIndex][checkerIndex]) .pipe(catchError(() => { Logging.log({ level: LogLevel.Error, message: this.errorStrings.ToolValidationResult.message.format(tool.parentModule.name, tool.displayName), source: 'ToolConditionValidator' }); return of({ show: false, ends: true }); }), expand((value) => { checkerIndex++; if (value.ends) { return EMPTY; } if (value.detail !== undefined && lastDetail == null) { lastDetail = value.detail; } if (value.message !== undefined && lastMessage == null) { lastMessage = value.message; } if (!value.show) { // failed result. increment collection index and reset checkerIndex. if (checkersCollection.length > ++collectionIndex) { // still has another condition, try next checker set. return this.runChecker(checkersCollection[collectionIndex][checkerIndex = 0]); } else { // no more condition, end to return 'false'. return of({ show: false, detail: lastDetail, message: lastMessage, ends: true }); } } if (checkersCollection[collectionIndex].length > checkerIndex) { // check next checker. return this.runChecker(checkersCollection[collectionIndex][checkerIndex]); } else { // all state/succeeded within the checker set. return of({ show: true, detail: lastDetail, message: lastMessage, ends: true }); } }), filter(combined => combined.ends), map(combined => { return { ...tool, ...{ show: combined.show, detail: combined.detail, message: combined.message, connectionId: connection ? connection.id : null } }; })) }; } runChecker(checker) { return checker.pipe(take(1), map(result => ({ ...result, ends: false }))); } localhostValidate(connection) { return this.caches.gatewayCache.createObservable({}) .pipe(map(instance => ({ show: !(instance.mode !== GatewayMode.Service && connection.properties && connection.properties.displayName === 'localhost') }))); } installationTypeValidate(connection, installationType) { if (MsftSme.getValue(MsftSme.self().Environment.configuration, 'gateway.disabled')) { return of({ show: true }); } if (!installationType || !Array.isArray(installationType) || installationType.length === 0) { // Default to standard if property doesnt exist in the manifest. installationType = ['Standard']; } return this.caches.gatewayCache.createObservable({ params: {} }) .pipe(catchError((error) => { const errorStrings = MsftSme.getStrings().MsftSmeShell.Core.Error; const notification = this.appContext.notification.create(connection.name); notification.showError(errorStrings.ToolValidationGatewayInventoryError.title, errorStrings.ToolValidationGatewayInventoryError.message.format(Net.getErrorMessage(error))); return of(null); }), map(instance => ({ show: installationType.some(t => instance.installationType === t) }))); } inventoryValidate(connection, condition) { const nodeName = connection.activeAlias ? connection.activeAlias : connection.name; return this.caches.serverCache.createObservable({ params: { name: nodeName } }) .pipe(catchError((error) => { const errorStrings = MsftSme.getStrings().MsftSmeShell.Core.Error; const notification = this.appContext.notification.create(connection.name); notification.showError(errorStrings.ToolValidationInventoryError.title, errorStrings.ToolValidationInventoryError.message.format(Net.getErrorMessage(error))); return of(null); }), filter((instance) => instance.serverName === nodeName), map(instance => ({ show: this.checkServerInventoryCondition(condition, instance) }))); } propertyValidate(connection, condition) { const propertyNames = Object.keys(condition); let result = true; for (const propertyName of propertyNames) { const propertyValue = connection.properties && connection.properties[propertyName]; const conditionItem = condition[propertyName]; if (conditionItem && !this.checkCondition(propertyValue, conditionItem)) { result = false; break; } } return of({ show: result }); } toolInventoryValidate(connection, id, scriptOrCommand) { const command = PowerShell.getPowerShellCommand(scriptOrCommand); return this.toolInventoryCache.query({ ...{ name: connection.activeAlias ? connection.activeAlias : connection.name, id: id }, ...command }) .pipe(map(inventory => ({ show: inventory.instance.state === EnvironmentModuleToolState.Available || inventory.instance.state === EnvironmentModuleToolState.NotConfigured, detail: inventory.instance.state, message: inventory.instance.message })), catchError((error) => { const message = Net.getErrorMessage(error); Logging.logError('ToolConditionValidator', `${id}: ${message}`); return of({ show: false, detail: EnvironmentModuleToolState.NotSupported, message }); })); } checkServerInventoryCondition(condition, instance) { for (const property of ToolConditionValidator.serverInventoryProperties) { const conditionItem = condition[property.conditionName]; if (conditionItem && !this.checkCondition(instance[property.dataName], conditionItem)) { return false; } } return true; } checkCondition(data, condition) { switch (condition.type) { case 'number': const numberValue = this.getNumberOrZero(data); const numberTest = this.getNumberOrZero(condition.value); switch (condition.operator) { case ToolConditionValidator.operators.gt: return numberValue > numberTest; case ToolConditionValidator.operators.ge: return numberValue >= numberTest; case ToolConditionValidator.operators.lt: return numberValue < numberTest; case ToolConditionValidator.operators.le: return numberValue <= numberTest; case ToolConditionValidator.operators.eq: return numberValue === numberTest; case ToolConditionValidator.operators.ne: return numberValue !== numberTest; case ToolConditionValidator.operators.is: return !!numberValue; case ToolConditionValidator.operators.not: return !numberValue; default: throw new Error(this.errorStrings.ToolValidationUnsupportedOperator.message.format(JSON.stringify(condition), JSON.stringify(data))); } case 'string': const stringValue = '' + data; const stringTest = '' + condition.value; switch (condition.operator) { case ToolConditionValidator.operators.gt: return stringValue.toLowerCase() > stringTest.toLowerCase(); case ToolConditionValidator.operators.ge: return stringValue.toLowerCase() >= stringTest.toLowerCase(); case ToolConditionValidator.operators.lt: return stringValue.toLowerCase() < stringTest.toLowerCase(); case ToolConditionValidator.operators.le: return stringValue.toLowerCase() <= stringTest.toLowerCase(); case ToolConditionValidator.operators.eq: return stringValue.toLowerCase() === stringTest.toLowerCase(); case ToolConditionValidator.operators.ne: return stringValue.toLowerCase() !== stringTest.toLowerCase(); case ToolConditionValidator.operators.is: return !!data; case ToolConditionValidator.operators.not: return !data; case ToolConditionValidator.operators.contains: return stringValue.toLowerCase().indexOf(stringTest.toLowerCase()) >= 0; case ToolConditionValidator.operators.notContains: return stringValue.toLowerCase().indexOf(stringTest.toLowerCase()) < 0; default: throw new Error(this.errorStrings.ToolValidationUnsupportedOperator.message.format(JSON.stringify(condition), JSON.stringify(data))); } case 'boolean': switch (condition.operator) { case ToolConditionValidator.operators.is: return !!data; case ToolConditionValidator.operators.not: return !data; default: throw new Error(this.errorStrings.ToolValidationUnsupportedOperator.message.format(JSON.stringify(condition), JSON.stringify(data))); } case 'version': const versionValue = data; const versionTest = condition.value; return this.compareVersion(versionValue, versionTest, condition.operator); case 'numberArray': const checkNumber = condition.value; if (!checkNumber || typeof checkNumber === 'string' || !checkNumber.length || typeof checkNumber[0] !== 'number') { throw new Error(this.errorStrings.ToolValidationUnsupportedDataType.message.format(JSON.stringify(condition), JSON.stringify(data))); } const numberArrayValue = this.getNumberOrZero(data); const numberArray = condition.value; for (const numberItem of numberArray) { const numberArrayTest = this.getNumberOrZero(numberItem); switch (condition.operator) { case ToolConditionValidator.operators.anyEq: if (numberArrayValue === numberArrayTest) { return true; } break; case ToolConditionValidator.operators.anyNe: if (numberArrayValue !== numberArrayTest) { return true; } break; default: throw new Error(this.errorStrings.ToolValidationUnsupportedOperator.message.format(JSON.stringify(condition), JSON.stringify(data))); } } return false; case 'stringArray': const checkString = condition.value; if (!checkString || typeof checkString === 'string' || !checkString.length || typeof checkString[0] !== 'string') { throw new Error(this.errorStrings.ToolValidationUnsupportedDataType.message.format(JSON.stringify(condition), JSON.stringify(data))); } const stringArrayValue = '' + data; const stringArray = condition.value; for (const stringArrayTest of stringArray) { switch (condition.operator) { case ToolConditionValidator.operators.anyEq: if (stringArrayValue.toLowerCase() === stringArrayTest.toLowerCase()) { return true; } break; case ToolConditionValidator.operators.anyNe: if (stringArrayValue.toLowerCase() !== stringArrayTest.toLowerCase()) { return true; } break; case ToolConditionValidator.operators.anyContains: if (stringArrayValue.toLowerCase().indexOf(stringArrayTest.toLowerCase()) >= 0) { return true; } break; case ToolConditionValidator.operators.anyNotContains: if (stringArrayValue.toLowerCase().indexOf(stringArrayTest.toLowerCase()) < 0) { return true; } break; default: throw new Error(this.errorStrings.ToolValidationUnsupportedOperator.message.format(JSON.stringify(condition), JSON.stringify(data))); } } return false; default: throw new Error(this.errorStrings.ToolValidationUnsupportedDataType.message.format(JSON.stringify(condition), JSON.stringify(data))); } } compareVersion(left, right, operator) { const leftSegments = left.split('.'); const rightSegments = right.split('.'); if (!leftSegments || leftSegments.length <= 0 || !rightSegments || rightSegments.length <= 0) { throw new Error(this.errorStrings.ToolValidationVersionFormat.message); } const count = Math.max(leftSegments.length, rightSegments.length); let status; for (let index = 0; index < count; index++) { if (rightSegments[index] === '*') { // quit comparison with wildcard. status = 0; break; } const leftSegment = this.getNumberOrZero(leftSegments[index]); const rightSegment = this.getNumberOrZero(rightSegments[index]); if (leftSegment === rightSegment) { // equal. status = 0; } else if (leftSegment > rightSegment) { // greater. status = 1; break; } else { // lesser. status = -1; break; } } switch (operator) { case ToolConditionValidator.operators.gt: return status > 0; case ToolConditionValidator.operators.ge: return status >= 0; case ToolConditionValidator.operators.lt: return status < 0; case ToolConditionValidator.operators.le: return status <= 0; case ToolConditionValidator.operators.eq: return status === 0; case ToolConditionValidator.operators.ne: return status !== 0; default: throw new Error(this.errorStrings.ToolValidationUnsupportedOperator.message.format(operator, left)); } } getNumberOrZero(data) { const result = Number(data); return isNaN(result) ? 0 : result; } } //# sourceMappingURL=tool-condition-validator.js.map // SIG // Begin signature block // SIG // MIIoKAYJKoZIhvcNAQcCoIIoGTCCKBUCAQExDzANBglg // SIG // hkgBZQMEAgEFADB3BgorBgEEAYI3AgEEoGkwZzAyBgor // SIG // BgEEAYI3AgEeMCQCAQEEEBDgyQbOONQRoqMAEEvTUJAC // SIG // AQACAQACAQACAQACAQAwMTANBglghkgBZQMEAgEFAAQg // SIG // 2l6FKyUHsUNjfSE5c+n8tvQ3HYhHPwJ24zAsK0vOXF+g // 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 // DQEJBDEiBCBZ6Ljmk6Nsa9JmWuisceQR2IruqqdXAOWa // SIG // 6E1ox1hJ+TBCBgorBgEEAYI3AgEMMTQwMqAUgBIATQBp // SIG // AGMAcgBvAHMAbwBmAHShGoAYaHR0cDovL3d3dy5taWNy // SIG // b3NvZnQuY29tMA0GCSqGSIb3DQEBAQUABIIBAF0/xlG/ // SIG // 50j1jBYD+2u9/Y2AC5HHTPpS+cqfed1sCMcWkGt/SczH // SIG // UHXjNWMYlKN9c23pQClDaFtGFFq9oo2Ii1buweJ2TWRo // SIG // 0wqE5KeX0QvAWbckvxFts4/K+SaQLAVZYJguhOhNFeat // SIG // d9hD1e03fTqYU+G5lOxMw82DOvf7va37q9po8qFhnjNj // SIG // j1SmCSd1JnSdw3o+ugxYdjR26iOab7aJbpUTptmgg+E4 // SIG // cG7Nm5miuSGUbVegeIRHjhBP1zzMM/G2Q1cr1AD70Pu4 // SIG // uHthJKnqM24BoW6tgjE83Y5nap45Gjq87nLrURT3RsA/ // SIG // bdTTs1l2QdrS6DyGD6J3P7YSY8ShgheUMIIXkAYKKwYB // SIG // BAGCNwMDATGCF4Awghd8BgkqhkiG9w0BBwKgghdtMIIX // SIG // aQIBAzEPMA0GCWCGSAFlAwQCAQUAMIIBUgYLKoZIhvcN // SIG // AQkQAQSgggFBBIIBPTCCATkCAQEGCisGAQQBhFkKAwEw // SIG // MTANBglghkgBZQMEAgEFAAQgK77awkSjCPuIQIMtZseU // SIG // waQDBq9trOmrJPvPJ5NgEmUCBmeuHCKiuRgTMjAyNTAy // SIG // MjAxNTI4MzkuMDUyWjAEgAIB9KCB0aSBzjCByzELMAkG // SIG // A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAO // SIG // BgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m // SIG // dCBDb3Jwb3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0 // SIG // IEFtZXJpY2EgT3BlcmF0aW9uczEnMCUGA1UECxMeblNo // SIG // aWVsZCBUU1MgRVNOOjM3MDMtMDVFMC1EOTQ3MSUwIwYD // SIG // VQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNl // SIG // oIIR6jCCByAwggUIoAMCAQICEzMAAAHqmiRy1Vk/YWMA // SIG // AQAAAeowDQYJKoZIhvcNAQELBQAwfDELMAkGA1UEBhMC // SIG // VVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcT // SIG // B1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jw // SIG // b3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUt // SIG // U3RhbXAgUENBIDIwMTAwHhcNMjMxMjA2MTg0NTMwWhcN // SIG // MjUwMzA1MTg0NTMwWjCByzELMAkGA1UEBhMCVVMxEzAR // SIG // BgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v // SIG // bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv // SIG // bjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2EgT3Bl // SIG // cmF0aW9uczEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNO // SIG // OjM3MDMtMDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3Nv // SIG // ZnQgVGltZS1TdGFtcCBTZXJ2aWNlMIICIjANBgkqhkiG // SIG // 9w0BAQEFAAOCAg8AMIICCgKCAgEAtQtf8Ug/IAfV+y7n // SIG // aKNq1m9pLKmheuULSZG0KZrHOhuG4OTDr+lj/7ieFzib // SIG // yl/3NbdHo+KFganRg+lW411+E9Cn8pU7pa8yrYMZ8WYe // SIG // 6tbg9A8v8ORtAyQz2+qMUK8+rzFdmd8vWcY32agZw36h // SIG // qJ/+FQx52YXWrNtrL0guRh8sLENifdDDOy+HnGPE5yyP // SIG // OZF101REm9PbcS9rRzGKwfihwstPHbN+mp+yHDhn0ZoR // SIG // 2xaD2uaJvWBqVSkvMXk+xAMFu1m1y/5aOafSkUSIwJbA // SIG // QRw9U3RgbnKxgt00F0k6fbOw45L7zRblGtASrM+lIgi8 // SIG // SRkEmYXdojiUxHydX8WJNp2OkgirFflZrVeWoj82P7Fq // SIG // BWOeNvs86wD6+Hpa76/bgenIvynIv/xDhEWRFEwT1zBP // SIG // 4mvrfI609st7oNeTEglboTrDa5rmRcGkQq0RA9Ms+Ffc // SIG // JTExhyCVueYjTNxz1SSdfbzkr6wj/ZbBHBMFmSENRQsj // SIG // zp5DNX7O/PNHWoQGuVJj6jJOVhCscwz1adPNV+UUOhxl // SIG // VM+mXYENI3E+fRBvgigz0Q+psfKL8yKUv6/8BBzyreZD // SIG // oWK48kB13PShyk1n16QFY9UsqreV+J6/jKXrm7/jfz40 // SIG // BD69ImCQ40sya6iC4QbOacrW+r8kfB1FTKfpgAOK14zs // SIG // ONr5B30CAwEAAaOCAUkwggFFMB0GA1UdDgQWBBQrgUUl // SIG // olHm6RdAVNTEyHKLBW5ZXjAfBgNVHSMEGDAWgBSfpxVd // SIG // AF5iXYP05dJlpxtTNRnpcjBfBgNVHR8EWDBWMFSgUqBQ // SIG // hk5odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3Bz // SIG // L2NybC9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENB // SIG // JTIwMjAxMCgxKS5jcmwwbAYIKwYBBQUHAQEEYDBeMFwG // SIG // CCsGAQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29mdC5j // SIG // b20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUyMFRpbWUt // SIG // U3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNydDAMBgNVHRMB // SIG // Af8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMA4G // SIG // A1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQsFAAOCAgEA // SIG // KIOtVl4/fv58VW19xt+yoL8qDQJ7rtsNx6FmY9x9GAnk // SIG // N2/SkmU4VU4VuIhXB6yp4RTAW1yV+LkCOd5Dlkmlgmld // SIG // 8Qs56Ubd3OP4Ep93bzv9Rj9zCZKSX4KOegoEvcyzoj99 // SIG // ZH5qVHT6npGW+IrzEei6D2+RzZatFmwacxW7bE4za08n // SIG // 6qnKgMHOq/fQ39lEE6g2tL88KQPAsYgINipWz8jMATj3 // SIG // K/YSU/LBqV/2YSw4ddXWXG1AM1x6NUSaK0kn7VWvYS1p // SIG // 88RsxBmnz1MC5qBE4oThi6iEJQqb6/eB4mpNBqtMGOpX // SIG // blEI5P5cWeBMwMP3BjHpPCd0HYjUvLvbo2IdQezS6+rd // SIG // yIJX0nA1d23VVnrdYrU1KClUSyIr0Q8AE+3UR9dwqt9o // SIG // 9iRuQWLv14rURPHHc2iZg1Qc2IZT5fUF7wvuqkfCOjSD // SIG // f/fdeG06v0uIOhReH9XYsVMROKpX1DzIsRq9BbeP0tD+ // SIG // H8JobPlh0Z+tjweI98wh4sSiQrEZ/SEdxMQUCkHTIuWr // SIG // oqgesUAQA1H/he4UimX2wPLBUha3i0qob4/qlEBfODXM // SIG // bmsaWyVlabDtfCC+EG7eOQs/0DGuxJjBjZ+2vDDN7k0D // SIG // pUMtLunP46tddYtSajI2sk3HkGTTATDORDHOQ6+Zt0+G // SIG // w4/VkzS4D/EhXtxKk2llTDkwggdxMIIFWaADAgECAhMz // 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 // BAsTHm5TaGllbGQgVFNTIEVTTjozNzAzLTA1RTAtRDk0 // SIG // NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAg // SIG // U2VydmljZaIjCgEBMAcGBSsOAwIaAxUAidse3EH46UbJ // SIG // CfFBiHLTgpJhJI+ggYMwgYCkfjB8MQswCQYDVQQGEwJV // SIG // UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMH // SIG // UmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv // SIG // cmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1T // SIG // dGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsFAAIFAOth // SIG // K9cwIhgPMjAyNTAyMjAwNDE5MzVaGA8yMDI1MDIyMTA0 // SIG // MTkzNVowdDA6BgorBgEEAYRZCgQBMSwwKjAKAgUA62Er // SIG // 1wIBADAHAgEAAgIe1zAHAgEAAgIUJDAKAgUA62J9VwIB // SIG // ADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMC // SIG // oAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3 // SIG // DQEBCwUAA4IBAQBwHC5Wp7wT3GmvVZTrBLC3U5Zooxnk // SIG // 3zQ5Vz5Q4SQgejguxV9KLVSKlz2YShpbjFbBgOx5Llhe // SIG // AYxp48wV5yJtN38z5nA+VxxmVwWhgUxagAWw+yoaEkeR // SIG // d6OnxSlxYNhYKfbkAifB2LZpXrr5kYByjfBnkHNIv4VB // SIG // JcwIg+oXNMIwi/HX4k0otB39l/ruYv7/4KeatcfgdiNc // SIG // Bi8ycdf8RYgBJ7Keiy85HqBjfVF5QD/w28knib9sr3T0 // SIG // qhZyqVdqamwO4wfO388PBKHuaRSFE5vyoNH59RJHqP5C // SIG // NLk9hcxhsNd+VzL9fVJnMympMqFQOM/xoh/rlrkPhK9o // SIG // UdfCMYIEDTCCBAkCAQEwgZMwfDELMAkGA1UEBhMCVVMx // SIG // EzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl // SIG // ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3Jh // SIG // dGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3Rh // SIG // bXAgUENBIDIwMTACEzMAAAHqmiRy1Vk/YWMAAQAAAeow // SIG // DQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzEN // SIG // BgsqhkiG9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQgPY4W // SIG // oxAP6i44cPS3CX7fdChoo9FwCxL4ruNvdRBkP4cwgfoG // SIG // CyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCApj6HV42Q0 // SIG // eIsINJbSwDVwYeRtbiqiiL6vLIynpLhmeDCBmDCBgKR+ // SIG // MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n // SIG // dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN // SIG // aWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1p // SIG // Y3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMzAAAB // SIG // 6pokctVZP2FjAAEAAAHqMCIEIGyjaIeIa7Ei9ibtJl0b // SIG // vz0NvJpIKDh1OAlHHM2qxjN9MA0GCSqGSIb3DQEBCwUA // SIG // BIICAH+1dTZ6l4Xi3Ej555h8jaxdoS7OQi0sLPGrySk6 // SIG // 4yXZPaje7pgYZmjAy78ud1dshs9pB8MIqIvX0u4tyQ1M // SIG // EiYxZE75naGyvKs/AxN2AXW0bFLMxoLJgcr1a/b0R3bX // SIG // jNkcjJ7A6C30fD4eCdN7p+Hh7ndsMJ/wVCmBB0A5vSU1 // SIG // xByMVGeu+1AY0njklypmsMFIy5LNzZwzt+mOxwZxXPaQ // SIG // d5oxrT/s71AU8EUDqyYTNFt0z6xFx9CXJbaDCxgm6+Jf // SIG // oRE0xZqCXMkPvXp0xqvprzDMFb4GmcpN20aq+rpmceXN // SIG // kaugH0HNdcVSCvgSRUhAZ+QjCVB8CGUqP7YDDnnwB4R2 // SIG // 3/yPoSL3MR8s2KWpJWP3XjIJ91k1NZjH+Eftkx6FDRlC // SIG // LJZgUW/fYj+PIxgA4kaBVPJbNKsTdywoWP7Xix1iCGLl // SIG // rqyRh4MCSiIigbY32omnCDAU7OlOuXKIoWSs6w81+b1o // SIG // uDrAdi+3C+nWkOs0d20bHyxnhtOy6td3G1wM9SoRBRjU // SIG // Dnq9CicWHWFN/+61N6M+X9LNaJ+DLWSljj4ca8ZHLhLd // SIG // ZM77QbtK6XrlneX8jO1Onzg4lRjG1h57cKd7uqsZEQB8 // SIG // ylNy5Z519sB8kLrkmpKj6f/89DdweiRABFDbQDgJ+nMT // SIG // 4XIbOgvH0mwbc9QCPHJENZ1XdOJk // SIG // End signature block