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.
- package/.eslintignore +5 -0
- package/.eslintrc.js +3 -2
- package/.github/workflows/nodejs.yml +1 -1
- package/AggregateReducer.js +19 -0
- package/AggregateReducer.test.js +82 -0
- package/CHANGELOG.md +22 -1
- package/README.md +158 -81
- package/Transducer.js +75 -26
- package/Transducer.test.js +117 -0
- package/_internal/ComparisonOperator.js +68 -0
- package/_internal/File.js +41 -0
- package/_internal/FlatMappingAsyncIterator.js +4 -4
- package/_internal/FlatMappingIterator.js +1 -1
- package/_internal/TimeInLoopSuite.js +138 -0
- package/_internal/areAnyValuesPromises.js +23 -0
- package/_internal/{arrayAll.js → arrayEvery.js} +4 -4
- package/_internal/arrayExtend.js +3 -2
- package/_internal/arrayFilter.js +3 -4
- package/_internal/{arrayAny.js → arraySome.js} +8 -8
- package/_internal/{asyncIteratorAll.js → asyncIteratorEvery.js} +4 -4
- package/_internal/{asyncIteratorAny.js → asyncIteratorSome.js} +4 -4
- package/_internal/curry4.test.js +25 -0
- package/_internal/curry5.test.js +29 -0
- package/_internal/curryArgs2.js +43 -0
- package/_internal/curryArgs3.test.js +21 -0
- package/_internal/curryArgs4.js +65 -0
- package/_internal/curryArgs4.test.js +25 -0
- package/_internal/equals.js +13 -0
- package/_internal/findAllFilePaths.js +22 -0
- package/_internal/funcApply.js +2 -0
- package/_internal/funcCall.js +14 -0
- package/_internal/functionArrayAll.js +26 -0
- package/_internal/functionArrayAllSeries.js +55 -0
- package/_internal/functionObjectAll.js +32 -0
- package/_internal/genericReduce.js +4 -22
- package/_internal/genericTransform.js +15 -15
- package/_internal/improvedGenericTransform.js +93 -0
- package/_internal/{iteratorAll.js → iteratorEvery.js} +4 -4
- package/_internal/{iteratorAny.js → iteratorSome.js} +6 -6
- package/_internal/leftResolverRightResolverCompare.js +19 -0
- package/_internal/leftResolverRightValueCompare.js +16 -0
- package/_internal/leftValueRightResolverCompare.js +16 -0
- package/_internal/objectReduce.js +0 -1
- package/_internal/pathResolve.js +6 -0
- package/_internal/{reducerAll.js → reducerEvery.js} +4 -4
- package/_internal/reducerFlatMap.js +4 -4
- package/_internal/reducerFlatten.js +1 -1
- package/_internal/{reducerAny.js → reducerSome.js} +4 -4
- package/_internal/timeInLoop.js +37 -0
- package/_internal/timeInLoop.test.js +18 -0
- package/_internal/timeInLoopAsync.js +35 -0
- package/_internal/timeInLoopAsync.test.js +22 -0
- package/all.js +106 -40
- package/and.js +54 -36
- package/archive/{FlatMappingIteratorCool.js → FlatMappingIterator-2020-09-28.js} +1 -1
- package/archive/_Promise-2023-05-29.js +93 -0
- package/archive/arrayMap2-2023-05-29.js +73 -0
- package/archive/benchmarks-v1.9.7/all.js +34 -0
- package/archive/benchmarks-v1.9.7/and.js +24 -0
- package/archive/benchmarks-v1.9.7/assign.js +174 -0
- package/archive/benchmarks-v1.9.7/curry.js +55 -0
- package/archive/benchmarks-v1.9.7/eq.js +25 -0
- package/archive/benchmarks-v1.9.7/filter.js +1322 -0
- package/archive/benchmarks-v1.9.7/flatMap.js +48 -0
- package/archive/benchmarks-v1.9.7/get.js +44 -0
- package/archive/benchmarks-v1.9.7/gt.js +25 -0
- package/archive/benchmarks-v1.9.7/gte.js +25 -0
- package/archive/benchmarks-v1.9.7/lt.js +25 -0
- package/archive/benchmarks-v1.9.7/lte.js +25 -0
- package/archive/benchmarks-v1.9.7/map.js +892 -0
- package/archive/benchmarks-v1.9.7/omit.js +28 -0
- package/archive/benchmarks-v1.9.7/or.js +51 -0
- package/archive/benchmarks-v1.9.7/pick.js +24 -0
- package/archive/benchmarks-v1.9.7/pipe.js +152 -0
- package/archive/benchmarks-v1.9.7/reduce.js +739 -0
- package/archive/benchmarks-v1.9.7/switchCase.js +256 -0
- package/archive/benchmarks-v1.9.7/tap.js +90 -0
- package/archive/benchmarks-v1.9.7/transform.js +218 -0
- package/archive/benchmarks-v1.9.7/tryCatch.js +108 -0
- package/assign.js +18 -8
- package/bench +65 -0
- package/benchmark-output/v1.9.7 +268 -0
- package/benchmarks/all.async.js +43 -0
- package/benchmarks/all.js +42 -33
- package/benchmarks/always.js +15 -0
- package/benchmarks/and.async.js +25 -0
- package/benchmarks/and.js +20 -19
- package/benchmarks/assign.async.js +27 -0
- package/benchmarks/assign.js +20 -167
- package/benchmarks/curry.js +34 -54
- package/benchmarks/eq.async.js +23 -0
- package/benchmarks/eq.js +17 -19
- package/benchmarks/every.async.js +19 -0
- package/benchmarks/every.js +19 -0
- package/benchmarks/filter.async.js +32 -0
- package/benchmarks/filter.js +27 -1311
- package/benchmarks/flatMap.async.js +26 -0
- package/benchmarks/flatMap.js +26 -36
- package/benchmarks/get.async.js +19 -0
- package/benchmarks/get.js +27 -32
- package/benchmarks/gt.async.js +23 -0
- package/benchmarks/gt.js +17 -19
- package/benchmarks/gte.async.js +23 -0
- package/benchmarks/gte.js +17 -19
- package/benchmarks/lt.async.js +23 -0
- package/benchmarks/lt.js +17 -19
- package/benchmarks/lte.async.js +23 -0
- package/benchmarks/lte.js +17 -19
- package/benchmarks/map.async.js +43 -0
- package/benchmarks/map.js +24 -876
- package/benchmarks/misc/Promise.js +26 -0
- package/benchmarks/misc/isPromise.js +30 -0
- package/benchmarks/misc/promiseAll.js +36 -0
- package/benchmarks/not.js +23 -0
- package/benchmarks/omit.js +30 -20
- package/benchmarks/or.async.js +25 -0
- package/benchmarks/or.js +23 -49
- package/benchmarks/pick.js +30 -16
- package/benchmarks/pipe.async.js +47 -0
- package/benchmarks/pipe.js +46 -151
- package/benchmarks/reduce.async.js +32 -0
- package/benchmarks/reduce.js +27 -728
- package/benchmarks/set.async.js +19 -0
- package/benchmarks/set.js +41 -0
- package/benchmarks/some.async.js +19 -0
- package/benchmarks/some.js +19 -0
- package/benchmarks/switchCase.async.js +27 -0
- package/benchmarks/switchCase.js +55 -256
- package/benchmarks/tap.js +10 -85
- package/benchmarks/thunkify.js +15 -0
- package/benchmarks/transform.async.js +27 -0
- package/benchmarks/transform.js +28 -206
- package/benchmarks/tryCatch.async.js +25 -0
- package/benchmarks/tryCatch.js +24 -100
- package/build +472 -0
- package/compose.js +46 -0
- package/dist/Transducer.es.js +481 -0
- package/dist/Transducer.es.min.js +7 -0
- package/dist/Transducer.js +488 -0
- package/dist/Transducer.min.js +8 -0
- package/dist/Transducer.mjs +481 -0
- package/dist/__.es.js +2 -2
- package/dist/__.es.min.js +2 -2
- package/dist/__.js +2 -2
- package/dist/__.min.js +2 -2
- package/dist/__.mjs +2 -2
- package/dist/all.es.js +173 -92
- package/dist/all.es.min.js +3 -3
- package/dist/all.js +173 -92
- package/dist/all.min.js +3 -3
- package/dist/all.mjs +173 -92
- package/dist/always.es.js +2 -2
- package/dist/always.es.min.js +2 -2
- package/dist/always.js +2 -2
- package/dist/always.min.js +2 -2
- package/dist/always.mjs +2 -2
- package/dist/and.es.js +101 -32
- package/dist/and.es.min.js +3 -3
- package/dist/and.js +101 -32
- package/dist/and.min.js +3 -3
- package/dist/and.mjs +101 -32
- package/dist/assign.es.js +37 -10
- package/dist/assign.es.min.js +3 -3
- package/dist/assign.js +37 -10
- package/dist/assign.min.js +3 -3
- package/dist/assign.mjs +37 -10
- package/dist/compose.es.js +71 -0
- package/dist/compose.es.min.js +7 -0
- package/dist/compose.js +78 -0
- package/dist/compose.min.js +8 -0
- package/dist/compose.mjs +71 -0
- package/dist/curry.es.js +2 -2
- package/dist/curry.es.min.js +2 -2
- package/dist/curry.js +2 -2
- package/dist/curry.min.js +2 -2
- package/dist/curry.mjs +2 -2
- package/dist/eq.es.js +172 -49
- package/dist/eq.es.min.js +3 -3
- package/dist/eq.js +172 -49
- package/dist/eq.min.js +3 -3
- package/dist/eq.mjs +172 -49
- package/dist/every.es.js +194 -0
- package/dist/every.es.min.js +7 -0
- package/dist/every.js +201 -0
- package/dist/every.min.js +8 -0
- package/dist/every.mjs +194 -0
- package/dist/filter.es.js +14 -108
- package/dist/filter.es.min.js +3 -3
- package/dist/filter.js +14 -108
- package/dist/filter.min.js +3 -3
- package/dist/filter.mjs +14 -108
- package/dist/flatMap.es.js +27 -214
- package/dist/flatMap.es.min.js +3 -3
- package/dist/flatMap.js +27 -214
- package/dist/flatMap.min.js +3 -3
- package/dist/flatMap.mjs +27 -214
- package/dist/{x/forEach.es.js → forEach.es.js} +52 -70
- package/dist/forEach.es.min.js +7 -0
- package/dist/{x/forEach.js → forEach.js} +52 -70
- package/dist/forEach.min.js +8 -0
- package/dist/{x/forEach.mjs → forEach.mjs} +52 -70
- package/dist/get.es.js +60 -5
- package/dist/get.es.min.js +3 -3
- package/dist/get.js +60 -5
- package/dist/get.min.js +3 -3
- package/dist/get.mjs +60 -5
- package/dist/gt.es.js +171 -48
- package/dist/gt.es.min.js +3 -3
- package/dist/gt.js +171 -48
- package/dist/gt.min.js +3 -3
- package/dist/gt.mjs +171 -48
- package/dist/gte.es.js +171 -48
- package/dist/gte.es.min.js +3 -3
- package/dist/gte.js +171 -48
- package/dist/gte.min.js +3 -3
- package/dist/gte.mjs +171 -48
- package/dist/lt.es.js +171 -48
- package/dist/lt.es.min.js +3 -3
- package/dist/lt.js +171 -48
- package/dist/lt.min.js +3 -3
- package/dist/lt.mjs +171 -48
- package/dist/lte.es.js +171 -48
- package/dist/lte.es.min.js +3 -3
- package/dist/lte.js +171 -48
- package/dist/lte.min.js +3 -3
- package/dist/lte.mjs +171 -48
- package/dist/map.es.js +12 -101
- package/dist/map.es.min.js +3 -3
- package/dist/map.js +12 -101
- package/dist/map.min.js +3 -3
- package/dist/map.mjs +12 -101
- package/dist/not.es.js +75 -14
- package/dist/not.es.min.js +3 -3
- package/dist/not.js +75 -14
- package/dist/not.min.js +3 -3
- package/dist/not.mjs +75 -14
- package/dist/omit.es.js +7 -2
- package/dist/omit.es.min.js +3 -3
- package/dist/omit.js +7 -2
- package/dist/omit.min.js +3 -3
- package/dist/omit.mjs +7 -2
- package/dist/or.es.js +99 -31
- package/dist/or.es.min.js +3 -3
- package/dist/or.js +99 -31
- package/dist/or.min.js +3 -3
- package/dist/or.mjs +99 -31
- package/dist/pick.es.js +8 -3
- package/dist/pick.es.min.js +3 -3
- package/dist/pick.js +8 -3
- package/dist/pick.min.js +3 -3
- package/dist/pick.mjs +8 -3
- package/dist/pipe.es.js +42 -46
- package/dist/pipe.es.min.js +3 -3
- package/dist/pipe.js +42 -46
- package/dist/pipe.min.js +3 -3
- package/dist/pipe.mjs +42 -46
- package/dist/reduce.es.js +52 -94
- package/dist/reduce.es.min.js +3 -3
- package/dist/reduce.js +52 -94
- package/dist/reduce.min.js +3 -3
- package/dist/reduce.mjs +52 -94
- package/dist/rubico.es.js +762 -883
- package/dist/rubico.es.min.js +3 -3
- package/dist/rubico.global.js +2794 -0
- package/dist/rubico.global.min.js +8 -0
- package/dist/rubico.js +739 -869
- package/dist/rubico.min.js +3 -3
- package/dist/rubico.mjs +762 -883
- package/dist/set.es.js +18 -3
- package/dist/set.es.min.js +3 -3
- package/dist/set.js +18 -3
- package/dist/set.min.js +3 -3
- package/dist/set.mjs +18 -3
- package/dist/{any.js → some.es.js} +64 -54
- package/dist/some.es.min.js +7 -0
- package/dist/{any.mjs → some.js} +72 -48
- package/dist/some.min.js +8 -0
- package/dist/{any.es.js → some.mjs} +65 -48
- package/dist/switchCase.es.js +55 -5
- package/dist/switchCase.es.min.js +3 -3
- package/dist/switchCase.js +55 -5
- package/dist/switchCase.min.js +3 -3
- package/dist/switchCase.mjs +55 -5
- package/dist/tap.es.js +2 -9
- package/dist/tap.es.min.js +3 -3
- package/dist/tap.js +2 -9
- package/dist/tap.min.js +3 -3
- package/dist/tap.mjs +2 -9
- package/dist/thunkify.es.js +45 -2
- package/dist/thunkify.es.min.js +3 -3
- package/dist/thunkify.js +45 -2
- package/dist/thunkify.min.js +3 -3
- package/dist/thunkify.mjs +45 -2
- package/dist/transform.es.js +35 -71
- package/dist/transform.es.min.js +3 -3
- package/dist/transform.js +35 -71
- package/dist/transform.min.js +3 -3
- package/dist/transform.mjs +35 -71
- package/dist/tryCatch.es.js +33 -17
- package/dist/tryCatch.es.min.js +3 -3
- package/dist/tryCatch.js +33 -17
- package/dist/tryCatch.min.js +3 -3
- package/dist/tryCatch.mjs +33 -17
- package/dist/x/append.es.js +2 -2
- package/dist/x/append.es.min.js +2 -2
- package/dist/x/append.js +2 -2
- package/dist/x/append.min.js +2 -2
- package/dist/x/append.mjs +2 -2
- package/dist/x/callProp.es.js +2 -2
- package/dist/x/callProp.es.min.js +2 -2
- package/dist/x/callProp.js +2 -2
- package/dist/x/callProp.min.js +2 -2
- package/dist/x/callProp.mjs +2 -2
- package/dist/x/defaultsDeep.es.js +2 -2
- package/dist/x/defaultsDeep.es.min.js +2 -2
- package/dist/x/defaultsDeep.js +2 -2
- package/dist/x/defaultsDeep.min.js +2 -2
- package/dist/x/defaultsDeep.mjs +2 -2
- package/dist/x/differenceWith.es.js +7 -7
- package/dist/x/differenceWith.es.min.js +3 -3
- package/dist/x/differenceWith.js +7 -7
- package/dist/x/differenceWith.min.js +2 -2
- package/dist/x/differenceWith.mjs +7 -7
- package/dist/x/filterOut.es.js +60 -117
- package/dist/x/filterOut.es.min.js +3 -3
- package/dist/x/filterOut.js +60 -117
- package/dist/x/filterOut.min.js +3 -3
- package/dist/x/filterOut.mjs +60 -117
- package/dist/x/find.es.js +2 -2
- package/dist/x/find.es.min.js +2 -2
- package/dist/x/find.js +2 -2
- package/dist/x/find.min.js +2 -2
- package/dist/x/find.mjs +2 -2
- package/dist/x/findIndex.es.js +2 -2
- package/dist/x/findIndex.es.min.js +2 -2
- package/dist/x/findIndex.js +2 -2
- package/dist/x/findIndex.min.js +2 -2
- package/dist/x/findIndex.mjs +2 -2
- package/dist/x/first.es.js +2 -2
- package/dist/x/first.es.min.js +2 -2
- package/dist/x/first.js +2 -2
- package/dist/x/first.min.js +2 -2
- package/dist/x/first.mjs +2 -2
- package/dist/x/flatten.es.js +20 -207
- package/dist/x/flatten.es.min.js +3 -3
- package/dist/x/flatten.js +20 -207
- package/dist/x/flatten.min.js +3 -3
- package/dist/x/flatten.mjs +20 -207
- package/dist/x/groupBy.es.js +52 -94
- package/dist/x/groupBy.es.min.js +3 -3
- package/dist/x/groupBy.js +52 -94
- package/dist/x/groupBy.min.js +3 -3
- package/dist/x/groupBy.mjs +52 -94
- package/dist/x/has.es.js +2 -2
- package/dist/x/has.es.min.js +2 -2
- package/dist/x/has.js +2 -2
- package/dist/x/has.min.js +2 -2
- package/dist/x/has.mjs +2 -2
- package/dist/x/identity.es.js +2 -2
- package/dist/x/identity.es.min.js +2 -2
- package/dist/x/identity.js +2 -2
- package/dist/x/identity.min.js +2 -2
- package/dist/x/identity.mjs +2 -2
- package/dist/x/includes.es.js +2 -2
- package/dist/x/includes.es.min.js +2 -2
- package/dist/x/includes.js +2 -2
- package/dist/x/includes.min.js +2 -2
- package/dist/x/includes.mjs +2 -2
- package/dist/x/isDeepEqual.es.js +2 -2
- package/dist/x/isDeepEqual.es.min.js +2 -2
- package/dist/x/isDeepEqual.js +2 -2
- package/dist/x/isDeepEqual.min.js +2 -2
- package/dist/x/isDeepEqual.mjs +2 -2
- package/dist/x/isEmpty.es.js +2 -2
- package/dist/x/isEmpty.es.min.js +2 -2
- package/dist/x/isEmpty.js +2 -2
- package/dist/x/isEmpty.min.js +2 -2
- package/dist/x/isEmpty.mjs +2 -2
- package/dist/x/isEqual.es.js +2 -2
- package/dist/x/isEqual.es.min.js +2 -2
- package/dist/x/isEqual.js +2 -2
- package/dist/x/isEqual.min.js +2 -2
- package/dist/x/isEqual.mjs +2 -2
- package/dist/x/isFunction.es.js +2 -2
- package/dist/x/isFunction.es.min.js +2 -2
- package/dist/x/isFunction.js +2 -2
- package/dist/x/isFunction.min.js +2 -2
- package/dist/x/isFunction.mjs +2 -2
- package/dist/x/isIn.es.js +2 -2
- package/dist/x/isIn.es.min.js +2 -2
- package/dist/x/isIn.js +2 -2
- package/dist/x/isIn.min.js +2 -2
- package/dist/x/isIn.mjs +2 -2
- package/dist/x/isObject.es.js +2 -2
- package/dist/x/isObject.es.min.js +2 -2
- package/dist/x/isObject.js +2 -2
- package/dist/x/isObject.min.js +2 -2
- package/dist/x/isObject.mjs +2 -2
- package/dist/x/isString.es.js +2 -2
- package/dist/x/isString.es.min.js +2 -2
- package/dist/x/isString.js +2 -2
- package/dist/x/isString.min.js +2 -2
- package/dist/x/isString.mjs +2 -2
- package/dist/x/keys.es.js +2 -2
- package/dist/x/keys.es.min.js +2 -2
- package/dist/x/keys.js +2 -2
- package/dist/x/keys.min.js +2 -2
- package/dist/x/keys.mjs +2 -2
- package/dist/x/last.es.js +2 -2
- package/dist/x/last.es.min.js +2 -2
- package/dist/x/last.js +2 -2
- package/dist/x/last.min.js +2 -2
- package/dist/x/last.mjs +2 -2
- package/dist/x/maxBy.es.js +59 -6
- package/dist/x/maxBy.es.min.js +3 -3
- package/dist/x/maxBy.js +59 -6
- package/dist/x/maxBy.min.js +3 -3
- package/dist/x/maxBy.mjs +59 -6
- package/dist/x/noop.es.js +2 -2
- package/dist/x/noop.es.min.js +2 -2
- package/dist/x/noop.js +2 -2
- package/dist/x/noop.min.js +2 -2
- package/dist/x/noop.mjs +2 -2
- package/dist/x/pluck.es.js +32 -103
- package/dist/x/pluck.es.min.js +3 -3
- package/dist/x/pluck.js +32 -103
- package/dist/x/pluck.min.js +3 -3
- package/dist/x/pluck.mjs +32 -103
- package/dist/x/prepend.es.js +2 -2
- package/dist/x/prepend.es.min.js +2 -2
- package/dist/x/prepend.js +2 -2
- package/dist/x/prepend.min.js +2 -2
- package/dist/x/prepend.mjs +2 -2
- package/dist/x/size.es.js +2 -2
- package/dist/x/size.es.min.js +2 -2
- package/dist/x/size.js +2 -2
- package/dist/x/size.min.js +2 -2
- package/dist/x/size.mjs +2 -2
- package/dist/x/trace.es.js +2 -9
- package/dist/x/trace.es.min.js +3 -3
- package/dist/x/trace.js +2 -9
- package/dist/x/trace.min.js +3 -3
- package/dist/x/trace.mjs +2 -9
- package/dist/x/unionWith.es.js +2 -2
- package/dist/x/unionWith.es.min.js +2 -2
- package/dist/x/unionWith.js +2 -2
- package/dist/x/unionWith.min.js +2 -2
- package/dist/x/unionWith.mjs +2 -2
- package/dist/x/uniq.es.js +2 -2
- package/dist/x/uniq.es.min.js +2 -2
- package/dist/x/uniq.js +2 -2
- package/dist/x/uniq.min.js +2 -2
- package/dist/x/uniq.mjs +2 -2
- package/dist/x/unless.es.js +2 -2
- package/dist/x/unless.es.min.js +2 -2
- package/dist/x/unless.js +2 -2
- package/dist/x/unless.min.js +2 -2
- package/dist/x/unless.mjs +2 -2
- package/dist/x/values.es.js +2 -2
- package/dist/x/values.es.min.js +2 -2
- package/dist/x/values.js +2 -2
- package/dist/x/values.min.js +2 -2
- package/dist/x/values.mjs +2 -2
- package/dist/x/when.es.js +2 -2
- package/dist/x/when.es.min.js +2 -2
- package/dist/x/when.js +2 -2
- package/dist/x/when.min.js +2 -2
- package/dist/x/when.mjs +2 -2
- package/dist-test.js +31 -2042
- package/eq.js +21 -63
- package/es.js +739 -869
- package/every.js +94 -0
- package/filter.js +23 -91
- package/flatMap.js +75 -70
- package/forEach.js +82 -0
- package/get.js +23 -6
- package/global.js +6 -4
- package/gt.js +15 -58
- package/gte.js +14 -57
- package/index.js +739 -869
- package/lt.js +17 -60
- package/lte.js +14 -57
- package/map.js +9 -94
- package/memory-usage/reduce.js +2 -2
- package/not.js +27 -45
- package/omit.js +4 -0
- package/or.js +50 -33
- package/package.json +7 -7
- package/pick.js +5 -1
- package/pipe.js +13 -50
- package/reduce.js +42 -51
- package/rubico.js +22 -11
- package/set.js +43 -10
- package/some.js +105 -0
- package/switchCase.js +13 -3
- package/tap.js +0 -24
- package/test.js +847 -1096
- package/thunkify.js +9 -0
- package/tmp.js +1 -0
- package/transform.js +31 -16
- package/tryCatch.js +19 -15
- package/x/differenceWith.js +3 -3
- package/x/filterOut.test.js +6 -6
- package/x/flatten.js +1 -1
- package/x/flatten.test.js +0 -8
- package/x/index.js +0 -2
- package/x/pluck.js +18 -27
- package/x/pluck.test.js +3 -7
- package/x/timeInLoop.js +16 -25
- package/x/timeInLoop.test.js +8 -4
- package/any.js +0 -82
- package/dist/__.min.mjs +0 -7
- package/dist/all.min.mjs +0 -7
- package/dist/always.min.mjs +0 -7
- package/dist/and.min.mjs +0 -7
- package/dist/any.es.min.js +0 -7
- package/dist/any.min.js +0 -8
- package/dist/any.min.mjs +0 -7
- package/dist/assign.min.mjs +0 -7
- package/dist/curry.min.mjs +0 -7
- package/dist/eq.min.mjs +0 -7
- package/dist/filter.min.mjs +0 -7
- package/dist/flatMap.min.mjs +0 -7
- package/dist/fork.es.js +0 -159
- package/dist/fork.es.min.js +0 -7
- package/dist/fork.js +0 -166
- package/dist/fork.min.js +0 -8
- package/dist/fork.min.mjs +0 -7
- package/dist/fork.mjs +0 -159
- package/dist/get.min.mjs +0 -7
- package/dist/gt.min.mjs +0 -7
- package/dist/gte.min.mjs +0 -7
- package/dist/lt.min.mjs +0 -7
- package/dist/lte.min.mjs +0 -7
- package/dist/map.min.mjs +0 -7
- package/dist/not.min.mjs +0 -7
- package/dist/omit.min.mjs +0 -7
- package/dist/or.min.mjs +0 -7
- package/dist/pick.min.mjs +0 -7
- package/dist/pipe.min.mjs +0 -7
- package/dist/reduce.min.mjs +0 -7
- package/dist/rubico.min.mjs +0 -7
- package/dist/set.min.mjs +0 -7
- package/dist/switchCase.min.mjs +0 -7
- package/dist/tap.min.mjs +0 -7
- package/dist/thunkify.min.mjs +0 -7
- package/dist/transform.min.mjs +0 -7
- package/dist/tryCatch.min.mjs +0 -7
- package/dist/x/append.min.mjs +0 -7
- package/dist/x/callProp.min.mjs +0 -7
- package/dist/x/defaultsDeep.min.mjs +0 -7
- package/dist/x/differenceWith.min.mjs +0 -7
- package/dist/x/filterOut.min.mjs +0 -7
- package/dist/x/find.min.mjs +0 -7
- package/dist/x/findIndex.min.mjs +0 -7
- package/dist/x/first.min.mjs +0 -7
- package/dist/x/flatten.min.mjs +0 -7
- package/dist/x/forEach.es.min.js +0 -7
- package/dist/x/forEach.min.js +0 -8
- package/dist/x/forEach.min.mjs +0 -7
- package/dist/x/groupBy.min.mjs +0 -7
- package/dist/x/has.min.mjs +0 -7
- package/dist/x/identity.min.mjs +0 -7
- package/dist/x/includes.min.mjs +0 -7
- package/dist/x/isDeepEqual.min.mjs +0 -7
- package/dist/x/isEmpty.min.mjs +0 -7
- package/dist/x/isEqual.min.mjs +0 -7
- package/dist/x/isFunction.min.mjs +0 -7
- package/dist/x/isIn.min.mjs +0 -7
- package/dist/x/isObject.min.mjs +0 -7
- package/dist/x/isString.min.mjs +0 -7
- package/dist/x/keys.min.mjs +0 -7
- package/dist/x/last.min.mjs +0 -7
- package/dist/x/maxBy.min.mjs +0 -7
- package/dist/x/noop.min.mjs +0 -7
- package/dist/x/pluck.min.mjs +0 -7
- package/dist/x/prepend.min.mjs +0 -7
- package/dist/x/size.min.mjs +0 -7
- package/dist/x/trace.min.mjs +0 -7
- package/dist/x/unionWith.min.mjs +0 -7
- package/dist/x/uniq.min.mjs +0 -7
- package/dist/x/unless.min.mjs +0 -7
- package/dist/x/values.min.mjs +0 -7
- package/dist/x/when.min.mjs +0 -7
- package/distributor.js +0 -495
- package/fork.js +0 -101
- package/x/forEach.js +0 -93
- package/x/forEach.test.js +0 -218
- /package/{benchmarks → archive/benchmarks-v1.9.7}/any.js +0 -0
- /package/{x/forEach.benchmark.js → archive/benchmarks-v1.9.7/forEach.js} +0 -0
- /package/{benchmarks → archive/benchmarks-v1.9.7}/fork.js +0 -0
- /package/{benchmarks → archive/benchmarks-v1.9.7}/integration.js +0 -0
- /package/{benchmarks → archive/benchmarks-v1.9.7}/misc.js +0 -0
package/.eslintignore
ADDED
package/.eslintrc.js
CHANGED
|
@@ -192,7 +192,7 @@ module.exports = {
|
|
|
192
192
|
'allowForLoopAfterthoughts': true
|
|
193
193
|
}
|
|
194
194
|
],
|
|
195
|
-
'no-process-env': '
|
|
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-
|
|
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',
|
|
@@ -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
|
-
##
|
|
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
|
|
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.
|
|
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
|
|
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
|
-
|
|
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`
|
|
60
|
+
import `rubico` in the browser.
|
|
47
61
|
```html [htmlmixed]
|
|
48
|
-
|
|
49
|
-
<script src="https://unpkg.com/rubico/dist/
|
|
50
|
-
|
|
51
|
-
|
|
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
|
|
73
|
-
|
|
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
|
|
87
|
-
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
119
|
-
|
|
120
|
-
|
|
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
|
-
|
|
113
|
+
|
|
114
|
+
// compose predicates
|
|
115
|
+
and, or, not, some, every,
|
|
116
|
+
|
|
117
|
+
// comparison operators
|
|
123
118
|
eq, gt, lt, gte, lte,
|
|
124
|
-
|
|
125
|
-
|
|
119
|
+
|
|
120
|
+
// partial application
|
|
121
|
+
thunkify, always, curry, __,
|
|
126
122
|
} = rubico
|
|
127
123
|
```
|
|
128
124
|
|
|
129
|
-
|
|
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
|
|
128
|
+
const helloPromise = Promise.resolve('hello')
|
|
133
129
|
|
|
134
|
-
|
|
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
|
-
|
|
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
|
-
|
|
146
|
-
|
|
147
|
-
//
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
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
|
-
|
|
153
|
-
|
|
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
|
|
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
|
|
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
|
-
|
|
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
|
-
*
|
|
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
|
|
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.
|
|
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
|
|
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
|
|
46
|
-
* //
|
|
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
|
|
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
|
|
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
|
-
*
|
|
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
|
|
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
|