asGen.js 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677
  1. 'use strict';
  2. const {none, final, isFinalValue, getFinalValue, many, isMany, getManyValues} = require('../defs');
  3. const next = async function*(value, fns, index) {
  4. for (let i = index; i <= fns.length; ++i) {
  5. if (value && typeof value.then == 'function') {
  6. // thenable
  7. value = await value;
  8. }
  9. if (value === none) break;
  10. if (isFinalValue(value)) {
  11. const val = getFinalValue(value);
  12. if (val !== none) yield val;
  13. break;
  14. }
  15. if (isMany(value)) {
  16. const values = getManyValues(value);
  17. if (i == fns.length) {
  18. yield* values;
  19. } else {
  20. for (let j = 0; j < values.length; ++j) {
  21. yield* next(values[j], fns, i);
  22. }
  23. }
  24. break;
  25. }
  26. if (value && typeof value.next == 'function') {
  27. // generator
  28. for (;;) {
  29. let data = value.next();
  30. if (data && typeof data.then == 'function') {
  31. data = await data;
  32. }
  33. if (data.done) break;
  34. if (i == fns.length) {
  35. yield data.value;
  36. } else {
  37. yield* next(data.value, fns, i);
  38. }
  39. }
  40. break;
  41. }
  42. if (i == fns.length) {
  43. yield value;
  44. break;
  45. }
  46. value = fns[i](value);
  47. }
  48. };
  49. const nop = async function*() {};
  50. const asGen = (...fns) => {
  51. fns = fns.filter(fn => fn);
  52. if (!fns.length) return nop;
  53. if (Symbol.asyncIterator && fns[0][Symbol.asyncIterator]) {
  54. fns[0] = fns[0][Symbol.asyncIterator];
  55. } else if (Symbol.iterator && fns[0][Symbol.iterator]) {
  56. fns[0] = fns[0][Symbol.iterator];
  57. }
  58. return async function*(value) {
  59. yield* next(value, fns, 0);
  60. };
  61. };
  62. asGen.next = next;
  63. asGen.none = none;
  64. asGen.final = final;
  65. asGen.isFinalValue = isFinalValue;
  66. asGen.getFinalValue = getFinalValue;
  67. asGen.many = many;
  68. asGen.isMany = isMany;
  69. asGen.getManyValues = getManyValues;
  70. module.exports = asGen;