2019-05-31 11:56:07 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2019-05-31 11:56:07 -04:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @fileoverview
|
|
|
|
* @suppress {missingRequire}
|
|
|
|
*/
|
|
|
|
|
|
|
|
Zone.__load_patch('fetch', (global: any, Zone: ZoneType, api: _ZonePrivate) => {
|
|
|
|
interface FetchTaskData extends TaskData {
|
|
|
|
fetchArgs?: any[];
|
|
|
|
}
|
|
|
|
let fetch = global['fetch'];
|
|
|
|
if (typeof fetch !== 'function') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const originalFetch = global[api.symbol('fetch')];
|
|
|
|
if (originalFetch) {
|
|
|
|
// restore unpatched fetch first
|
|
|
|
fetch = originalFetch;
|
|
|
|
}
|
|
|
|
const ZoneAwarePromise = global.Promise;
|
|
|
|
const symbolThenPatched = api.symbol('thenPatched');
|
|
|
|
const fetchTaskScheduling = api.symbol('fetchTaskScheduling');
|
|
|
|
const fetchTaskAborting = api.symbol('fetchTaskAborting');
|
|
|
|
const OriginalAbortController = global['AbortController'];
|
|
|
|
const supportAbort = typeof OriginalAbortController === 'function';
|
|
|
|
let abortNative: Function|null = null;
|
|
|
|
if (supportAbort) {
|
|
|
|
global['AbortController'] = function() {
|
|
|
|
const abortController = new OriginalAbortController();
|
|
|
|
const signal = abortController.signal;
|
|
|
|
signal.abortController = abortController;
|
|
|
|
return abortController;
|
|
|
|
};
|
|
|
|
abortNative = api.patchMethod(
|
|
|
|
OriginalAbortController.prototype, 'abort',
|
|
|
|
(delegate: Function) => (self: any, args: any) => {
|
|
|
|
if (self.task) {
|
|
|
|
return self.task.zone.cancelTask(self.task);
|
|
|
|
}
|
|
|
|
return delegate.apply(self, args);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
const placeholder = function() {};
|
|
|
|
global['fetch'] = function() {
|
|
|
|
const args = Array.prototype.slice.call(arguments);
|
|
|
|
const options = args.length > 1 ? args[1] : null;
|
|
|
|
const signal = options && options.signal;
|
|
|
|
return new Promise((res, rej) => {
|
|
|
|
const task = Zone.current.scheduleMacroTask(
|
2020-04-13 19:40:21 -04:00
|
|
|
'fetch', placeholder, {fetchArgs: args} as FetchTaskData,
|
2019-05-31 11:56:07 -04:00
|
|
|
() => {
|
|
|
|
let fetchPromise;
|
|
|
|
let zone = Zone.current;
|
|
|
|
try {
|
|
|
|
(zone as any)[fetchTaskScheduling] = true;
|
|
|
|
fetchPromise = fetch.apply(this, args);
|
|
|
|
} catch (error) {
|
|
|
|
rej(error);
|
|
|
|
return;
|
|
|
|
} finally {
|
|
|
|
(zone as any)[fetchTaskScheduling] = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(fetchPromise instanceof ZoneAwarePromise)) {
|
|
|
|
let ctor = fetchPromise.constructor;
|
|
|
|
if (!ctor[symbolThenPatched]) {
|
|
|
|
api.patchThen(ctor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fetchPromise.then(
|
|
|
|
(resource: any) => {
|
|
|
|
if (task.state !== 'notScheduled') {
|
|
|
|
task.invoke();
|
|
|
|
}
|
|
|
|
res(resource);
|
|
|
|
},
|
|
|
|
(error: any) => {
|
|
|
|
if (task.state !== 'notScheduled') {
|
|
|
|
task.invoke();
|
|
|
|
}
|
|
|
|
rej(error);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
() => {
|
|
|
|
if (!supportAbort) {
|
|
|
|
rej('No AbortController supported, can not cancel fetch');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (signal && signal.abortController && !signal.aborted &&
|
|
|
|
typeof signal.abortController.abort === 'function' && abortNative) {
|
|
|
|
try {
|
|
|
|
(Zone.current as any)[fetchTaskAborting] = true;
|
|
|
|
abortNative.call(signal.abortController);
|
|
|
|
} finally {
|
|
|
|
(Zone.current as any)[fetchTaskAborting] = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
rej('cancel fetch need a AbortController.signal');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (signal && signal.abortController) {
|
|
|
|
signal.abortController.task = task;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
});
|