Source: lib/util/tXml.js

  1. /*! @license
  2. * tXml
  3. * Copyright 2015 Tobias Nickel
  4. * SPDX-License-Identifier: MIT
  5. */
  6. goog.provide('shaka.util.TXml');
  7. goog.require('shaka.util.StringUtils');
  8. goog.require('shaka.log');
  9. /**
  10. * This code is a modified version of the tXml library.
  11. *
  12. * @author: Tobias Nickel
  13. * created: 06.04.2015
  14. * https://github.com/TobiasNickel/tXml
  15. */
  16. /**
  17. * Permission is hereby granted, free of charge, to any person obtaining a copy
  18. * of this software and associated documentation files (the "Software"), to deal
  19. * in the Software without restriction, including without limitation the rights
  20. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  21. * copies of the Software, and to permit persons to whom the Software is
  22. * furnished to do so, subject to the following conditions:
  23. *
  24. * The above copyright notice and this permission notice shall be included in
  25. * all copies or substantial portions of the Software.
  26. *
  27. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  28. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  29. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  30. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  31. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  32. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  33. * SOFTWARE.
  34. */
  35. shaka.util.TXml = class {
  36. /**
  37. * Parse some data
  38. * @param {BufferSource} data
  39. * @param {string=} expectedRootElemName
  40. * @return {shaka.extern.xml.Node | null}
  41. */
  42. static parseXml(data, expectedRootElemName) {
  43. const xmlString = shaka.util.StringUtils.fromBytesAutoDetect(data);
  44. return shaka.util.TXml.parseXmlString(xmlString, expectedRootElemName);
  45. }
  46. /**
  47. * Parse some data
  48. * @param {string} xmlString
  49. * @param {string=} expectedRootElemName
  50. * @return {shaka.extern.xml.Node | null}
  51. */
  52. static parseXmlString(xmlString, expectedRootElemName) {
  53. const result = shaka.util.TXml.parse(xmlString);
  54. if (!expectedRootElemName && result.length) {
  55. return result[0];
  56. }
  57. const rootNode = result.find((n) => n.tagName === expectedRootElemName);
  58. if (rootNode) {
  59. return rootNode;
  60. }
  61. shaka.log.error('parseXml root element not found!');
  62. return null;
  63. }
  64. /**
  65. * Parse some data
  66. * @param {string} schema
  67. * @return {string}
  68. */
  69. static getKnownNameSpace(schema) {
  70. if (shaka.util.TXml.knownNameSpaces_.has(schema)) {
  71. return shaka.util.TXml.knownNameSpaces_.get(schema);
  72. }
  73. return '';
  74. }
  75. /**
  76. * Parse some data
  77. * @param {string} schema
  78. * @param {string} NS
  79. */
  80. static setKnownNameSpace(schema, NS) {
  81. shaka.util.TXml.knownNameSpaces_.set(schema, NS);
  82. }
  83. /**
  84. * parseXML / html into a DOM Object,
  85. * with no validation and some failure tolerance
  86. * @param {string} S your XML to parse
  87. * @return {Array.<shaka.extern.xml.Node>}
  88. */
  89. static parse(S) {
  90. let pos = 0;
  91. const openBracket = '<';
  92. const openBracketCC = '<'.charCodeAt(0);
  93. const closeBracket = '>';
  94. const closeBracketCC = '>'.charCodeAt(0);
  95. const minusCC = '-'.charCodeAt(0);
  96. const slashCC = '/'.charCodeAt(0);
  97. const exclamationCC = '!'.charCodeAt(0);
  98. const singleQuoteCC = '\''.charCodeAt(0);
  99. const doubleQuoteCC = '"'.charCodeAt(0);
  100. const openCornerBracketCC = '['.charCodeAt(0);
  101. /**
  102. * parsing a list of entries
  103. */
  104. function parseChildren(tagName, preserveSpace = false) {
  105. /** @type {Array.<shaka.extern.xml.Node | string>} */
  106. const children = [];
  107. while (S[pos]) {
  108. if (S.charCodeAt(pos) == openBracketCC) {
  109. if (S.charCodeAt(pos + 1) === slashCC) {
  110. const closeStart = pos + 2;
  111. pos = S.indexOf(closeBracket, pos);
  112. const closeTag = S.substring(closeStart, pos);
  113. let indexOfCloseTag = closeTag.indexOf(tagName);
  114. if (indexOfCloseTag == -1) {
  115. // handle VTT closing tags like <c.lime></c>
  116. const indexOfPeriod = tagName.indexOf('.');
  117. if (indexOfPeriod > 0) {
  118. const shortTag = tagName.substring(0, indexOfPeriod);
  119. indexOfCloseTag = closeTag.indexOf(shortTag);
  120. }
  121. }
  122. // eslint-disable-next-line no-restricted-syntax
  123. if (indexOfCloseTag == -1) {
  124. const parsedText = S.substring(0, pos).split('\n');
  125. throw new Error(
  126. 'Unexpected close tag\nLine: ' + (parsedText.length - 1) +
  127. '\nColumn: ' +
  128. (parsedText[parsedText.length - 1].length + 1) +
  129. '\nChar: ' + S[pos],
  130. );
  131. }
  132. if (pos + 1) {
  133. pos += 1;
  134. }
  135. return children;
  136. } else if (S.charCodeAt(pos + 1) === exclamationCC) {
  137. if (S.charCodeAt(pos + 2) == minusCC) {
  138. while (pos !== -1 && !(S.charCodeAt(pos) === closeBracketCC &&
  139. S.charCodeAt(pos - 1) == minusCC &&
  140. S.charCodeAt(pos - 2) == minusCC &&
  141. pos != -1)) {
  142. pos = S.indexOf(closeBracket, pos + 1);
  143. }
  144. if (pos === -1) {
  145. pos = S.length;
  146. }
  147. } else if (
  148. S.charCodeAt(pos + 2) === openCornerBracketCC &&
  149. S.charCodeAt(pos + 8) === openCornerBracketCC &&
  150. S.substr(pos + 3, 5).toLowerCase() === 'cdata'
  151. ) {
  152. // cdata
  153. const cdataEndIndex = S.indexOf(']]>', pos);
  154. if (cdataEndIndex == -1) {
  155. children.push(S.substr(pos + 9));
  156. pos = S.length;
  157. } else {
  158. children.push(S.substring(pos + 9, cdataEndIndex));
  159. pos = cdataEndIndex + 3;
  160. }
  161. continue;
  162. }
  163. pos++;
  164. continue;
  165. }
  166. const node = parseNode(preserveSpace);
  167. children.push(node);
  168. if (typeof node === 'string') {
  169. return children;
  170. }
  171. if (node.tagName[0] === '?' && node.children) {
  172. children.push(...node.children);
  173. node.children = [];
  174. }
  175. } else {
  176. const text = parseText();
  177. if (preserveSpace) {
  178. if (text.length > 0) {
  179. children.push(text);
  180. }
  181. } else if (children.length &&
  182. text.length == 1 && text[0] == '\n') {
  183. children.push(text);
  184. } else {
  185. const trimmed = text.trim();
  186. if (trimmed.length > 0) {
  187. children.push(text);
  188. }
  189. }
  190. pos++;
  191. }
  192. }
  193. return children;
  194. }
  195. /**
  196. * returns the text outside of texts until the first '<'
  197. */
  198. function parseText() {
  199. const start = pos;
  200. pos = S.indexOf(openBracket, pos) - 1;
  201. if (pos === -2) {
  202. pos = S.length;
  203. }
  204. return S.slice(start, pos + 1);
  205. }
  206. /**
  207. * returns text until the first nonAlphabetic letter
  208. */
  209. const nameSpacer = '\r\n\t>/= ';
  210. /**
  211. * Parse text in current context
  212. * @return {string}
  213. */
  214. function parseName() {
  215. const start = pos;
  216. while (nameSpacer.indexOf(S[pos]) === -1 && S[pos]) {
  217. pos++;
  218. }
  219. return S.slice(start, pos);
  220. }
  221. /**
  222. * Parse text in current context
  223. * @param {boolean} preserveSpace Preserve the space between nodes
  224. * @return {shaka.extern.xml.Node | string}
  225. */
  226. function parseNode(preserveSpace) {
  227. pos++;
  228. const tagName = parseName();
  229. const attributes = {};
  230. let children = [];
  231. // parsing attributes
  232. while (S.charCodeAt(pos) !== closeBracketCC && S[pos]) {
  233. const c = S.charCodeAt(pos);
  234. // abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
  235. if ((c > 64 && c < 91) || (c > 96 && c < 123)) {
  236. const name = parseName();
  237. // search beginning of the string
  238. let code = S.charCodeAt(pos);
  239. while (code && code !== singleQuoteCC && code !== doubleQuoteCC &&
  240. !((code > 64 && code < 91) || (code > 96 && code < 123)) &&
  241. code !== closeBracketCC) {
  242. pos++;
  243. code = S.charCodeAt(pos);
  244. }
  245. let value = parseString();
  246. if (code === singleQuoteCC || code === doubleQuoteCC) {
  247. if (pos === -1) {
  248. /** @type {shaka.extern.xml.Node} */
  249. const node = {
  250. tagName,
  251. attributes,
  252. children,
  253. parent: null,
  254. };
  255. for (let i = 0; i < children.length; i++) {
  256. if (typeof children[i] !== 'string') {
  257. children[i].parent = node;
  258. }
  259. }
  260. return node;
  261. }
  262. } else {
  263. value = null;
  264. pos--;
  265. }
  266. if (name.startsWith('xmlns:')) {
  267. const segs = name.split(':');
  268. shaka.util.TXml.setKnownNameSpace(
  269. /** @type {string} */ (value), segs[1]);
  270. }
  271. if (tagName === 'tt' &&
  272. name === 'xml:space' &&
  273. value === 'preserve') {
  274. preserveSpace = true;
  275. }
  276. attributes[name] = value;
  277. }
  278. pos++;
  279. }
  280. if (S.charCodeAt(pos - 1) !== slashCC) {
  281. pos++;
  282. const contents = parseChildren(tagName, preserveSpace);
  283. children = contents;
  284. } else {
  285. pos++;
  286. }
  287. /** @type {shaka.extern.xml.Node} */
  288. const node = {
  289. tagName,
  290. attributes,
  291. children,
  292. parent: null,
  293. };
  294. const childrenLength = children.length;
  295. for (let i = 0; i < childrenLength; i++) {
  296. const childrenValue = children[i];
  297. if (typeof childrenValue !== 'string') {
  298. childrenValue.parent = node;
  299. } else if (i == childrenLength - 1 && childrenValue == '\n') {
  300. children.pop();
  301. }
  302. }
  303. return node;
  304. }
  305. /**
  306. * Parse string in current context
  307. * @return {string}
  308. */
  309. function parseString() {
  310. const startChar = S[pos];
  311. const startpos = pos + 1;
  312. pos = S.indexOf(startChar, startpos);
  313. return S.slice(startpos, pos);
  314. }
  315. return parseChildren('');
  316. }
  317. /**
  318. * Verifies if the element is a TXml node.
  319. * @param {!shaka.extern.xml.Node} elem The XML element.
  320. * @return {!boolean} Is the element a TXml node
  321. */
  322. static isNode(elem) {
  323. return !!(elem.tagName);
  324. }
  325. /**
  326. * Checks if a node is of type text.
  327. * @param {!shaka.extern.xml.Node | string} elem The XML element.
  328. * @return {boolean} True if it is a text node.
  329. */
  330. static isText(elem) {
  331. return typeof elem === 'string';
  332. }
  333. /**
  334. * gets child XML elements.
  335. * @param {!shaka.extern.xml.Node} elem The parent XML element.
  336. * @return {!Array.<!shaka.extern.xml.Node>} The child XML elements.
  337. */
  338. static getChildNodes(elem) {
  339. const found = [];
  340. if (!elem.children) {
  341. return [];
  342. }
  343. for (const child of elem.children) {
  344. if (typeof child !== 'string') {
  345. found.push(child);
  346. }
  347. }
  348. return found;
  349. }
  350. /**
  351. * Finds child XML elements.
  352. * @param {!shaka.extern.xml.Node} elem The parent XML element.
  353. * @param {string} name The child XML element's tag name.
  354. * @return {!Array.<!shaka.extern.xml.Node>} The child XML elements.
  355. */
  356. static findChildren(elem, name) {
  357. const found = [];
  358. if (!elem.children) {
  359. return [];
  360. }
  361. for (const child of elem.children) {
  362. if (child.tagName === name) {
  363. found.push(child);
  364. }
  365. }
  366. return found;
  367. }
  368. /**
  369. * Gets inner text.
  370. * @param {!shaka.extern.xml.Node | string} node The XML element.
  371. * @return {?string} The text contents, or null if there are none.
  372. */
  373. static getTextContents(node) {
  374. const StringUtils = shaka.util.StringUtils;
  375. if (typeof node === 'string') {
  376. return StringUtils.htmlUnescape(node);
  377. }
  378. const textContent = node.children.reduce(
  379. (acc, curr) => (typeof curr === 'string' ? acc + curr : acc),
  380. '',
  381. );
  382. if (textContent === '') {
  383. return null;
  384. }
  385. return StringUtils.htmlUnescape(textContent);
  386. }
  387. /**
  388. * Gets the text contents of a node.
  389. * @param {!shaka.extern.xml.Node} node The XML element.
  390. * @return {?string} The text contents, or null if there are none.
  391. */
  392. static getContents(node) {
  393. if (!Array.from(node.children).every(
  394. (n) => typeof n === 'string' )) {
  395. return null;
  396. }
  397. // Read merged text content from all text nodes.
  398. let text = shaka.util.TXml.getTextContents(node);
  399. if (text) {
  400. text = text.trim();
  401. }
  402. return text;
  403. }
  404. /**
  405. * Finds child XML elements recursively.
  406. * @param {!shaka.extern.xml.Node} elem The parent XML element.
  407. * @param {string} name The child XML element's tag name.
  408. * @param {!Array.<!shaka.extern.xml.Node>} found accumulator for found nodes
  409. * @return {!Array.<!shaka.extern.xml.Node>} The child XML elements.
  410. */
  411. static getElementsByTagName(elem, name, found = []) {
  412. if (elem.tagName === name) {
  413. found.push(elem);
  414. }
  415. if (elem.children) {
  416. for (const child of elem.children) {
  417. shaka.util.TXml.getElementsByTagName(child, name, found);
  418. }
  419. }
  420. return found;
  421. }
  422. /**
  423. * Finds a child XML element.
  424. * @param {!shaka.extern.xml.Node} elem The parent XML element.
  425. * @param {string} name The child XML element's tag name.
  426. * @return {shaka.extern.xml.Node | null} The child XML element,
  427. * or null if a child XML element
  428. * does not exist with the given tag name OR if there exists more than one
  429. * child XML element with the given tag name.
  430. */
  431. static findChild(elem, name) {
  432. const children = shaka.util.TXml.findChildren(elem, name);
  433. if (children.length != 1) {
  434. return null;
  435. }
  436. return children[0];
  437. }
  438. /**
  439. * Finds a namespace-qualified child XML element.
  440. * @param {!shaka.extern.xml.Node} elem The parent XML element.
  441. * @param {string} ns The child XML element's namespace URI.
  442. * @param {string} name The child XML element's local name.
  443. * @return {shaka.extern.xml.Node | null} The child XML element, or null
  444. * if a child XML element
  445. * does not exist with the given tag name OR if there exists more than one
  446. * child XML element with the given tag name.
  447. */
  448. static findChildNS(elem, ns, name) {
  449. const children = shaka.util.TXml.findChildrenNS(elem, ns, name);
  450. if (children.length != 1) {
  451. return null;
  452. }
  453. return children[0];
  454. }
  455. /**
  456. * Parses an attribute by its name.
  457. * @param {!shaka.extern.xml.Node} elem The XML element.
  458. * @param {string} name The attribute name.
  459. * @param {function(string): (T|null)} parseFunction A function that parses
  460. * the attribute.
  461. * @param {(T|null)=} defaultValue The attribute's default value, if not
  462. * specified, the attibute's default value is null.
  463. * @return {(T|null)} The parsed attribute on success, or the attribute's
  464. * default value if the attribute does not exist or could not be parsed.
  465. * @template T
  466. */
  467. static parseAttr(elem, name, parseFunction, defaultValue = null) {
  468. let parsedValue = null;
  469. const value = elem.attributes[name];
  470. if (value != null) {
  471. parsedValue = parseFunction(value);
  472. }
  473. return parsedValue == null ? defaultValue : parsedValue;
  474. }
  475. /**
  476. * Gets a namespace-qualified attribute.
  477. * @param {!shaka.extern.xml.Node} elem The element to get from.
  478. * @param {string} ns The namespace URI.
  479. * @param {string} name The local name of the attribute.
  480. * @return {?string} The attribute's value, or null if not present.
  481. */
  482. static getAttributeNS(elem, ns, name) {
  483. const schemaNS = shaka.util.TXml.getKnownNameSpace(ns);
  484. // Think this is equivalent
  485. const attribute = elem.attributes[`${schemaNS}:${name}`];
  486. return attribute || null;
  487. }
  488. /**
  489. * Finds namespace-qualified child XML elements.
  490. * @param {!shaka.extern.xml.Node} elem The parent XML element.
  491. * @param {string} ns The child XML element's namespace URI.
  492. * @param {string} name The child XML element's local name.
  493. * @return {!Array.<!shaka.extern.xml.Node>} The child XML elements.
  494. */
  495. static findChildrenNS(elem, ns, name) {
  496. const schemaNS = shaka.util.TXml.getKnownNameSpace(ns);
  497. const found = [];
  498. if (elem.children) {
  499. for (const child of elem.children) {
  500. if (child && child.tagName === `${schemaNS}:${name}`) {
  501. found.push(child);
  502. }
  503. }
  504. }
  505. return found;
  506. }
  507. /**
  508. * Gets a namespace-qualified attribute.
  509. * @param {!shaka.extern.xml.Node} elem The element to get from.
  510. * @param {!Array.<string>} nsList The lis of namespace URIs.
  511. * @param {string} name The local name of the attribute.
  512. * @return {?string} The attribute's value, or null if not present.
  513. */
  514. static getAttributeNSList(elem, nsList, name) {
  515. for (const ns of nsList) {
  516. const attr = shaka.util.TXml.getAttributeNS(
  517. elem, ns, name,
  518. );
  519. if (attr) {
  520. return attr;
  521. }
  522. }
  523. return null;
  524. }
  525. /**
  526. * Parses an XML date string.
  527. * @param {string} dateString
  528. * @return {?number} The parsed date in seconds on success; otherwise, return
  529. * null.
  530. */
  531. static parseDate(dateString) {
  532. if (!dateString) {
  533. return null;
  534. }
  535. // Times in the manifest should be in UTC. If they don't specify a timezone,
  536. // Date.parse() will use the local timezone instead of UTC. So manually add
  537. // the timezone if missing ('Z' indicates the UTC timezone).
  538. // Format: YYYY-MM-DDThh:mm:ss.ssssss
  539. if (/^\d+-\d+-\d+T\d+:\d+:\d+(\.\d+)?$/.test(dateString)) {
  540. dateString += 'Z';
  541. }
  542. const result = Date.parse(dateString);
  543. return isNaN(result) ? null : (result / 1000.0);
  544. }
  545. /**
  546. * Parses an XML duration string.
  547. * Negative values are not supported. Years and months are treated as exactly
  548. * 365 and 30 days respectively.
  549. * @param {string} durationString The duration string, e.g., "PT1H3M43.2S",
  550. * which means 1 hour, 3 minutes, and 43.2 seconds.
  551. * @return {?number} The parsed duration in seconds on success; otherwise,
  552. * return null.
  553. * @see {@link http://www.datypic.com/sc/xsd/t-xsd_duration.html}
  554. */
  555. static parseDuration(durationString) {
  556. if (!durationString) {
  557. return null;
  558. }
  559. const re = '^P(?:([0-9]*)Y)?(?:([0-9]*)M)?(?:([0-9]*)D)?' +
  560. '(?:T(?:([0-9]*)H)?(?:([0-9]*)M)?(?:([0-9.]*)S)?)?$';
  561. const matches = new RegExp(re).exec(durationString);
  562. if (!matches) {
  563. shaka.log.warning('Invalid duration string:', durationString);
  564. return null;
  565. }
  566. // Note: Number(null) == 0 but Number(undefined) == NaN.
  567. const years = Number(matches[1] || null);
  568. const months = Number(matches[2] || null);
  569. const days = Number(matches[3] || null);
  570. const hours = Number(matches[4] || null);
  571. const minutes = Number(matches[5] || null);
  572. const seconds = Number(matches[6] || null);
  573. // Assume a year always has 365 days and a month always has 30 days.
  574. const d = (60 * 60 * 24 * 365) * years +
  575. (60 * 60 * 24 * 30) * months +
  576. (60 * 60 * 24) * days +
  577. (60 * 60) * hours +
  578. 60 * minutes +
  579. seconds;
  580. return isFinite(d) ? d : null;
  581. }
  582. /**
  583. * Parses a range string.
  584. * @param {string} rangeString The range string, e.g., "101-9213".
  585. * @return {?{start: number, end: number}} The parsed range on success;
  586. * otherwise, return null.
  587. */
  588. static parseRange(rangeString) {
  589. const matches = /([0-9]+)-([0-9]+)/.exec(rangeString);
  590. if (!matches) {
  591. return null;
  592. }
  593. const start = Number(matches[1]);
  594. if (!isFinite(start)) {
  595. return null;
  596. }
  597. const end = Number(matches[2]);
  598. if (!isFinite(end)) {
  599. return null;
  600. }
  601. return {start: start, end: end};
  602. }
  603. /**
  604. * Parses an integer.
  605. * @param {string} intString The integer string.
  606. * @return {?number} The parsed integer on success; otherwise, return null.
  607. */
  608. static parseInt(intString) {
  609. const n = Number(intString);
  610. return (n % 1 === 0) ? n : null;
  611. }
  612. /**
  613. * Parses a positive integer.
  614. * @param {string} intString The integer string.
  615. * @return {?number} The parsed positive integer on success; otherwise,
  616. * return null.
  617. */
  618. static parsePositiveInt(intString) {
  619. const n = Number(intString);
  620. return (n % 1 === 0) && (n > 0) ? n : null;
  621. }
  622. /**
  623. * Parses a non-negative integer.
  624. * @param {string} intString The integer string.
  625. * @return {?number} The parsed non-negative integer on success; otherwise,
  626. * return null.
  627. */
  628. static parseNonNegativeInt(intString) {
  629. const n = Number(intString);
  630. return (n % 1 === 0) && (n >= 0) ? n : null;
  631. }
  632. /**
  633. * Parses a floating point number.
  634. * @param {string} floatString The floating point number string.
  635. * @return {?number} The parsed floating point number on success; otherwise,
  636. * return null. May return -Infinity or Infinity.
  637. */
  638. static parseFloat(floatString) {
  639. const n = Number(floatString);
  640. return !isNaN(n) ? n : null;
  641. }
  642. /**
  643. * Parses a boolean.
  644. * @param {string} booleanString The boolean string.
  645. * @return {boolean} The boolean
  646. */
  647. static parseBoolean(booleanString) {
  648. if (!booleanString) {
  649. return false;
  650. }
  651. return booleanString.toLowerCase() === 'true';
  652. }
  653. /**
  654. * Evaluate a division expressed as a string.
  655. * @param {string} exprString
  656. * The expression to evaluate, e.g. "200/2". Can also be a single number.
  657. * @return {?number} The evaluated expression as floating point number on
  658. * success; otherwise return null.
  659. */
  660. static evalDivision(exprString) {
  661. let res;
  662. let n;
  663. if ((res = exprString.match(/^(\d+)\/(\d+)$/))) {
  664. n = Number(res[1]) / Number(res[2]);
  665. } else {
  666. n = Number(exprString);
  667. }
  668. return !isNaN(n) ? n : null;
  669. }
  670. /**
  671. * Parse xPath strings for segments and id targets.
  672. * @param {string} exprString
  673. * @return {!Array<!shaka.util.TXml.PathNode>}
  674. */
  675. static parseXpath(exprString) {
  676. const StringUtils = shaka.util.StringUtils;
  677. const returnPaths = [];
  678. // Split string by paths but ignore '/' in quotes
  679. const paths = StringUtils.htmlUnescape(exprString)
  680. .split(/\/+(?=(?:[^'"]*['"][^'"]*['"])*[^'"]*$)/);
  681. for (const path of paths) {
  682. const nodeName = path.match(/^([\w]+)/);
  683. if (nodeName) {
  684. // We only want the id attribute in which case
  685. // /'(.*?)'/ will suffice to get it.
  686. const idAttr = path.match(/(@id='(.*?)')/);
  687. const position = path.match(/\[(\d+)\]/);
  688. returnPaths.push({
  689. name: nodeName[0],
  690. id: idAttr ?
  691. idAttr[0].match(/'(.*?)'/)[0].replace(/'/gm, '') : null,
  692. // position is counted from 1, so make it readable for devs
  693. position: position ? Number(position[1]) - 1 : null,
  694. attribute: path.split('/@')[1] || null,
  695. });
  696. } else if (path.startsWith('@') && returnPaths.length) {
  697. returnPaths[returnPaths.length - 1].attribute = path.slice(1);
  698. }
  699. }
  700. return returnPaths;
  701. }
  702. /**
  703. * Modifies nodes in specified array by adding or removing nodes
  704. * and updating attributes.
  705. * @param {!Array<shaka.extern.xml.Node>} nodes
  706. * @param {!shaka.extern.xml.Node} patchNode
  707. */
  708. static modifyNodes(nodes, patchNode) {
  709. const TXml = shaka.util.TXml;
  710. const paths = TXml.parseXpath(patchNode.attributes['sel'] || '');
  711. if (!paths.length) {
  712. return;
  713. }
  714. const lastNode = paths[paths.length - 1];
  715. const position = patchNode.attributes['pos'] || null;
  716. let index = lastNode.position;
  717. if (index === null) {
  718. index = position === 'prepend' ? 0 : nodes.length;
  719. } else if (position === 'prepend') {
  720. --index;
  721. } else if (position === 'after') {
  722. ++index;
  723. }
  724. const action = patchNode.tagName;
  725. const attribute = lastNode.attribute;
  726. // Modify attribute
  727. if (attribute) {
  728. TXml.modifyNodeAttribute(nodes[index], action, attribute,
  729. TXml.getContents(patchNode) || '');
  730. // Rearrange nodes
  731. } else {
  732. if (action === 'remove' || action === 'replace') {
  733. nodes.splice(index, 1);
  734. }
  735. if (action === 'add' || action === 'replace') {
  736. const newNodes = patchNode.children;
  737. nodes.splice(index, 0, ...newNodes);
  738. }
  739. }
  740. }
  741. /**
  742. * @param {!shaka.extern.xml.Node} node
  743. * @param {string} action
  744. * @param {string} attribute
  745. * @param {string} value
  746. */
  747. static modifyNodeAttribute(node, action, attribute, value) {
  748. if (action === 'remove') {
  749. delete node.attributes[attribute];
  750. } else if (action === 'add' || action === 'replace') {
  751. node.attributes[attribute] = value;
  752. }
  753. }
  754. /**
  755. * Converts a tXml node to DOM element.
  756. * @param {shaka.extern.xml.Node} node
  757. * @param {boolean=} doParents
  758. * @param {boolean=} doChildren
  759. * @return {!Element}
  760. */
  761. static txmlNodeToDomElement(node, doParents = true, doChildren = true) {
  762. const TXml = shaka.util.TXml;
  763. const element = document.createElement(node.tagName);
  764. for (const k in node.attributes) {
  765. const v = node.attributes[k];
  766. element.setAttribute(k, v);
  767. }
  768. if (doParents && node.parent && node.parent.tagName != '?xml') {
  769. const parentElement = TXml.txmlNodeToDomElement(
  770. node.parent, /* doParents= */ true, /* doChildren= */ false);
  771. parentElement.appendChild(element);
  772. }
  773. if (doChildren) {
  774. for (const child of node.children) {
  775. let childElement;
  776. if (typeof child == 'string') {
  777. childElement = new Text(child);
  778. } else {
  779. childElement = TXml.txmlNodeToDomElement(
  780. child, /* doParents= */ false, /* doChildren= */ true);
  781. }
  782. element.appendChild(childElement);
  783. }
  784. }
  785. return element;
  786. }
  787. };
  788. shaka.util.TXml.knownNameSpaces_ = new Map([]);
  789. /**
  790. * @typedef {{
  791. * name: string,
  792. * id: ?string,
  793. * position: ?number,
  794. * attribute: ?string
  795. * }}
  796. */
  797. shaka.util.TXml.PathNode;