Source: utils.js

  1. /**
  2. * #### Import members from **@edx/frontend-platform**
  3. *
  4. * @module Utilities
  5. */
  6. import camelCase from 'lodash.camelcase';
  7. import snakeCase from 'lodash.snakecase';
  8. /**
  9. * This is the underlying function used by camelCaseObject, snakeCaseObject, and convertKeyNames
  10. * above.
  11. *
  12. * Given an object (or array) and a modification function, will perform the function on each key it
  13. * encounters on the object and its tree of children.
  14. *
  15. * The modification function must take a string as an argument and returns a string.
  16. *
  17. * Example:
  18. *
  19. * ```
  20. * (key) => {
  21. * if (key === 'edX') {
  22. * return 'Open edX';
  23. * }
  24. * return key;
  25. * }
  26. * ```
  27. *
  28. * This function will turn any key that matches 'edX' into 'Open edX'. All other keys will be
  29. * passed through unmodified.
  30. *
  31. * Can accept arrays as well as objects, and will perform its conversion on any objects it finds in
  32. * the array.
  33. *
  34. * @param {Object} object
  35. * @param {function} modify
  36. * @returns {Object}
  37. */
  38. export function modifyObjectKeys(object, modify) {
  39. // If the passed in object is not an Object, return it.
  40. if (
  41. object === undefined
  42. || object === null
  43. || (typeof object !== 'object' && !Array.isArray(object))
  44. ) {
  45. return object;
  46. }
  47. if (Array.isArray(object)) {
  48. return object.map(value => modifyObjectKeys(value, modify));
  49. }
  50. // Otherwise, process all its keys.
  51. const result = {};
  52. Object.entries(object).forEach(([key, value]) => {
  53. result[modify(key)] = modifyObjectKeys(value, modify);
  54. });
  55. return result;
  56. }
  57. /**
  58. * Performs a deep conversion to camelCase on all keys in the provided object and its tree of
  59. * children. Uses [lodash.camelcase](https://lodash.com/docs/4.17.15#camelCase) on each key. This
  60. * is commonly used to convert snake_case keys in models from a backend server into camelCase keys
  61. * for use in the JavaScript client.
  62. *
  63. * Can accept arrays as well as objects, and will perform its conversion on any objects it finds in
  64. * the array.
  65. *
  66. * @param {Array|Object} object
  67. * @returns {Array|Object}
  68. */
  69. export function camelCaseObject(object) {
  70. return modifyObjectKeys(object, camelCase);
  71. }
  72. /**
  73. * Performs a deep conversion to snake_case on all keys in the provided object and its tree of
  74. * children. Uses [lodash.snakecase](https://lodash.com/docs/4.17.15#snakeCase) on each key. This
  75. * is commonly used to convert camelCase keys from the JavaScript app into snake_case keys expected
  76. * by backend servers.
  77. *
  78. * Can accept arrays as well as objects, and will perform its conversion on any objects it finds in
  79. * the array.
  80. *
  81. * @param {Array|Object} object
  82. * @returns {Array|Object}
  83. */
  84. export function snakeCaseObject(object) {
  85. return modifyObjectKeys(object, snakeCase);
  86. }
  87. /**
  88. * Given a map of key-value pairs, performs a deep conversion key names in the specified object
  89. * _from_ the key _to_ the value. This is useful for updating names in an API request to the names
  90. * used throughout a client application if they happen to differ. It can also be used in the
  91. * reverse - formatting names from the client application to names expected by an API.
  92. *
  93. * ```
  94. * import { convertKeyNames } from '@edx/frontend-base';
  95. *
  96. * // This object can be of any shape or depth with subobjects/arrays.
  97. * const myObject = {
  98. * myKey: 'my value',
  99. * }
  100. *
  101. * const result = convertKeyNames(myObject, { myKey: 'their_key' });
  102. *
  103. * console.log(result) // { their_key: 'my value' }
  104. * ```
  105. *
  106. * Can accept arrays as well as objects, and will perform its conversion on any objects it finds in
  107. * the array.
  108. *
  109. * @param {Array|Object} object
  110. * @param {Object} nameMap
  111. * @returns {Array|Object}
  112. */
  113. export function convertKeyNames(object, nameMap) {
  114. const transformer = key => (nameMap[key] === undefined ? key : nameMap[key]);
  115. return modifyObjectKeys(object, transformer);
  116. }
  117. /**
  118. * Given a string URL return an element that has been parsed via href.
  119. * This element has the possibility to return different part of the URL.
  120. parser.protocol; // => "http:"
  121. parser.hostname; // => "example.com"
  122. parser.port; // => "3000"
  123. parser.pathname; // => "/pathname/"
  124. parser.search; // => "?search=test"
  125. parser.hash; // => "#hash"
  126. parser.host; // => "example.com:3000"
  127. * https://gist.github.com/jlong/2428561
  128. *
  129. * @param {string}
  130. * @returns {Object}
  131. */
  132. export function parseURL(url) {
  133. if (typeof document !== 'undefined') {
  134. const parser = document.createElement('a');
  135. parser.href = url;
  136. return parser;
  137. }
  138. return {};
  139. }
  140. /**
  141. * Given a string URL return the path of the URL
  142. *
  143. *
  144. * @param {string}
  145. * @returns {string}
  146. */
  147. export function getPath(url) {
  148. return typeof document !== 'undefined' ? parseURL(url)?.pathname : '';
  149. }
  150. /**
  151. * *Deprecated*: A method which converts the supplied query string into an object of
  152. * key-value pairs and returns it. Defaults to the current query string - should perform like
  153. * [window.searchParams](https://developer.mozilla.org/en-US/docs/Web/API/URL/searchParams)
  154. *
  155. * @deprecated
  156. * @param {string} [search=global.location.search]
  157. * @returns {Object}
  158. */
  159. export function getQueryParameters(search = global.location.search) {
  160. const keyValueFragments = search
  161. .slice(search.indexOf('?') + 1)
  162. .split('&')
  163. .filter(hash => hash !== '');
  164. return keyValueFragments.reduce((params, keyValueFragment) => {
  165. const split = keyValueFragment.indexOf('=');
  166. const key = keyValueFragment.slice(0, split);
  167. const value = keyValueFragment.slice(split + 1);
  168. return Object.assign(params, { [key]: decodeURIComponent(value) });
  169. }, {});
  170. }
  171. /**
  172. * This function helps catch a certain class of misconfiguration in which configuration variables
  173. * are not properly defined and/or supplied to a consumer that requires them. Any key that exists
  174. * is still set to "undefined" indicates a misconfiguration further up in the application, and
  175. * should be flagged as an error, and is logged to 'warn'.
  176. *
  177. * Keys that are intended to be falsy should be defined using null, 0, false, etc.
  178. *
  179. * @param {Object} object
  180. * @param {string} requester A human-readable identifier for the code which called this function.
  181. * Used when throwing errors to aid in debugging.
  182. */
  183. export function ensureDefinedConfig(object, requester) {
  184. Object.keys(object).forEach((key) => {
  185. if (object[key] === undefined) {
  186. // eslint-disable-next-line no-console
  187. console.warn(`Module configuration error: ${key} is required by ${requester}.`);
  188. }
  189. });
  190. }