rx-player
Version:
Canal+ HTML5 Video Player
445 lines (417 loc) • 15.6 kB
text/typescript
/**
* Copyright 2015 CANAL+ Group
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { CustomLoaderError, isKnownError, NetworkErrorTypes } from "../../../errors";
import log from "../../../log";
import type { ICdnMetadata } from "../../../parsers/manifest";
import cancellableSleep from "../../../utils/cancellable_sleep";
import getFuzzedDelay from "../../../utils/get_fuzzed_delay";
import getTimestamp from "../../../utils/monotonic_timestamp";
import noop from "../../../utils/noop";
import { RequestError } from "../../../utils/request";
import type { CancellationSignal } from "../../../utils/task_canceller";
import TaskCanceller from "../../../utils/task_canceller";
import type CdnPrioritizer from "../cdn_prioritizer";
/**
* Called on a loader error.
* Returns whether the loader request should be retried.
*
* TODO the notion of retrying or not could be transport-specific (e.g. 412 are
* mainly used for Smooth contents) and thus as part of the transport code (e.g.
* by rejecting with an error always having a `canRetry` property?).
* Or not, to ponder.
*
* @param {Error} error
* @returns {Boolean} - If true, the request can be retried.
*/
function shouldRetry(error: unknown): boolean {
if (error instanceof RequestError) {
if (error.type === NetworkErrorTypes.ERROR_HTTP_CODE) {
return (
error.status >= 500 ||
error.status === 404 ||
error.status === 415 || // some CDN seems to use that code when
// requesting low-latency segments too much
// in advance
error.status === 412
);
}
return (
error.type === NetworkErrorTypes.TIMEOUT ||
error.type === NetworkErrorTypes.ERROR_EVENT
);
} else if (error instanceof CustomLoaderError) {
if (typeof error.canRetry === "boolean") {
return error.canRetry;
}
if (error.xhr !== undefined) {
return (
error.xhr.status >= 500 ||
error.xhr.status === 404 ||
error.xhr.status === 415 || // some CDN seems to use that code when
// requesting low-latency segments too much
// in advance
error.xhr.status === 412
);
}
return false;
}
return isKnownError(error) && error.code === "INTEGRITY_ERROR";
}
/** Settings to give to the backoff functions to configure their behavior. */
export interface IBackoffSettings {
/**
* Initial delay to wait if a request fails before making a new request, in
* milliseconds.
*/
baseDelay: number;
/**
* Maximum delay to wait if a request fails before making a new request, in
* milliseconds.
*/
maxDelay: number;
/**
* Maximum number of retries to perform on "regular" errors (e.g. due to HTTP
* status, integrity errors, timeouts...).
*/
maxRetry: number;
/** Callback called when a request is retried. */
onRetry: (err: unknown) => void;
}
/**
* Specific algorithm used to perform segment and manifest requests.
*
* Here how it works:
*
* 1. You give it one or multiple of the CDN available for the resource you
* want to request (from the most important one to the least important),
* a callback doing the request with the chosen CDN in argument, and some
* options.
*
* 2. it tries to call the request callback with the most prioritized CDN
* first:
* - if it works as expected, it resolves the returned Promise with that
* request's response.
* - if it fails, it calls ther `onRetry` callback given with the
* corresponding error, un-prioritize that CDN and try with the new
* most prioritized CDN.
*
* Each CDN might be retried multiple times, depending on the nature of the
* error and the Configuration given.
*
* Multiple retries of the same CDN are done after a delay to avoid
* overwhelming it, this is what we call a "backoff". That delay raises
* exponentially as multiple consecutive errors are encountered on this
* CDN.
*
* @param {Array.<string>|null} cdns - The different CDN on which the
* wanted resource is available. `scheduleRequestWithCdns` will call the
* `performRequest` callback with the right element from that array if different
* from `null`.
*
* Can be set to `null` when that resource is not reachable through a CDN, in
* which case the `performRequest` callback may be called with `null`.
* @param {Object|null} cdnPrioritizer - Interface allowing to give the priority
* between multiple CDNs.
* @param {Function} performRequest - Callback implementing the request in
* itself. Resolving when the resource request succeed and rejecting with the
* corresponding error when the request failed.
* @param {Object} options - Configuration allowing to tweak the number on which
* the algorithm behind `scheduleRequestWithCdns` bases itself.
* @param {Object} cancellationSignal - CancellationSignal allowing to cancel
* the logic of `scheduleRequestWithCdns`.
* To trigger if the resource is not needed anymore.
* @returns {Promise} - Promise resolving, with the corresponding
* `performRequest`'s data, when the resource request succeed and rejecting in
* the following scenarios:
* - `scheduleRequestWithCdns` has been cancelled due to `cancellationSignal`
* being triggered. In that case a `CancellationError` is thrown.
*
* - The resource request(s) failed and will not be retried anymore.
*/
export async function scheduleRequestWithCdns<T>(
cdns: ICdnMetadata[] | null,
cdnPrioritizer: CdnPrioritizer | null,
performRequest: (
cdn: ICdnMetadata | null,
cancellationSignal: CancellationSignal,
) => Promise<T>,
options: IBackoffSettings,
cancellationSignal: CancellationSignal,
): Promise<T> {
if (cancellationSignal.cancellationError !== null) {
return Promise.reject(cancellationSignal.cancellationError);
}
const { baseDelay, maxDelay, maxRetry, onRetry } = options;
if (cdns !== null && cdns.length === 0) {
log.warn("Fetchers: no CDN given to `scheduleRequestWithCdns`.");
}
const missedAttempts: Map<ICdnMetadata | null, ICdnAttemptMetadata> = new Map();
const initialCdnToRequest = getCdnToRequest();
if (initialCdnToRequest === undefined) {
throw new Error("No CDN to request");
}
return requestCdn(initialCdnToRequest);
/**
* Returns what is now the most prioritary CDN to request the wanted resource.
*
* A return value of `null` indicates that the resource can be requested
* through another mean than by doing an HTTP request.
*
* A return value of `undefined` indicates that there's no CDN left to request
* the resource.
* @returns {Object|null|undefined}
*/
function getCdnToRequest(): ICdnMetadata | null | undefined {
if (cdns === null) {
const nullAttemptObject = missedAttempts.get(null);
if (nullAttemptObject !== undefined && nullAttemptObject.isBlacklisted) {
return undefined;
}
return null;
} else if (cdnPrioritizer === null) {
return getPrioritaryRequestableCdnFromSortedList(cdns);
} else {
const prioritized = cdnPrioritizer.getCdnPreferenceForResource(cdns);
return getPrioritaryRequestableCdnFromSortedList(prioritized);
}
}
/**
* Perform immediately the request for the given CDN.
*
* If it fails, forbid the CDN from being used - optionally and in some
* conditions, only temporarily, then try the next CDN according to
* previously-set delays (with a potential sleep before to respect them).
*
* Reject if both the request fails and there's no CDN left to use.
* @param {string|null} cdn
* @returns {Promise}
*/
async function requestCdn(cdn: ICdnMetadata | null): Promise<T> {
try {
const res = await performRequest(cdn, cancellationSignal);
return res;
} catch (error: unknown) {
if (TaskCanceller.isCancellationError(error)) {
throw error;
}
if (cdn !== null && cdnPrioritizer !== null) {
// We failed requesting the resource on this CDN.
// Globally give priority to the next CDN through the CdnPrioritizer.
cdnPrioritizer.downgradeCdn(cdn);
}
let missedAttemptsObj = missedAttempts.get(cdn);
if (missedAttemptsObj === undefined) {
missedAttemptsObj = {
errorCounter: 1,
blockedUntil: undefined,
isBlacklisted: false,
};
missedAttempts.set(cdn, missedAttemptsObj);
} else {
missedAttemptsObj.errorCounter++;
}
if (!shouldRetry(error)) {
missedAttemptsObj.blockedUntil = undefined;
missedAttemptsObj.isBlacklisted = true;
return retryWithNextCdn(error);
}
if (missedAttemptsObj.errorCounter > maxRetry) {
missedAttemptsObj.blockedUntil = undefined;
missedAttemptsObj.isBlacklisted = true;
} else {
const errorCounter = missedAttemptsObj.errorCounter;
const delay = Math.min(baseDelay * Math.pow(2, errorCounter - 1), maxDelay);
const fuzzedDelay = getFuzzedDelay(delay);
missedAttemptsObj.blockedUntil = getTimestamp() + fuzzedDelay;
}
return retryWithNextCdn(error);
}
}
/**
* After a request error, find the new most prioritary CDN and perform the
* request with it, optionally after a delay.
*
* If there's no CDN left to test, reject the original request error.
* @param {*} prevRequestError
* @returns {Promise}
*/
async function retryWithNextCdn(prevRequestError: unknown): Promise<T> {
const nextCdn = getCdnToRequest();
if (cancellationSignal.isCancelled()) {
throw cancellationSignal.cancellationError;
}
if (nextCdn === undefined) {
throw prevRequestError;
}
onRetry(prevRequestError);
if (cancellationSignal.isCancelled()) {
throw cancellationSignal.cancellationError;
}
return waitPotentialBackoffAndRequest(nextCdn, prevRequestError);
}
/**
* Request the corresponding CDN after the optional backoff needed before
* requesting it.
*
* If a new CDN become prioritary in the meantime, request it instead, again
* awaiting its optional backoff delay if it exists.
* @param {string|null} nextWantedCdn
* @param {*} prevRequestError
* @returns {Promise}
*/
function waitPotentialBackoffAndRequest(
nextWantedCdn: ICdnMetadata | null,
prevRequestError: unknown,
): Promise<T> {
const nextCdnAttemptObj = missedAttempts.get(nextWantedCdn);
if (nextCdnAttemptObj === undefined || nextCdnAttemptObj.blockedUntil === undefined) {
return requestCdn(nextWantedCdn);
}
const now = getTimestamp();
const blockedFor = nextCdnAttemptObj.blockedUntil - now;
if (blockedFor <= 0) {
return requestCdn(nextWantedCdn);
}
const canceller = new TaskCanceller();
const unlinkCanceller = canceller.linkToSignal(cancellationSignal);
return new Promise<T>((res, rej) => {
cdnPrioritizer?.addEventListener(
"priorityChange",
() => {
const updatedPrioritaryCdn = getCdnToRequest();
if (cancellationSignal.isCancelled()) {
throw cancellationSignal.cancellationError;
}
if (updatedPrioritaryCdn === undefined) {
return cleanAndReject(prevRequestError);
}
if (updatedPrioritaryCdn !== nextWantedCdn) {
canceller.cancel();
waitPotentialBackoffAndRequest(updatedPrioritaryCdn, prevRequestError).then(
cleanAndResolve,
cleanAndReject,
);
}
},
canceller.signal,
);
cancellableSleep(blockedFor, canceller.signal).then(
() => requestCdn(nextWantedCdn).then(cleanAndResolve, cleanAndReject),
noop,
);
function cleanAndResolve(response: T) {
unlinkCanceller();
res(response);
}
function cleanAndReject(err: unknown) {
unlinkCanceller();
rej(err);
}
});
}
/**
* Takes in input the list of CDN that can be used to request the resource, in
* a general preference order.
*
* Returns the actual most prioritary Cdn to request, based on the current
* attempts already done for that resource.
*
* Returns `undefined` if there's no Cdn left to request the resource.
* @param {Array.<Object>} sortedCdns
* @returns {Object|undefined}
*/
function getPrioritaryRequestableCdnFromSortedList(
sortedCdns: ICdnMetadata[],
): ICdnMetadata | undefined {
if (missedAttempts.size === 0) {
return sortedCdns[0];
}
const now = getTimestamp();
return sortedCdns
.filter((c) => missedAttempts.get(c)?.isBlacklisted !== true)
.reduce(
(
acc: [ICdnMetadata, number | undefined] | undefined,
x: ICdnMetadata,
): [ICdnMetadata, number | undefined] => {
let blockedUntil = missedAttempts.get(x)?.blockedUntil;
if (blockedUntil !== undefined && blockedUntil <= now) {
blockedUntil = undefined;
}
if (acc === undefined) {
return [x, blockedUntil];
}
if (acc[1] === undefined) {
return acc;
}
if (blockedUntil === undefined) {
return [x, undefined];
}
if (blockedUntil < acc[1]) {
return [x, blockedUntil];
}
return acc;
},
undefined,
)?.[0];
}
}
/**
* Lightweight version of the request algorithm, this time with only a simple
* Promise given.
* @param {Function} performRequest
* @param {Object} options
* @returns {Promise}
*/
export function scheduleRequestPromise<T>(
performRequest: () => Promise<T>,
options: IBackoffSettings,
cancellationSignal: CancellationSignal,
): Promise<T> {
// same than for a single unknown CDN
return scheduleRequestWithCdns(null, null, performRequest, options, cancellationSignal);
}
/**
* Metadata associated to attempt(s) of requesting a resource through the same
* CDN.
*
* Each `ICdnAttemptMetadata` object should concern only one CDN.
*/
interface ICdnAttemptMetadata {
/**
* Count the amount of consecutive times an error has been encountered while
* requesting this resource though the concerned CDN.
*
* For example `1` means that the request through this CDN failed for the
* first consecutive time.
* `2` means that after requesting this CDN two consecutive times, the request
* still failed.
* etc.
*/
errorCounter: number;
/**
* Monotonically-raising timestamp, until which it should be forbidden to
* request this CDN.
* Enforcing this delay allows to prevent making too much requests to a given
* CDN.
*
* `undefined` when either there is no enforced delay or when the CDN is
* blacklisted anyway (@see isBlacklisted)
*/
blockedUntil: number | undefined;
/** If `true`, that request should not be requested at all anymore. */
isBlacklisted: boolean;
}