@microsoft/windows-admin-center-sdk
Version:
Microsoft - Windows Admin Center Shell
934 lines (931 loc) • 44.6 kB
JavaScript
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