rubico 1.9.7 → 2.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (593) hide show
  1. package/.eslintignore +5 -0
  2. package/.eslintrc.js +3 -2
  3. package/.github/workflows/nodejs.yml +1 -1
  4. package/AggregateReducer.js +19 -0
  5. package/AggregateReducer.test.js +82 -0
  6. package/CHANGELOG.md +22 -1
  7. package/README.md +158 -81
  8. package/Transducer.js +75 -26
  9. package/Transducer.test.js +117 -0
  10. package/_internal/ComparisonOperator.js +68 -0
  11. package/_internal/File.js +41 -0
  12. package/_internal/FlatMappingAsyncIterator.js +4 -4
  13. package/_internal/FlatMappingIterator.js +1 -1
  14. package/_internal/TimeInLoopSuite.js +138 -0
  15. package/_internal/areAnyValuesPromises.js +23 -0
  16. package/_internal/{arrayAll.js → arrayEvery.js} +4 -4
  17. package/_internal/arrayExtend.js +3 -2
  18. package/_internal/arrayFilter.js +3 -4
  19. package/_internal/{arrayAny.js → arraySome.js} +8 -8
  20. package/_internal/{asyncIteratorAll.js → asyncIteratorEvery.js} +4 -4
  21. package/_internal/{asyncIteratorAny.js → asyncIteratorSome.js} +4 -4
  22. package/_internal/curry4.test.js +25 -0
  23. package/_internal/curry5.test.js +29 -0
  24. package/_internal/curryArgs2.js +43 -0
  25. package/_internal/curryArgs3.test.js +21 -0
  26. package/_internal/curryArgs4.js +65 -0
  27. package/_internal/curryArgs4.test.js +25 -0
  28. package/_internal/equals.js +13 -0
  29. package/_internal/findAllFilePaths.js +22 -0
  30. package/_internal/funcApply.js +2 -0
  31. package/_internal/funcCall.js +14 -0
  32. package/_internal/functionArrayAll.js +26 -0
  33. package/_internal/functionArrayAllSeries.js +55 -0
  34. package/_internal/functionObjectAll.js +32 -0
  35. package/_internal/genericReduce.js +4 -22
  36. package/_internal/genericTransform.js +15 -15
  37. package/_internal/improvedGenericTransform.js +93 -0
  38. package/_internal/{iteratorAll.js → iteratorEvery.js} +4 -4
  39. package/_internal/{iteratorAny.js → iteratorSome.js} +6 -6
  40. package/_internal/leftResolverRightResolverCompare.js +19 -0
  41. package/_internal/leftResolverRightValueCompare.js +16 -0
  42. package/_internal/leftValueRightResolverCompare.js +16 -0
  43. package/_internal/objectReduce.js +0 -1
  44. package/_internal/pathResolve.js +6 -0
  45. package/_internal/{reducerAll.js → reducerEvery.js} +4 -4
  46. package/_internal/reducerFlatMap.js +4 -4
  47. package/_internal/reducerFlatten.js +1 -1
  48. package/_internal/{reducerAny.js → reducerSome.js} +4 -4
  49. package/_internal/timeInLoop.js +37 -0
  50. package/_internal/timeInLoop.test.js +18 -0
  51. package/_internal/timeInLoopAsync.js +35 -0
  52. package/_internal/timeInLoopAsync.test.js +22 -0
  53. package/all.js +106 -40
  54. package/and.js +54 -36
  55. package/archive/{FlatMappingIteratorCool.js → FlatMappingIterator-2020-09-28.js} +1 -1
  56. package/archive/_Promise-2023-05-29.js +93 -0
  57. package/archive/arrayMap2-2023-05-29.js +73 -0
  58. package/archive/benchmarks-v1.9.7/all.js +34 -0
  59. package/archive/benchmarks-v1.9.7/and.js +24 -0
  60. package/archive/benchmarks-v1.9.7/assign.js +174 -0
  61. package/archive/benchmarks-v1.9.7/curry.js +55 -0
  62. package/archive/benchmarks-v1.9.7/eq.js +25 -0
  63. package/archive/benchmarks-v1.9.7/filter.js +1322 -0
  64. package/archive/benchmarks-v1.9.7/flatMap.js +48 -0
  65. package/archive/benchmarks-v1.9.7/get.js +44 -0
  66. package/archive/benchmarks-v1.9.7/gt.js +25 -0
  67. package/archive/benchmarks-v1.9.7/gte.js +25 -0
  68. package/archive/benchmarks-v1.9.7/lt.js +25 -0
  69. package/archive/benchmarks-v1.9.7/lte.js +25 -0
  70. package/archive/benchmarks-v1.9.7/map.js +892 -0
  71. package/archive/benchmarks-v1.9.7/omit.js +28 -0
  72. package/archive/benchmarks-v1.9.7/or.js +51 -0
  73. package/archive/benchmarks-v1.9.7/pick.js +24 -0
  74. package/archive/benchmarks-v1.9.7/pipe.js +152 -0
  75. package/archive/benchmarks-v1.9.7/reduce.js +739 -0
  76. package/archive/benchmarks-v1.9.7/switchCase.js +256 -0
  77. package/archive/benchmarks-v1.9.7/tap.js +90 -0
  78. package/archive/benchmarks-v1.9.7/transform.js +218 -0
  79. package/archive/benchmarks-v1.9.7/tryCatch.js +108 -0
  80. package/assign.js +18 -8
  81. package/bench +65 -0
  82. package/benchmark-output/v1.9.7 +268 -0
  83. package/benchmarks/all.async.js +43 -0
  84. package/benchmarks/all.js +42 -33
  85. package/benchmarks/always.js +15 -0
  86. package/benchmarks/and.async.js +25 -0
  87. package/benchmarks/and.js +20 -19
  88. package/benchmarks/assign.async.js +27 -0
  89. package/benchmarks/assign.js +20 -167
  90. package/benchmarks/curry.js +34 -54
  91. package/benchmarks/eq.async.js +23 -0
  92. package/benchmarks/eq.js +17 -19
  93. package/benchmarks/every.async.js +19 -0
  94. package/benchmarks/every.js +19 -0
  95. package/benchmarks/filter.async.js +32 -0
  96. package/benchmarks/filter.js +27 -1311
  97. package/benchmarks/flatMap.async.js +26 -0
  98. package/benchmarks/flatMap.js +26 -36
  99. package/benchmarks/get.async.js +19 -0
  100. package/benchmarks/get.js +27 -32
  101. package/benchmarks/gt.async.js +23 -0
  102. package/benchmarks/gt.js +17 -19
  103. package/benchmarks/gte.async.js +23 -0
  104. package/benchmarks/gte.js +17 -19
  105. package/benchmarks/lt.async.js +23 -0
  106. package/benchmarks/lt.js +17 -19
  107. package/benchmarks/lte.async.js +23 -0
  108. package/benchmarks/lte.js +17 -19
  109. package/benchmarks/map.async.js +43 -0
  110. package/benchmarks/map.js +24 -876
  111. package/benchmarks/misc/Promise.js +26 -0
  112. package/benchmarks/misc/isPromise.js +30 -0
  113. package/benchmarks/misc/promiseAll.js +36 -0
  114. package/benchmarks/not.js +23 -0
  115. package/benchmarks/omit.js +30 -20
  116. package/benchmarks/or.async.js +25 -0
  117. package/benchmarks/or.js +23 -49
  118. package/benchmarks/pick.js +30 -16
  119. package/benchmarks/pipe.async.js +47 -0
  120. package/benchmarks/pipe.js +46 -151
  121. package/benchmarks/reduce.async.js +32 -0
  122. package/benchmarks/reduce.js +27 -728
  123. package/benchmarks/set.async.js +19 -0
  124. package/benchmarks/set.js +41 -0
  125. package/benchmarks/some.async.js +19 -0
  126. package/benchmarks/some.js +19 -0
  127. package/benchmarks/switchCase.async.js +27 -0
  128. package/benchmarks/switchCase.js +55 -256
  129. package/benchmarks/tap.js +10 -85
  130. package/benchmarks/thunkify.js +15 -0
  131. package/benchmarks/transform.async.js +27 -0
  132. package/benchmarks/transform.js +28 -206
  133. package/benchmarks/tryCatch.async.js +25 -0
  134. package/benchmarks/tryCatch.js +24 -100
  135. package/build +472 -0
  136. package/compose.js +46 -0
  137. package/dist/Transducer.es.js +481 -0
  138. package/dist/Transducer.es.min.js +7 -0
  139. package/dist/Transducer.js +488 -0
  140. package/dist/Transducer.min.js +8 -0
  141. package/dist/Transducer.mjs +481 -0
  142. package/dist/__.es.js +2 -2
  143. package/dist/__.es.min.js +2 -2
  144. package/dist/__.js +2 -2
  145. package/dist/__.min.js +2 -2
  146. package/dist/__.mjs +2 -2
  147. package/dist/all.es.js +173 -92
  148. package/dist/all.es.min.js +3 -3
  149. package/dist/all.js +173 -92
  150. package/dist/all.min.js +3 -3
  151. package/dist/all.mjs +173 -92
  152. package/dist/always.es.js +2 -2
  153. package/dist/always.es.min.js +2 -2
  154. package/dist/always.js +2 -2
  155. package/dist/always.min.js +2 -2
  156. package/dist/always.mjs +2 -2
  157. package/dist/and.es.js +101 -32
  158. package/dist/and.es.min.js +3 -3
  159. package/dist/and.js +101 -32
  160. package/dist/and.min.js +3 -3
  161. package/dist/and.mjs +101 -32
  162. package/dist/assign.es.js +37 -10
  163. package/dist/assign.es.min.js +3 -3
  164. package/dist/assign.js +37 -10
  165. package/dist/assign.min.js +3 -3
  166. package/dist/assign.mjs +37 -10
  167. package/dist/compose.es.js +71 -0
  168. package/dist/compose.es.min.js +7 -0
  169. package/dist/compose.js +78 -0
  170. package/dist/compose.min.js +8 -0
  171. package/dist/compose.mjs +71 -0
  172. package/dist/curry.es.js +2 -2
  173. package/dist/curry.es.min.js +2 -2
  174. package/dist/curry.js +2 -2
  175. package/dist/curry.min.js +2 -2
  176. package/dist/curry.mjs +2 -2
  177. package/dist/eq.es.js +172 -49
  178. package/dist/eq.es.min.js +3 -3
  179. package/dist/eq.js +172 -49
  180. package/dist/eq.min.js +3 -3
  181. package/dist/eq.mjs +172 -49
  182. package/dist/every.es.js +194 -0
  183. package/dist/every.es.min.js +7 -0
  184. package/dist/every.js +201 -0
  185. package/dist/every.min.js +8 -0
  186. package/dist/every.mjs +194 -0
  187. package/dist/filter.es.js +14 -108
  188. package/dist/filter.es.min.js +3 -3
  189. package/dist/filter.js +14 -108
  190. package/dist/filter.min.js +3 -3
  191. package/dist/filter.mjs +14 -108
  192. package/dist/flatMap.es.js +27 -214
  193. package/dist/flatMap.es.min.js +3 -3
  194. package/dist/flatMap.js +27 -214
  195. package/dist/flatMap.min.js +3 -3
  196. package/dist/flatMap.mjs +27 -214
  197. package/dist/{x/forEach.es.js → forEach.es.js} +52 -70
  198. package/dist/forEach.es.min.js +7 -0
  199. package/dist/{x/forEach.js → forEach.js} +52 -70
  200. package/dist/forEach.min.js +8 -0
  201. package/dist/{x/forEach.mjs → forEach.mjs} +52 -70
  202. package/dist/get.es.js +60 -5
  203. package/dist/get.es.min.js +3 -3
  204. package/dist/get.js +60 -5
  205. package/dist/get.min.js +3 -3
  206. package/dist/get.mjs +60 -5
  207. package/dist/gt.es.js +171 -48
  208. package/dist/gt.es.min.js +3 -3
  209. package/dist/gt.js +171 -48
  210. package/dist/gt.min.js +3 -3
  211. package/dist/gt.mjs +171 -48
  212. package/dist/gte.es.js +171 -48
  213. package/dist/gte.es.min.js +3 -3
  214. package/dist/gte.js +171 -48
  215. package/dist/gte.min.js +3 -3
  216. package/dist/gte.mjs +171 -48
  217. package/dist/lt.es.js +171 -48
  218. package/dist/lt.es.min.js +3 -3
  219. package/dist/lt.js +171 -48
  220. package/dist/lt.min.js +3 -3
  221. package/dist/lt.mjs +171 -48
  222. package/dist/lte.es.js +171 -48
  223. package/dist/lte.es.min.js +3 -3
  224. package/dist/lte.js +171 -48
  225. package/dist/lte.min.js +3 -3
  226. package/dist/lte.mjs +171 -48
  227. package/dist/map.es.js +12 -101
  228. package/dist/map.es.min.js +3 -3
  229. package/dist/map.js +12 -101
  230. package/dist/map.min.js +3 -3
  231. package/dist/map.mjs +12 -101
  232. package/dist/not.es.js +75 -14
  233. package/dist/not.es.min.js +3 -3
  234. package/dist/not.js +75 -14
  235. package/dist/not.min.js +3 -3
  236. package/dist/not.mjs +75 -14
  237. package/dist/omit.es.js +7 -2
  238. package/dist/omit.es.min.js +3 -3
  239. package/dist/omit.js +7 -2
  240. package/dist/omit.min.js +3 -3
  241. package/dist/omit.mjs +7 -2
  242. package/dist/or.es.js +99 -31
  243. package/dist/or.es.min.js +3 -3
  244. package/dist/or.js +99 -31
  245. package/dist/or.min.js +3 -3
  246. package/dist/or.mjs +99 -31
  247. package/dist/pick.es.js +8 -3
  248. package/dist/pick.es.min.js +3 -3
  249. package/dist/pick.js +8 -3
  250. package/dist/pick.min.js +3 -3
  251. package/dist/pick.mjs +8 -3
  252. package/dist/pipe.es.js +42 -46
  253. package/dist/pipe.es.min.js +3 -3
  254. package/dist/pipe.js +42 -46
  255. package/dist/pipe.min.js +3 -3
  256. package/dist/pipe.mjs +42 -46
  257. package/dist/reduce.es.js +52 -94
  258. package/dist/reduce.es.min.js +3 -3
  259. package/dist/reduce.js +52 -94
  260. package/dist/reduce.min.js +3 -3
  261. package/dist/reduce.mjs +52 -94
  262. package/dist/rubico.es.js +762 -883
  263. package/dist/rubico.es.min.js +3 -3
  264. package/dist/rubico.global.js +2794 -0
  265. package/dist/rubico.global.min.js +8 -0
  266. package/dist/rubico.js +739 -869
  267. package/dist/rubico.min.js +3 -3
  268. package/dist/rubico.mjs +762 -883
  269. package/dist/set.es.js +18 -3
  270. package/dist/set.es.min.js +3 -3
  271. package/dist/set.js +18 -3
  272. package/dist/set.min.js +3 -3
  273. package/dist/set.mjs +18 -3
  274. package/dist/{any.js → some.es.js} +64 -54
  275. package/dist/some.es.min.js +7 -0
  276. package/dist/{any.mjs → some.js} +72 -48
  277. package/dist/some.min.js +8 -0
  278. package/dist/{any.es.js → some.mjs} +65 -48
  279. package/dist/switchCase.es.js +55 -5
  280. package/dist/switchCase.es.min.js +3 -3
  281. package/dist/switchCase.js +55 -5
  282. package/dist/switchCase.min.js +3 -3
  283. package/dist/switchCase.mjs +55 -5
  284. package/dist/tap.es.js +2 -9
  285. package/dist/tap.es.min.js +3 -3
  286. package/dist/tap.js +2 -9
  287. package/dist/tap.min.js +3 -3
  288. package/dist/tap.mjs +2 -9
  289. package/dist/thunkify.es.js +45 -2
  290. package/dist/thunkify.es.min.js +3 -3
  291. package/dist/thunkify.js +45 -2
  292. package/dist/thunkify.min.js +3 -3
  293. package/dist/thunkify.mjs +45 -2
  294. package/dist/transform.es.js +35 -71
  295. package/dist/transform.es.min.js +3 -3
  296. package/dist/transform.js +35 -71
  297. package/dist/transform.min.js +3 -3
  298. package/dist/transform.mjs +35 -71
  299. package/dist/tryCatch.es.js +33 -17
  300. package/dist/tryCatch.es.min.js +3 -3
  301. package/dist/tryCatch.js +33 -17
  302. package/dist/tryCatch.min.js +3 -3
  303. package/dist/tryCatch.mjs +33 -17
  304. package/dist/x/append.es.js +2 -2
  305. package/dist/x/append.es.min.js +2 -2
  306. package/dist/x/append.js +2 -2
  307. package/dist/x/append.min.js +2 -2
  308. package/dist/x/append.mjs +2 -2
  309. package/dist/x/callProp.es.js +2 -2
  310. package/dist/x/callProp.es.min.js +2 -2
  311. package/dist/x/callProp.js +2 -2
  312. package/dist/x/callProp.min.js +2 -2
  313. package/dist/x/callProp.mjs +2 -2
  314. package/dist/x/defaultsDeep.es.js +2 -2
  315. package/dist/x/defaultsDeep.es.min.js +2 -2
  316. package/dist/x/defaultsDeep.js +2 -2
  317. package/dist/x/defaultsDeep.min.js +2 -2
  318. package/dist/x/defaultsDeep.mjs +2 -2
  319. package/dist/x/differenceWith.es.js +7 -7
  320. package/dist/x/differenceWith.es.min.js +3 -3
  321. package/dist/x/differenceWith.js +7 -7
  322. package/dist/x/differenceWith.min.js +2 -2
  323. package/dist/x/differenceWith.mjs +7 -7
  324. package/dist/x/filterOut.es.js +60 -117
  325. package/dist/x/filterOut.es.min.js +3 -3
  326. package/dist/x/filterOut.js +60 -117
  327. package/dist/x/filterOut.min.js +3 -3
  328. package/dist/x/filterOut.mjs +60 -117
  329. package/dist/x/find.es.js +2 -2
  330. package/dist/x/find.es.min.js +2 -2
  331. package/dist/x/find.js +2 -2
  332. package/dist/x/find.min.js +2 -2
  333. package/dist/x/find.mjs +2 -2
  334. package/dist/x/findIndex.es.js +2 -2
  335. package/dist/x/findIndex.es.min.js +2 -2
  336. package/dist/x/findIndex.js +2 -2
  337. package/dist/x/findIndex.min.js +2 -2
  338. package/dist/x/findIndex.mjs +2 -2
  339. package/dist/x/first.es.js +2 -2
  340. package/dist/x/first.es.min.js +2 -2
  341. package/dist/x/first.js +2 -2
  342. package/dist/x/first.min.js +2 -2
  343. package/dist/x/first.mjs +2 -2
  344. package/dist/x/flatten.es.js +20 -207
  345. package/dist/x/flatten.es.min.js +3 -3
  346. package/dist/x/flatten.js +20 -207
  347. package/dist/x/flatten.min.js +3 -3
  348. package/dist/x/flatten.mjs +20 -207
  349. package/dist/x/groupBy.es.js +52 -94
  350. package/dist/x/groupBy.es.min.js +3 -3
  351. package/dist/x/groupBy.js +52 -94
  352. package/dist/x/groupBy.min.js +3 -3
  353. package/dist/x/groupBy.mjs +52 -94
  354. package/dist/x/has.es.js +2 -2
  355. package/dist/x/has.es.min.js +2 -2
  356. package/dist/x/has.js +2 -2
  357. package/dist/x/has.min.js +2 -2
  358. package/dist/x/has.mjs +2 -2
  359. package/dist/x/identity.es.js +2 -2
  360. package/dist/x/identity.es.min.js +2 -2
  361. package/dist/x/identity.js +2 -2
  362. package/dist/x/identity.min.js +2 -2
  363. package/dist/x/identity.mjs +2 -2
  364. package/dist/x/includes.es.js +2 -2
  365. package/dist/x/includes.es.min.js +2 -2
  366. package/dist/x/includes.js +2 -2
  367. package/dist/x/includes.min.js +2 -2
  368. package/dist/x/includes.mjs +2 -2
  369. package/dist/x/isDeepEqual.es.js +2 -2
  370. package/dist/x/isDeepEqual.es.min.js +2 -2
  371. package/dist/x/isDeepEqual.js +2 -2
  372. package/dist/x/isDeepEqual.min.js +2 -2
  373. package/dist/x/isDeepEqual.mjs +2 -2
  374. package/dist/x/isEmpty.es.js +2 -2
  375. package/dist/x/isEmpty.es.min.js +2 -2
  376. package/dist/x/isEmpty.js +2 -2
  377. package/dist/x/isEmpty.min.js +2 -2
  378. package/dist/x/isEmpty.mjs +2 -2
  379. package/dist/x/isEqual.es.js +2 -2
  380. package/dist/x/isEqual.es.min.js +2 -2
  381. package/dist/x/isEqual.js +2 -2
  382. package/dist/x/isEqual.min.js +2 -2
  383. package/dist/x/isEqual.mjs +2 -2
  384. package/dist/x/isFunction.es.js +2 -2
  385. package/dist/x/isFunction.es.min.js +2 -2
  386. package/dist/x/isFunction.js +2 -2
  387. package/dist/x/isFunction.min.js +2 -2
  388. package/dist/x/isFunction.mjs +2 -2
  389. package/dist/x/isIn.es.js +2 -2
  390. package/dist/x/isIn.es.min.js +2 -2
  391. package/dist/x/isIn.js +2 -2
  392. package/dist/x/isIn.min.js +2 -2
  393. package/dist/x/isIn.mjs +2 -2
  394. package/dist/x/isObject.es.js +2 -2
  395. package/dist/x/isObject.es.min.js +2 -2
  396. package/dist/x/isObject.js +2 -2
  397. package/dist/x/isObject.min.js +2 -2
  398. package/dist/x/isObject.mjs +2 -2
  399. package/dist/x/isString.es.js +2 -2
  400. package/dist/x/isString.es.min.js +2 -2
  401. package/dist/x/isString.js +2 -2
  402. package/dist/x/isString.min.js +2 -2
  403. package/dist/x/isString.mjs +2 -2
  404. package/dist/x/keys.es.js +2 -2
  405. package/dist/x/keys.es.min.js +2 -2
  406. package/dist/x/keys.js +2 -2
  407. package/dist/x/keys.min.js +2 -2
  408. package/dist/x/keys.mjs +2 -2
  409. package/dist/x/last.es.js +2 -2
  410. package/dist/x/last.es.min.js +2 -2
  411. package/dist/x/last.js +2 -2
  412. package/dist/x/last.min.js +2 -2
  413. package/dist/x/last.mjs +2 -2
  414. package/dist/x/maxBy.es.js +59 -6
  415. package/dist/x/maxBy.es.min.js +3 -3
  416. package/dist/x/maxBy.js +59 -6
  417. package/dist/x/maxBy.min.js +3 -3
  418. package/dist/x/maxBy.mjs +59 -6
  419. package/dist/x/noop.es.js +2 -2
  420. package/dist/x/noop.es.min.js +2 -2
  421. package/dist/x/noop.js +2 -2
  422. package/dist/x/noop.min.js +2 -2
  423. package/dist/x/noop.mjs +2 -2
  424. package/dist/x/pluck.es.js +32 -103
  425. package/dist/x/pluck.es.min.js +3 -3
  426. package/dist/x/pluck.js +32 -103
  427. package/dist/x/pluck.min.js +3 -3
  428. package/dist/x/pluck.mjs +32 -103
  429. package/dist/x/prepend.es.js +2 -2
  430. package/dist/x/prepend.es.min.js +2 -2
  431. package/dist/x/prepend.js +2 -2
  432. package/dist/x/prepend.min.js +2 -2
  433. package/dist/x/prepend.mjs +2 -2
  434. package/dist/x/size.es.js +2 -2
  435. package/dist/x/size.es.min.js +2 -2
  436. package/dist/x/size.js +2 -2
  437. package/dist/x/size.min.js +2 -2
  438. package/dist/x/size.mjs +2 -2
  439. package/dist/x/trace.es.js +2 -9
  440. package/dist/x/trace.es.min.js +3 -3
  441. package/dist/x/trace.js +2 -9
  442. package/dist/x/trace.min.js +3 -3
  443. package/dist/x/trace.mjs +2 -9
  444. package/dist/x/unionWith.es.js +2 -2
  445. package/dist/x/unionWith.es.min.js +2 -2
  446. package/dist/x/unionWith.js +2 -2
  447. package/dist/x/unionWith.min.js +2 -2
  448. package/dist/x/unionWith.mjs +2 -2
  449. package/dist/x/uniq.es.js +2 -2
  450. package/dist/x/uniq.es.min.js +2 -2
  451. package/dist/x/uniq.js +2 -2
  452. package/dist/x/uniq.min.js +2 -2
  453. package/dist/x/uniq.mjs +2 -2
  454. package/dist/x/unless.es.js +2 -2
  455. package/dist/x/unless.es.min.js +2 -2
  456. package/dist/x/unless.js +2 -2
  457. package/dist/x/unless.min.js +2 -2
  458. package/dist/x/unless.mjs +2 -2
  459. package/dist/x/values.es.js +2 -2
  460. package/dist/x/values.es.min.js +2 -2
  461. package/dist/x/values.js +2 -2
  462. package/dist/x/values.min.js +2 -2
  463. package/dist/x/values.mjs +2 -2
  464. package/dist/x/when.es.js +2 -2
  465. package/dist/x/when.es.min.js +2 -2
  466. package/dist/x/when.js +2 -2
  467. package/dist/x/when.min.js +2 -2
  468. package/dist/x/when.mjs +2 -2
  469. package/dist-test.js +31 -2042
  470. package/eq.js +21 -63
  471. package/es.js +739 -869
  472. package/every.js +94 -0
  473. package/filter.js +23 -91
  474. package/flatMap.js +75 -70
  475. package/forEach.js +82 -0
  476. package/get.js +23 -6
  477. package/global.js +6 -4
  478. package/gt.js +15 -58
  479. package/gte.js +14 -57
  480. package/index.js +739 -869
  481. package/lt.js +17 -60
  482. package/lte.js +14 -57
  483. package/map.js +9 -94
  484. package/memory-usage/reduce.js +2 -2
  485. package/not.js +27 -45
  486. package/omit.js +4 -0
  487. package/or.js +50 -33
  488. package/package.json +7 -7
  489. package/pick.js +5 -1
  490. package/pipe.js +13 -50
  491. package/reduce.js +42 -51
  492. package/rubico.js +22 -11
  493. package/set.js +43 -10
  494. package/some.js +105 -0
  495. package/switchCase.js +13 -3
  496. package/tap.js +0 -24
  497. package/test.js +847 -1096
  498. package/thunkify.js +9 -0
  499. package/tmp.js +1 -0
  500. package/transform.js +31 -16
  501. package/tryCatch.js +19 -15
  502. package/x/differenceWith.js +3 -3
  503. package/x/filterOut.test.js +6 -6
  504. package/x/flatten.js +1 -1
  505. package/x/flatten.test.js +0 -8
  506. package/x/index.js +0 -2
  507. package/x/pluck.js +18 -27
  508. package/x/pluck.test.js +3 -7
  509. package/x/timeInLoop.js +16 -25
  510. package/x/timeInLoop.test.js +8 -4
  511. package/any.js +0 -82
  512. package/dist/__.min.mjs +0 -7
  513. package/dist/all.min.mjs +0 -7
  514. package/dist/always.min.mjs +0 -7
  515. package/dist/and.min.mjs +0 -7
  516. package/dist/any.es.min.js +0 -7
  517. package/dist/any.min.js +0 -8
  518. package/dist/any.min.mjs +0 -7
  519. package/dist/assign.min.mjs +0 -7
  520. package/dist/curry.min.mjs +0 -7
  521. package/dist/eq.min.mjs +0 -7
  522. package/dist/filter.min.mjs +0 -7
  523. package/dist/flatMap.min.mjs +0 -7
  524. package/dist/fork.es.js +0 -159
  525. package/dist/fork.es.min.js +0 -7
  526. package/dist/fork.js +0 -166
  527. package/dist/fork.min.js +0 -8
  528. package/dist/fork.min.mjs +0 -7
  529. package/dist/fork.mjs +0 -159
  530. package/dist/get.min.mjs +0 -7
  531. package/dist/gt.min.mjs +0 -7
  532. package/dist/gte.min.mjs +0 -7
  533. package/dist/lt.min.mjs +0 -7
  534. package/dist/lte.min.mjs +0 -7
  535. package/dist/map.min.mjs +0 -7
  536. package/dist/not.min.mjs +0 -7
  537. package/dist/omit.min.mjs +0 -7
  538. package/dist/or.min.mjs +0 -7
  539. package/dist/pick.min.mjs +0 -7
  540. package/dist/pipe.min.mjs +0 -7
  541. package/dist/reduce.min.mjs +0 -7
  542. package/dist/rubico.min.mjs +0 -7
  543. package/dist/set.min.mjs +0 -7
  544. package/dist/switchCase.min.mjs +0 -7
  545. package/dist/tap.min.mjs +0 -7
  546. package/dist/thunkify.min.mjs +0 -7
  547. package/dist/transform.min.mjs +0 -7
  548. package/dist/tryCatch.min.mjs +0 -7
  549. package/dist/x/append.min.mjs +0 -7
  550. package/dist/x/callProp.min.mjs +0 -7
  551. package/dist/x/defaultsDeep.min.mjs +0 -7
  552. package/dist/x/differenceWith.min.mjs +0 -7
  553. package/dist/x/filterOut.min.mjs +0 -7
  554. package/dist/x/find.min.mjs +0 -7
  555. package/dist/x/findIndex.min.mjs +0 -7
  556. package/dist/x/first.min.mjs +0 -7
  557. package/dist/x/flatten.min.mjs +0 -7
  558. package/dist/x/forEach.es.min.js +0 -7
  559. package/dist/x/forEach.min.js +0 -8
  560. package/dist/x/forEach.min.mjs +0 -7
  561. package/dist/x/groupBy.min.mjs +0 -7
  562. package/dist/x/has.min.mjs +0 -7
  563. package/dist/x/identity.min.mjs +0 -7
  564. package/dist/x/includes.min.mjs +0 -7
  565. package/dist/x/isDeepEqual.min.mjs +0 -7
  566. package/dist/x/isEmpty.min.mjs +0 -7
  567. package/dist/x/isEqual.min.mjs +0 -7
  568. package/dist/x/isFunction.min.mjs +0 -7
  569. package/dist/x/isIn.min.mjs +0 -7
  570. package/dist/x/isObject.min.mjs +0 -7
  571. package/dist/x/isString.min.mjs +0 -7
  572. package/dist/x/keys.min.mjs +0 -7
  573. package/dist/x/last.min.mjs +0 -7
  574. package/dist/x/maxBy.min.mjs +0 -7
  575. package/dist/x/noop.min.mjs +0 -7
  576. package/dist/x/pluck.min.mjs +0 -7
  577. package/dist/x/prepend.min.mjs +0 -7
  578. package/dist/x/size.min.mjs +0 -7
  579. package/dist/x/trace.min.mjs +0 -7
  580. package/dist/x/unionWith.min.mjs +0 -7
  581. package/dist/x/uniq.min.mjs +0 -7
  582. package/dist/x/unless.min.mjs +0 -7
  583. package/dist/x/values.min.mjs +0 -7
  584. package/dist/x/when.min.mjs +0 -7
  585. package/distributor.js +0 -495
  586. package/fork.js +0 -101
  587. package/x/forEach.js +0 -93
  588. package/x/forEach.test.js +0 -218
  589. /package/{benchmarks → archive/benchmarks-v1.9.7}/any.js +0 -0
  590. /package/{x/forEach.benchmark.js → archive/benchmarks-v1.9.7/forEach.js} +0 -0
  591. /package/{benchmarks → archive/benchmarks-v1.9.7}/fork.js +0 -0
  592. /package/{benchmarks → archive/benchmarks-v1.9.7}/integration.js +0 -0
  593. /package/{benchmarks → archive/benchmarks-v1.9.7}/misc.js +0 -0
package/.eslintignore ADDED
@@ -0,0 +1,5 @@
1
+ archive
2
+ dist
3
+ es.js
4
+ index.js
5
+ benchmarks
package/.eslintrc.js CHANGED
@@ -192,7 +192,7 @@ module.exports = {
192
192
  'allowForLoopAfterthoughts': true
193
193
  }
194
194
  ],
195
- 'no-process-env': 'error',
195
+ 'no-process-env': 'off',
196
196
  'no-process-exit': 'error',
197
197
  'no-promise-executor-return': 'off',
198
198
  'no-proto': 'error',
@@ -209,7 +209,8 @@ module.exports = {
209
209
  'no-sequences': 'off',
210
210
  'no-shadow': 'off',
211
211
  'no-spaced-func': 'error',
212
- 'no-sync': 'error',
212
+ 'no-sparse-arrays': 'off',
213
+ 'no-sync': 'off',
213
214
  'no-tabs': 'error',
214
215
  'no-template-curly-in-string': 'error',
215
216
  'no-ternary': 'off',
@@ -16,7 +16,7 @@ jobs:
16
16
 
17
17
  strategy:
18
18
  matrix:
19
- node-version: [14.x, 15.x, 16.x, 17.x, 18.x]
19
+ node-version: [16.x, 18.x, 20.x]
20
20
 
21
21
  steps:
22
22
  - uses: actions/checkout@v2
@@ -0,0 +1,19 @@
1
+ const reducerConcat = require('./_internal/reducerConcat')
2
+ const identity = require('./_internal/identity')
3
+
4
+ /**
5
+ * @name AggregateReducer
6
+ *
7
+ * @synopsis
8
+ * ```coffeescript [specscript]
9
+ * AggregateReducer(reducers Array<reducer function>) -> aggregateReducer function
10
+ * ```
11
+ */
12
+ const AggregateReducer = function (reducers) {
13
+ if (reducers.length == 0) {
14
+ return identity
15
+ }
16
+ return reducers.reduce(reducerConcat, identity)
17
+ }
18
+
19
+ module.exports = AggregateReducer
@@ -0,0 +1,82 @@
1
+ const assert = require('assert')
2
+ const reduce = require('./reduce')
3
+ const AggregateReducer = require('./AggregateReducer')
4
+
5
+ describe('AggregateReducer', () => {
6
+ it('Combines reducers', async () => {
7
+ const reducerA = (state, action) => {
8
+ if (action.type == 'A') {
9
+ return { ...state, A: true }
10
+ }
11
+ return state
12
+ }
13
+ const reducerB = (state, action) => {
14
+ if (action.type == 'B') {
15
+ return { ...state, B: true }
16
+ }
17
+ return state
18
+ }
19
+ const reducerC = (state, action) => {
20
+ if (action.type == 'C') {
21
+ return { ...state, C: true }
22
+ }
23
+ return state
24
+ }
25
+
26
+ const combinedReducer = AggregateReducer([
27
+ reducerA,
28
+ reducerB,
29
+ reducerC,
30
+ ])
31
+
32
+ const stateA = [{ type: 'A' }].reduce(combinedReducer, {})
33
+ assert.deepEqual(stateA, { A: true })
34
+
35
+ const stateAB = [{ type: 'A' }, { type: 'B' }].reduce(combinedReducer, {})
36
+ assert.deepEqual(stateAB, { A: true, B: true })
37
+
38
+ const stateABC = [{ type: 'A' }, { type: 'B' }, { type: 'C' }].reduce(combinedReducer, {})
39
+ assert.deepEqual(stateABC, { A: true, B: true, C: true })
40
+ })
41
+
42
+ it('Combines async reducers', async () => {
43
+ const reducerA = async (state, action) => {
44
+ if (action.type == 'A') {
45
+ return { ...state, A: true }
46
+ }
47
+ return state
48
+ }
49
+ const reducerB = async (state, action) => {
50
+ if (action.type == 'B') {
51
+ return { ...state, B: true }
52
+ }
53
+ return state
54
+ }
55
+ const reducerC = async (state, action) => {
56
+ if (action.type == 'C') {
57
+ return { ...state, C: true }
58
+ }
59
+ return state
60
+ }
61
+
62
+ const combinedReducer = AggregateReducer([
63
+ reducerA,
64
+ reducerB,
65
+ reducerC,
66
+ ])
67
+
68
+ const stateA = await reduce([{ type: 'A' }], combinedReducer, {})
69
+ assert.deepEqual(stateA, { A: true })
70
+
71
+ const stateAB = await reduce([{ type: 'A' }, { type: 'B' }], combinedReducer, {})
72
+ assert.deepEqual(stateAB, { A: true, B: true })
73
+
74
+ const stateABC = await reduce([{ type: 'A' }, { type: 'B' }, { type: 'C' }], combinedReducer, {})
75
+ assert.deepEqual(stateABC, { A: true, B: true, C: true })
76
+ })
77
+
78
+ it('Identity', async () => {
79
+ const identity = AggregateReducer([])
80
+ assert.equal(1, identity(1))
81
+ })
82
+ })
package/CHANGELOG.md CHANGED
@@ -1,4 +1,25 @@
1
- ## v1.9.3 - latest
1
+ ## v2.0.0
2
+ * Refactor all transducer functionality into Transducer.js from `pipe`, `map`, `filter`, and `flatMap`
3
+ * Remove GeneratorFunction and AsyncGeneratorFunction functionality in `map`, `filter`, `flatMap`, and `reduce`
4
+ * Refactor reducer chaining functionality into AggregateReducer.js from `reduce`
5
+ * Remove Duplex stream handling from `flatMap`
6
+ * Remove binary handling from `flatMap`
7
+ * Promises as arguments for all core functions
8
+ * Eager API for all core functions
9
+ * Automate benchmarks
10
+ * Remove the .sync property functions
11
+ * Remove the .withIndex property functions
12
+ * Deprecate map.own
13
+ * `fork` renamed to `all`
14
+ * `all` renamed to `every`
15
+ * `any` renamed to `some`
16
+ * new `dist-test` only validates syntax with `require`
17
+ * migrate `forEach` into core API.
18
+ * refactor transducer functionality from `forEach` into `Transducer.forEach`
19
+ * async performance improvements
20
+ * `_internal/arrayExtend` stops treating strings like arrays so `transform` creates arrays of strings as expected
21
+
22
+ ## v1.9.3
2
23
  * property value passed to set may be a resolver - [#224](https://github.com/a-synchronous/rubico/pull/224)
3
24
  * rubico/x/isIn - counterpart to includes - [#228](https://github.com/a-synchronous/rubico/pull/228) - [lulldev](https://github.com/lulldev)
4
25
  * rubico/x/maxBy - finds the item that is the max by a property denoted by path - [#229](https://github.com/a-synchronous/rubico/pull/229)
package/README.md CHANGED
@@ -19,90 +19,68 @@ const isOdd = number => number % 2 == 1
19
19
 
20
20
  const asyncSquare = async number => number ** 2
21
21
 
22
- const squaredOdds = pipe([
22
+ const numbers = [1, 2, 3, 4, 5]
23
+
24
+ pipe(numbers, [
23
25
  filter(isOdd),
24
26
  map(asyncSquare),
27
+ console.log, // [1, 9, 25]
25
28
  ])
26
-
27
- squaredOdds([1, 2, 3, 4, 5]).then(console.log) // [1, 9, 25]
28
29
  ```
29
30
 
30
31
  # Installation
31
- [Core build](https://unpkg.com/rubico/index.js) ([~6.1 kB minified and gzipped](https://unpkg.com/rubico/dist/rubico.min.js))
32
+ [Core build](https://unpkg.com/rubico/index.js) ([~6.8 kB minified and gzipped](https://unpkg.com/rubico/dist/rubico.min.js))
32
33
 
33
34
  with `npm`
34
35
  ```bash
35
36
  npm i rubico
36
37
  ```
37
38
 
38
- require `rubico` in Node.js
39
+ require `rubico` in CommonJS.
39
40
  ```javascript
41
+ // import rubico core globally
42
+ require('rubico/global')
43
+
44
+ // import rubico core as rubico
40
45
  const rubico = require('rubico')
46
+
47
+ // import an operator from rubico core
41
48
  const pipe = require('rubico/pipe')
42
- const tap = require('rubico/tap')
49
+
50
+ // import rubico/x as x
51
+ const x = require('rubico/x')
52
+
53
+ // import an operator from rubico/x
43
54
  const defaultsDeep = require('rubico/x/defaultsDeep')
55
+
56
+ // import rubico's Transducer module
57
+ const Transducer = require('rubico/Transducer')
44
58
  ```
45
59
 
46
- import `rubico` globally
60
+ import `rubico` in the browser.
47
61
  ```html [htmlmixed]
48
- <script src="https://unpkg.com/rubico"></script>
49
- <script src="https://unpkg.com/rubico/dist/pipe.js"></script>
50
- <script src="https://unpkg.com/rubico/dist/tap.js"></script>
51
- <script src="https://unpkg.com/rubico/dist/x/defaultsDeep.js"></script>
52
- <script>
53
- console.log(rubico) // { pipe, tap, ... }
54
- console.log(pipe) // [Function: pipe]
55
- console.log(tap) // [Function: tap]
56
- console.log(defaultsDeep) // [Function: defaultsDeep]
57
- </script>
58
-
59
- <!-- minified -->
62
+ <!-- import rubico core globally -->
63
+ <script src="https://unpkg.com/rubico/dist/global.min.js"></script>
64
+
65
+ <!-- import rubico core as rubico -->
60
66
  <script src="https://unpkg.com/rubico/dist/rubico.min.js"></script>
67
+
68
+ <!-- import an operator from rubico core -->
61
69
  <script src="https://unpkg.com/rubico/dist/pipe.min.js"></script>
62
- <script src="https://unpkg.com/rubico/dist/tap.min.js"></script>
63
- <script src="https://unpkg.com/rubico/dist/x/defaultsDeep.min.js"></script>
64
- <script>
65
- console.log(rubico) // { pipe, tap, ... }
66
- console.log(pipe) // [Function: pipe]
67
- console.log(tap) // [Function: tap]
68
- console.log(defaultsDeep) // [Function: defaultsDeep]
69
- </script>
70
- ```
71
70
 
72
- import `rubico` in the browser via ES ([JavaScript](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Modules)) Modules
73
- ```javascript
74
- import rubico from 'https://unpkg.com/rubico/dist/rubico.es.js'
75
- import pipe from 'https://unpkg.com/rubico/dist/pipe.es.js'
76
- import tap from 'https://unpkg.com/rubico/dist/tap.es.js'
77
- import defaultsDeep from 'https://unpkg.com/rubico/dist/x/defaultsDeep.es.js'
78
-
79
- // minified
80
- import rubico from 'https://unpkg.com/rubico/dist/rubico.es.min.js'
81
- import pipe from 'https://unpkg.com/rubico/dist/pipe.es.min.js'
82
- import tap from 'https://unpkg.com/rubico/dist/tap.es.min.js'
83
- import defaultsDeep from 'https://unpkg.com/rubico/dist/x/defaultsDeep.es.min.js'
84
- ```
71
+ <!-- import an operator from rubico/x -->
72
+ <script src="https://unpkg.com/rubico/dist/x/defaultsDeep.js"></script>
85
73
 
86
- import `rubico` as ESM in Node 16 and up
87
- ```javascript
88
- import rubico from 'rubico/dist/rubico.mjs'
89
- import pipe from 'rubico/dist/pipe.mjs'
90
- import tap from 'rubico/dist/tap.mjs'
91
- import defaultsDeep from 'rubico/dist/x/defaultsDeep.mjs'
92
-
93
- // minified
94
- import rubico from 'rubico/dist/rubico.min.mjs'
95
- import pipe from 'rubico/dist/pipe.min.mjs'
96
- import tap from 'rubico/dist/tap.min.mjs'
97
- import defaultsDeep from 'rubico/dist/x/defaultsDeep.min.mjs'
74
+ <!-- import rubico's Transducer module -->
75
+ <script src="https://unpkg.com/rubico/dist/Transducer.min.js"></script>
98
76
  ```
99
77
 
100
78
  # Motivation
101
79
 
102
80
  A note from the author
103
- > At a certain point in my career, I grew frustrated with the entanglement of my own code. While looking for something better, I found functional programming. I was excited by the idea of functional composition, but disillusioned by the redundancy of effectful types. I started Rubico to capitalize on the prior while rebuking the latter. Many iterations since then, the library has grown into something I personally enjoy using, and continue to use to this day.
81
+ > At a certain point in my career, I grew frustrated with the entanglement of my own code. While looking for something better, I found functional programming. I was excited by the idea of functional composition, but disillusioned by the redundancy of effectful types. I started rubico to capitalize on the prior while rebuking the latter. Many iterations since then, the library has grown into something I personally enjoy using, and continue to use to this day.
104
82
 
105
- Rubico is founded on the following principles:
83
+ rubico is founded on the following principles:
106
84
  * asynchronous code should be simple
107
85
  * functional style should not care about async
108
86
  * functional transformations should be composable, performant, and simple to express
@@ -111,54 +89,153 @@ When you import this library, you obtain the freedom that comes from having thos
111
89
 
112
90
  # Introduction
113
91
 
114
- Rubico is a module of twenty-nine operators for async-enabled functional programming in JavaScript.
92
+ rubico is a library for async-enabled functional programming in JavaScript that supports a simple and composable functional style in asynchronous environments. The core exports a small number of functions that can be grouped into categories for programming operations.
115
93
 
116
94
  ```javascript
117
95
  const {
118
- pipe, tap,
119
- switchCase, tryCatch,
120
- fork, assign, get, set, pick, omit,
96
+ // compose functions
97
+ pipe, compose,
98
+
99
+ // compose effects
100
+ tap, forEach,
101
+
102
+ // control flow
103
+ switchCase,
104
+
105
+ // handle errors
106
+ tryCatch,
107
+
108
+ // compose objects
109
+ all, assign, get, set, pick, omit,
110
+
111
+ // transform data
121
112
  map, filter, reduce, transform, flatMap,
122
- and, or, not, any, all,
113
+
114
+ // compose predicates
115
+ and, or, not, some, every,
116
+
117
+ // comparison operators
123
118
  eq, gt, lt, gte, lte,
124
- thunkify, always,
125
- curry, __,
119
+
120
+ // partial application
121
+ thunkify, always, curry, __,
126
122
  } = rubico
127
123
  ```
128
124
 
129
- These operators act sensibly on a wide range of vanilla JavaScript types to create declarative, extensible, and async-enabled function compositions.
125
+ With async-enabled, or [a]synchronous, functional programming, functions provided to the rubico operators may be asynchronous and return a Promise. Any promises provided to the operators in argument position are also resolved before continuing with the operation.
130
126
 
131
127
  ```javascript [playground]
132
- const { pipe, map } = rubico
128
+ const helloPromise = Promise.resolve('hello')
133
129
 
134
- const toTodosUrl = id => `https://jsonplaceholder.typicode.com/todos/${id}`
130
+ pipe(helloPromise, [ // helloPromise is resolved for 'hello'
131
+ async greeting => `${greeting} world`,
132
+ // the Promise returned from the async function is resolved
133
+ // and the resolved value is passed to console.log
135
134
 
136
- const logTodoByID = pipe([ // fetch a Todo and log it
137
- toTodosUrl,
138
- fetch,
139
- response => response.json(),
140
- console.log,
135
+ console.log, // hello world
141
136
  ])
137
+ ```
138
+
139
+ Most operators support two kinds of API: one eager and one tacit. With the eager API, the operator consumes the data as the first argument followed by the common arguments, executing eagerly with one call. With the tacit API, the operator takes the common arguments without the data argument, returning a lazily evaluated operator that executes once called with the data argument. Operators having both an eager and tacit API supports a natural and composable code style.
140
+
141
+ ```javascript [playground]
142
+ const myObj = { a: 1, b: 2, c: 3 }
143
+
144
+ // the first use of map is eager
145
+ const myDuplicatedSquaredObject = map(myObj, pipe([
146
+ number => [number, number],
147
+
148
+ // the second use of map is tacit
149
+ map(number => number ** 2),
150
+ ]))
151
+
152
+ console.log(myDuplicatedSquaredObject)
153
+ // { a: [1, 1], b: [4, 4], c: [9, 9] }
154
+ ```
155
+
156
+ The rubico operators are versatile and act on a wide range of vanilla JavaScript types to create declarative, extensible, and async-enabled function compositions. The same operator `map` can act on an array and also act on a `Map` data structure.
157
+
158
+ ```javascript [playground]
159
+ const { pipe, tap, map, filter } = rubico
160
+
161
+ const toTodosUrl = id => `https://jsonplaceholder.typicode.com/todos/${id}`
142
162
 
143
163
  const todoIDs = [1, 2, 3, 4, 5]
144
164
 
145
- map(logTodoByID)(todoIDs) // fetch Todos per id of TodoIDs and log them
146
- // { userId: 1, id: 4, title: 'et porro tempora', completed: true }
147
- // { userId: 1, id: 1, title: 'delectus aut autem', completed: false }
148
- // { userId: 1, id: 3, title: 'fugiat veniam minus', completed: false }
149
- // { userId: 1, id: 2, title: 'quis ut nam facilis...', completed: false }
150
- // { userId: 1, id: 5, title: 'laboriosam mollitia...', completed: false }
165
+ pipe(todoIDs, [
166
+
167
+ // fetch todos per id of todoIDs
168
+ map(pipe([
169
+ toTodosUrl,
170
+ fetch,
171
+ response => response.json(),
172
+
173
+ tap(console.log),
174
+ // { userId: 1, id: 4, title: 'et porro tempora', completed: true }
175
+ // { userId: 1, id: 1, title: 'delectus aut autem', completed: false }
176
+ // { userId: 1, id: 3, title: 'fugiat veniam minus', completed: false }
177
+ // { userId: 1, id: 2, title: 'quis ut nam facilis...', completed: false }
178
+ // { userId: 1, id: 5, title: 'laboriosam mollitia...', completed: false }
179
+ ])),
180
+
181
+ // group the todos by userId in a new Map
182
+ function createUserTodosMap(todos) {
183
+ const userTodosMap = new Map()
184
+ for (const todo of todos) {
185
+ const { userId } = todo
186
+ if (userTodosMap.has(userId)) {
187
+ userTodosMap.get(userId).push(todo)
188
+ } else {
189
+ userTodosMap.set(userId, [todo])
190
+ }
191
+ }
192
+ return userTodosMap
193
+ },
194
+
195
+ // filter for completed todos
196
+ // map iterates through each value (array of todos) of the userTodosMap
197
+ // filter iterates through each todo of the arrays of todos
198
+ map(filter(function didComplete(todo) {
199
+ return todo.completed
200
+ })),
201
+
202
+ tap(console.log),
203
+ // Map(1) {
204
+ // 1 => [ { userId: 1, id: 4, title: 'et porro tempora', completed: true } ]
205
+ // }
206
+ ])
207
+ ```
208
+
209
+ rubico also offers transducers in its `Transducer` module. You can consume these transducers with the `transform` and `compose` operators. Due to the way transducers are implemented, you should use `compose` over `pipe` to chain a left-to-right composition of transducers.
151
210
 
152
- // same as above but with limited concurrency
153
- // map.pool(2, logTodoByID)(todoIDs)
211
+ ```javascript [playground]
212
+ const isOdd = number => number % 2 == 1
213
+
214
+ const asyncSquare = async number => number ** 2
215
+
216
+ const generateNumbers = function* () {
217
+ yield 1
218
+ yield 2
219
+ yield 3
220
+ yield 4
221
+ yield 5
222
+ }
223
+
224
+ pipe(generateNumbers(), [
225
+ transform(compose([
226
+ Transducer.filter(isOdd),
227
+ Transducer.map(asyncSquare),
228
+ ]), []),
229
+ console.log, // [1, 9, 25]
230
+ ])
154
231
  ```
155
232
 
156
- For advanced asynchronous use cases, check out some of Rubico's property functions:
233
+ For advanced asynchronous use cases, check out the property functions on some of the operators like `map`, e.g.
157
234
  * `map` - apply a mapper function concurrently
158
235
  * `map.pool` - apply a mapper function concurrently with a concurrency limit
159
236
  * `map.series` - apply a mapper function serially
160
237
 
161
- For even more advanced functions, please visit `rubico/x`. You can find the full method documentation at [rubico.land/docs](https://rubico.land/docs).
238
+ For more advanced functions, please visit `rubico/x`. You can find the full documentation at [rubico.land/docs](https://rubico.land/docs).
162
239
 
163
240
  # Contributing
164
241
  Your feedback and contributions are welcome. If you have a suggestion, please raise an issue. Prior to that, please search through the issues first in case your suggestion has been made already. If you decide to work on an issue, or feel like taking initiative and contributing anything at all, feel free to create a pull request and I will get back to you shortly.
@@ -166,7 +243,7 @@ Your feedback and contributions are welcome. If you have a suggestion, please ra
166
243
  Pull requests should provide some basic context and link the relevant issue. Here is an [example pull request](https://github.com/a-synchronous/rubico/pull/12). If you are interested in contributing, the [help wanted](https://github.com/a-synchronous/rubico/issues?q=is%3Aissue+is%3Aopen+label%3A%22help+wanted%22) tag is a good place to start.
167
244
 
168
245
  # License
169
- Rubico is [MIT Licensed](https://github.com/a-synchronous/rubico/blob/master/LICENSE).
246
+ rubico is [MIT Licensed](https://github.com/a-synchronous/rubico/blob/master/LICENSE).
170
247
 
171
248
  # Support
172
249
  * minimum Node.js version: 12
package/Transducer.js CHANGED
@@ -1,3 +1,11 @@
1
+ const funcConcat = require('./_internal/funcConcat')
2
+ const reducerMap = require('./_internal/reducerMap')
3
+ const reducerFilter = require('./_internal/reducerFilter')
4
+ const reducerFlatMap = require('./_internal/reducerFlatMap')
5
+ const reducerForEach = require('./_internal/reducerForEach')
6
+ const curry2 = require('./_internal/curry2')
7
+ const __ = require('./_internal/placeholder')
8
+
1
9
  /**
2
10
  * @name Transducer
3
11
  *
@@ -6,45 +14,39 @@
6
14
  */
7
15
  const Transducer = {}
8
16
 
9
- /**
10
- * @name Transducer.pipe
11
- *
12
- * @description
13
- * Composes transducers
14
- */
15
- Transducer.pipe = function () {}
16
-
17
17
  /**
18
18
  * @name Transducer.map
19
19
  *
20
+ * @synopsis
21
+ * ```coffeescript [specscript]
22
+ * type Reducer = (accumulator any, item any)=>(nextAccumulator Promise|any)
23
+ * type Transducer = Reducer=>Reducer
24
+ *
25
+ * Transducer.map(mapperFunc function) -> mappingTransducer Transducer
26
+ * ```
27
+ *
20
28
  * @description
21
- * Create a mapping transducer by supplying `map` with a reducer. A reducer is a variadic function that depicts a relationship between an accumulator and any number of arguments. A transducer is a function that accepts a reducer as an argument and returns another reducer.
29
+ * Creates a mapping transducer with a provided reducer. A reducer is a variadic function that depicts a relationship between an accumulator and any number of arguments. A transducer is a function that accepts a reducer as an argument and returns another reducer.
22
30
  *
23
31
  * ```coffeescript [specscript]
24
- * Reducer<T> = (any, T)=>Promise|any
32
+ * type Reducer = (accumulator any, item any)=>(nextAccumulator Promise|any)
25
33
  *
26
- * Transducer = Reducer=>Reducer
34
+ * type Transducer = Reducer=>Reducer
27
35
  * ```
28
36
  *
29
- * The transducer signature enables chaining functionality for reducers. `map` is core to this mechanism, and provides a way via transducers to transform items of reducers. To `map`, reducers are just another category.
37
+ * The transducer signature enables chaining functionality for reducers. `map` is core to this mechanism, and provides a way via transducers to transform the items of reducers.
30
38
  *
31
39
  * ```javascript [playground]
32
40
  * const square = number => number ** 2
33
41
  *
34
42
  * const concat = (array, item) => array.concat(item)
35
43
  *
36
- * const mapSquare = map(square)
37
- * // mapSquare could potentially be a transducer, but at this point, it is
38
- * // undifferentiated and not necessarily locked in to transducer behavior.
39
- *
40
- * console.log(
41
- * mapSquare([1, 2, 3, 4, 5]),
42
- * ) // [1, 4, 9, 16, 25]
44
+ * const mapSquare = Transducer.map(square)
45
+ * // mapSquare is a transducer
43
46
  *
44
47
  * const squareConcatReducer = mapSquare(concat)
45
- * // now mapSquare is passed the function concat, so it assumes transducer
46
- * // position. squareConcatReducer is a reducer with chained functionality
47
- * // square and concat.
48
+ * // now mapSquare is passed the reducer function concat; squareConcatReducer
49
+ * // is a reducer with chained functionality square and concat
48
50
  *
49
51
  * console.log(
50
52
  * [1, 2, 3, 4, 5].reduce(squareConcatReducer, []),
@@ -55,12 +57,33 @@ Transducer.pipe = function () {}
55
57
  * ) // '1491625'
56
58
  * ```
57
59
  *
60
+ * Create reducers with chained functionality by using the `Transducer.map` eager API.
61
+ *
62
+ * ```javascript [playground]
63
+ * const square = number => number ** 2
64
+ *
65
+ * const concat = (array, item) => array.concat(item)
66
+ *
67
+ * const squareConcatReducer = Transducer.map(concat, square)
68
+ * // now mapSquare is passed the reducer function concat; squareConcatReducer
69
+ * // is a reducer with chained functionality square and concat
70
+ * ```
58
71
  */
59
- Transducer.map = function map() {}
72
+ Transducer.map = function transducerMap(mapper) {
73
+ return curry2(reducerMap, __, mapper)
74
+ }
60
75
 
61
76
  /**
62
77
  * @name Transducer.filter
63
78
  *
79
+ * @synopsis
80
+ * ```coffeescript [specscript]
81
+ * type Reducer = (accumulator any, item any)=>(nextAccumulator Promise|any)
82
+ * type Transducer = Reducer=>Reducer
83
+ *
84
+ * Transducer.filter(predicate function) -> filteringTransducer Transducer
85
+ * ```
86
+ *
64
87
  * @description
65
88
  * A reducer in filterable position creates a filtering reducer - one that skips items of the reducer's reducing operation if they test falsy by the predicate. It is possible to use an asynchronous predicate when filtering a reducer, however the implementation of `reduce` must support asynchronous operations. This library provides such an implementation as `reduce`.
66
89
  *
@@ -76,14 +99,19 @@ Transducer.map = function map() {}
76
99
  * ) // [1, 3, 5]
77
100
  * ```
78
101
  */
79
- Transducer.filter = function filter() {}
102
+ Transducer.filter = function transducerFilter(predicate) {
103
+ return curry2(reducerFilter, __, predicate)
104
+ }
80
105
 
81
106
  /**
82
107
  * @name Transducer.flatMap
83
108
  *
84
109
  * @synopsis
85
110
  * ```coffeescript [specscript]
86
- * flatMap(flatMapper)(reducer) -> Reducer<T>
111
+ * type Reducer = (accumulator any, item any)=>(nextAccumulator Promise|any)
112
+ * type Transducer = Reducer=>Reducer
113
+ *
114
+ * Transducer.flatMap(flatMapper) -> flatMappingTransducer Transducer
87
115
  * ```
88
116
  *
89
117
  * @description
@@ -108,6 +136,27 @@ Transducer.filter = function filter() {}
108
136
  *
109
137
  * In the case above, each item of the array of numbers returned by `powers` is called with the reducer `arrayConcat` for flattening into the final result.
110
138
  */
111
- Transducer.flatMap = function flatMap() {}
139
+ Transducer.flatMap = function transducerFlatMap(flatMapper) {
140
+ return curry2(reducerFlatMap, __, flatMapper)
141
+ }
142
+
143
+ /**
144
+ * @name Transducer.forEach
145
+ *
146
+ * @synopsis
147
+ * ```coffeescript [specscript]
148
+ * type Reducer = (accumulator any, item any)=>(nextAccumulator Promise|any)
149
+ * type Transducer = Reducer=>Reducer
150
+ *
151
+ * Transducer.forEach(func function) -> forEachTransducer Transducer
152
+ * ```
153
+ */
154
+ Transducer.forEach = function transducerForEach(func) {
155
+ return curry2(reducerForEach, __, func)
156
+ }
157
+
158
+ Transducer.passthrough = function transducerPassthrough(reducer) {
159
+ return reducer
160
+ }
112
161
 
113
162
  module.exports = Transducer