Home Intro Source Mebo GitHub

src/Utils/HierarchicalCollection.js

  1. const assert = require('assert');
  2. const TypeCheck = require('js-typecheck');
  3. const deepMerge = require('./deepMerge');
  4.  
  5. // symbols used for private members to avoid any potential clashing
  6. // caused by re-implementations
  7. const _collection = Symbol('collection');
  8.  
  9.  
  10. /**
  11. * Provides an interface to deal with data in a multi dimension plain object.
  12. *
  13. * It automatically creates the intermediate levels when assigning a value under
  14. * nested levels to the collection. Also, if the value already exists
  15. * under the collection then it allows the value to be merged with the
  16. * existing values.
  17. * @private
  18. */
  19. class HierarchicalCollection{
  20.  
  21. /**
  22. * Creates a new instance
  23. */
  24. constructor(){
  25. this[_collection] = {};
  26. }
  27.  
  28. /**
  29. * Inserts a new value to the collection
  30. *
  31. * @param {string} path - path about where the value should be stored (the levels
  32. * must be separated by '.').
  33. * @param {*} value - value that is going to be stored under the collection
  34. * @param {boolean} [merge=true] - this option is used to decide in case of the
  35. * last level is already existing under the collection, if the value should be
  36. * either merged (default) or overridden.
  37. */
  38. insert(path, value, merge=true){
  39. assert(TypeCheck.isString(path), 'path needs to be defined as string');
  40. assert(path.length, 'path cannot be empty');
  41. assert((/^([\w_\.\-])+$/gi).test(path), `Illegal path name: ${path}`); // eslint-disable-line no-useless-escape
  42.  
  43. let currentLevel = this[_collection];
  44. let finalLevel = path;
  45.  
  46. // building the intermediate levels if necessary
  47. if (path.indexOf('.') !== -1){
  48. const levels = path.split('.');
  49. for (const level of levels.slice(0, -1)){
  50. if (!(level in currentLevel)){
  51. currentLevel[level] = {};
  52. }
  53.  
  54. currentLevel = currentLevel[level];
  55. }
  56. finalLevel = levels.slice(-1)[0];
  57. }
  58.  
  59. // assigning value
  60. if (merge && TypeCheck.isPlainObject(value) && finalLevel in currentLevel){
  61. const merged = deepMerge(currentLevel[finalLevel], value);
  62. Object.assign(currentLevel[finalLevel], merged);
  63. }
  64. else{
  65. currentLevel[finalLevel] = value;
  66. }
  67. }
  68.  
  69. /**
  70. * Returns a value under the collection
  71. *
  72. * @param {string} path - path about where the value is localized (the levels
  73. * must be separated by '.'). In case of empty string the entire collection
  74. * is returned.
  75. * @param {*} [defaultValue] - default value returned in case a value was
  76. * not found for the path
  77. * @return {*}
  78. */
  79. query(path, defaultValue=undefined){
  80. assert(TypeCheck.isString(path), 'path needs to be defined as string');
  81.  
  82. let currentLevel = this[_collection];
  83.  
  84. // returning the entire collection
  85. if (!path.length){
  86. return currentLevel;
  87. }
  88.  
  89. // no intermediate levels
  90. if (path.indexOf('.') === -1){
  91. if (!(path in currentLevel)){
  92. return defaultValue;
  93. }
  94. return currentLevel[path];
  95. }
  96.  
  97. // otherwise find the value going through the intermediate levels
  98. const levels = path.split('.');
  99. for (const level of levels){
  100. if (!(level in currentLevel)){
  101. return defaultValue;
  102. }
  103.  
  104. currentLevel = currentLevel[level];
  105. }
  106.  
  107. return currentLevel;
  108. }
  109.  
  110. /**
  111. * Returns a list of the root levels
  112. *
  113. * @return {Array<string>}
  114. */
  115. root(){
  116. return Object.keys(this[_collection]);
  117. }
  118. }
  119.  
  120. module.exports = HierarchicalCollection;