Reference Source

js/base/collection_base.js

  1. // NPM IMPORTS
  2. import assert from 'assert'
  3. import _ from 'lodash'
  4.  
  5. // COMMON IMPORTS
  6. import T from '../utils/types'
  7. import Errorable from './errorable'
  8. import Instance from './instance'
  9.  
  10.  
  11. /**
  12. * Contextual constant for this file logs.
  13. * @private
  14. */
  15. const context = 'common/base/collection_base'
  16.  
  17.  
  18.  
  19. /**
  20. * Base class for all collections classes.
  21. * @abstract
  22. *
  23. * @author Luc BORIES
  24. * @license Apache-2.0
  25. *
  26. * @example
  27. API:
  28. * ->set_all(arg_items):nothing - Set all collection items.
  29. * ->get_all(arg_types):array - Get all collection items or filter items with given type.
  30. * ->get_all_names(arg_types):array - Get all items names with or without a filter on items types.
  31. * ->get_all_ids():array - Get all items ids with or without a filter on items types.
  32. *
  33. * ->item(arg_name):Instance - Get an item by its name.
  34. *
  35. * ->get_count():number - Get all items count.
  36. * ->get_first():object|undefined - Get first item.
  37. * ->get_last():object|undefined - Get last item.
  38. *
  39. * ->add(arg_item):nothing - Add an item to the collection.
  40. * ->add_first(arg_item):nothing - Add an item to the collection at the first position.
  41. * ->remove(arg_item):nothing - Remove an item from the collection.
  42. * ->has(arg_item):boolean - Test if an item is inside the collection.
  43. *
  44. * ->find_by_name(arg_name):Instance|undefined - Find an item by its name into the collection.
  45. * ->find_by_id(arg_id):Instance|undefined - Find an item by its id into the collection.
  46. * ->find_by_attr(arg_attr_name, arg_attr_value):Instance|undefined - Find an item by one of its attributes into the collection.
  47. * ->find_by_filter(arg_filter_function):Instance|undefined - Find an item by a filter function.
  48. *
  49. * ->filter_by_attr(arg_attr_name, arg_attr_value):array - Filter items by one of theirs attributes into the collection.
  50. * ->filter_by_filter(arg_filter_function):array - Filter items by a filter function.
  51. *
  52. * ->get_accepted_types():array - Get all collection accepted types.
  53. * ->set_accepted_types(arg_types):nothing - Set all collection accepted types.
  54. * ->add_accepted_type(arg_type):nothing - Add one collection accepted type.
  55. * ->has_accepted_type(arg_type):boolean - Test if collection has given accepted type.
  56. *
  57. * ->forEach(arg_cb):nothing - forEach wrapper on ordered items.
  58. */
  59. export default class CollectionBase extends Errorable
  60. {
  61. /**
  62. * Create a collection of Instance objects.
  63. *
  64. * @returns {nothing}
  65. */
  66. constructor()
  67. {
  68. super(context, undefined)
  69.  
  70. /**
  71. * Class type flag.
  72. * @type {boolean}
  73. */
  74. this.is_collection_base = true
  75.  
  76. /**
  77. * Items array.
  78. * @type {array}
  79. */
  80. this._items_array = []
  81.  
  82. /**
  83. * Items names map.
  84. * @type {object}
  85. */
  86. this._items_by_name = {}
  87.  
  88. /**
  89. * Items ids map.
  90. * @type {object}
  91. */
  92. this._items_by_id = {}
  93.  
  94. /**
  95. * Accepted types array.
  96. * @type {array}
  97. */
  98. this._accepted_types = ['*']
  99. }
  100.  
  101.  
  102.  
  103. /**
  104. * Format string dump.
  105. *
  106. * @returns{string}
  107. */
  108. toString()
  109. {
  110. let str = '['
  111. _.forEach(this._items_array, (item, index)=>(index > 0 ? ',' : '') + item.get_name() )
  112. return str + ']'
  113. }
  114. /**
  115. * Set all collection items.
  116. *
  117. * @param {Instance|array} arg_items - collection items: one or many Instance objects.
  118. *
  119. * @returns {nothing}
  120. */
  121. set_all(arg_items)
  122. {
  123. // DEBUG
  124. let str = '['
  125. _.forEach(arg_items, (item, index)=> str += (index > 0 ? ',' : '') + (item.get_name ? item.get_name() : 'bad item of type ' + (typeof item) ) )
  126. str += ']'
  127. console.log('set_all', str, typeof arg_items)
  128.  
  129. // RESET STORES
  130. this._items_array = []
  131. this._items_by_name = {}
  132. this._items_by_id = {}
  133.  
  134. // ONE INSTANCE IS GIVEN
  135. if ( T.isObject(arg_items) && arg_items instanceof Instance )
  136. {
  137. this._add(arg_items)
  138. return
  139. }
  140. // AN OBJECT OR AN ARRAY IS GIVEN
  141. if ( T.isObject(arg_items) || T.isArray(arg_items) )
  142. {
  143. _.forEach(arg_items,
  144. (item)=>{
  145. if ( T.isObject(item) && item instanceof Instance )
  146. {
  147. this._add(item)
  148. }
  149. }
  150. )
  151. return
  152. }
  153.  
  154. console.error(context + '::bad given items type (not an Instance, object, array)')
  155. }
  156.  
  157.  
  158.  
  159. /**
  160. * Set all collection items.
  161. * @private
  162. *
  163. * @param {array} arg_items - Instance objects array.
  164. *
  165. * @returns {nothing}
  166. */
  167. _set_all(arg_items)
  168. {
  169. this._items_array = arg_items
  170. }
  171.  
  172.  
  173.  
  174. /**
  175. * Get all collection items.
  176. * @private
  177. *
  178. * @returns {array}
  179. */
  180. _get_all()
  181. {
  182. return this._items_array
  183. }
  184. /**
  185. * Get all collection items or filter items with given type.
  186. *
  187. * @param {array|string|nothing} arg_types - type or types for items filtering.
  188. *
  189. * @returns {array} - all or filtered items, empty array if not found.
  190. */
  191. get_all(arg_types)
  192. {
  193. // NO TYPE FILTER
  194. if (! arg_types)
  195. {
  196. return _.toArray( this._items_array )
  197. }
  198.  
  199. // ONE TYPE FILTER
  200. if ( T.isString(arg_types) )
  201. {
  202. return _.filter(this._items_array, item => item.get_types() == arg_types )
  203. }
  204.  
  205. // MANY TYPES FILTER
  206. if ( T.isArray(arg_types) )
  207. {
  208. return _.filter(this._items_array, item => arg_types.indexOf( item.get_types() ) >= 0 )
  209. }
  210.  
  211. return []
  212. }
  213. /**
  214. * Get all items names with or without a filter on items types.
  215. *
  216. * @param {array|string|nothing} arg_types - type or types for items filtering.
  217. *
  218. * @returns {array} - all or filtered items names, empty array if not found.
  219. */
  220. get_all_names(arg_types)
  221. {
  222. // NO TYPE FILTER
  223. if (! arg_types)
  224. {
  225. return _.map(this._items_array, (item) => item.get_name() )
  226. }
  227.  
  228. // ONE TYPE FILTER
  229. if ( T.isString(arg_types) )
  230. {
  231. return _.filter( this._items_array, item => item.get_types() == arg_types ).map( (item) => item.get_name() )
  232. }
  233.  
  234. // MANY TYPES FILTER
  235. if ( T.isArray(arg_types) )
  236. {
  237. return _.filter( this._items_array, item => arg_types.indexOf( item.get_types() ) >= 0 ).map( (item) => item.get_name() )
  238. }
  239.  
  240. return []
  241. }
  242.  
  243. /**
  244. * Get all items ids with or without a filter on items types.
  245. *
  246. * @returns {array} - all items ids.
  247. */
  248. get_all_ids()
  249. {
  250. return _.map(this._items_array, (item) => item.get_id() )
  251. }
  252. /**
  253. * Get an item by its name.
  254. *
  255. * @param {string} arg_name - instance name.
  256. *
  257. * @returns {Instance|undefined}
  258. */
  259. item(arg_name)
  260. {
  261. return this._items_by_name ? this._items_by_name[arg_name] : undefined
  262. }
  263. /**
  264. * Get an item by its name.
  265. *
  266. * @param {string} arg_name - instance name.
  267. *
  268. * @returns {Instance|undefined}
  269. */
  270. get(arg_name)
  271. {
  272. return this._items_by_name ? this._items_by_name[arg_name] : undefined
  273. }
  274. /**
  275. * Default iterator operator.
  276. */
  277. // * [Symbol.iterator]() {
  278. // for (let item of this.$items)
  279. // {
  280. // yield item
  281. // }
  282. // }
  283. // [Symbol.iterator]()
  284. // {
  285. // let step = 0
  286. // const count = this.$items.length
  287. // const iterator = {
  288. // next()
  289. // {
  290. // if (step < count)
  291. // {
  292. // const item = this.$items[step]
  293. // step++
  294. // return { value:item, done:false }
  295. // }
  296. // return { value:undefined, done:true }
  297. // }
  298. // }
  299. // return iterator
  300. // }
  301. // NOT COMPATIBLE WITH NODE 0.10
  302. // [Symbol.iterator]()
  303. // {
  304. // return this.$items.iterator()
  305. // }
  306. /**
  307. * Get all items count.
  308. *
  309. * @returns {number} - all items count.
  310. */
  311. get_count()
  312. {
  313. return _.size(this._items_array)
  314. }
  315. /**
  316. * Get first item.
  317. *
  318. * @returns {object|undefined} - first collection items or undefined if collection is empty.
  319. */
  320. get_first()
  321. {
  322. return _.first(this._items_array)
  323. }
  324. /**
  325. * Get last item.
  326. *
  327. * @returns {object|undefined} - last collection items or null if collection is empty.
  328. */
  329. get_last()
  330. {
  331. return _.last(this._items_array)
  332. }
  333.  
  334.  
  335.  
  336. /**
  337. * Add an item to the collection.
  338. *
  339. * @param {Instance} arg_item - Instance item.
  340. *
  341. * @returns {nothing}
  342. */
  343. add(arg_item)
  344. {
  345. if ( T.isObject(arg_item) && arg_item instanceof Instance )
  346. {
  347. if ( this.has_accepted_type('*') || this.has_accepted_type( arg_item.get_type() ) )
  348. {
  349. this._add(arg_item)
  350. return
  351. }
  352. this.error('not accepted type [' + arg_item.get_type() + '] for instance [' + arg_item.get_name() + ']')
  353. return
  354. }
  355. this.error('bad item: not an instance object')
  356. }
  357. /**
  358. * Add an item to the collection at the first position.
  359. *
  360. * @param {Instance} arg_item - Instance item.
  361. *
  362. * @returns {nothing}
  363. */
  364. add_first(arg_item)
  365. {
  366. if ( T.isObject(arg_item) && arg_item instanceof Instance )
  367. {
  368. if ( this.has_accepted_type('*') || this.has_accepted_type(arg_item.$type) )
  369. {
  370. this._add_first(arg_item)
  371. return
  372. }
  373. this.error('not accepted type [' + arg_item.$type + '] for instance [' + arg_item.$name + ']')
  374. return
  375. }
  376. this.error('bad item: not an instance object')
  377. }
  378. /**
  379. * Remove an item from the collection.
  380. *
  381. * @param {Instance} arg_item - Instance item.
  382. *
  383. * @returns {nothing}
  384. */
  385. remove(arg_item)
  386. {
  387. if ( T.isObject(arg_item) && arg_item instanceof Instance )
  388. {
  389. const name = arg_item.get_name()
  390. if (name in this._items_by_name)
  391. {
  392. this._remove(arg_item)
  393. return
  394. }
  395. }
  396. this.error('bad item: not an instance object or not found')
  397. }
  398.  
  399. /**
  400. * Test if an item is inside the collection.
  401. *
  402. * @param {Instance} arg_item - Instance item.
  403. *
  404. * @returns {boolean}
  405. */
  406. has(arg_item)
  407. {
  408. if ( T.isObject(arg_item) && arg_item instanceof Instance )
  409. {
  410. return this._has(arg_item)
  411. }
  412. return false
  413. }
  414.  
  415. /**
  416. * Add an item to the collection without type checks (unsafe).
  417. * @private
  418. *
  419. * @param {Instance} arg_item - Instance item.
  420. *
  421. * @returns {nothing}
  422. */
  423. _add(arg_item)
  424. {
  425. if( this._has(arg_item) )
  426. {
  427. return
  428. }
  429.  
  430. const name = arg_item.get_name()
  431. const id = arg_item.get_id()
  432.  
  433. this._items_array.push(arg_item)
  434. this._items_by_name[name] = arg_item
  435. this._items_by_id[id] = arg_item
  436. }
  437.  
  438. /**
  439. * Add an item to the collection without type checks at first position (unsafe).
  440. * @private
  441. *
  442. * @param {Instance} arg_item - Instance item.
  443. *
  444. * @returns {nothing}
  445. */
  446. _add_first(arg_item)
  447. {
  448. if( this._has(arg_item) )
  449. {
  450. return
  451. }
  452. const name = arg_item.get_name()
  453. const id = arg_item.get_id()
  454.  
  455. this._items_array = [arg_item].concat(this._items_array)
  456. this._items_by_name[name] = arg_item
  457. this._items_by_id[id] = arg_item
  458. }
  459.  
  460. /**
  461. * Remove an item from the collection without type checks (unsafe).
  462. * @private
  463. *
  464. * @param {Instance} arg_item - Instance item.
  465. *
  466. * @returns {nothing}
  467. */
  468. _remove(arg_item)
  469. {
  470. const name = arg_item.get_name()
  471. const id = arg_item.get_id()
  472. const index = this._items_array.indexOf(arg_item)
  473.  
  474. this._items_array.splice(index, 1)
  475. delete this._items_by_name[name]
  476. delete this._items_by_id[id]
  477. }
  478.  
  479. /**
  480. * Test if an item is inside the collection without type checks (unsafe).
  481. * @private
  482. *
  483. * @param {Instance} arg_item - Instance item.
  484. *
  485. * @returns {boolean}
  486. */
  487. _has(arg_item)
  488. {
  489. const name = arg_item.get_name()
  490. return (name in this._items_by_name)
  491. }
  492. /**
  493. * Find an item by its name into the collection.
  494. *
  495. * TODO: optimize with a map index.
  496. *
  497. * @param {string} arg_name - instance name.
  498. *
  499. * @returns {Instance|undefined}
  500. */
  501. find_by_name(arg_name)
  502. {
  503. return this._items_by_name[arg_name]
  504. }
  505. /**
  506. * Find an item by its id into the collection.
  507. *
  508. * @param {string} arg_id - instance id.
  509. *
  510. * @returns {Instance|undefined}
  511. */
  512. find_by_id(arg_id)
  513. {
  514. return this._items_by_id[arg_id]
  515. }
  516. /**
  517. * Find an item by one of its attributes into the collection.
  518. *
  519. * @param {string} arg_attr_name - instance attribute name.
  520. * @param {any} arg_attr_value - instance attribute value.
  521. *
  522. * @returns {Instance|undefined}
  523. */
  524. find_by_attr(arg_attr_name, arg_attr_value)
  525. {
  526. return _.find(this._items_array, item => (arg_attr_name in item) && item[arg_attr_name] == arg_attr_value)
  527. }
  528. /**
  529. * Find an item by a filter function.
  530. *
  531. * @param {string} arg_filter_function - function to apply on instance, returns a boolean.
  532. *
  533. * @returns {Instance|undefined}
  534. */
  535. find_by_filter(arg_filter_function)
  536. {
  537. return _.find(this._items_array, item => arg_filter_function(item) )
  538. }
  539. /**
  540. * Filter items by one of theirs attributes into the collection.
  541. *
  542. * @param {string} arg_attr_name - instance attribute name.
  543. * @param {any} arg_attr_value - instance attribute value.
  544. *
  545. * @returns {array}
  546. */
  547. filter_by_attr(arg_attr_name, arg_attr_value)
  548. {
  549. return _.filter(this._items_array, item => (arg_attr_name in item) && item[arg_attr_name] == arg_attr_value)
  550. }
  551.  
  552. /**
  553. * Filter items by a filter function.
  554. *
  555. * @param {string} arg_filter_function - function to apply on instance, returns a boolean.
  556. *
  557. * @returns {array}
  558. */
  559. filter_by_filter(arg_filter_function)
  560. {
  561. return _.filter(this._items_array, item => arg_filter_function(item) )
  562. }
  563. /**
  564. * Get all collection accepted types.
  565. *
  566. * @returns {array} - array of types strings.
  567. */
  568. get_accepted_types()
  569. {
  570. this._accepted_types
  571. }
  572. /**
  573. * Set all collection accepted types.
  574. *
  575. * @param {array} arg_types - accepted types strings array.
  576. *
  577. * @returns {nothing}
  578. */
  579. set_accepted_types(arg_types)
  580. {
  581. assert(T.isArray(arg_types), context + ':bad accepted types array')
  582. this._accepted_types = arg_types
  583. }
  584. /**
  585. * Add one collection accepted type.
  586. *
  587. * @param {string} arg_type - accepted types string.
  588. *
  589. * @returns {nothing}
  590. */
  591. add_accepted_type(arg_type)
  592. {
  593. assert(T.isString(arg_type), context + ':bad accepted type string')
  594. this._accepted_types.push(arg_type)
  595. }
  596. /**
  597. * Test if collection has given accepted type.
  598. *
  599. * @param {string} arg_type - accepted types string.
  600. *
  601. * @returns {boolean}
  602. */
  603. has_accepted_type(arg_type)
  604. {
  605. return this._accepted_types.indexOf(arg_type) > -1
  606. }
  607. /**
  608. * forEach wrapper on ordered items.
  609. *
  610. * @param {function} arg_cb - callback to call on each item.
  611. *
  612. * @returns {nothing}
  613. */
  614. forEach(arg_cb)
  615. {
  616. _.forEach(this._items_array, arg_cb)
  617. }
  618. }