rx-player
Version:
Canal+ HTML5 Video Player
355 lines (333 loc) • 11.1 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 } from "../../errors";
import type { ISegmentLoader as ICustomSegmentLoader } from "../../public_types";
import assert from "../../utils/assert";
import request from "../../utils/request";
import type { CancellationError, CancellationSignal } from "../../utils/task_canceller";
import type {
ISegmentContext,
ISegmentLoaderCallbacks,
ISegmentLoaderOptions,
ISegmentLoaderResultSegmentCreated,
ISegmentLoaderResultSegmentLoaded,
} from "../types";
import addQueryString from "../utils/add_query_string";
import byteRange from "../utils/byte_range";
import checkISOBMFFIntegrity from "../utils/check_isobmff_integrity";
import isMP4EmbeddedTrack from "./is_mp4_embedded_track";
import { createAudioInitSegment, createVideoInitSegment } from "./isobmff";
/**
* Segment loader triggered if there was no custom-defined one in the API.
* @param {string} initialUrl
* @param {Object} context
* @param {Object} loaderOptions
* @param {Object} callbacks
* @param {Object} cancelSignal
* @param {boolean} checkMediaSegmentIntegrity
* @returns {Promise}
*/
async function regularSegmentLoader(
initialUrl: string,
context: ISegmentContext,
callbacks: ISegmentLoaderCallbacks<Uint8Array<ArrayBuffer> | ArrayBuffer | null>,
loaderOptions: ISegmentLoaderOptions,
cancelSignal: CancellationSignal,
checkMediaSegmentIntegrity?: boolean | undefined,
): Promise<
ISegmentLoaderResultSegmentLoaded<Uint8Array<ArrayBuffer> | ArrayBuffer | null>
> {
const cmcdHeaders =
loaderOptions.cmcdPayload?.type === "headers"
? loaderOptions.cmcdPayload.value
: undefined;
const range = context.segment.range;
let headers;
if (Array.isArray(range)) {
headers = {
...cmcdHeaders,
Range: byteRange(range),
};
} else if (cmcdHeaders !== undefined) {
headers = cmcdHeaders;
}
const url =
loaderOptions.cmcdPayload?.type === "query"
? addQueryString(initialUrl, loaderOptions.cmcdPayload.value)
: initialUrl;
const data = await request({
url,
responseType: "arraybuffer",
headers,
timeout: loaderOptions.timeout,
connectionTimeout: loaderOptions.connectionTimeout,
cancelSignal,
onProgress: callbacks.onProgress,
});
const isMP4 = isMP4EmbeddedTrack(context.mimeType);
if (!isMP4 || checkMediaSegmentIntegrity !== true) {
return { resultType: "segment-loaded" as const, resultData: data };
}
const dataU8 = new Uint8Array(data.responseData);
checkISOBMFFIntegrity(dataU8, context.segment.isInit);
return {
resultType: "segment-loaded" as const,
resultData: { ...data, responseData: dataU8 },
};
}
/**
* Defines the url for the request, load the right loader (custom/default
* one).
*/
const generateSegmentLoader =
({
checkMediaSegmentIntegrity,
segmentLoader,
}: {
checkMediaSegmentIntegrity?: boolean | undefined;
segmentLoader?: ICustomSegmentLoader | undefined;
}) =>
(
url: string | null,
context: ISegmentContext,
loaderOptions: ISegmentLoaderOptions,
cancelSignal: CancellationSignal,
callbacks: ISegmentLoaderCallbacks<Uint8Array<ArrayBuffer> | ArrayBuffer | null>,
): Promise<
| ISegmentLoaderResultSegmentLoaded<Uint8Array<ArrayBuffer> | ArrayBuffer | null>
| ISegmentLoaderResultSegmentCreated<Uint8Array<ArrayBuffer> | ArrayBuffer | null>
> => {
const { segment } = context;
if (segment.isInit) {
if (
segment.privateInfos === undefined ||
segment.privateInfos.smoothInitSegment === undefined
) {
throw new Error("Smooth: Invalid segment format");
}
const smoothInitPrivateInfos = segment.privateInfos.smoothInitSegment;
let responseData: Uint8Array<ArrayBuffer>;
const {
codecPrivateData,
timescale,
height,
width,
protection = { keyId: undefined, keySystems: undefined },
} = smoothInitPrivateInfos;
if (codecPrivateData === undefined) {
throw new Error("Smooth: no codec private data.");
}
switch (context.type) {
case "video": {
responseData = createVideoInitSegment(
timescale,
width ?? 0,
height ?? 0,
72,
72,
4, // vRes, hRes, nal
codecPrivateData,
protection.keyId,
);
break;
}
case "audio": {
const {
channels = 0,
bitsPerSample = 0,
packetSize = 0,
samplingRate = 0,
} = smoothInitPrivateInfos;
responseData = createAudioInitSegment(
timescale,
channels,
bitsPerSample,
packetSize,
samplingRate,
codecPrivateData,
protection.keyId,
);
break;
}
default:
if (
(__ENVIRONMENT__.CURRENT_ENV as number) === (__ENVIRONMENT__.DEV as number)
) {
assert(false, "responseData should have been set");
}
responseData = new Uint8Array(0);
}
return Promise.resolve({
resultType: "segment-created" as const,
resultData: responseData,
});
} else if (url === null) {
return Promise.resolve({
resultType: "segment-created" as const,
resultData: null,
});
} else {
if (typeof segmentLoader !== "function") {
return regularSegmentLoader(
url,
context,
callbacks,
loaderOptions,
cancelSignal,
checkMediaSegmentIntegrity,
);
}
return new Promise((res, rej) => {
/** `true` when the custom segmentLoader should not be active anymore. */
let hasFinished = false;
/**
* Callback triggered when the custom segment loader has a response.
* @param {Object} _args
*/
const resolve = (_args: {
data: ArrayBuffer | Uint8Array;
size?: number | undefined;
duration?: number | undefined;
}) => {
if (hasFinished || cancelSignal.isCancelled()) {
return;
}
hasFinished = true;
cancelSignal.deregister(abortCustomLoader);
let data: ArrayBuffer | Uint8Array<ArrayBuffer>;
if (_args.data instanceof Uint8Array) {
if (_args.data.buffer instanceof ArrayBuffer) {
// Typescript is not so smart here for now
data = _args.data as Uint8Array<ArrayBuffer>;
} else {
data = _args.data.slice();
}
} else {
data = _args.data;
}
const isMP4 = isMP4EmbeddedTrack(context.mimeType);
if (!isMP4 || checkMediaSegmentIntegrity !== true) {
res({
resultType: "segment-loaded" as const,
resultData: {
responseData: data,
size: _args.size,
requestDuration: _args.duration,
},
});
}
const dataU8 = data instanceof Uint8Array ? data : new Uint8Array(data);
checkISOBMFFIntegrity(dataU8, context.segment.isInit);
res({
resultType: "segment-loaded" as const,
resultData: {
responseData: dataU8,
size: _args.size,
requestDuration: _args.duration,
},
});
};
/**
* Callback triggered when the custom segment loader fails
* @param {*} err - The corresponding error encountered
*/
const reject = (err: unknown) => {
if (hasFinished || cancelSignal.isCancelled()) {
return;
}
hasFinished = true;
cancelSignal.deregister(abortCustomLoader);
// Format error and send it
const castedErr = err as
| null
| undefined
| { message?: string; canRetry?: boolean; xhr?: XMLHttpRequest };
const message =
castedErr?.message ??
"Unknown error when fetching a Smooth segment through a " +
"custom segmentLoader.";
const emittedErr = new CustomLoaderError(
message,
castedErr?.canRetry ?? false,
castedErr?.xhr,
);
rej(emittedErr);
};
const progress = (_args: {
duration: number;
size: number;
totalSize?: number | undefined;
}) => {
if (hasFinished || cancelSignal.isCancelled()) {
return;
}
callbacks.onProgress({
duration: _args.duration,
size: _args.size,
totalSize: _args.totalSize,
});
};
const fallback = () => {
if (hasFinished || cancelSignal.isCancelled()) {
return;
}
hasFinished = true;
cancelSignal.deregister(abortCustomLoader);
regularSegmentLoader(
url,
context,
callbacks,
loaderOptions,
cancelSignal,
checkMediaSegmentIntegrity,
).then(res, rej);
};
const customCallbacks = { reject, resolve, fallback, progress };
let byteRanges: Array<[number, number]> | undefined;
if (context.segment.range !== undefined) {
byteRanges = [context.segment.range];
if (context.segment.indexRange !== undefined) {
byteRanges.push(context.segment.indexRange);
}
}
const args = {
isInit: context.segment.isInit,
timeout: loaderOptions.timeout,
byteRanges,
trackType: context.type,
url,
cmcdPayload: loaderOptions.cmcdPayload,
};
const abort = segmentLoader(args, customCallbacks);
cancelSignal.register(abortCustomLoader);
/**
* The logic to run when the custom loader is cancelled while pending.
* @param {Error} err
*/
function abortCustomLoader(err: CancellationError) {
if (hasFinished) {
return;
}
hasFinished = true;
if (!hasFinished && typeof abort === "function") {
abort();
}
rej(err);
}
});
}
};
export default generateSegmentLoader;