Source: lib/util/manifest_parser_utils.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.util.ManifestParserUtils');
  7. goog.require('goog.Uri');
  8. goog.require('shaka.util.BufferUtils');
  9. goog.require('shaka.util.Error');
  10. goog.require('shaka.util.StringUtils');
  11. goog.require('shaka.util.Uint8ArrayUtils');
  12. /**
  13. * @summary Utility functions for manifest parsing.
  14. */
  15. shaka.util.ManifestParserUtils = class {
  16. /**
  17. * Resolves an array of relative URIs to the given base URIs. This will result
  18. * in M*N number of URIs.
  19. *
  20. * Note: This method is slow in SmartTVs and Consoles. It should only be
  21. * called when necessary.
  22. *
  23. * @param {!Array.<string>} baseUris
  24. * @param {!Array.<string>} relativeUris
  25. * @return {!Array.<string>}
  26. */
  27. static resolveUris(baseUris, relativeUris) {
  28. if (relativeUris.length == 0) {
  29. return baseUris;
  30. }
  31. if (baseUris.length == 1 && relativeUris.length == 1) {
  32. const baseUri = new goog.Uri(baseUris[0]);
  33. const relativeUri = new goog.Uri(relativeUris[0]);
  34. return [baseUri.resolve(relativeUri).toString()];
  35. }
  36. const relativeAsGoog = relativeUris.map((uri) => new goog.Uri(uri));
  37. // For each base URI, this code resolves it with every relative URI.
  38. // The result is a single array containing all the resolved URIs.
  39. const resolvedUris = [];
  40. for (const baseStr of baseUris) {
  41. const base = new goog.Uri(baseStr);
  42. for (const relative of relativeAsGoog) {
  43. resolvedUris.push(base.resolve(relative).toString());
  44. }
  45. }
  46. return resolvedUris;
  47. }
  48. /**
  49. * Creates a DrmInfo object from the given info.
  50. *
  51. * @param {string} keySystem
  52. * @param {string} encryptionScheme
  53. * @param {Array.<shaka.extern.InitDataOverride>} initData
  54. * @return {shaka.extern.DrmInfo}
  55. */
  56. static createDrmInfo(keySystem, encryptionScheme, initData) {
  57. return {
  58. keySystem,
  59. encryptionScheme,
  60. licenseServerUri: '',
  61. distinctiveIdentifierRequired: false,
  62. persistentStateRequired: false,
  63. audioRobustness: '',
  64. videoRobustness: '',
  65. serverCertificate: null,
  66. serverCertificateUri: '',
  67. sessionType: '',
  68. initData: initData || [],
  69. keyIds: new Set(),
  70. };
  71. }
  72. /**
  73. * Creates a DrmInfo object from ClearKeys.
  74. *
  75. * @param {!Map.<string, string>} clearKeys
  76. * @param {string=} encryptionScheme
  77. * @return {shaka.extern.DrmInfo}
  78. */
  79. static createDrmInfoFromClearKeys(clearKeys, encryptionScheme = 'cenc') {
  80. const StringUtils = shaka.util.StringUtils;
  81. const Uint8ArrayUtils = shaka.util.Uint8ArrayUtils;
  82. const keys = [];
  83. const keyIds = [];
  84. const originalKeyIds = [];
  85. clearKeys.forEach((key, keyId) => {
  86. let kid = keyId;
  87. if (kid.length != 22) {
  88. kid = Uint8ArrayUtils.toBase64(
  89. Uint8ArrayUtils.fromHex(keyId), false);
  90. }
  91. let k = key;
  92. if (k.length != 22) {
  93. k = Uint8ArrayUtils.toBase64(
  94. Uint8ArrayUtils.fromHex(key), false);
  95. }
  96. const keyObj = {
  97. kty: 'oct',
  98. kid: kid,
  99. k: k,
  100. };
  101. keys.push(keyObj);
  102. keyIds.push(keyObj.kid);
  103. originalKeyIds.push(keyId);
  104. });
  105. const jwkSet = {keys: keys};
  106. const license = JSON.stringify(jwkSet);
  107. // Use the keyids init data since is suggested by EME.
  108. // Suggestion: https://bit.ly/2JYcNTu
  109. // Format: https://www.w3.org/TR/eme-initdata-keyids/
  110. const initDataStr = JSON.stringify({'kids': keyIds});
  111. const initData =
  112. shaka.util.BufferUtils.toUint8(StringUtils.toUTF8(initDataStr));
  113. const initDatas = [{initData: initData, initDataType: 'keyids'}];
  114. return {
  115. keySystem: 'org.w3.clearkey',
  116. encryptionScheme,
  117. licenseServerUri: 'data:application/json;base64,' + window.btoa(license),
  118. distinctiveIdentifierRequired: false,
  119. persistentStateRequired: false,
  120. audioRobustness: '',
  121. videoRobustness: '',
  122. serverCertificate: null,
  123. serverCertificateUri: '',
  124. sessionType: '',
  125. initData: initDatas,
  126. keyIds: new Set(originalKeyIds),
  127. };
  128. }
  129. /**
  130. * Attempts to guess which codecs from the codecs list belong to a given
  131. * content type.
  132. * Assumes that at least one codec is correct, and throws if none are.
  133. *
  134. * @param {string} contentType
  135. * @param {!Array.<string>} codecs
  136. * @return {string}
  137. */
  138. static guessCodecs(contentType, codecs) {
  139. if (codecs.length == 1) {
  140. return codecs[0];
  141. }
  142. const match = shaka.util.ManifestParserUtils.guessCodecsSafe(
  143. contentType, codecs);
  144. // A failure is specifically denoted by null; an empty string represents a
  145. // valid match of no codec.
  146. if (match != null) {
  147. return match;
  148. }
  149. // Unable to guess codecs.
  150. throw new shaka.util.Error(
  151. shaka.util.Error.Severity.CRITICAL,
  152. shaka.util.Error.Category.MANIFEST,
  153. shaka.util.Error.Code.HLS_COULD_NOT_GUESS_CODECS,
  154. codecs);
  155. }
  156. /**
  157. * Attempts to guess which codecs from the codecs list belong to a given
  158. * content type. Does not assume a single codec is anything special, and does
  159. * not throw if it fails to match.
  160. *
  161. * @param {string} contentType
  162. * @param {!Array.<string>} codecs
  163. * @return {?string} or null if no match is found
  164. */
  165. static guessCodecsSafe(contentType, codecs) {
  166. const formats = shaka.util.ManifestParserUtils
  167. .CODEC_REGEXPS_BY_CONTENT_TYPE_[contentType];
  168. for (const format of formats) {
  169. for (const codec of codecs) {
  170. if (format.test(codec.trim())) {
  171. return codec.trim();
  172. }
  173. }
  174. }
  175. // Text does not require a codec string.
  176. if (contentType == shaka.util.ManifestParserUtils.ContentType.TEXT) {
  177. return '';
  178. }
  179. return null;
  180. }
  181. /**
  182. * Attempts to guess which codecs from the codecs list belong to a given
  183. * content.
  184. *
  185. * @param {string} contentType
  186. * @param {!Array.<string>} codecs
  187. * @return {!Array.<string>}
  188. */
  189. static guessAllCodecsSafe(contentType, codecs) {
  190. const allCodecs = [];
  191. const formats = shaka.util.ManifestParserUtils
  192. .CODEC_REGEXPS_BY_CONTENT_TYPE_[contentType];
  193. for (const format of formats) {
  194. for (const codec of codecs) {
  195. if (format.test(codec.trim())) {
  196. allCodecs.push(codec.trim());
  197. }
  198. }
  199. }
  200. return allCodecs;
  201. }
  202. };
  203. /**
  204. * @enum {string}
  205. */
  206. shaka.util.ManifestParserUtils.ContentType = {
  207. VIDEO: 'video',
  208. AUDIO: 'audio',
  209. TEXT: 'text',
  210. IMAGE: 'image',
  211. APPLICATION: 'application',
  212. };
  213. /**
  214. * @enum {string}
  215. */
  216. shaka.util.ManifestParserUtils.TextStreamKind = {
  217. SUBTITLE: 'subtitle',
  218. CLOSED_CAPTION: 'caption',
  219. };
  220. /**
  221. * Specifies how tolerant the player is of inaccurate segment start times and
  222. * end times within a manifest. For example, gaps or overlaps between segments
  223. * in a SegmentTimeline which are greater than or equal to this value will
  224. * result in a warning message.
  225. *
  226. * @const {number}
  227. */
  228. shaka.util.ManifestParserUtils.GAP_OVERLAP_TOLERANCE_SECONDS = 1 / 15;
  229. /**
  230. * A list of regexps to detect well-known video codecs.
  231. *
  232. * @const {!Array.<!RegExp>}
  233. * @private
  234. */
  235. shaka.util.ManifestParserUtils.VIDEO_CODEC_REGEXPS_ = [
  236. /^avc/,
  237. /^hev/,
  238. /^hvc/,
  239. /^vvc/,
  240. /^vvi/,
  241. /^vp0?[89]/,
  242. /^av01/,
  243. /^dvh/, // Dolby Vision based in HEVC
  244. /^dva/, // Dolby Vision based in AVC
  245. /^dav/, // Dolby Vision based in AV1
  246. ];
  247. /**
  248. * A list of regexps to detect well-known audio codecs.
  249. *
  250. * @const {!Array.<!RegExp>}
  251. * @private
  252. */
  253. shaka.util.ManifestParserUtils.AUDIO_CODEC_REGEXPS_ = [
  254. /^vorbis$/,
  255. /^Opus$/, // correct codec string according to RFC 6381 section 3.3
  256. /^opus$/, // some manifests wrongfully use this
  257. /^fLaC$/, // correct codec string according to RFC 6381 section 3.3
  258. /^flac$/, // some manifests wrongfully use this
  259. /^mp4a/,
  260. /^[ae]c-3$/,
  261. /^ac-4/,
  262. /^dts[cex]$/, // DTS Digital Surround (dtsc), DTS Express (dtse), DTS:X (dtsx)
  263. /^iamf/,
  264. /^mhm[12]/, // MPEG-H Audio LC
  265. ];
  266. /**
  267. * A list of regexps to detect well-known text codecs.
  268. *
  269. * @const {!Array.<!RegExp>}
  270. * @private
  271. */
  272. shaka.util.ManifestParserUtils.TEXT_CODEC_REGEXPS_ = [
  273. /^vtt$/,
  274. /^wvtt/,
  275. /^stpp/,
  276. ];
  277. /**
  278. * @const {!Object.<string, !Array.<!RegExp>>}
  279. */
  280. shaka.util.ManifestParserUtils.CODEC_REGEXPS_BY_CONTENT_TYPE_ = {
  281. 'audio': shaka.util.ManifestParserUtils.AUDIO_CODEC_REGEXPS_,
  282. 'video': shaka.util.ManifestParserUtils.VIDEO_CODEC_REGEXPS_,
  283. 'text': shaka.util.ManifestParserUtils.TEXT_CODEC_REGEXPS_,
  284. };