Source: lib/util/config_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.ConfigUtils');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.log');
  9. goog.require('shaka.util.ObjectUtils');
  10. /** @export */
  11. shaka.util.ConfigUtils = class {
  12. /**
  13. * @param {!Object} destination
  14. * @param {!Object} source
  15. * @param {!Object} template supplies default values
  16. * @param {!Object} overrides
  17. * Supplies override type checking. When the current path matches
  18. * the key in this object, each sub-value must match the type in this
  19. * object. If this contains an Object, it is used as the template.
  20. * @param {string} path to this part of the config
  21. * @return {boolean}
  22. * @export
  23. */
  24. static mergeConfigObjects(destination, source, template, overrides, path) {
  25. goog.asserts.assert(destination, 'Destination config must not be null!');
  26. /**
  27. * @type {boolean}
  28. * If true, don't validate the keys in the next level.
  29. */
  30. const ignoreKeys = path in overrides;
  31. let isValid = true;
  32. for (const k in source) {
  33. const subPath = path + '.' + k;
  34. const subTemplate = ignoreKeys ? overrides[path] : template[k];
  35. // The order of these checks is important.
  36. if (!ignoreKeys && !(k in template)) {
  37. shaka.log.alwaysError('Invalid config, unrecognized key ' + subPath);
  38. isValid = false;
  39. } else if (source[k] === undefined) {
  40. // An explicit 'undefined' value causes the key to be deleted from the
  41. // destination config and replaced with a default from the template if
  42. // possible.
  43. if (subTemplate === undefined || ignoreKeys) {
  44. // There is nothing in the template, so delete.
  45. delete destination[k];
  46. } else {
  47. // There is something in the template, so go back to that.
  48. destination[k] = shaka.util.ObjectUtils.cloneObject(subTemplate);
  49. }
  50. } else if (subTemplate.constructor == Object &&
  51. source[k] &&
  52. source[k].constructor == Object) {
  53. // These are plain Objects with no other constructor.
  54. if (!destination[k]) {
  55. // Initialize the destination with the template so that normal
  56. // merging and type-checking can happen.
  57. destination[k] = shaka.util.ObjectUtils.cloneObject(subTemplate);
  58. }
  59. const subMergeValid = shaka.util.ConfigUtils.mergeConfigObjects(
  60. destination[k], source[k], subTemplate, overrides, subPath);
  61. isValid = isValid && subMergeValid;
  62. } else if (typeof source[k] != typeof subTemplate ||
  63. source[k] == null ||
  64. // Function cosntructors are not informative, and differ
  65. // between sync and async functions. So don't look at
  66. // constructor for function types.
  67. (typeof source[k] != 'function' &&
  68. source[k].constructor != subTemplate.constructor)) {
  69. // The source is the wrong type. This check allows objects to be
  70. // nulled, but does not allow null for any non-object fields.
  71. shaka.log.alwaysError('Invalid config, wrong type for ' + subPath);
  72. isValid = false;
  73. } else if (typeof template[k] == 'function' &&
  74. template[k].length != source[k].length) {
  75. shaka.log.alwaysWarn(
  76. 'Unexpected number of arguments for ' + subPath);
  77. destination[k] = source[k];
  78. } else {
  79. destination[k] = source[k];
  80. }
  81. }
  82. return isValid;
  83. }
  84. /**
  85. * Convert config from ('fieldName', value) format to a partial config object.
  86. *
  87. * E. g. from ('manifest.retryParameters.maxAttempts', 1) to
  88. * { manifest: { retryParameters: { maxAttempts: 1 }}}.
  89. *
  90. * @param {string} fieldName
  91. * @param {*} value
  92. * @return {!Object}
  93. * @export
  94. */
  95. static convertToConfigObject(fieldName, value) {
  96. const configObject = {};
  97. let last = configObject;
  98. let searchIndex = 0;
  99. let nameStart = 0;
  100. while (true) { // eslint-disable-line no-constant-condition
  101. const idx = fieldName.indexOf('.', searchIndex);
  102. if (idx < 0) {
  103. break;
  104. }
  105. if (idx == 0 || fieldName[idx - 1] != '\\') {
  106. const part = fieldName.substring(nameStart, idx).replace(/\\\./g, '.');
  107. last[part] = {};
  108. last = last[part];
  109. nameStart = idx + 1;
  110. }
  111. searchIndex = idx + 1;
  112. }
  113. last[fieldName.substring(nameStart).replace(/\\\./g, '.')] = value;
  114. return configObject;
  115. }
  116. /**
  117. * Reference the input parameters so the compiler doesn't remove them from
  118. * the calling function. Return whatever value is specified.
  119. *
  120. * This allows an empty or default implementation of a config callback that
  121. * still bears the complete function signature even in compiled mode.
  122. *
  123. * The caller should look something like this:
  124. *
  125. * const callback = (a, b, c, d) => {
  126. * return referenceParametersAndReturn(
  127. [a, b, c, d],
  128. a); // Can be anything, doesn't need to be one of the parameters
  129. * };
  130. *
  131. * @param {!Array.<?>} parameters
  132. * @param {T} returnValue
  133. * @return {T}
  134. * @template T
  135. * @noinline
  136. */
  137. static referenceParametersAndReturn(parameters, returnValue) {
  138. return parameters && returnValue;
  139. }
  140. };