gen.js 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. 'use strict';
  2. const defs = 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 === defs.none) break;
  10. if (value === defs.stop) throw new defs.Stop();
  11. if (defs.isFinalValue(value)) {
  12. yield defs.getFinalValue(value);
  13. break;
  14. }
  15. if (defs.isMany(value)) {
  16. const values = defs.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. const f = fns[i];
  47. value = f(value);
  48. }
  49. };
  50. const gen = (...fns) => {
  51. fns = fns
  52. .filter(fn => fn)
  53. .flat(Infinity)
  54. .map(fn => (defs.isFunctionList(fn) ? defs.getFunctionList(fn) : fn))
  55. .flat(Infinity);
  56. if (!fns.length) {
  57. fns = [x => x];
  58. }
  59. let flushed = false;
  60. let g = async function* (value) {
  61. if (flushed) throw Error('Call to a flushed pipe.');
  62. if (value !== defs.none) {
  63. yield* next(value, fns, 0);
  64. } else {
  65. flushed = true;
  66. for (let i = 0; i < fns.length; ++i) {
  67. const f = fns[i];
  68. if (defs.isFlushable(f)) {
  69. yield* next(f(defs.none), fns, i + 1);
  70. }
  71. }
  72. }
  73. };
  74. const needToFlush = fns.some(fn => defs.isFlushable(fn));
  75. if (needToFlush) g = defs.flushable(g);
  76. return defs.setFunctionList(g, fns);
  77. };
  78. module.exports = gen;
  79. module.exports.next = next;