@microsoft/windows-admin-center-sdk
Version:
Microsoft - Windows Admin Center Shell
665 lines (662 loc) • 30.2 kB
JavaScript
import { Observable, throwError } from 'rxjs';
import { ajax } from 'rxjs/ajax';
import { catchError, mergeMap, switchMap } from 'rxjs/operators';
import { Cookie } from './cookie';
import { ErrorExtended } from './error-extended';
import { headerConstants } from './http-constants';
/**
* Retry options for an http request
*/
export class HttpRetryOptions {
maxRetry = 0;
handlers = [];
}
/**
* Enum for http method types
*/
export var HttpMethod;
(function (HttpMethod) {
HttpMethod["Get"] = "GET";
HttpMethod["Post"] = "POST";
HttpMethod["Put"] = "PUT";
HttpMethod["Delete"] = "DELETE";
HttpMethod["Patch"] = "PATCH";
HttpMethod["Head"] = "HEAD";
HttpMethod["Options"] = "OPTIONS";
})(HttpMethod || (HttpMethod = {}));
/**
* The Http observable based class.
*/
export class Http {
/**
* Default ajax options must be used for CORS.
*/
static defaultHttpOptions = { withCredentials: true, crossDomain: true };
/**
* The collection of set of monitors.
*/
static monitorSets = [];
/**
* The default retry options.
*/
defaultRetryOptions = new HttpRetryOptions();
/**
* Register the set of monitors.
*
* @param monitorSet The set of monitors.
*/
static registerMonitorSet(monitorSet) {
const found = Http.monitorSets.find(monitors => monitors.name === monitorSet.name);
if (found) {
return;
}
Http.monitorSets.push(monitorSet);
}
/**
* Unregister the set of monitors.
*
* @param name The name of set of monitors.
* @returns boolean true if unregistered the named set.
*/
static unregisterMonitors(name) {
const found = Http.monitorSets.find(monitors => monitors.name === name);
if (found) {
Http.monitorSets.remove(found);
return true;
}
return false;
}
/**
* The common request method.
* Adds default responseType, contentType, Accept values if they are not already included in the request
*
* @param request the request options.
* @param options the retry options.
*/
request(request, retryOptions) {
if (!request) {
request = {};
}
if (!retryOptions) {
retryOptions = this.defaultRetryOptions;
}
if (!request.headers) {
request.headers = {};
}
if (!request.responseType) {
request.responseType = 'json';
}
if (request.headers && !request.headers[headerConstants.CONTENT_TYPE]) {
request.headers[headerConstants.CONTENT_TYPE] = 'application/json; charset=utf-8';
}
// Fix for file transfer
if (request.body instanceof FormData) {
delete request.headers[headerConstants.CONTENT_TYPE];
}
if (request.method === HttpMethod.Get || request.method === HttpMethod.Delete || request.method === HttpMethod.Head) {
delete request.headers[headerConstants.CONTENT_TYPE];
request.body = undefined;
}
if (request.headers && !request.headers[headerConstants.ACCEPT]) {
request.headers[headerConstants.ACCEPT] = 'application/json, text/plain, */*';
}
if (request.headers && !request.headers[headerConstants.CROSS_SITE_REQUEST_FORGERY_TOKEN]) {
const token = Cookie.getCrossSiteRequestForgeryToken();
if (token) {
request.headers[headerConstants.CROSS_SITE_REQUEST_FORGERY_TOKEN] = token;
}
}
if (retryOptions.maxRetry > 0 && retryOptions.handlers && retryOptions.handlers.length > 0) {
return this.requestWithHandlers(request, retryOptions);
}
return this.monitorAjax(request);
}
/**
* Performs a request without modification.
* If the result is an error, we will retry with the handlers in options
*
* @param request the request options.
* @param options the retry options.
* @param count the current iteration of the retry cycle.
*/
requestWithHandlers(request, retryOptions, count = 0) {
return this.monitorAjax(request)
.pipe(catchError((error, caught) => {
// original request is replaced with latest instance. it must take current error request.
const caughtRequest = caught.source.request || (caught.source.source && caught.source.source.value) || request;
if (++count > retryOptions.maxRetry) {
return throwError(() => error);
}
const handler = retryOptions.handlers.find(handler2 => handler2.canHandle(error.status, error));
if (handler) {
if (handler.handleNoRetry) {
return handler.handleNoRetry(error.status, caughtRequest, error)
.pipe(mergeMap(() => throwError(() => error)));
}
else if (handler.handle) {
return handler.handle(error.status, caughtRequest, error)
.pipe(catchError(handlerError => {
// if the handler throws, return the original error with an inserted
// property for the handler error
error['handlerError'] = handlerError;
return throwError(() => error);
}), switchMap(() => this.monitorAjax(caughtRequest)));
}
}
return throwError(() => error);
}));
}
/**
* Performs a request with `get` http method.
*
* @param url the url.
* @param request the request options.
* @param options the retry options.
*/
get(url, request, options) {
request = request ? request : {};
request.url = url;
request.method = HttpMethod.Get;
return this.request(request, options);
}
/**
* Performs a request with `post` http method.
*
* @param url the url.
* @param body the body content.
* @param request the request options.
* @param options the retry options.
*/
post(url, body, request, options) {
request = request ? request : {};
request.url = url;
request.method = HttpMethod.Post;
request.body = body;
return this.request(request, options);
}
/**
* Performs a request with `put` http method.
*
* @param url the url.
* @param body the body content.
* @param request the request options.
* @param options the retry options.
*/
put(url, body, request, options) {
request = request ? request : {};
request.url = url;
request.method = HttpMethod.Put;
request.body = body;
return this.request(request, options);
}
/**
* Performs a request with `delete` http method.
*
* @param url the url.
* @param request the request options.
* @param options the retry options.
*/
delete(url, request, options) {
request = request ? request : {};
request.url = url;
request.method = HttpMethod.Delete;
return this.request(request, options);
}
/**
* Performs a request with `patch` http method.
*
* @param url the url.
* @param body the body content.
* @param request the request options.
* @param options the retry options.
*/
patch(url, body, request, options) {
request = request ? request : {};
request.url = url;
request.method = HttpMethod.Patch;
request.body = body;
return this.request(request, options);
}
/**
* Performs a request with `head` http method.
*
* @param url the url.
* @param request the request options.
* @param options the retry options.
*/
head(url, request, options) {
request = request ? request : {};
request.url = url;
request.method = HttpMethod.Head;
return this.request(request, options);
}
/**
* Performs a request with `options` http method.
*
* @param url the url.
* @param request the request options.
* @param options the retry options.
*/
options(url, request, options) {
request = request ? request : {};
request.url = url;
request.method = HttpMethod.Options;
return this.request(request, options);
}
/**
* Performs a request with 'get' http method with cache control.
*
* @param url the uri for GET call.
* @return the observable for GET result data.
*/
getNoCache(url, noCache = true, responseType = '', withCredentials = true) {
const publish = new Observable(observer => {
const request = new XMLHttpRequest();
const handler = () => {
if (request.readyState === XMLHttpRequest.DONE) {
if (request.status === 200) {
try {
let response;
if (responseType === '') {
response = JSON.parse(request.response);
}
else {
response = request.response;
}
observer.next({
status: request.status,
response
});
observer.complete();
}
catch (e) {
observer.error(e);
}
}
else {
const error = new ErrorExtended(request.statusText);
error.extendedSource = ErrorExtended.sources.getNoCache;
error.extended = { status: request.status, url };
observer.error(error);
}
}
};
request.open('Get', url);
request.withCredentials = withCredentials;
request.responseType = responseType;
request.setRequestHeader(headerConstants.ACCEPT, 'application/json, text/plain, */*');
if (noCache) {
request.setRequestHeader('Cache-control', 'no-cache');
}
request.onreadystatechange = handler;
request.send();
});
return publish;
}
/**
* Performs a request with 'delete' http method without waiting for the response.
*
* @param url the uri for GET call.
*/
deleteQuick(url, headers) {
const request = new XMLHttpRequest();
request.open('Delete', url);
request.withCredentials = true;
request.responseType = 'json';
request.setRequestHeader(headerConstants.ACCEPT, 'application/json, text/plain, */*');
request.setRequestHeader(headerConstants.CONTENT_TYPE, 'application/json; charset=utf-8');
request.setRequestHeader(headerConstants.CACHE_CONTROL, 'no-cache');
let xsrf = false;
if (headers) {
for (const key in headers) {
if (key) {
request.setRequestHeader(key, headers[key]);
if (key.toLowerCase() === headerConstants.CROSS_SITE_REQUEST_FORGERY_TOKEN.toLowerCase()) {
xsrf = true;
}
}
}
}
const token = Cookie.getCrossSiteRequestForgeryToken();
if (token && !xsrf) {
request.setRequestHeader(headerConstants.CROSS_SITE_REQUEST_FORGERY_TOKEN, token);
}
// not watching any response.
request.send();
}
monitorAjax(request) {
let monitored = ajax;
for (const monitorSet of Http.monitorSets) {
monitored = this.monitor(monitored, monitorSet);
}
return monitored(request);
}
monitor(target, monitorSet) {
return function (request) {
return monitorSet.preMonitor(request)
.pipe(switchMap(monitoredRequest => {
if (monitoredRequest instanceof Error) {
return throwError(() => monitoredRequest);
}
return target(monitoredRequest).pipe(catchError((error) => {
return monitorSet.errorMonitor(error);
}), switchMap(response => monitorSet.successMonitor(response)));
}));
};
}
}
//# sourceMappingURL=http.js.map
// SIG // Begin signature block
// SIG // MIIoKAYJKoZIhvcNAQcCoIIoGTCCKBUCAQExDzANBglg
// SIG // hkgBZQMEAgEFADB3BgorBgEEAYI3AgEEoGkwZzAyBgor
// SIG // BgEEAYI3AgEeMCQCAQEEEBDgyQbOONQRoqMAEEvTUJAC
// SIG // AQACAQACAQACAQACAQAwMTANBglghkgBZQMEAgEFAAQg
// SIG // t/n63NHUUM8x9e3ysYnXzQISCh0YmcKe/fRMTtNytMmg
// 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 // DQEJBDEiBCDjhMyOBSSPDn4+NgGe0UlVKXrc3CHZIZsQ
// SIG // 52P2eiCiajBCBgorBgEEAYI3AgEMMTQwMqAUgBIATQBp
// SIG // AGMAcgBvAHMAbwBmAHShGoAYaHR0cDovL3d3dy5taWNy
// SIG // b3NvZnQuY29tMA0GCSqGSIb3DQEBAQUABIIBAKR+mReM
// SIG // WXmaA4nr+TVUbaERofljvxwlMpHmWemglnSLHtnTHGAr
// SIG // K9NM50SUAHiLVoPOHe36hsz63F+qnFSlumolHZzZmwkc
// SIG // yWBvEDsekRBNAUFbOnOeho1XK1Fcgad1IORpuQxSInrH
// SIG // UNlCMe17DpxSCn7DOEt+9S1tyJQv6uLIkYPo6XuVDNxv
// SIG // 9ynoJE87esNl4iDRj9iBqoiXB/+FPgKmiIsE3jW98P2n
// SIG // x/v5dHwGCPLuTfcKFmqPYQ6hoBlzsOs+3P2BYRTnc3+8
// SIG // VJf69+WFoR+TQku4crHAMJLfh2iXK05d/K/WbmIJsNnJ
// SIG // mEZEZHFHVejVPXMjRaxEZQl2w+ihgheUMIIXkAYKKwYB
// SIG // BAGCNwMDATGCF4Awghd8BgkqhkiG9w0BBwKgghdtMIIX
// SIG // aQIBAzEPMA0GCWCGSAFlAwQCAQUAMIIBUgYLKoZIhvcN
// SIG // AQkQAQSgggFBBIIBPTCCATkCAQEGCisGAQQBhFkKAwEw
// SIG // MTANBglghkgBZQMEAgEFAAQge3cHYXHhb/jeywNGtf3z
// SIG // eFXWYJOJMzvVvqRf2y3YC2cCBmet2JkD2BgTMjAyNTAy
// SIG // MjAxNTI4MzMuOTAyWjAEgAIB9KCB0aSBzjCByzELMAkG
// SIG // A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAO
// SIG // BgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29m
// SIG // dCBDb3Jwb3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0
// SIG // IEFtZXJpY2EgT3BlcmF0aW9uczEnMCUGA1UECxMeblNo
// SIG // aWVsZCBUU1MgRVNOOkE0MDAtMDVFMC1EOTQ3MSUwIwYD
// SIG // VQQDExxNaWNyb3NvZnQgVGltZS1TdGFtcCBTZXJ2aWNl
// SIG // oIIR6jCCByAwggUIoAMCAQICEzMAAAHs4CukgtCRUoAA
// SIG // AQAAAewwDQYJKoZIhvcNAQELBQAwfDELMAkGA1UEBhMC
// SIG // VVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcT
// SIG // B1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jw
// SIG // b3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUt
// SIG // U3RhbXAgUENBIDIwMTAwHhcNMjMxMjA2MTg0NTM4WhcN
// SIG // MjUwMzA1MTg0NTM4WjCByzELMAkGA1UEBhMCVVMxEzAR
// SIG // BgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v
// SIG // bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv
// SIG // bjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2EgT3Bl
// SIG // cmF0aW9uczEnMCUGA1UECxMeblNoaWVsZCBUU1MgRVNO
// SIG // OkE0MDAtMDVFMC1EOTQ3MSUwIwYDVQQDExxNaWNyb3Nv
// SIG // ZnQgVGltZS1TdGFtcCBTZXJ2aWNlMIICIjANBgkqhkiG
// SIG // 9w0BAQEFAAOCAg8AMIICCgKCAgEAsEf0bgk24MVFlZv1
// SIG // XbpdtrsHRGZtCKABbOqCK9/VSvyLT/NHJ/vE5rT+u4mm
// SIG // weA5gCifRh+nSRoRDyaWOL0ykUjsK0TcVSCqDz3lBd3+
// SIG // FchxHKP7tUFGnZcA9d9jbmQsW54ejItpSxu6Q77M2ajB
// SIG // u0tzAotm5Np77RinXgCC/h++4C+K9NU0lm+67BNiW9T/
// SIG // zemP1tQqg4tfyG9/80all7eM8b3SBnD40uGSskBBd0hG
// SIG // QKuFyI4sqMDx2qjW2cXX9pFjv2o3X01PObfd+AlwIp29
// SIG // KPrkPSrWijS1VXDX+UKUuH+vzLFzryBbgmDEXSg46Zr6
// SIG // MAHi/tY9u2wsQgaQ0B61pHz82af1/m7fQuxOYTz+h1Ua
// SIG // KgWEe7tYFH+RhKvua9RwNI2o59EOjr32HJBNB3Tr+ilm
// SIG // vrAJiRuzw702Wnu+4aJs8eiD6oIFaTWbgpO/Un1Zpyrv
// SIG // RefFAJ1OfE6gxxMxrEJzFECrLUt845+klNDSxBTQnrZb
// SIG // mipKlg0VSxFm7t9vSBId7alz138ukYf8Am8HvUgiSKKr
// SIG // QXsQaz8kGANl2s9XyvcrE7MdJAPVdScFVeOCGvXPjMLQ
// SIG // EerKinQIEaP27P17vILmvCw3uilsrve+HvZhlu2TvJ2q
// SIG // wxawE9RFxhw7nsoEir79iu8AfJQIDBiY+9wkL6/o6qFs
// SIG // Mel3cnkCAwEAAaOCAUkwggFFMB0GA1UdDgQWBBT0WtBH
// SIG // ZP4r9cIWELFfFIBH+EyFhjAfBgNVHSMEGDAWgBSfpxVd
// SIG // AF5iXYP05dJlpxtTNRnpcjBfBgNVHR8EWDBWMFSgUqBQ
// SIG // hk5odHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3Bz
// SIG // L2NybC9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENB
// SIG // JTIwMjAxMCgxKS5jcmwwbAYIKwYBBQUHAQEEYDBeMFwG
// SIG // CCsGAQUFBzAChlBodHRwOi8vd3d3Lm1pY3Jvc29mdC5j
// SIG // b20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUyMFRpbWUt
// SIG // U3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNydDAMBgNVHRMB
// SIG // Af8EAjAAMBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMA4G
// SIG // A1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQsFAAOCAgEA
// SIG // krzEpDEq745Qz2oPAEW9DhawELUizA6TdFGNxY7z4cBi
// SIG // g664sZp7jH465lY0atbvCIZA7xhf2332xU6/iAJw0noP
// SIG // Ewfc3xv+Mm5J7qKZJW3ho27ezC8aX4aJQhEchHNtDzGS
// SIG // ic/Ur837jtZ+ca6yzi/JtJ5r+ZAXL/stQFyeUHC4nJoX
// SIG // tiKd/w+uxHeqD6kCNN5g42GktTUIQTbbue8Dyl2dRKDU
// SIG // 6AZPGwOvN/cNdfW/mvVk6KiLJHURqD+cYwyL/pnNLwR4
// SIG // WRpCVb3yIZuAKfM6bQu8VQJctI3jr+XVBjAmIGY76E5o
// SIG // HeOW6gMLp3Zj5Rrq+3pXlmHnS0H+7Ny+fqn2mP8RIf/b
// SIG // qNe0pzP4B1UhgM7563hoTqwdi7XSqFUnuS22KYoV3LQ3
// SIG // u+omLS/pocVzxKc3Wt2yZYT0zkNyjhGQKVREQaOcpbVo
// SIG // zwlpV8cgqZeY4/Z2NJ33dO9W3pp6LvAN61Ga3YCiGrrb
// SIG // B+0hzojnm2RqjbvuttrybWt3gGLAgGsQHAfQYiT5Wu12
// SIG // nfaq02HU+OVZQmE7QUmOKFUbHnUgA7/fY7/4mCABstWw
// SIG // srbmtKP0Kr/Xqyps0Ak1TF2g3NuQ0y3DBia0bmtytMYr
// SIG // 3bZ6AXsc1Sa+sl6jPgWtsISFUbxnK4gZCl9BSRXlu69v
// SIG // V1/pNHuA5xuogRykI3nOlTcwggdxMIIFWaADAgECAhMz
// 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 // BAsTHm5TaGllbGQgVFNTIEVTTjpBNDAwLTA1RTAtRDk0
// SIG // NzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAg
// SIG // U2VydmljZaIjCgEBMAcGBSsOAwIaAxUAjhz7YFXc/RFt
// SIG // IjzS/wV6iaKlTH+ggYMwgYCkfjB8MQswCQYDVQQGEwJV
// SIG // UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMH
// SIG // UmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv
// SIG // cmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1T
// SIG // dGFtcCBQQ0EgMjAxMDANBgkqhkiG9w0BAQsFAAIFAOth
// SIG // kQQwIhgPMjAyNTAyMjAxMTMxMTZaGA8yMDI1MDIyMTEx
// SIG // MzExNlowdDA6BgorBgEEAYRZCgQBMSwwKjAKAgUA62GR
// SIG // BAIBADAHAgEAAgISuTAHAgEAAgITFzAKAgUA62LihAIB
// SIG // ADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEEAYRZCgMC
// SIG // oAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3
// SIG // DQEBCwUAA4IBAQAuUY2t4eE7i88PXLaEjJJxs86dP00o
// SIG // 71UhvQLqvZc95KCDITUy1cSjQCTN7sjQmNZ9dZ4tsvS/
// SIG // JlabGY9rlSvv8JpTcpGLJE5wUaEp08HFh33nFn3iZzX0
// SIG // Gp9AcwXGiHqje5HcyN9ddVIZZre4gluUmrRBAPDN1G40
// SIG // BdBbddVcncwSc1EgStAYTaIVEafOXuHveLImjTexBrfI
// SIG // Gr2vjS6Jbp87X0YCfF4147nKvKe4fKRkDjT2JYIpRSg2
// SIG // 3+4phwYM7qI6rhnbW5FvZ4FGFvjPpH2Rz6uM4p0TNO6V
// SIG // /G56G7tRdm1Js1QwIaeiYlFBxLAydmRbO52v30FLvbUi
// SIG // m4xTMYIEDTCCBAkCAQEwgZMwfDELMAkGA1UEBhMCVVMx
// SIG // EzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl
// SIG // ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3Jh
// SIG // dGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3Rh
// SIG // bXAgUENBIDIwMTACEzMAAAHs4CukgtCRUoAAAQAAAeww
// SIG // DQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzEN
// SIG // BgsqhkiG9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQgXNG+
// SIG // VrB1k3I5VPphBcHCvkBN+QXQyxkzL+8FBkwnLhswgfoG
// SIG // CyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCAnCeb1an03
// SIG // yIcdtUAQWysqP8XIkCF2qDFlC3owBNUKgzCBmDCBgKR+
// SIG // MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n
// SIG // dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVN
// SIG // aWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1p
// SIG // Y3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwAhMzAAAB
// SIG // 7OArpILQkVKAAAEAAAHsMCIEIJMWfWH4rigb0xVwIT3U
// SIG // 52hJl/r0nZiZ9lzyPXRL+C1OMA0GCSqGSIb3DQEBCwUA
// SIG // BIICAIYjRSm+Dk+LCR1w1kSGTJM3v3AG2m9e/8OYIx3i
// SIG // vBTjXQ/6/TUUHyyD2DVHnGnTe1mLag+d1UNq2gbcTIU1
// SIG // 0Di5oFPM0lQFYPsDGpU3pIfkUIlcqFcXUY9uQuQ5dFGo
// SIG // qlVL6tUbNTNjnwuBTmlo7wBr0GthVts7MSlCq0d6DNPP
// SIG // 4EgK9RWznrALqlqBHpXG7Ziy56HodkpLbcNGXzTTuuNK
// SIG // nHzDkENXAFKl+wXQ9icQBQhywvxzlIgwm7ieSCbA5EmY
// SIG // 04vqPFmrAZEf3nutncmV3PkExSd9e1zJc9POWdN9zofY
// SIG // NqOsClHKoPvIMb1kSWVbUnmOBZDUj+rXTfeAnipkgkFx
// SIG // hZgPviwYf/PjOIkLQ/t3C6bdf6xEJLjH+GLNZtpopGKM
// SIG // W+CP37Z58mozb5YHPFhNDGyYXnGbtflL5rNYu9gCRLaZ
// SIG // 8ui+mIs+3ahPdOOl5lJ1g973RORRtSJpRnf80+VJmge+
// SIG // 0V9E0UDX122iR8TO7ysfcWZ9OeL0I7eAEvG5vhLJXt81
// SIG // 5//d4Xf9IO1mUdrGXVCfnfhD9e63bNX6CObdDCxw6pY5
// SIG // CrGeGZJSYlOFtsRoXz8sCS61KGs12Rcdyk+Xj63D0+5l
// SIG // H3SAud572wCX6MOuIvIerlSBa288wtQN2VrFkyvCHUhy
// SIG // k9qzhKeNqoulJVQRE9iE9e3itVAu
// SIG // End signature block