@mmapp/react-compiler 0.1.0-alpha.1 → 0.1.0-alpha.4

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 (346) hide show
  1. package/ATOM-PIPELINE.md +144 -0
  2. package/README.md +88 -40
  3. package/dist/auth-3UK75242.mjs +17 -0
  4. package/dist/babel/index.d.mts +2 -2
  5. package/dist/babel/index.d.ts +2 -2
  6. package/dist/babel/index.js +2816 -279
  7. package/dist/babel/index.mjs +2 -2
  8. package/dist/chunk-3USIFFE4.mjs +2190 -0
  9. package/dist/chunk-45YMGEVT.mjs +186 -0
  10. package/dist/chunk-4FN2AISW.mjs +148 -0
  11. package/dist/chunk-4OPI5L7G.mjs +2593 -0
  12. package/dist/chunk-4RYTKOOJ.mjs +186 -0
  13. package/dist/chunk-52XHYD2V.mjs +214 -0
  14. package/dist/chunk-5FTDWKHH.mjs +244 -0
  15. package/dist/chunk-5GUFFFGL.mjs +148 -0
  16. package/dist/chunk-5RKTOVR5.mjs +244 -0
  17. package/dist/chunk-5YDMOO4X.mjs +214 -0
  18. package/dist/chunk-64ZWEMLJ.mjs +148 -0
  19. package/dist/chunk-6XP4KSWQ.mjs +2190 -0
  20. package/dist/chunk-72QWL54I.mjs +175 -0
  21. package/dist/chunk-7B4TRI7C.mjs +4835 -0
  22. package/dist/chunk-7JRAEFRB.mjs +7510 -0
  23. package/dist/chunk-7T6Q5KAA.mjs +7506 -0
  24. package/dist/chunk-7ZKGHTNB.mjs +4952 -0
  25. package/dist/chunk-ABYPKRSB.mjs +215 -0
  26. package/dist/chunk-BZEXUPDH.mjs +175 -0
  27. package/dist/chunk-CIESM3BP.mjs +33 -0
  28. package/dist/chunk-DE3ZGQAC.mjs +148 -0
  29. package/dist/chunk-DMCY3BBG.mjs +1933 -0
  30. package/dist/chunk-DPIK3PJS.mjs +244 -0
  31. package/dist/chunk-E5IVH4RE.mjs +186 -0
  32. package/dist/chunk-E6FZNUR5.mjs +4953 -0
  33. package/dist/chunk-EJRBDQDP.mjs +2607 -0
  34. package/dist/chunk-ELO4TXJL.mjs +186 -0
  35. package/dist/chunk-EO6SYNCG.mjs +175 -0
  36. package/dist/chunk-FKRO52XH.mjs +3446 -0
  37. package/dist/chunk-FL4YAKU6.mjs +4941 -0
  38. package/dist/chunk-FYT47UBU.mjs +5076 -0
  39. package/dist/chunk-GCLGPOJZ.mjs +148 -0
  40. package/dist/chunk-GXB4JOP7.mjs +5072 -0
  41. package/dist/chunk-HFXOUMTD.mjs +175 -0
  42. package/dist/chunk-HRYR54PT.mjs +175 -0
  43. package/dist/chunk-HWIZ47US.mjs +214 -0
  44. package/dist/chunk-IB7MNPQL.mjs +4953 -0
  45. package/dist/chunk-ICSIHQCG.mjs +148 -0
  46. package/dist/chunk-J3M4GUS7.mjs +161 -0
  47. package/dist/chunk-J7JUAHS4.mjs +186 -0
  48. package/dist/chunk-JLA5VNQ3.mjs +186 -0
  49. package/dist/chunk-JQLWFCTM.mjs +214 -0
  50. package/dist/chunk-JRGFBWTN.mjs +2918 -0
  51. package/dist/chunk-KFJJCQAL.mjs +148 -0
  52. package/dist/chunk-KJUIIEQE.mjs +186 -0
  53. package/dist/chunk-KNWTHRVQ.mjs +175 -0
  54. package/dist/chunk-KSG4XSZF.mjs +175 -0
  55. package/dist/chunk-LF5N6DOU.mjs +175 -0
  56. package/dist/chunk-LJQCM2IM.mjs +214 -0
  57. package/dist/chunk-NTB7OEX2.mjs +2918 -0
  58. package/dist/chunk-NW6555WJ.mjs +186 -0
  59. package/dist/chunk-O4AUS7EU.mjs +148 -0
  60. package/dist/chunk-OMZE6VLQ.mjs +214 -0
  61. package/dist/chunk-OPJKP747.mjs +7506 -0
  62. package/dist/chunk-P4BR7WVO.mjs +2190 -0
  63. package/dist/chunk-QQHVYH2X.mjs +244 -0
  64. package/dist/chunk-R2DD5GTY.mjs +186 -0
  65. package/dist/chunk-S5QLWLLT.mjs +186 -0
  66. package/dist/chunk-SCWGT2FY.mjs +2190 -0
  67. package/dist/chunk-SMKJUSB3.mjs +2190 -0
  68. package/dist/chunk-THFYE5ZX.mjs +244 -0
  69. package/dist/chunk-UDDTWG5J.mjs +734 -0
  70. package/dist/chunk-VCAY2KGM.mjs +175 -0
  71. package/dist/chunk-VLTKQDJ3.mjs +244 -0
  72. package/dist/chunk-WBYMW4NQ.mjs +3450 -0
  73. package/dist/chunk-WECAV6QB.mjs +148 -0
  74. package/dist/chunk-WMKBXUCE.mjs +3228 -0
  75. package/dist/chunk-WVYY32LD.mjs +939 -0
  76. package/dist/chunk-XAJ5BKKL.mjs +4947 -0
  77. package/dist/chunk-XDVM4YHX.mjs +3450 -0
  78. package/dist/chunk-XG2X7AEA.mjs +175 -0
  79. package/dist/chunk-XG7Z23NQ.mjs +148 -0
  80. package/dist/chunk-XWZAOCQ7.mjs +2607 -0
  81. package/dist/chunk-Y6MA7ULW.mjs +148 -0
  82. package/dist/chunk-YMS7Q7LG.mjs +214 -0
  83. package/dist/chunk-Z2G5RZ4H.mjs +186 -0
  84. package/dist/chunk-ZA37XTGA.mjs +175 -0
  85. package/dist/chunk-ZE3KCHBM.mjs +2918 -0
  86. package/dist/cli/index.js +14720 -7199
  87. package/dist/cli/index.mjs +224 -183
  88. package/dist/codemod/cli.js +1 -1
  89. package/dist/codemod/cli.mjs +2 -2
  90. package/dist/codemod/index.d.mts +3 -3
  91. package/dist/codemod/index.d.ts +3 -3
  92. package/dist/codemod/index.js +1 -1
  93. package/dist/codemod/index.mjs +2 -2
  94. package/dist/config-PL24KEWL.mjs +219 -0
  95. package/dist/deploy-YAJGW6II.mjs +9 -0
  96. package/dist/dev-server-CrQ041KP.d.mts +79 -0
  97. package/dist/dev-server-CrQ041KP.d.ts +79 -0
  98. package/dist/dev-server-RmGHIntF.d.mts +113 -0
  99. package/dist/dev-server-RmGHIntF.d.ts +113 -0
  100. package/dist/dev-server.d.mts +2 -2
  101. package/dist/dev-server.d.ts +2 -2
  102. package/dist/dev-server.js +6424 -1597
  103. package/dist/dev-server.mjs +5 -5
  104. package/dist/envelope-ChEkuHij.d.mts +265 -0
  105. package/dist/envelope-ChEkuHij.d.ts +265 -0
  106. package/dist/envelope.d.mts +2 -2
  107. package/dist/envelope.d.ts +2 -2
  108. package/dist/envelope.js +2814 -277
  109. package/dist/envelope.mjs +3 -3
  110. package/dist/index-CEKyyazf.d.mts +104 -0
  111. package/dist/index-CEKyyazf.d.ts +104 -0
  112. package/dist/index.d.mts +168 -9
  113. package/dist/index.d.ts +168 -9
  114. package/dist/index.js +5606 -681
  115. package/dist/index.mjs +217 -9
  116. package/dist/init-7FJENUDK.mjs +407 -0
  117. package/{src/cli/init.ts → dist/init-7JQMAAXS.mjs} +70 -95
  118. package/dist/init-DQDX3QK6.mjs +369 -0
  119. package/dist/init-EHO4VQ22.mjs +369 -0
  120. package/dist/init-UC3FWPIW.mjs +367 -0
  121. package/dist/init-UNSMVKIK.mjs +366 -0
  122. package/dist/init-UNV5XIDE.mjs +367 -0
  123. package/dist/project-compiler-2P4N4DR7.mjs +10 -0
  124. package/dist/project-compiler-D2LCC27O.mjs +10 -0
  125. package/dist/project-compiler-EJ3GANJE.mjs +10 -0
  126. package/dist/project-compiler-LOQKVRZJ.mjs +10 -0
  127. package/dist/project-compiler-NNK32MPG.mjs +10 -0
  128. package/dist/project-compiler-OP2VVGJQ.mjs +10 -0
  129. package/dist/project-compiler-RQ6OQKRM.mjs +10 -0
  130. package/dist/project-compiler-VWNNCHGO.mjs +10 -0
  131. package/dist/project-compiler-XVAAU4C5.mjs +10 -0
  132. package/dist/project-compiler-YES5FGMD.mjs +10 -0
  133. package/dist/project-compiler-ZB4RUYVL.mjs +10 -0
  134. package/dist/project-compiler-ZKMQDLGU.mjs +10 -0
  135. package/dist/project-decompiler-FLXCEJHS.mjs +7 -0
  136. package/dist/project-decompiler-U55HQUHW.mjs +7 -0
  137. package/dist/project-decompiler-US7GAVIC.mjs +7 -0
  138. package/dist/project-decompiler-VLPR22QF.mjs +7 -0
  139. package/dist/pull-FUS5QYZS.mjs +109 -0
  140. package/dist/pull-KOL2QAYQ.mjs +109 -0
  141. package/dist/pull-LD5ENLGY.mjs +109 -0
  142. package/dist/pull-P44LDRWB.mjs +109 -0
  143. package/dist/seed-KOGEPGOJ.mjs +154 -0
  144. package/dist/server-VW6UPCHO.mjs +277 -0
  145. package/dist/testing/index.d.mts +8 -8
  146. package/dist/testing/index.d.ts +8 -8
  147. package/dist/testing/index.js +2824 -287
  148. package/dist/testing/index.mjs +2 -2
  149. package/dist/verify-BYHUKARQ.mjs +1833 -0
  150. package/dist/verify-OQDEQYMS.mjs +1833 -0
  151. package/dist/verify-SEIXUGN4.mjs +1833 -0
  152. package/dist/vite/index.d.mts +1 -1
  153. package/dist/vite/index.d.ts +1 -1
  154. package/dist/vite/index.js +2817 -280
  155. package/dist/vite/index.mjs +3 -3
  156. package/examples/authentication/main.workflow.tsx +1 -1
  157. package/examples/authentication/mm.config.ts +1 -1
  158. package/examples/authentication/pages/LoginPage.tsx +2 -2
  159. package/examples/authentication/pages/SignupPage.tsx +2 -2
  160. package/examples/counter.workflow.tsx +1 -1
  161. package/examples/dashboard.workflow.tsx +1 -1
  162. package/examples/invoice-approval/actions/invoice.server.ts +1 -1
  163. package/examples/invoice-approval/main.workflow.tsx +1 -1
  164. package/examples/invoice-approval/mm.config.ts +1 -1
  165. package/examples/invoice-approval/pages/InvoiceDetailPage.tsx +1 -1
  166. package/examples/invoice-approval/pages/InvoiceFormPage.tsx +1 -1
  167. package/examples/invoice-approval/pages/InvoiceListPage.tsx +1 -1
  168. package/examples/todo-app.workflow.tsx +1 -1
  169. package/examples/uber-app/actions/matching.server.ts +1 -1
  170. package/examples/uber-app/actions/notifications.server.ts +1 -1
  171. package/examples/uber-app/actions/payments.server.ts +1 -1
  172. package/examples/uber-app/actions/pricing.server.ts +1 -1
  173. package/examples/uber-app/app/admin/analytics.tsx +2 -2
  174. package/examples/uber-app/app/admin/fleet.tsx +21 -21
  175. package/examples/uber-app/app/admin/surge-pricing.tsx +2 -2
  176. package/examples/uber-app/app/driver/dashboard.tsx +2 -2
  177. package/examples/uber-app/app/driver/earnings.tsx +2 -2
  178. package/examples/uber-app/app/driver/navigation.tsx +2 -2
  179. package/examples/uber-app/app/driver/ride-acceptance.tsx +2 -2
  180. package/examples/uber-app/app/rider/home.tsx +2 -2
  181. package/examples/uber-app/app/rider/payment-methods.tsx +2 -2
  182. package/examples/uber-app/app/rider/ride-history.tsx +2 -2
  183. package/examples/uber-app/app/rider/ride-tracking.tsx +2 -2
  184. package/examples/uber-app/components/DriverCard.tsx +1 -1
  185. package/examples/uber-app/components/MapView.tsx +3 -3
  186. package/examples/uber-app/components/RatingStars.tsx +2 -2
  187. package/examples/uber-app/components/RideCard.tsx +1 -1
  188. package/examples/uber-app/mm.config.ts +1 -1
  189. package/examples/uber-app/workflows/dispute-resolution.workflow.tsx +2 -2
  190. package/examples/uber-app/workflows/driver-onboarding.workflow.tsx +2 -2
  191. package/examples/uber-app/workflows/payment-processing.workflow.tsx +2 -2
  192. package/examples/uber-app/workflows/ride-request.workflow.tsx +2 -2
  193. package/package.json +10 -4
  194. package/compile-blueprint-chat.mjs +0 -99
  195. package/compile-blueprint-glass-console.mjs +0 -98
  196. package/compile-chat-defs.mjs +0 -92
  197. package/examples/uber-app/tests/payment.test.tsx +0 -129
  198. package/examples/uber-app/tests/ride-flow.test.tsx +0 -123
  199. package/package.json.backup +0 -86
  200. package/scripts/decompile.ts +0 -226
  201. package/scripts/seed-auth.ts +0 -267
  202. package/scripts/seed-uber.ts +0 -248
  203. package/scripts/validate-uber.ts +0 -119
  204. package/seed-blueprint-chat.mjs +0 -444
  205. package/seed-blueprint-glass-console.mjs +0 -445
  206. package/seed-compiled.mjs +0 -318
  207. package/src/RoundTripValidator.ts +0 -400
  208. package/src/__tests__/atom-rendering-coverage.test.ts +0 -680
  209. package/src/__tests__/auth-module-compilation.test.ts +0 -247
  210. package/src/__tests__/auth-template-compilation.test.ts +0 -589
  211. package/src/__tests__/change-extractor.test.ts +0 -142
  212. package/src/__tests__/cli-pull.test.ts +0 -73
  213. package/src/__tests__/cli-test.test.ts +0 -72
  214. package/src/__tests__/component-extractor.test.ts +0 -331
  215. package/src/__tests__/context-extractor.test.ts +0 -145
  216. package/src/__tests__/decompiler.test.ts +0 -718
  217. package/src/__tests__/define-blueprint.test.ts +0 -133
  218. package/src/__tests__/definition-validator.test.ts +0 -519
  219. package/src/__tests__/during-extractor.test.ts +0 -152
  220. package/src/__tests__/effect-extractor.test.ts +0 -107
  221. package/src/__tests__/event-emission.test.ts +0 -127
  222. package/src/__tests__/examples.test.ts +0 -236
  223. package/src/__tests__/full-blueprint-coverage.test.ts +0 -1221
  224. package/src/__tests__/golden-suite.test.ts +0 -403
  225. package/src/__tests__/grammar-island-extractor.test.ts +0 -289
  226. package/src/__tests__/instance-key.test.ts +0 -82
  227. package/src/__tests__/ir-migration.test.ts +0 -255
  228. package/src/__tests__/lock-file.test.ts +0 -117
  229. package/src/__tests__/model-extractor.test.ts +0 -195
  230. package/src/__tests__/model-field-acl.test.ts +0 -237
  231. package/src/__tests__/model-hooks.test.ts +0 -130
  232. package/src/__tests__/model-ref-resolution.test.ts +0 -268
  233. package/src/__tests__/model-roundtrip.test.ts +0 -502
  234. package/src/__tests__/model-runtime.test.ts +0 -112
  235. package/src/__tests__/model-transitions.test.ts +0 -183
  236. package/src/__tests__/nrt-action-trace.test.ts +0 -391
  237. package/src/__tests__/pipeline-hardening.test.ts +0 -413
  238. package/src/__tests__/project-compiler.test.ts +0 -546
  239. package/src/__tests__/project-decompiler.test.ts +0 -343
  240. package/src/__tests__/query-compilation.test.ts +0 -145
  241. package/src/__tests__/round-trip/PLAN.md +0 -158
  242. package/src/__tests__/round-trip/README.md +0 -52
  243. package/src/__tests__/round-trip/RESULTS.md +0 -86
  244. package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +0 -55
  245. package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +0 -11
  246. package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +0 -54
  247. package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +0 -79
  248. package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +0 -12
  249. package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +0 -50
  250. package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +0 -25
  251. package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +0 -11
  252. package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +0 -32
  253. package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +0 -79
  254. package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +0 -10
  255. package/src/__tests__/round-trip/round-trip.test.ts +0 -2598
  256. package/src/__tests__/round-trip-ir.test.ts +0 -300
  257. package/src/__tests__/round-trip.test.ts +0 -1212
  258. package/src/__tests__/route-merging.test.ts +0 -372
  259. package/src/__tests__/router-composition.test.ts +0 -489
  260. package/src/__tests__/router-extractor.test.ts +0 -176
  261. package/src/__tests__/server-action-extractor.test.ts +0 -128
  262. package/src/__tests__/smart-type-inference.test.ts +0 -365
  263. package/src/__tests__/source-envelope.test.ts +0 -284
  264. package/src/__tests__/source-fidelity.test.ts +0 -516
  265. package/src/__tests__/state-extractor.test.ts +0 -115
  266. package/src/__tests__/strict-mode.test.ts +0 -227
  267. package/src/__tests__/transition-effect-extractor.test.ts +0 -119
  268. package/src/__tests__/transition-extractor.test.ts +0 -68
  269. package/src/__tests__/ts-to-expression.test.ts +0 -462
  270. package/src/__tests__/type-generator.test.ts +0 -201
  271. package/src/__tests__/uber-validation.test.ts +0 -502
  272. package/src/action-compiler.ts +0 -361
  273. package/src/babel/emitters/experience-transform.ts +0 -199
  274. package/src/babel/emitters/ir-to-tsx-emitter.ts +0 -110
  275. package/src/babel/emitters/pure-form-emitter.ts +0 -1023
  276. package/src/babel/emitters/runtime-glue-emitter.ts +0 -39
  277. package/src/babel/extractors/change-extractor.ts +0 -199
  278. package/src/babel/extractors/component-extractor.ts +0 -907
  279. package/src/babel/extractors/computed-extractor.ts +0 -262
  280. package/src/babel/extractors/context-extractor.ts +0 -277
  281. package/src/babel/extractors/during-extractor.ts +0 -295
  282. package/src/babel/extractors/effect-extractor.ts +0 -340
  283. package/src/babel/extractors/event-extractor.ts +0 -235
  284. package/src/babel/extractors/grammar-island-extractor.ts +0 -302
  285. package/src/babel/extractors/model-extractor.ts +0 -1018
  286. package/src/babel/extractors/router-extractor.ts +0 -303
  287. package/src/babel/extractors/server-action-extractor.ts +0 -173
  288. package/src/babel/extractors/server-action-hook-extractor.ts +0 -72
  289. package/src/babel/extractors/server-state-extractor.ts +0 -88
  290. package/src/babel/extractors/state-extractor.ts +0 -214
  291. package/src/babel/extractors/transition-effect-extractor.ts +0 -176
  292. package/src/babel/extractors/transition-extractor.ts +0 -143
  293. package/src/babel/index.ts +0 -24
  294. package/src/babel/transpilers/ts-to-expression.ts +0 -674
  295. package/src/babel/visitor.ts +0 -807
  296. package/src/cli/auth.ts +0 -255
  297. package/src/cli/build.ts +0 -288
  298. package/src/cli/deploy.ts +0 -206
  299. package/src/cli/index.ts +0 -328
  300. package/src/cli/installer.ts +0 -261
  301. package/src/cli/lock-file.ts +0 -94
  302. package/src/cli/mmrc.ts +0 -22
  303. package/src/cli/pull.ts +0 -172
  304. package/src/cli/registry-client.ts +0 -175
  305. package/src/cli/test.ts +0 -397
  306. package/src/cli/type-generator.ts +0 -243
  307. package/src/codemod/__tests__/forward.test.ts +0 -239
  308. package/src/codemod/__tests__/reverse.test.ts +0 -145
  309. package/src/codemod/__tests__/round-trip.test.ts +0 -137
  310. package/src/codemod/annotation.ts +0 -97
  311. package/src/codemod/classify.ts +0 -197
  312. package/src/codemod/cli.ts +0 -207
  313. package/src/codemod/control-flow.ts +0 -409
  314. package/src/codemod/forward.ts +0 -244
  315. package/src/codemod/import-manager.ts +0 -171
  316. package/src/codemod/index.ts +0 -120
  317. package/src/codemod/reverse.ts +0 -197
  318. package/src/codemod/rules.ts +0 -174
  319. package/src/codemod/state-transform.ts +0 -126
  320. package/src/decompiler/ast-builder.ts +0 -538
  321. package/src/decompiler/config-generator.ts +0 -151
  322. package/src/decompiler/index.ts +0 -315
  323. package/src/decompiler/project-decompiler.ts +0 -1776
  324. package/src/decompiler/project.ts +0 -862
  325. package/src/decompiler/split-strategy.ts +0 -140
  326. package/src/decompiler/state-emitter.ts +0 -1053
  327. package/src/decompiler/sx-emitter.ts +0 -318
  328. package/src/decompiler/workspace-hydrator.ts +0 -189
  329. package/src/dev-server.ts +0 -238
  330. package/src/envelope/fs-tree.ts +0 -217
  331. package/src/envelope/source-envelope.ts +0 -264
  332. package/src/envelope.ts +0 -315
  333. package/src/incremental-compiler.ts +0 -401
  334. package/src/index.ts +0 -99
  335. package/src/model-compiler.ts +0 -277
  336. package/src/project-compiler.ts +0 -1629
  337. package/src/route-extractor.ts +0 -333
  338. package/src/testing/index.ts +0 -32
  339. package/src/testing/snapshot.ts +0 -252
  340. package/src/testing/test-utils.ts +0 -226
  341. package/src/types.ts +0 -68
  342. package/src/vite/index.ts +0 -288
  343. package/test-compile.mjs +0 -142
  344. package/tsconfig.json +0 -25
  345. package/tsup.config.ts +0 -23
  346. package/vitest.config.ts +0 -9
@@ -1,502 +0,0 @@
1
- /**
2
- * Uber App Validation Test — Phase 6 UBER VALIDATION
3
- *
4
- * Reads the 32-file Uber example app, compiles all files through the
5
- * ProjectCompiler and standalone modules, and validates the compiled IR
6
- * has the correct structure.
7
- */
8
-
9
- import { describe, it, expect } from 'vitest';
10
- import * as fs from 'fs';
11
- import * as path from 'path';
12
- import { compileProject } from '../project-compiler';
13
- import { compileModel, compileModels } from '../model-compiler';
14
- import { extractRoutes } from '../route-extractor';
15
- import { compileActions } from '../action-compiler';
16
- import { decompileProjectEnhanced } from '../decompiler/project-decompiler';
17
-
18
- // =============================================================================
19
- // Helpers
20
- // =============================================================================
21
-
22
- const UBER_APP_DIR = path.resolve(__dirname, '../../examples/uber-app');
23
-
24
- /** Recursively read all files from a directory into a Record<string, string>. */
25
- function readAllFiles(dir: string, prefix = ''): Record<string, string> {
26
- const result: Record<string, string> = {};
27
- const entries = fs.readdirSync(dir, { withFileTypes: true });
28
- for (const entry of entries) {
29
- const relPath = prefix ? prefix + '/' + entry.name : entry.name;
30
- const absPath = path.join(dir, entry.name);
31
- if (entry.isDirectory()) {
32
- Object.assign(result, readAllFiles(absPath, relPath));
33
- } else if (/\.(ts|tsx)$/.test(entry.name)) {
34
- result[relPath] = fs.readFileSync(absPath, 'utf-8');
35
- }
36
- }
37
- return result;
38
- }
39
-
40
- /** Filter files by path prefix. */
41
- function filterFiles(files: Record<string, string>, prefix: string): Record<string, string> {
42
- const result: Record<string, string> = {};
43
- for (const [k, v] of Object.entries(files)) {
44
- if (k.startsWith(prefix)) result[k] = v;
45
- }
46
- return result;
47
- }
48
-
49
- // =============================================================================
50
- // Load Uber app files
51
- // =============================================================================
52
-
53
- const allFiles = readAllFiles(UBER_APP_DIR);
54
-
55
- describe('Uber App Validation', () => {
56
- // =========================================================================
57
- // File inventory
58
- // =========================================================================
59
- it('should discover all 32 files', () => {
60
- const fileCount = Object.keys(allFiles).length;
61
- expect(fileCount).toBeGreaterThanOrEqual(30);
62
- // Verify key file categories exist
63
- const models = Object.keys(allFiles).filter(f => f.startsWith('models/'));
64
- const workflows = Object.keys(allFiles).filter(f => f.startsWith('workflows/'));
65
- const actions = Object.keys(allFiles).filter(f => f.startsWith('actions/'));
66
- const appPages = Object.keys(allFiles).filter(f => f.startsWith('app/'));
67
- expect(models).toHaveLength(6);
68
- expect(workflows).toHaveLength(4);
69
- expect(actions).toHaveLength(4);
70
- expect(appPages).toHaveLength(11);
71
- });
72
-
73
- // =========================================================================
74
- // Full ProjectCompiler
75
- // =========================================================================
76
- describe('ProjectCompiler — full compilation', () => {
77
- const result = compileProject(allFiles);
78
-
79
- it('should compile without fatal errors', () => {
80
- // Some files may produce warnings, but no unrecoverable errors
81
- const fatalErrors = result.errors.filter(e => !e.message.includes('Cannot find'));
82
- // We allow import resolution errors since examples use mock imports
83
- expect(result.ir).toBeDefined();
84
- expect(result.ir.slug).toBe('uber-rideshare');
85
- });
86
-
87
- it('should resolve config from mm.config.ts', () => {
88
- expect(result.ir.name).toBe('Uber Rideshare');
89
- expect(result.ir.version).toBe('2.0.0');
90
- expect(result.ir.category).toBe('blueprint');
91
- });
92
-
93
- it('should produce child definitions for workflows and models', () => {
94
- expect(result.childDefinitions.length).toBeGreaterThanOrEqual(10);
95
- const slugs = result.childDefinitions.map(c => c.slug);
96
- // Should have model children
97
- expect(slugs.some(s => s.includes('user') || s.includes('User'))).toBe(true);
98
- expect(slugs.some(s => s.includes('ride') || s.includes('Ride'))).toBe(true);
99
- // Should have a router child
100
- expect(slugs.some(s => s.includes('router'))).toBe(true);
101
- });
102
-
103
- it('should extract route table from app/ pages', () => {
104
- expect(result.routeTable.length).toBeGreaterThanOrEqual(11);
105
- const paths = result.routeTable.map(r => r.path);
106
- expect(paths.some(p => p.includes('rider'))).toBe(true);
107
- expect(paths.some(p => p.includes('driver'))).toBe(true);
108
- expect(paths.some(p => p.includes('admin'))).toBe(true);
109
- });
110
-
111
- it('should extract server actions', () => {
112
- expect(result.serverActions.length).toBeGreaterThanOrEqual(4);
113
- const names = result.serverActions.map(a => a.name);
114
- expect(names).toContain('findBestDriver');
115
- expect(names).toContain('calculateEstimatedFare');
116
- expect(names).toContain('authorizePayment');
117
- });
118
-
119
- it('should have composition metadata on parent IR', () => {
120
- const meta = result.ir.metadata as Record<string, unknown>;
121
- expect(meta.composition).toBeDefined();
122
- const comp = meta.composition as Record<string, number>;
123
- expect(comp.modelCount).toBe(6);
124
- expect(comp.workflowCount).toBe(4);
125
- expect(comp.routeCount).toBeGreaterThanOrEqual(11);
126
- expect(comp.serverActionCount).toBeGreaterThanOrEqual(4);
127
- });
128
-
129
- it('should produce Phase 2 model results', () => {
130
- expect(result.modelResults).toBeDefined();
131
- expect(result.modelResults!.size).toBe(6);
132
- });
133
-
134
- it('should produce Phase 2 action result', () => {
135
- expect(result.actionResult).toBeDefined();
136
- expect(result.actionResult!.actions.length).toBeGreaterThanOrEqual(4);
137
- });
138
-
139
- it('should produce Phase 2 route result', () => {
140
- expect(result.routeResult).toBeDefined();
141
- expect(result.routeResult!.routes.length).toBeGreaterThanOrEqual(11);
142
- });
143
-
144
- it('should capture component definitions from components/*.tsx', () => {
145
- expect(result.componentDefinitions).toBeDefined();
146
- const names = Object.keys(result.componentDefinitions);
147
- expect(names).toContain('MapView');
148
- expect(names).toContain('DriverCard');
149
- expect(names).toContain('RatingStars');
150
- expect(names).toContain('RideCard');
151
- });
152
-
153
- it('should capture component experience trees', () => {
154
- const mapView = result.componentDefinitions['MapView'];
155
- expect(mapView).toBeDefined();
156
- expect(mapView.experience).toBeDefined();
157
- expect(mapView.experience.component).toBeDefined();
158
- });
159
-
160
- it('should capture component props', () => {
161
- const mapView = result.componentDefinitions['MapView'];
162
- expect(mapView.props.length).toBeGreaterThan(0);
163
- expect(mapView.props).toContain('pickupLocation');
164
- expect(mapView.props).toContain('dropoffLocation');
165
- });
166
-
167
- it('should store componentDefinitions in parent IR metadata', () => {
168
- const meta = result.ir.metadata as Record<string, unknown>;
169
- expect(meta.componentDefinitions).toBeDefined();
170
- const defs = meta.componentDefinitions as Record<string, unknown>;
171
- expect(Object.keys(defs)).toContain('MapView');
172
- expect(Object.keys(defs)).toContain('DriverCard');
173
- });
174
-
175
- it('should include componentCount in composition metadata', () => {
176
- const meta = result.ir.metadata as Record<string, unknown>;
177
- const comp = meta.composition as Record<string, number>;
178
- expect(comp.componentCount).toBe(4);
179
- });
180
-
181
- it('should capture server action bodies', () => {
182
- const findBestDriver = result.serverActions.find(a => a.name === 'findBestDriver');
183
- expect(findBestDriver).toBeDefined();
184
- expect(findBestDriver!.body).toBeDefined();
185
- expect(findBestDriver!.body).toContain('export async function findBestDriver');
186
- expect(findBestDriver!.body).toContain('haversineKm');
187
- });
188
-
189
- it('should capture server action return types', () => {
190
- const findBestDriver = result.serverActions.find(a => a.name === 'findBestDriver');
191
- expect(findBestDriver).toBeDefined();
192
- expect(findBestDriver!.returnType).toContain('Promise');
193
- });
194
-
195
- it('should capture bodies for all exported server actions', () => {
196
- const actionsWithBodies = result.serverActions.filter(a => a.body);
197
- expect(actionsWithBodies.length).toBeGreaterThanOrEqual(10);
198
- // Every action should have a body
199
- for (const action of result.serverActions) {
200
- expect(action.body).toBeDefined();
201
- expect(action.body!.length).toBeGreaterThan(0);
202
- }
203
- });
204
- });
205
-
206
- // =========================================================================
207
- // ModelCompiler — standalone
208
- // =========================================================================
209
- describe('ModelCompiler — standalone', () => {
210
- const modelFiles = filterFiles(allFiles, 'models/');
211
-
212
- it('should compile all 6 models', () => {
213
- const results = compileModels(modelFiles);
214
- expect(results.size).toBe(6);
215
- });
216
-
217
- it('should produce correct field counts per model', () => {
218
- const results = compileModels(modelFiles);
219
- const fieldCounts: Record<string, number> = {};
220
- for (const [filename, r] of results) {
221
- fieldCounts[r.interfaceName] = r.fieldNames.length;
222
- }
223
- // Each model has a substantial number of fields
224
- expect(fieldCounts['User']).toBeGreaterThanOrEqual(20);
225
- expect(fieldCounts['Ride']).toBeGreaterThanOrEqual(40);
226
- expect(fieldCounts['Vehicle']).toBeGreaterThanOrEqual(15);
227
- expect(fieldCounts['Payment']).toBeGreaterThanOrEqual(30);
228
- expect(fieldCounts['Rating']).toBeGreaterThanOrEqual(15);
229
- expect(fieldCounts['Location']).toBeGreaterThanOrEqual(25);
230
- });
231
-
232
- it('should have states on all models', () => {
233
- const results = compileModels(modelFiles);
234
- for (const [, r] of results) {
235
- // Models always have at least one state (draft or derived)
236
- expect(r.stateNames.length).toBeGreaterThan(0);
237
- }
238
- });
239
-
240
- it('should detect fieldOptions on all models', () => {
241
- const results = compileModels(modelFiles);
242
- for (const [, r] of results) {
243
- expect(r.hasFieldOptions).toBe(true);
244
- expect(Object.keys(r.fieldOptions).length).toBeGreaterThan(0);
245
- }
246
- });
247
-
248
- it('should set category=data on all models', () => {
249
- const results = compileModels(modelFiles);
250
- for (const [, r] of results) {
251
- expect(r.ir.category).toBe('data');
252
- }
253
- });
254
-
255
- it('should compile individual models correctly', () => {
256
- const rideSource = modelFiles['models/ride.model.ts'];
257
- const result = compileModel('models/ride.model.ts', rideSource);
258
- expect(result.interfaceName).toBe('Ride');
259
- expect(result.ir.category).toBe('data');
260
- // Model has many fields from the Ride interface
261
- expect(result.fieldNames.length).toBeGreaterThanOrEqual(40);
262
- // fieldOptions should be detected
263
- expect(result.hasFieldOptions).toBe(true);
264
- expect(result.fieldOptions).toHaveProperty('vehicleType');
265
- });
266
- });
267
-
268
- // =========================================================================
269
- // RouteExtractor — standalone
270
- // =========================================================================
271
- describe('RouteExtractor — standalone', () => {
272
- const appFiles = filterFiles(allFiles, 'app/');
273
-
274
- it('should extract 11 routes', () => {
275
- const result = extractRoutes(appFiles);
276
- expect(result.routes.length).toBe(11);
277
- });
278
-
279
- it('should produce correct route paths', () => {
280
- const result = extractRoutes(appFiles);
281
- const paths = result.routes.map(r => r.path).sort();
282
- expect(paths).toContain('/rider/home');
283
- expect(paths).toContain('/rider/ride-tracking');
284
- expect(paths).toContain('/rider/payment-methods');
285
- expect(paths).toContain('/rider/ride-history');
286
- expect(paths).toContain('/driver/dashboard');
287
- expect(paths).toContain('/driver/earnings');
288
- expect(paths).toContain('/driver/navigation');
289
- expect(paths).toContain('/driver/ride-acceptance');
290
- expect(paths).toContain('/admin/analytics');
291
- expect(paths).toContain('/admin/fleet');
292
- expect(paths).toContain('/admin/surge-pricing');
293
- });
294
-
295
- it('should generate a router IR workflow', () => {
296
- const result = extractRoutes(appFiles);
297
- expect(result.routerIR).toBeDefined();
298
- expect(result.routerIR.slug).toContain('router');
299
- expect(result.routerIR.states.length).toBe(11);
300
- // Should have one START state
301
- const starts = result.routerIR.states.filter(s => s.type === 'START');
302
- expect(starts.length).toBe(1);
303
- });
304
-
305
- it('should generate navigation transitions', () => {
306
- const result = extractRoutes(appFiles);
307
- // n*(n-1) transitions for n routes
308
- expect(result.routerIR.transitions.length).toBe(11 * 10);
309
- });
310
-
311
- it('should assign correct state names', () => {
312
- const result = extractRoutes(appFiles);
313
- const stateNames = result.routes.map(r => r.stateName);
314
- expect(stateNames).toContain('RIDER_HOME');
315
- expect(stateNames).toContain('DRIVER_DASHBOARD');
316
- expect(stateNames).toContain('ADMIN_ANALYTICS');
317
- });
318
- });
319
-
320
- // =========================================================================
321
- // ActionCompiler — standalone
322
- // =========================================================================
323
- describe('ActionCompiler — standalone', () => {
324
- const actionFiles = filterFiles(allFiles, 'actions/');
325
-
326
- it('should compile all 4 action files', () => {
327
- const result = compileActions(actionFiles, { blueprintSlug: 'uber-rideshare' });
328
- expect(result.byFile.size).toBe(4);
329
- });
330
-
331
- it('should extract action registrations', () => {
332
- const result = compileActions(actionFiles, { blueprintSlug: 'uber-rideshare' });
333
- expect(result.actions.length).toBeGreaterThanOrEqual(10);
334
- const names = result.actions.map(a => a.name);
335
- // Matching
336
- expect(names).toContain('findBestDriver');
337
- expect(names).toContain('batchMatchDrivers');
338
- // Pricing
339
- expect(names).toContain('calculateEstimatedFare');
340
- expect(names).toContain('calculateActualFare');
341
- expect(names).toContain('calculateCancellationFee');
342
- expect(names).toContain('calculateSurge');
343
- // Payments
344
- expect(names).toContain('authorizePayment');
345
- expect(names).toContain('capturePayment');
346
- expect(names).toContain('settlePayment');
347
- expect(names).toContain('processRefund');
348
- expect(names).toContain('processDriverPayout');
349
- // Notifications
350
- expect(names).toContain('notifyRideMatched');
351
- expect(names).toContain('notifyDriverArrived');
352
- });
353
-
354
- it('should generate correct endpoints', () => {
355
- const result = compileActions(actionFiles, { blueprintSlug: 'uber-rideshare' });
356
- const pricing = result.actions.find(a => a.name === 'calculateEstimatedFare');
357
- expect(pricing).toBeDefined();
358
- expect(pricing!.endpoint).toBe('/api/v1/actions/uber-rideshare/pricing/calculateEstimatedFare');
359
- expect(pricing!.group).toBe('pricing');
360
- expect(pricing!.actionId).toBe('pricing:calculateEstimatedFare');
361
- });
362
-
363
- it('should group actions by file', () => {
364
- const result = compileActions(actionFiles, { blueprintSlug: 'uber-rideshare' });
365
- expect(result.byGroup.has('matching')).toBe(true);
366
- expect(result.byGroup.has('pricing')).toBe(true);
367
- expect(result.byGroup.has('payments')).toBe(true);
368
- expect(result.byGroup.has('notifications')).toBe(true);
369
- });
370
-
371
- it('should mark async actions correctly', () => {
372
- const result = compileActions(actionFiles, { blueprintSlug: 'uber-rideshare' });
373
- // All exported functions in server actions should be async
374
- for (const action of result.actions) {
375
- expect(action.async).toBe(true);
376
- }
377
- });
378
- });
379
-
380
- // =========================================================================
381
- // Round-trip: compile → structure check
382
- // =========================================================================
383
- describe('Round-trip consistency', () => {
384
- it('should produce identical results on double compilation', () => {
385
- const result1 = compileProject(allFiles);
386
- const result2 = compileProject(allFiles);
387
-
388
- // Same number of child definitions
389
- expect(result1.childDefinitions.length).toBe(result2.childDefinitions.length);
390
- // Same route table
391
- expect(result1.routeTable.length).toBe(result2.routeTable.length);
392
- // Same server actions
393
- expect(result1.serverActions.length).toBe(result2.serverActions.length);
394
- // Same parent slug
395
- expect(result1.ir.slug).toBe(result2.ir.slug);
396
- // Same error count
397
- expect(result1.errors.length).toBe(result2.errors.length);
398
- });
399
-
400
- it('should preserve model field counts through project compilation', () => {
401
- const standalone = compileModels(filterFiles(allFiles, 'models/'));
402
- const project = compileProject(allFiles);
403
-
404
- // Model results from project should match standalone
405
- expect(project.modelResults).toBeDefined();
406
- expect(project.modelResults!.size).toBe(standalone.size);
407
-
408
- for (const [filename, standaloneResult] of standalone) {
409
- const projectResult = project.modelResults!.get(filename);
410
- expect(projectResult).toBeDefined();
411
- expect(projectResult!.fieldNames.length).toBe(standaloneResult.fieldNames.length);
412
- expect(projectResult!.interfaceName).toBe(standaloneResult.interfaceName);
413
- }
414
- });
415
-
416
- it('should preserve action counts through project compilation', () => {
417
- const standalone = compileActions(filterFiles(allFiles, 'actions/'), {
418
- blueprintSlug: 'uber-rideshare',
419
- });
420
- const project = compileProject(allFiles);
421
-
422
- expect(project.actionResult).toBeDefined();
423
- expect(project.actionResult!.actions.length).toBe(standalone.actions.length);
424
- });
425
-
426
- it('should preserve route counts through project compilation', () => {
427
- const standalone = extractRoutes(filterFiles(allFiles, 'app/'));
428
- const project = compileProject(allFiles);
429
-
430
- expect(project.routeResult).toBeDefined();
431
- expect(project.routeResult!.routes.length).toBe(standalone.routes.length);
432
- });
433
- });
434
-
435
- // =========================================================================
436
- // Decompiler round-trip: compile → IR → decompile → verify output
437
- // =========================================================================
438
- describe('Decompiler round-trip', () => {
439
- const compiled = compileProject(allFiles);
440
-
441
- // Build a simplified IR for decompiler testing — strip page experience
442
- // to avoid bindJSXTree errors from complex $instance bindings
443
- function makeDecompilerInput() {
444
- const ir = { ...compiled.ir } as any;
445
- // Remove the complex page experience (has $instance bindings the decompiler can't resolve)
446
- delete ir.experience;
447
- // Set a simple experience tree for the main file
448
- ir.experience = {
449
- id: 'root',
450
- component: 'Stack',
451
- children: [
452
- { id: 'h1', component: 'Heading', config: { value: 'Uber Rideshare' } },
453
- ],
454
- };
455
- return ir;
456
- }
457
-
458
- it('should decompile and reconstruct component files', () => {
459
- const decompileResult = decompileProjectEnhanced(makeDecompilerInput());
460
- const paths = decompileResult.files.map(f => f.path);
461
-
462
- expect(paths.some(p => p === 'components/MapView.tsx')).toBe(true);
463
- expect(paths.some(p => p === 'components/DriverCard.tsx')).toBe(true);
464
- expect(paths.some(p => p === 'components/RatingStars.tsx')).toBe(true);
465
- expect(paths.some(p => p === 'components/RideCard.tsx')).toBe(true);
466
- });
467
-
468
- it('should decompile and reconstruct server action files', () => {
469
- const decompileResult = decompileProjectEnhanced(makeDecompilerInput());
470
- const actionFiles = decompileResult.files.filter(f => f.role === 'server-action');
471
-
472
- expect(actionFiles.length).toBeGreaterThanOrEqual(1);
473
- // At least one action file should contain preserved function bodies
474
- const hasPreservedBodies = actionFiles.some(f =>
475
- f.content.includes('export async function')
476
- );
477
- expect(hasPreservedBodies).toBe(true);
478
- });
479
-
480
- it('should reconstruct server action files with full function bodies', () => {
481
- const decompileResult = decompileProjectEnhanced(makeDecompilerInput());
482
- const actionFiles = decompileResult.files.filter(f => f.role === 'server-action');
483
-
484
- // Find an action file that contains matching.server.ts content
485
- const matchingFile = actionFiles.find(f =>
486
- f.content.includes('findBestDriver')
487
- );
488
- expect(matchingFile).toBeDefined();
489
- expect(matchingFile!.content).toContain('haversineKm');
490
- expect(matchingFile!.content).toContain('export async function findBestDriver');
491
- });
492
-
493
- it('should reconstruct component files with JSX', () => {
494
- const decompileResult = decompileProjectEnhanced(makeDecompilerInput());
495
- const mapViewFile = decompileResult.files.find(f => f.path === 'components/MapView.tsx');
496
- expect(mapViewFile).toBeDefined();
497
- // Should contain JSX from the decompiled experience tree
498
- expect(mapViewFile!.content.length).toBeGreaterThan(50);
499
- expect(mapViewFile!.content).toContain('MapView');
500
- });
501
- });
502
- });