@mondaydotcomorg/atp-compiler 0.17.14

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 (285) hide show
  1. package/README.md +382 -0
  2. package/__tests__/integration/all-array-methods-native.test.ts +321 -0
  3. package/__tests__/integration/all-callback-types.test.ts +406 -0
  4. package/__tests__/integration/comprehensive-edge-cases.test.ts +701 -0
  5. package/__tests__/integration/native-behavior-verification.test.ts +340 -0
  6. package/__tests__/integration/semantic-correctness.test.ts +354 -0
  7. package/__tests__/integration/threshold-tests.test.ts +529 -0
  8. package/__tests__/unit/batch-optimizer.test.ts +253 -0
  9. package/__tests__/unit/checkpoint-manager.test.ts +145 -0
  10. package/__tests__/unit/detector.test.ts +346 -0
  11. package/dist/atp-compiler/src/index.d.ts +6 -0
  12. package/dist/atp-compiler/src/index.d.ts.map +1 -0
  13. package/dist/atp-compiler/src/index.js +6 -0
  14. package/dist/atp-compiler/src/index.js.map +1 -0
  15. package/dist/atp-compiler/src/runtime/batch-parallel.d.ts +3 -0
  16. package/dist/atp-compiler/src/runtime/batch-parallel.d.ts.map +1 -0
  17. package/dist/atp-compiler/src/runtime/batch-parallel.js +13 -0
  18. package/dist/atp-compiler/src/runtime/batch-parallel.js.map +1 -0
  19. package/dist/atp-compiler/src/runtime/checkpoint-manager.d.ts +19 -0
  20. package/dist/atp-compiler/src/runtime/checkpoint-manager.d.ts.map +1 -0
  21. package/dist/atp-compiler/src/runtime/checkpoint-manager.js +81 -0
  22. package/dist/atp-compiler/src/runtime/checkpoint-manager.js.map +1 -0
  23. package/dist/atp-compiler/src/runtime/context.d.ts +8 -0
  24. package/dist/atp-compiler/src/runtime/context.d.ts.map +1 -0
  25. package/dist/atp-compiler/src/runtime/context.js +25 -0
  26. package/dist/atp-compiler/src/runtime/context.js.map +1 -0
  27. package/dist/atp-compiler/src/runtime/errors.d.ts +38 -0
  28. package/dist/atp-compiler/src/runtime/errors.d.ts.map +1 -0
  29. package/dist/atp-compiler/src/runtime/errors.js +61 -0
  30. package/dist/atp-compiler/src/runtime/errors.js.map +1 -0
  31. package/dist/atp-compiler/src/runtime/index.d.ts +16 -0
  32. package/dist/atp-compiler/src/runtime/index.d.ts.map +1 -0
  33. package/dist/atp-compiler/src/runtime/index.js +20 -0
  34. package/dist/atp-compiler/src/runtime/index.js.map +1 -0
  35. package/dist/atp-compiler/src/runtime/resumable-arrays.d.ts +9 -0
  36. package/dist/atp-compiler/src/runtime/resumable-arrays.d.ts.map +1 -0
  37. package/dist/atp-compiler/src/runtime/resumable-arrays.js +179 -0
  38. package/dist/atp-compiler/src/runtime/resumable-arrays.js.map +1 -0
  39. package/dist/atp-compiler/src/runtime/resumable-loops.d.ts +4 -0
  40. package/dist/atp-compiler/src/runtime/resumable-loops.d.ts.map +1 -0
  41. package/dist/atp-compiler/src/runtime/resumable-loops.js +61 -0
  42. package/dist/atp-compiler/src/runtime/resumable-loops.js.map +1 -0
  43. package/dist/atp-compiler/src/runtime/resumable-parallel.d.ts +3 -0
  44. package/dist/atp-compiler/src/runtime/resumable-parallel.d.ts.map +1 -0
  45. package/dist/atp-compiler/src/runtime/resumable-parallel.js +44 -0
  46. package/dist/atp-compiler/src/runtime/resumable-parallel.js.map +1 -0
  47. package/dist/atp-compiler/src/transformer/array-transformer-batch.d.ts +13 -0
  48. package/dist/atp-compiler/src/transformer/array-transformer-batch.d.ts.map +1 -0
  49. package/dist/atp-compiler/src/transformer/array-transformer-batch.js +55 -0
  50. package/dist/atp-compiler/src/transformer/array-transformer-batch.js.map +1 -0
  51. package/dist/atp-compiler/src/transformer/array-transformer-sequential.d.ts +6 -0
  52. package/dist/atp-compiler/src/transformer/array-transformer-sequential.d.ts.map +1 -0
  53. package/dist/atp-compiler/src/transformer/array-transformer-sequential.js +23 -0
  54. package/dist/atp-compiler/src/transformer/array-transformer-sequential.js.map +1 -0
  55. package/dist/atp-compiler/src/transformer/array-transformer-utils.d.ts +18 -0
  56. package/dist/atp-compiler/src/transformer/array-transformer-utils.d.ts.map +1 -0
  57. package/dist/atp-compiler/src/transformer/array-transformer-utils.js +69 -0
  58. package/dist/atp-compiler/src/transformer/array-transformer-utils.js.map +1 -0
  59. package/dist/atp-compiler/src/transformer/array-transformer-wrappers.d.ts +26 -0
  60. package/dist/atp-compiler/src/transformer/array-transformer-wrappers.d.ts.map +1 -0
  61. package/dist/atp-compiler/src/transformer/array-transformer-wrappers.js +88 -0
  62. package/dist/atp-compiler/src/transformer/array-transformer-wrappers.js.map +1 -0
  63. package/dist/atp-compiler/src/transformer/array-transformer.d.ts +12 -0
  64. package/dist/atp-compiler/src/transformer/array-transformer.d.ts.map +1 -0
  65. package/dist/atp-compiler/src/transformer/array-transformer.js +47 -0
  66. package/dist/atp-compiler/src/transformer/array-transformer.js.map +1 -0
  67. package/dist/atp-compiler/src/transformer/batch-detector.d.ts +16 -0
  68. package/dist/atp-compiler/src/transformer/batch-detector.d.ts.map +1 -0
  69. package/dist/atp-compiler/src/transformer/batch-detector.js +131 -0
  70. package/dist/atp-compiler/src/transformer/batch-detector.js.map +1 -0
  71. package/dist/atp-compiler/src/transformer/batch-optimizer.d.ts +27 -0
  72. package/dist/atp-compiler/src/transformer/batch-optimizer.d.ts.map +1 -0
  73. package/dist/atp-compiler/src/transformer/batch-optimizer.js +244 -0
  74. package/dist/atp-compiler/src/transformer/batch-optimizer.js.map +1 -0
  75. package/dist/atp-compiler/src/transformer/detector.d.ts +9 -0
  76. package/dist/atp-compiler/src/transformer/detector.d.ts.map +1 -0
  77. package/dist/atp-compiler/src/transformer/detector.js +141 -0
  78. package/dist/atp-compiler/src/transformer/detector.js.map +1 -0
  79. package/dist/atp-compiler/src/transformer/index.d.ts +22 -0
  80. package/dist/atp-compiler/src/transformer/index.d.ts.map +1 -0
  81. package/dist/atp-compiler/src/transformer/index.js +132 -0
  82. package/dist/atp-compiler/src/transformer/index.js.map +1 -0
  83. package/dist/atp-compiler/src/transformer/loop-transformer.d.ts +25 -0
  84. package/dist/atp-compiler/src/transformer/loop-transformer.d.ts.map +1 -0
  85. package/dist/atp-compiler/src/transformer/loop-transformer.js +193 -0
  86. package/dist/atp-compiler/src/transformer/loop-transformer.js.map +1 -0
  87. package/dist/atp-compiler/src/transformer/promise-transformer.d.ts +17 -0
  88. package/dist/atp-compiler/src/transformer/promise-transformer.d.ts.map +1 -0
  89. package/dist/atp-compiler/src/transformer/promise-transformer.js +132 -0
  90. package/dist/atp-compiler/src/transformer/promise-transformer.js.map +1 -0
  91. package/dist/atp-compiler/src/transformer/utils.d.ts +15 -0
  92. package/dist/atp-compiler/src/transformer/utils.d.ts.map +1 -0
  93. package/dist/atp-compiler/src/transformer/utils.js +118 -0
  94. package/dist/atp-compiler/src/transformer/utils.js.map +1 -0
  95. package/dist/atp-compiler/src/types.d.ts +57 -0
  96. package/dist/atp-compiler/src/types.d.ts.map +1 -0
  97. package/dist/atp-compiler/src/types.js +23 -0
  98. package/dist/atp-compiler/src/types.js.map +1 -0
  99. package/dist/protocol/src/auth.d.ts +173 -0
  100. package/dist/protocol/src/auth.d.ts.map +1 -0
  101. package/dist/protocol/src/auth.js +202 -0
  102. package/dist/protocol/src/auth.js.map +1 -0
  103. package/dist/protocol/src/index.d.ts +7 -0
  104. package/dist/protocol/src/index.d.ts.map +1 -0
  105. package/dist/protocol/src/index.js +7 -0
  106. package/dist/protocol/src/index.js.map +1 -0
  107. package/dist/protocol/src/oauth.d.ts +63 -0
  108. package/dist/protocol/src/oauth.d.ts.map +1 -0
  109. package/dist/protocol/src/oauth.js +5 -0
  110. package/dist/protocol/src/oauth.js.map +1 -0
  111. package/dist/protocol/src/providers.d.ts +167 -0
  112. package/dist/protocol/src/providers.d.ts.map +1 -0
  113. package/dist/protocol/src/providers.js +33 -0
  114. package/dist/protocol/src/providers.js.map +1 -0
  115. package/dist/protocol/src/schemas.d.ts +6 -0
  116. package/dist/protocol/src/schemas.d.ts.map +1 -0
  117. package/dist/protocol/src/schemas.js +51 -0
  118. package/dist/protocol/src/schemas.js.map +1 -0
  119. package/dist/protocol/src/types.d.ts +489 -0
  120. package/dist/protocol/src/types.d.ts.map +1 -0
  121. package/dist/protocol/src/types.js +88 -0
  122. package/dist/protocol/src/types.js.map +1 -0
  123. package/dist/protocol/src/validation.d.ts +76 -0
  124. package/dist/protocol/src/validation.d.ts.map +1 -0
  125. package/dist/protocol/src/validation.js +129 -0
  126. package/dist/protocol/src/validation.js.map +1 -0
  127. package/dist/provenance/src/ast/instrumentor.d.ts +37 -0
  128. package/dist/provenance/src/ast/instrumentor.d.ts.map +1 -0
  129. package/dist/provenance/src/ast/instrumentor.js +299 -0
  130. package/dist/provenance/src/ast/instrumentor.js.map +1 -0
  131. package/dist/provenance/src/index.d.ts +7 -0
  132. package/dist/provenance/src/index.d.ts.map +1 -0
  133. package/dist/provenance/src/index.js +7 -0
  134. package/dist/provenance/src/index.js.map +1 -0
  135. package/dist/provenance/src/policies/engine.d.ts +71 -0
  136. package/dist/provenance/src/policies/engine.d.ts.map +1 -0
  137. package/dist/provenance/src/policies/engine.js +433 -0
  138. package/dist/provenance/src/policies/engine.js.map +1 -0
  139. package/dist/provenance/src/registry.d.ts +94 -0
  140. package/dist/provenance/src/registry.d.ts.map +1 -0
  141. package/dist/provenance/src/registry.js +445 -0
  142. package/dist/provenance/src/registry.js.map +1 -0
  143. package/dist/provenance/src/tokens.d.ts +49 -0
  144. package/dist/provenance/src/tokens.d.ts.map +1 -0
  145. package/dist/provenance/src/tokens.js +239 -0
  146. package/dist/provenance/src/tokens.js.map +1 -0
  147. package/dist/provenance/src/types.d.ts +150 -0
  148. package/dist/provenance/src/types.d.ts.map +1 -0
  149. package/dist/provenance/src/types.js +47 -0
  150. package/dist/provenance/src/types.js.map +1 -0
  151. package/dist/runtime/src/approval/handler.d.ts +12 -0
  152. package/dist/runtime/src/approval/handler.d.ts.map +1 -0
  153. package/dist/runtime/src/approval/handler.js +17 -0
  154. package/dist/runtime/src/approval/handler.js.map +1 -0
  155. package/dist/runtime/src/approval/index.d.ts +17 -0
  156. package/dist/runtime/src/approval/index.d.ts.map +1 -0
  157. package/dist/runtime/src/approval/index.js +94 -0
  158. package/dist/runtime/src/approval/index.js.map +1 -0
  159. package/dist/runtime/src/approval/types.d.ts +21 -0
  160. package/dist/runtime/src/approval/types.d.ts.map +1 -0
  161. package/dist/runtime/src/approval/types.js +5 -0
  162. package/dist/runtime/src/approval/types.js.map +1 -0
  163. package/dist/runtime/src/cache/backends.d.ts +39 -0
  164. package/dist/runtime/src/cache/backends.d.ts.map +1 -0
  165. package/dist/runtime/src/cache/backends.js +167 -0
  166. package/dist/runtime/src/cache/backends.js.map +1 -0
  167. package/dist/runtime/src/cache/index.d.ts +32 -0
  168. package/dist/runtime/src/cache/index.d.ts.map +1 -0
  169. package/dist/runtime/src/cache/index.js +103 -0
  170. package/dist/runtime/src/cache/index.js.map +1 -0
  171. package/dist/runtime/src/cache/types.d.ts +20 -0
  172. package/dist/runtime/src/cache/types.d.ts.map +1 -0
  173. package/dist/runtime/src/cache/types.js +2 -0
  174. package/dist/runtime/src/cache/types.js.map +1 -0
  175. package/dist/runtime/src/embedding/index.d.ts +39 -0
  176. package/dist/runtime/src/embedding/index.d.ts.map +1 -0
  177. package/dist/runtime/src/embedding/index.js +162 -0
  178. package/dist/runtime/src/embedding/index.js.map +1 -0
  179. package/dist/runtime/src/embedding/types.d.ts +28 -0
  180. package/dist/runtime/src/embedding/types.d.ts.map +1 -0
  181. package/dist/runtime/src/embedding/types.js +5 -0
  182. package/dist/runtime/src/embedding/types.js.map +1 -0
  183. package/dist/runtime/src/embedding/utils.d.ts +11 -0
  184. package/dist/runtime/src/embedding/utils.d.ts.map +1 -0
  185. package/dist/runtime/src/embedding/utils.js +30 -0
  186. package/dist/runtime/src/embedding/utils.js.map +1 -0
  187. package/dist/runtime/src/embedding/vector-store.d.ts +64 -0
  188. package/dist/runtime/src/embedding/vector-store.d.ts.map +1 -0
  189. package/dist/runtime/src/embedding/vector-store.js +142 -0
  190. package/dist/runtime/src/embedding/vector-store.js.map +1 -0
  191. package/dist/runtime/src/index.d.ts +18 -0
  192. package/dist/runtime/src/index.d.ts.map +1 -0
  193. package/dist/runtime/src/index.js +17 -0
  194. package/dist/runtime/src/index.js.map +1 -0
  195. package/dist/runtime/src/llm/callback.d.ts +13 -0
  196. package/dist/runtime/src/llm/callback.d.ts.map +1 -0
  197. package/dist/runtime/src/llm/callback.js +19 -0
  198. package/dist/runtime/src/llm/callback.js.map +1 -0
  199. package/dist/runtime/src/llm/index.d.ts +29 -0
  200. package/dist/runtime/src/llm/index.d.ts.map +1 -0
  201. package/dist/runtime/src/llm/index.js +118 -0
  202. package/dist/runtime/src/llm/index.js.map +1 -0
  203. package/dist/runtime/src/llm/replay.d.ts +47 -0
  204. package/dist/runtime/src/llm/replay.d.ts.map +1 -0
  205. package/dist/runtime/src/llm/replay.js +114 -0
  206. package/dist/runtime/src/llm/replay.js.map +1 -0
  207. package/dist/runtime/src/llm/types.d.ts +24 -0
  208. package/dist/runtime/src/llm/types.d.ts.map +1 -0
  209. package/dist/runtime/src/llm/types.js +2 -0
  210. package/dist/runtime/src/llm/types.js.map +1 -0
  211. package/dist/runtime/src/log/index.d.ts +12 -0
  212. package/dist/runtime/src/log/index.d.ts.map +1 -0
  213. package/dist/runtime/src/log/index.js +166 -0
  214. package/dist/runtime/src/log/index.js.map +1 -0
  215. package/dist/runtime/src/log/types.d.ts +19 -0
  216. package/dist/runtime/src/log/types.d.ts.map +1 -0
  217. package/dist/runtime/src/log/types.js +5 -0
  218. package/dist/runtime/src/log/types.js.map +1 -0
  219. package/dist/runtime/src/metadata/decorators.d.ts +27 -0
  220. package/dist/runtime/src/metadata/decorators.d.ts.map +1 -0
  221. package/dist/runtime/src/metadata/decorators.js +38 -0
  222. package/dist/runtime/src/metadata/decorators.js.map +1 -0
  223. package/dist/runtime/src/metadata/generated.d.ts +18 -0
  224. package/dist/runtime/src/metadata/generated.d.ts.map +1 -0
  225. package/dist/runtime/src/metadata/generated.js +290 -0
  226. package/dist/runtime/src/metadata/generated.js.map +1 -0
  227. package/dist/runtime/src/metadata/index.d.ts +11 -0
  228. package/dist/runtime/src/metadata/index.d.ts.map +1 -0
  229. package/dist/runtime/src/metadata/index.js +45 -0
  230. package/dist/runtime/src/metadata/index.js.map +1 -0
  231. package/dist/runtime/src/metadata/types.d.ts +22 -0
  232. package/dist/runtime/src/metadata/types.d.ts.map +1 -0
  233. package/dist/runtime/src/metadata/types.js +6 -0
  234. package/dist/runtime/src/metadata/types.js.map +1 -0
  235. package/dist/runtime/src/pause/index.d.ts +11 -0
  236. package/dist/runtime/src/pause/index.d.ts.map +1 -0
  237. package/dist/runtime/src/pause/index.js +15 -0
  238. package/dist/runtime/src/pause/index.js.map +1 -0
  239. package/dist/runtime/src/pause/types.d.ts +46 -0
  240. package/dist/runtime/src/pause/types.d.ts.map +1 -0
  241. package/dist/runtime/src/pause/types.js +57 -0
  242. package/dist/runtime/src/pause/types.js.map +1 -0
  243. package/dist/runtime/src/progress/index.d.ts +19 -0
  244. package/dist/runtime/src/progress/index.d.ts.map +1 -0
  245. package/dist/runtime/src/progress/index.js +61 -0
  246. package/dist/runtime/src/progress/index.js.map +1 -0
  247. package/dist/runtime/src/progress/types.d.ts +7 -0
  248. package/dist/runtime/src/progress/types.d.ts.map +1 -0
  249. package/dist/runtime/src/progress/types.js +2 -0
  250. package/dist/runtime/src/progress/types.js.map +1 -0
  251. package/dist/runtime/src/registry.d.ts +16 -0
  252. package/dist/runtime/src/registry.d.ts.map +1 -0
  253. package/dist/runtime/src/registry.js +16 -0
  254. package/dist/runtime/src/registry.js.map +1 -0
  255. package/dist/runtime/src/utils.d.ts +11 -0
  256. package/dist/runtime/src/utils.d.ts.map +1 -0
  257. package/dist/runtime/src/utils.js +31 -0
  258. package/dist/runtime/src/utils.js.map +1 -0
  259. package/dist/tsconfig.tsbuildinfo +1 -0
  260. package/jest.config.js +29 -0
  261. package/package.json +56 -0
  262. package/project.json +31 -0
  263. package/src/index.ts +6 -0
  264. package/src/runtime/batch-parallel.ts +22 -0
  265. package/src/runtime/checkpoint-manager.ts +105 -0
  266. package/src/runtime/context.ts +33 -0
  267. package/src/runtime/errors.ts +79 -0
  268. package/src/runtime/index.ts +35 -0
  269. package/src/runtime/resumable-arrays.ts +253 -0
  270. package/src/runtime/resumable-loops.ts +93 -0
  271. package/src/runtime/resumable-parallel.ts +57 -0
  272. package/src/transformer/array-transformer-batch.ts +86 -0
  273. package/src/transformer/array-transformer-sequential.ts +38 -0
  274. package/src/transformer/array-transformer-utils.ts +80 -0
  275. package/src/transformer/array-transformer-wrappers.ts +165 -0
  276. package/src/transformer/array-transformer.ts +76 -0
  277. package/src/transformer/batch-detector.ts +166 -0
  278. package/src/transformer/batch-optimizer.ts +320 -0
  279. package/src/transformer/detector.ts +171 -0
  280. package/src/transformer/index.ts +155 -0
  281. package/src/transformer/loop-transformer.ts +285 -0
  282. package/src/transformer/promise-transformer.ts +194 -0
  283. package/src/transformer/utils.ts +147 -0
  284. package/src/types.ts +101 -0
  285. package/tsconfig.json +12 -0
@@ -0,0 +1,171 @@
1
+ import { parse } from '@babel/parser';
2
+ import _traverse from '@babel/traverse';
3
+ // @ts-ignore - CommonJS/ESM compatibility
4
+ const traverse =
5
+ typeof (_traverse as any).default === 'function' ? (_traverse as any).default : _traverse;
6
+ import * as t from '@babel/types';
7
+ import type { DetectionResult, AsyncPattern } from '../types.js';
8
+ import { containsAwait, isArrayMethod, isPausableCallExpression } from './utils.js';
9
+
10
+ export class AsyncIterationDetector {
11
+ detect(code: string): DetectionResult {
12
+ const patterns: AsyncPattern[] = [];
13
+ let batchableParallel = false;
14
+
15
+ try {
16
+ const ast = parse(code, {
17
+ sourceType: 'module',
18
+ plugins: ['typescript'],
19
+ allowAwaitOutsideFunction: true,
20
+ allowReturnOutsideFunction: true,
21
+ });
22
+
23
+ traverse(ast, {
24
+ ForOfStatement: (path: any) => {
25
+ if (containsAwait(path.node.body)) {
26
+ patterns.push('for-of-await');
27
+ }
28
+ },
29
+
30
+ WhileStatement: (path: any) => {
31
+ if (containsAwait(path.node.body)) {
32
+ patterns.push('while-await');
33
+ }
34
+ },
35
+
36
+ CallExpression: (path: any) => {
37
+ const node = path.node;
38
+
39
+ if (isArrayMethod(node, 'map')) {
40
+ const callback = node.arguments[0];
41
+ if (callback && t.isFunction(callback) && callback.async) {
42
+ patterns.push('map-async');
43
+ }
44
+ }
45
+
46
+ if (isArrayMethod(node, 'forEach')) {
47
+ const callback = node.arguments[0];
48
+ if (callback && t.isFunction(callback) && callback.async) {
49
+ patterns.push('forEach-async');
50
+ }
51
+ }
52
+
53
+ if (isArrayMethod(node, 'filter')) {
54
+ const callback = node.arguments[0];
55
+ if (callback && t.isFunction(callback) && callback.async) {
56
+ patterns.push('filter-async');
57
+ }
58
+ }
59
+
60
+ if (isArrayMethod(node, 'reduce')) {
61
+ const callback = node.arguments[0];
62
+ if (callback && t.isFunction(callback) && callback.async) {
63
+ patterns.push('reduce-async');
64
+ }
65
+ }
66
+
67
+ if (isArrayMethod(node, 'find')) {
68
+ const callback = node.arguments[0];
69
+ if (callback && t.isFunction(callback) && callback.async) {
70
+ patterns.push('find-async');
71
+ }
72
+ }
73
+
74
+ if (isArrayMethod(node, 'some')) {
75
+ const callback = node.arguments[0];
76
+ if (callback && t.isFunction(callback) && callback.async) {
77
+ patterns.push('some-async');
78
+ }
79
+ }
80
+
81
+ if (isArrayMethod(node, 'every')) {
82
+ const callback = node.arguments[0];
83
+ if (callback && t.isFunction(callback) && callback.async) {
84
+ patterns.push('every-async');
85
+ }
86
+ }
87
+
88
+ if (isArrayMethod(node, 'flatMap')) {
89
+ const callback = node.arguments[0];
90
+ if (callback && t.isFunction(callback) && callback.async) {
91
+ patterns.push('flatMap-async');
92
+ }
93
+ }
94
+
95
+ if (this.isPromiseAll(node)) {
96
+ patterns.push('promise-all');
97
+ if (this.canBatchPromiseAll(node)) {
98
+ batchableParallel = true;
99
+ }
100
+ }
101
+
102
+ if (this.isPromiseAllSettled(node)) {
103
+ patterns.push('promise-allSettled');
104
+ }
105
+ },
106
+ });
107
+
108
+ return {
109
+ needsTransform: patterns.length > 0,
110
+ patterns: [...new Set(patterns)],
111
+ batchableParallel,
112
+ };
113
+ } catch (error) {
114
+ return {
115
+ needsTransform: false,
116
+ patterns: [],
117
+ batchableParallel: false,
118
+ };
119
+ }
120
+ }
121
+
122
+ private isPromiseAll(node: t.CallExpression): boolean {
123
+ const callee = node.callee;
124
+ return (
125
+ t.isMemberExpression(callee) &&
126
+ t.isIdentifier(callee.object, { name: 'Promise' }) &&
127
+ t.isIdentifier(callee.property, { name: 'all' })
128
+ );
129
+ }
130
+
131
+ private isPromiseAllSettled(node: t.CallExpression): boolean {
132
+ const callee = node.callee;
133
+ return (
134
+ t.isMemberExpression(callee) &&
135
+ t.isIdentifier(callee.object, { name: 'Promise' }) &&
136
+ t.isIdentifier(callee.property, { name: 'allSettled' })
137
+ );
138
+ }
139
+
140
+ private canBatchPromiseAll(node: t.CallExpression): boolean {
141
+ const arrayArg = node.arguments[0];
142
+
143
+ if (!t.isArrayExpression(arrayArg)) {
144
+ return false;
145
+ }
146
+
147
+ if (arrayArg.elements.length === 0) {
148
+ return false;
149
+ }
150
+
151
+ return arrayArg.elements.every((el) => {
152
+ if (!el || t.isSpreadElement(el)) {
153
+ return false;
154
+ }
155
+
156
+ return this.isDirectPausableCall(el);
157
+ });
158
+ }
159
+
160
+ private isDirectPausableCall(node: t.Node): boolean {
161
+ if (t.isAwaitExpression(node)) {
162
+ node = node.argument;
163
+ }
164
+
165
+ if (!t.isCallExpression(node)) {
166
+ return false;
167
+ }
168
+
169
+ return isPausableCallExpression(node);
170
+ }
171
+ }
@@ -0,0 +1,155 @@
1
+ import { parse } from '@babel/parser';
2
+ import _traverse from '@babel/traverse';
3
+ const traverse = (_traverse as any).default || _traverse;
4
+ import _generate from '@babel/generator';
5
+ const generate = (_generate as any).default || _generate;
6
+ import * as t from '@babel/types';
7
+ import { AsyncIterationDetector } from './detector.js';
8
+ import { LoopTransformer } from './loop-transformer.js';
9
+ import { ArrayTransformer } from './array-transformer.js';
10
+ import { PromiseTransformer } from './promise-transformer.js';
11
+ import type { TransformResult, CompilerConfig, TransformMetadata } from '../types.js';
12
+ import { DEFAULT_COMPILER_CONFIG } from '../types.js';
13
+ import { TransformationError } from '../runtime/errors.js';
14
+ import { resetIdCounter } from '../runtime/context.js';
15
+
16
+ export class ATPCompiler {
17
+ private config: CompilerConfig;
18
+ private detector: AsyncIterationDetector;
19
+ private loopTransformer: LoopTransformer;
20
+ private arrayTransformer: ArrayTransformer;
21
+ private promiseTransformer: PromiseTransformer;
22
+
23
+ constructor(config: Partial<CompilerConfig> = {}) {
24
+ this.config = { ...DEFAULT_COMPILER_CONFIG, ...config };
25
+ this.detector = new AsyncIterationDetector();
26
+ this.loopTransformer = new LoopTransformer(this.config.batchSizeThreshold);
27
+ this.arrayTransformer = new ArrayTransformer(this.config.batchSizeThreshold);
28
+ this.promiseTransformer = new PromiseTransformer(this.config.enableBatchParallel);
29
+ }
30
+
31
+ detect(code: string) {
32
+ return this.detector.detect(code);
33
+ }
34
+
35
+ transform(code: string): TransformResult {
36
+ resetIdCounter();
37
+
38
+ const detection = this.detector.detect(code);
39
+
40
+ if (!detection.needsTransform) {
41
+ return {
42
+ code,
43
+ transformed: false,
44
+ patterns: [],
45
+ metadata: {
46
+ loopCount: 0,
47
+ arrayMethodCount: 0,
48
+ parallelCallCount: 0,
49
+ batchableCount: 0,
50
+ },
51
+ };
52
+ }
53
+
54
+ try {
55
+ const ast = parse(code, {
56
+ sourceType: 'module',
57
+ plugins: ['typescript'],
58
+ allowAwaitOutsideFunction: true,
59
+ allowReturnOutsideFunction: true,
60
+ });
61
+
62
+ this.loopTransformer.resetTransformCount();
63
+ this.arrayTransformer.resetTransformCount();
64
+ this.promiseTransformer.resetTransformCount();
65
+
66
+ traverse(ast, {
67
+ ForOfStatement: (path: any) => {
68
+ this.loopTransformer.transformForOfLoop(path);
69
+ },
70
+
71
+ WhileStatement: (path: any) => {
72
+ this.loopTransformer.transformWhileLoop(path);
73
+ },
74
+
75
+ ForStatement: (path: any) => {
76
+ this.loopTransformer.transformForLoop(path);
77
+ },
78
+
79
+ CallExpression: (path: any) => {
80
+ if (this.isArrayMethodCall(path.node)) {
81
+ this.arrayTransformer.transformArrayMethod(path);
82
+ } else if (this.isPromiseAllCall(path.node)) {
83
+ this.promiseTransformer.transformPromiseAll(path);
84
+ } else if (this.isPromiseAllSettledCall(path.node)) {
85
+ this.promiseTransformer.transformPromiseAllSettled(path);
86
+ }
87
+ },
88
+ });
89
+
90
+ const output = generate(ast, {
91
+ sourceMaps: false,
92
+ retainLines: true,
93
+ comments: true,
94
+ });
95
+
96
+ const metadata: TransformMetadata = {
97
+ loopCount: this.loopTransformer.getTransformCount(),
98
+ arrayMethodCount: this.arrayTransformer.getTransformCount(),
99
+ parallelCallCount: this.promiseTransformer.getTransformCount(),
100
+ batchableCount: detection.batchableParallel ? 1 : 0,
101
+ };
102
+
103
+ return {
104
+ code: output.code,
105
+ transformed: true,
106
+ patterns: detection.patterns,
107
+ metadata,
108
+ };
109
+ } catch (error) {
110
+ const message = error instanceof Error ? error.message : String(error);
111
+ throw new TransformationError(message, code, 'unknown');
112
+ }
113
+ }
114
+
115
+ private isArrayMethodCall(node: t.CallExpression): boolean {
116
+ if (!t.isMemberExpression(node.callee)) {
117
+ return false;
118
+ }
119
+
120
+ const property = node.callee.property;
121
+ if (!t.isIdentifier(property)) {
122
+ return false;
123
+ }
124
+
125
+ const arrayMethods = ['map', 'forEach', 'filter', 'reduce', 'find', 'some', 'every', 'flatMap'];
126
+
127
+ return arrayMethods.includes(property.name);
128
+ }
129
+
130
+ private isPromiseAllCall(node: t.CallExpression): boolean {
131
+ const callee = node.callee;
132
+ return (
133
+ t.isMemberExpression(callee) &&
134
+ t.isIdentifier(callee.object, { name: 'Promise' }) &&
135
+ t.isIdentifier(callee.property, { name: 'all' })
136
+ );
137
+ }
138
+
139
+ private isPromiseAllSettledCall(node: t.CallExpression): boolean {
140
+ const callee = node.callee;
141
+ return (
142
+ t.isMemberExpression(callee) &&
143
+ t.isIdentifier(callee.object, { name: 'Promise' }) &&
144
+ t.isIdentifier(callee.property, { name: 'allSettled' })
145
+ );
146
+ }
147
+ }
148
+
149
+ export * from './detector.js';
150
+ export * from './batch-detector.js';
151
+ export * from './batch-optimizer.js';
152
+ export * from './loop-transformer.js';
153
+ export * from './array-transformer.js';
154
+ export * from './promise-transformer.js';
155
+ export * from './utils.js';
@@ -0,0 +1,285 @@
1
+ import * as t from '@babel/types';
2
+ import { generateUniqueId } from '../runtime/context.js';
3
+ import { containsAwait } from './utils.js';
4
+ import { BatchOptimizer } from './batch-optimizer.js';
5
+ import { BatchParallelDetector } from './batch-detector.js';
6
+
7
+ export class LoopTransformer {
8
+ private transformCount = 0;
9
+ private batchOptimizer: BatchOptimizer;
10
+ private batchDetector: BatchParallelDetector;
11
+ private batchSizeThreshold: number;
12
+
13
+ constructor(batchSizeThreshold: number = 10) {
14
+ this.batchOptimizer = new BatchOptimizer();
15
+ this.batchDetector = new BatchParallelDetector();
16
+ this.batchSizeThreshold = batchSizeThreshold;
17
+ }
18
+
19
+ transformForOfLoop(path: any): boolean {
20
+ const node = path.node as t.ForOfStatement;
21
+
22
+ if (!containsAwait(node.body)) {
23
+ return false;
24
+ }
25
+
26
+ const batchResult = this.batchOptimizer.canBatchForOfLoop(node);
27
+
28
+ if (batchResult.canBatch) {
29
+ const decision = this.batchOptimizer.makeSmartBatchDecision(
30
+ 'for...of',
31
+ batchResult,
32
+ node.right,
33
+ this.batchSizeThreshold
34
+ );
35
+
36
+ if (decision.shouldBatch) {
37
+ return this.transformForOfToBatch(path, node);
38
+ }
39
+ }
40
+
41
+ return this.transformForOfToSequential(path, node);
42
+ }
43
+
44
+ /**
45
+ * Transform simple for...of to batch parallel
46
+ */
47
+ private transformForOfToBatch(path: any, node: t.ForOfStatement): boolean {
48
+ const loopId = generateUniqueId('for_of_batch');
49
+ const left = node.left;
50
+ const right = node.right;
51
+
52
+ let paramName: string;
53
+ if (t.isVariableDeclaration(left)) {
54
+ const id = left.declarations[0]?.id;
55
+ paramName = t.isIdentifier(id) ? id.name : 'item';
56
+ } else if (t.isIdentifier(left)) {
57
+ paramName = left.name;
58
+ } else {
59
+ paramName = 'item';
60
+ }
61
+
62
+ const bodyStatements = t.isBlockStatement(node.body) ? node.body.body : [node.body];
63
+ const callback = t.arrowFunctionExpression(
64
+ [t.identifier(paramName)],
65
+ t.blockStatement(bodyStatements),
66
+ true
67
+ );
68
+
69
+ const llmCall = this.findLLMCallInBody(node.body);
70
+ if (!llmCall) {
71
+ return this.transformForOfToSequential(path, node);
72
+ }
73
+
74
+ const callInfo = this.batchDetector.extractCallInfo(llmCall);
75
+ if (!callInfo) {
76
+ return this.transformForOfToSequential(path, node);
77
+ }
78
+
79
+ const payloadNode = this.batchDetector.extractPayloadNode(llmCall);
80
+ if (!payloadNode) {
81
+ return this.transformForOfToSequential(path, node);
82
+ }
83
+
84
+ const batchCallsArray = t.callExpression(t.memberExpression(right, t.identifier('map')), [
85
+ t.arrowFunctionExpression(
86
+ [t.identifier(paramName)],
87
+ t.objectExpression([
88
+ t.objectProperty(t.identifier('type'), t.stringLiteral(callInfo.type)),
89
+ t.objectProperty(t.identifier('operation'), t.stringLiteral(callInfo.operation)),
90
+ t.objectProperty(t.identifier('payload'), payloadNode),
91
+ ])
92
+ ),
93
+ ]);
94
+
95
+ const batchCall = t.awaitExpression(
96
+ t.callExpression(
97
+ t.memberExpression(t.identifier('__runtime'), t.identifier('batchParallel')),
98
+ [batchCallsArray, t.stringLiteral(loopId)]
99
+ )
100
+ );
101
+
102
+ path.replaceWith(t.expressionStatement(batchCall));
103
+ this.transformCount++;
104
+ return true;
105
+ }
106
+
107
+ /**
108
+ * Transform for...of to sequential with checkpoints (fallback)
109
+ */
110
+ private transformForOfToSequential(path: any, node: t.ForOfStatement): boolean {
111
+ const loopId = generateUniqueId('for_of');
112
+ const left = node.left;
113
+ const right = node.right;
114
+
115
+ let paramName: string;
116
+ if (t.isVariableDeclaration(left)) {
117
+ const id = left.declarations[0]?.id;
118
+ paramName = t.isIdentifier(id) ? id.name : 'item';
119
+ } else if (t.isIdentifier(left)) {
120
+ paramName = left.name;
121
+ } else {
122
+ paramName = 'item';
123
+ }
124
+
125
+ const bodyStatements = t.isBlockStatement(node.body) ? node.body.body : [node.body];
126
+
127
+ const callbackFn = t.arrowFunctionExpression(
128
+ [t.identifier(paramName), t.identifier('__index')],
129
+ t.blockStatement(bodyStatements),
130
+ true
131
+ );
132
+
133
+ const runtimeCall = t.awaitExpression(
134
+ t.callExpression(
135
+ t.memberExpression(t.identifier('__runtime'), t.identifier('resumableForOf')),
136
+ [right, callbackFn, t.stringLiteral(loopId)]
137
+ )
138
+ );
139
+
140
+ path.replaceWith(t.expressionStatement(runtimeCall));
141
+ this.transformCount++;
142
+ return true;
143
+ }
144
+
145
+ /**
146
+ * Find LLM call in loop body
147
+ */
148
+ private findLLMCallInBody(body: t.Statement | t.BlockStatement): t.CallExpression | null {
149
+ let found: t.CallExpression | null = null;
150
+
151
+ const visit = (node: t.Node) => {
152
+ if (found) return;
153
+
154
+ if (t.isAwaitExpression(node) && t.isCallExpression(node.argument)) {
155
+ const call = node.argument;
156
+ if (t.isMemberExpression(call.callee)) {
157
+ found = call;
158
+ return;
159
+ }
160
+ }
161
+
162
+ Object.keys(node).forEach((key) => {
163
+ const value = (node as any)[key];
164
+ if (Array.isArray(value)) {
165
+ value.forEach((item) => {
166
+ if (item && typeof item === 'object' && item.type) {
167
+ visit(item);
168
+ }
169
+ });
170
+ } else if (value && typeof value === 'object' && value.type) {
171
+ visit(value);
172
+ }
173
+ });
174
+ };
175
+
176
+ visit(body);
177
+ return found;
178
+ }
179
+
180
+ transformWhileLoop(path: any): boolean {
181
+ const node = path.node as t.WhileStatement;
182
+
183
+ if (!containsAwait(node.body)) {
184
+ return false;
185
+ }
186
+
187
+ const loopId = generateUniqueId('while');
188
+
189
+ const conditionFn = t.arrowFunctionExpression([], node.test, false);
190
+
191
+ const bodyStatements = t.isBlockStatement(node.body) ? node.body.body : [node.body];
192
+
193
+ const bodyFn = t.arrowFunctionExpression(
194
+ [t.identifier('__iteration')],
195
+ t.blockStatement(bodyStatements),
196
+ true
197
+ );
198
+
199
+ const runtimeCall = t.awaitExpression(
200
+ t.callExpression(
201
+ t.memberExpression(t.identifier('__runtime'), t.identifier('resumableWhile')),
202
+ [conditionFn, bodyFn, t.stringLiteral(loopId)]
203
+ )
204
+ );
205
+
206
+ path.replaceWith(t.expressionStatement(runtimeCall));
207
+ this.transformCount++;
208
+ return true;
209
+ }
210
+
211
+ transformForLoop(path: any): boolean {
212
+ const node = path.node as t.ForStatement;
213
+
214
+ if (!containsAwait(node.body)) {
215
+ return false;
216
+ }
217
+
218
+ if (!node.init || !node.test || !node.update) {
219
+ return false;
220
+ }
221
+
222
+ const loopId = generateUniqueId('for');
223
+
224
+ let initValue: t.Expression = t.numericLiteral(0);
225
+ let loopVar = '__i';
226
+
227
+ if (t.isVariableDeclaration(node.init)) {
228
+ const decl = node.init.declarations[0];
229
+ if (decl && t.isIdentifier(decl.id) && decl.init) {
230
+ loopVar = decl.id.name;
231
+ initValue = decl.init;
232
+ }
233
+ }
234
+
235
+ const conditionFn = t.arrowFunctionExpression([t.identifier(loopVar)], node.test, false);
236
+
237
+ const bodyStatements = t.isBlockStatement(node.body) ? node.body.body : [node.body];
238
+
239
+ const bodyFn = t.arrowFunctionExpression(
240
+ [t.identifier(loopVar)],
241
+ t.blockStatement(bodyStatements),
242
+ true
243
+ );
244
+
245
+ let incrementFn: t.Expression;
246
+ if (t.isUpdateExpression(node.update)) {
247
+ if (node.update.operator === '++') {
248
+ incrementFn = t.arrowFunctionExpression(
249
+ [t.identifier(loopVar)],
250
+ t.binaryExpression('+', t.identifier(loopVar), t.numericLiteral(1)),
251
+ false
252
+ );
253
+ } else if (node.update.operator === '--') {
254
+ incrementFn = t.arrowFunctionExpression(
255
+ [t.identifier(loopVar)],
256
+ t.binaryExpression('-', t.identifier(loopVar), t.numericLiteral(1)),
257
+ false
258
+ );
259
+ } else {
260
+ return false;
261
+ }
262
+ } else {
263
+ return false;
264
+ }
265
+
266
+ const runtimeCall = t.awaitExpression(
267
+ t.callExpression(
268
+ t.memberExpression(t.identifier('__runtime'), t.identifier('resumableForLoop')),
269
+ [initValue, conditionFn, incrementFn, bodyFn, t.stringLiteral(loopId)]
270
+ )
271
+ );
272
+
273
+ path.replaceWith(t.expressionStatement(runtimeCall));
274
+ this.transformCount++;
275
+ return true;
276
+ }
277
+
278
+ getTransformCount(): number {
279
+ return this.transformCount;
280
+ }
281
+
282
+ resetTransformCount(): void {
283
+ this.transformCount = 0;
284
+ }
285
+ }