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

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 (265) hide show
  1. package/ATOM-PIPELINE.md +144 -0
  2. package/README.md +88 -40
  3. package/dist/babel/index.js +2814 -277
  4. package/dist/babel/index.mjs +2 -2
  5. package/dist/chunk-3USIFFE4.mjs +2190 -0
  6. package/dist/chunk-45YMGEVT.mjs +186 -0
  7. package/dist/chunk-4FN2AISW.mjs +148 -0
  8. package/dist/chunk-4OPI5L7G.mjs +2593 -0
  9. package/dist/chunk-4RYTKOOJ.mjs +186 -0
  10. package/dist/chunk-52XHYD2V.mjs +214 -0
  11. package/dist/chunk-5GUFFFGL.mjs +148 -0
  12. package/dist/chunk-5RKTOVR5.mjs +244 -0
  13. package/dist/chunk-5YDMOO4X.mjs +214 -0
  14. package/dist/chunk-64ZWEMLJ.mjs +148 -0
  15. package/dist/chunk-6XP4KSWQ.mjs +2190 -0
  16. package/dist/chunk-72QWL54I.mjs +175 -0
  17. package/dist/chunk-7B4TRI7C.mjs +4835 -0
  18. package/dist/chunk-7ZKGHTNB.mjs +4952 -0
  19. package/dist/chunk-CIESM3BP.mjs +33 -0
  20. package/dist/chunk-DE3ZGQAC.mjs +148 -0
  21. package/dist/chunk-DMCY3BBG.mjs +1933 -0
  22. package/dist/chunk-DPIK3PJS.mjs +244 -0
  23. package/dist/chunk-E5IVH4RE.mjs +186 -0
  24. package/dist/chunk-E6FZNUR5.mjs +4953 -0
  25. package/dist/chunk-EJRBDQDP.mjs +2607 -0
  26. package/dist/chunk-ELO4TXJL.mjs +186 -0
  27. package/dist/chunk-EO6SYNCG.mjs +175 -0
  28. package/dist/chunk-FKRO52XH.mjs +3446 -0
  29. package/dist/chunk-FL4YAKU6.mjs +4941 -0
  30. package/dist/chunk-FYT47UBU.mjs +5076 -0
  31. package/dist/chunk-GCLGPOJZ.mjs +148 -0
  32. package/dist/chunk-GXB4JOP7.mjs +5072 -0
  33. package/dist/chunk-HFXOUMTD.mjs +175 -0
  34. package/dist/chunk-HWIZ47US.mjs +214 -0
  35. package/dist/chunk-IB7MNPQL.mjs +4953 -0
  36. package/dist/chunk-ICSIHQCG.mjs +148 -0
  37. package/dist/chunk-J7JUAHS4.mjs +186 -0
  38. package/dist/chunk-JLA5VNQ3.mjs +186 -0
  39. package/dist/chunk-JQLWFCTM.mjs +214 -0
  40. package/dist/chunk-KFJJCQAL.mjs +148 -0
  41. package/dist/chunk-KJUIIEQE.mjs +186 -0
  42. package/dist/chunk-KNWTHRVQ.mjs +175 -0
  43. package/dist/chunk-KSG4XSZF.mjs +175 -0
  44. package/dist/chunk-LF5N6DOU.mjs +175 -0
  45. package/dist/chunk-LJQCM2IM.mjs +214 -0
  46. package/dist/chunk-NTB7OEX2.mjs +2918 -0
  47. package/dist/chunk-NW6555WJ.mjs +186 -0
  48. package/dist/chunk-OMZE6VLQ.mjs +214 -0
  49. package/dist/chunk-OPJKP747.mjs +7506 -0
  50. package/dist/chunk-P4BR7WVO.mjs +2190 -0
  51. package/dist/chunk-QQHVYH2X.mjs +244 -0
  52. package/dist/chunk-S5QLWLLT.mjs +186 -0
  53. package/dist/chunk-SCWGT2FY.mjs +2190 -0
  54. package/dist/chunk-SMKJUSB3.mjs +2190 -0
  55. package/dist/chunk-THFYE5ZX.mjs +244 -0
  56. package/dist/chunk-VCAY2KGM.mjs +175 -0
  57. package/dist/chunk-WBYMW4NQ.mjs +3450 -0
  58. package/dist/chunk-WECAV6QB.mjs +148 -0
  59. package/dist/chunk-WMKBXUCE.mjs +3228 -0
  60. package/dist/chunk-XAJ5BKKL.mjs +4947 -0
  61. package/dist/chunk-XG2X7AEA.mjs +175 -0
  62. package/dist/chunk-XG7Z23NQ.mjs +148 -0
  63. package/dist/chunk-XWZAOCQ7.mjs +2607 -0
  64. package/dist/chunk-Y6MA7ULW.mjs +148 -0
  65. package/dist/chunk-YMS7Q7LG.mjs +214 -0
  66. package/dist/chunk-ZA37XTGA.mjs +175 -0
  67. package/dist/cli/index.js +13189 -6838
  68. package/dist/cli/index.mjs +140 -22
  69. package/dist/codemod/cli.mjs +1 -1
  70. package/dist/codemod/index.mjs +1 -1
  71. package/dist/config-PL24KEWL.mjs +219 -0
  72. package/dist/dev-server-RmGHIntF.d.mts +113 -0
  73. package/dist/dev-server-RmGHIntF.d.ts +113 -0
  74. package/dist/dev-server.d.mts +1 -1
  75. package/dist/dev-server.d.ts +1 -1
  76. package/dist/dev-server.js +4135 -440
  77. package/dist/dev-server.mjs +5 -5
  78. package/dist/envelope.js +2812 -275
  79. package/dist/envelope.mjs +3 -3
  80. package/dist/index.d.mts +161 -2
  81. package/dist/index.d.ts +161 -2
  82. package/dist/index.js +4429 -428
  83. package/dist/index.mjs +217 -9
  84. package/{src/cli/init.ts → dist/init-7JQMAAXS.mjs} +70 -95
  85. package/dist/init-DQDX3QK6.mjs +369 -0
  86. package/dist/init-EHO4VQ22.mjs +369 -0
  87. package/dist/init-UC3FWPIW.mjs +367 -0
  88. package/dist/init-UNSMVKIK.mjs +366 -0
  89. package/dist/init-UNV5XIDE.mjs +367 -0
  90. package/dist/project-compiler-2P4N4DR7.mjs +10 -0
  91. package/dist/project-compiler-D2LCC27O.mjs +10 -0
  92. package/dist/project-compiler-EJ3GANJE.mjs +10 -0
  93. package/dist/project-compiler-LOQKVRZJ.mjs +10 -0
  94. package/dist/project-compiler-OP2VVGJQ.mjs +10 -0
  95. package/dist/project-compiler-RQ6OQKRM.mjs +10 -0
  96. package/dist/project-compiler-VWNNCHGO.mjs +10 -0
  97. package/dist/project-compiler-XVAAU4C5.mjs +10 -0
  98. package/dist/project-compiler-YES5FGMD.mjs +10 -0
  99. package/dist/project-compiler-ZKMQDLGU.mjs +10 -0
  100. package/dist/project-decompiler-FLXCEJHS.mjs +7 -0
  101. package/dist/project-decompiler-US7GAVIC.mjs +7 -0
  102. package/dist/project-decompiler-VLPR22QF.mjs +7 -0
  103. package/dist/pull-FUS5QYZS.mjs +109 -0
  104. package/dist/pull-LD5ENLGY.mjs +109 -0
  105. package/dist/pull-P44LDRWB.mjs +109 -0
  106. package/dist/testing/index.js +2822 -285
  107. package/dist/testing/index.mjs +2 -2
  108. package/dist/verify-SEIXUGN4.mjs +1833 -0
  109. package/dist/vite/index.js +2815 -278
  110. package/dist/vite/index.mjs +3 -3
  111. package/examples/uber-app/app/admin/fleet.tsx +19 -19
  112. package/package.json +16 -6
  113. package/compile-blueprint-chat.mjs +0 -99
  114. package/compile-blueprint-glass-console.mjs +0 -98
  115. package/compile-chat-defs.mjs +0 -92
  116. package/examples/uber-app/tests/payment.test.tsx +0 -129
  117. package/examples/uber-app/tests/ride-flow.test.tsx +0 -123
  118. package/package.json.backup +0 -86
  119. package/scripts/decompile.ts +0 -226
  120. package/scripts/seed-auth.ts +0 -267
  121. package/scripts/seed-uber.ts +0 -248
  122. package/scripts/validate-uber.ts +0 -119
  123. package/seed-blueprint-chat.mjs +0 -444
  124. package/seed-blueprint-glass-console.mjs +0 -445
  125. package/seed-compiled.mjs +0 -318
  126. package/src/RoundTripValidator.ts +0 -400
  127. package/src/__tests__/atom-rendering-coverage.test.ts +0 -680
  128. package/src/__tests__/auth-module-compilation.test.ts +0 -247
  129. package/src/__tests__/auth-template-compilation.test.ts +0 -589
  130. package/src/__tests__/change-extractor.test.ts +0 -142
  131. package/src/__tests__/cli-pull.test.ts +0 -73
  132. package/src/__tests__/cli-test.test.ts +0 -72
  133. package/src/__tests__/component-extractor.test.ts +0 -331
  134. package/src/__tests__/context-extractor.test.ts +0 -145
  135. package/src/__tests__/decompiler.test.ts +0 -718
  136. package/src/__tests__/define-blueprint.test.ts +0 -133
  137. package/src/__tests__/definition-validator.test.ts +0 -519
  138. package/src/__tests__/during-extractor.test.ts +0 -152
  139. package/src/__tests__/effect-extractor.test.ts +0 -107
  140. package/src/__tests__/event-emission.test.ts +0 -127
  141. package/src/__tests__/examples.test.ts +0 -236
  142. package/src/__tests__/full-blueprint-coverage.test.ts +0 -1221
  143. package/src/__tests__/golden-suite.test.ts +0 -403
  144. package/src/__tests__/grammar-island-extractor.test.ts +0 -289
  145. package/src/__tests__/instance-key.test.ts +0 -82
  146. package/src/__tests__/ir-migration.test.ts +0 -255
  147. package/src/__tests__/lock-file.test.ts +0 -117
  148. package/src/__tests__/model-extractor.test.ts +0 -195
  149. package/src/__tests__/model-field-acl.test.ts +0 -237
  150. package/src/__tests__/model-hooks.test.ts +0 -130
  151. package/src/__tests__/model-ref-resolution.test.ts +0 -268
  152. package/src/__tests__/model-roundtrip.test.ts +0 -502
  153. package/src/__tests__/model-runtime.test.ts +0 -112
  154. package/src/__tests__/model-transitions.test.ts +0 -183
  155. package/src/__tests__/nrt-action-trace.test.ts +0 -391
  156. package/src/__tests__/pipeline-hardening.test.ts +0 -413
  157. package/src/__tests__/project-compiler.test.ts +0 -546
  158. package/src/__tests__/project-decompiler.test.ts +0 -343
  159. package/src/__tests__/query-compilation.test.ts +0 -145
  160. package/src/__tests__/round-trip/PLAN.md +0 -158
  161. package/src/__tests__/round-trip/README.md +0 -52
  162. package/src/__tests__/round-trip/RESULTS.md +0 -86
  163. package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +0 -55
  164. package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +0 -11
  165. package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +0 -54
  166. package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +0 -79
  167. package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +0 -12
  168. package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +0 -50
  169. package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +0 -25
  170. package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +0 -11
  171. package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +0 -32
  172. package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +0 -79
  173. package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +0 -10
  174. package/src/__tests__/round-trip/round-trip.test.ts +0 -2598
  175. package/src/__tests__/round-trip-ir.test.ts +0 -300
  176. package/src/__tests__/round-trip.test.ts +0 -1212
  177. package/src/__tests__/route-merging.test.ts +0 -372
  178. package/src/__tests__/router-composition.test.ts +0 -489
  179. package/src/__tests__/router-extractor.test.ts +0 -176
  180. package/src/__tests__/server-action-extractor.test.ts +0 -128
  181. package/src/__tests__/smart-type-inference.test.ts +0 -365
  182. package/src/__tests__/source-envelope.test.ts +0 -284
  183. package/src/__tests__/source-fidelity.test.ts +0 -516
  184. package/src/__tests__/state-extractor.test.ts +0 -115
  185. package/src/__tests__/strict-mode.test.ts +0 -227
  186. package/src/__tests__/transition-effect-extractor.test.ts +0 -119
  187. package/src/__tests__/transition-extractor.test.ts +0 -68
  188. package/src/__tests__/ts-to-expression.test.ts +0 -462
  189. package/src/__tests__/type-generator.test.ts +0 -201
  190. package/src/__tests__/uber-validation.test.ts +0 -502
  191. package/src/action-compiler.ts +0 -361
  192. package/src/babel/emitters/experience-transform.ts +0 -199
  193. package/src/babel/emitters/ir-to-tsx-emitter.ts +0 -110
  194. package/src/babel/emitters/pure-form-emitter.ts +0 -1023
  195. package/src/babel/emitters/runtime-glue-emitter.ts +0 -39
  196. package/src/babel/extractors/change-extractor.ts +0 -199
  197. package/src/babel/extractors/component-extractor.ts +0 -907
  198. package/src/babel/extractors/computed-extractor.ts +0 -262
  199. package/src/babel/extractors/context-extractor.ts +0 -277
  200. package/src/babel/extractors/during-extractor.ts +0 -295
  201. package/src/babel/extractors/effect-extractor.ts +0 -340
  202. package/src/babel/extractors/event-extractor.ts +0 -235
  203. package/src/babel/extractors/grammar-island-extractor.ts +0 -302
  204. package/src/babel/extractors/model-extractor.ts +0 -1018
  205. package/src/babel/extractors/router-extractor.ts +0 -303
  206. package/src/babel/extractors/server-action-extractor.ts +0 -173
  207. package/src/babel/extractors/server-action-hook-extractor.ts +0 -72
  208. package/src/babel/extractors/server-state-extractor.ts +0 -88
  209. package/src/babel/extractors/state-extractor.ts +0 -214
  210. package/src/babel/extractors/transition-effect-extractor.ts +0 -176
  211. package/src/babel/extractors/transition-extractor.ts +0 -143
  212. package/src/babel/index.ts +0 -24
  213. package/src/babel/transpilers/ts-to-expression.ts +0 -674
  214. package/src/babel/visitor.ts +0 -807
  215. package/src/cli/auth.ts +0 -255
  216. package/src/cli/build.ts +0 -288
  217. package/src/cli/deploy.ts +0 -206
  218. package/src/cli/index.ts +0 -328
  219. package/src/cli/installer.ts +0 -261
  220. package/src/cli/lock-file.ts +0 -94
  221. package/src/cli/mmrc.ts +0 -22
  222. package/src/cli/pull.ts +0 -172
  223. package/src/cli/registry-client.ts +0 -175
  224. package/src/cli/test.ts +0 -397
  225. package/src/cli/type-generator.ts +0 -243
  226. package/src/codemod/__tests__/forward.test.ts +0 -239
  227. package/src/codemod/__tests__/reverse.test.ts +0 -145
  228. package/src/codemod/__tests__/round-trip.test.ts +0 -137
  229. package/src/codemod/annotation.ts +0 -97
  230. package/src/codemod/classify.ts +0 -197
  231. package/src/codemod/cli.ts +0 -207
  232. package/src/codemod/control-flow.ts +0 -409
  233. package/src/codemod/forward.ts +0 -244
  234. package/src/codemod/import-manager.ts +0 -171
  235. package/src/codemod/index.ts +0 -120
  236. package/src/codemod/reverse.ts +0 -197
  237. package/src/codemod/rules.ts +0 -174
  238. package/src/codemod/state-transform.ts +0 -126
  239. package/src/decompiler/ast-builder.ts +0 -538
  240. package/src/decompiler/config-generator.ts +0 -151
  241. package/src/decompiler/index.ts +0 -315
  242. package/src/decompiler/project-decompiler.ts +0 -1776
  243. package/src/decompiler/project.ts +0 -862
  244. package/src/decompiler/split-strategy.ts +0 -140
  245. package/src/decompiler/state-emitter.ts +0 -1053
  246. package/src/decompiler/sx-emitter.ts +0 -318
  247. package/src/decompiler/workspace-hydrator.ts +0 -189
  248. package/src/dev-server.ts +0 -238
  249. package/src/envelope/fs-tree.ts +0 -217
  250. package/src/envelope/source-envelope.ts +0 -264
  251. package/src/envelope.ts +0 -315
  252. package/src/incremental-compiler.ts +0 -401
  253. package/src/index.ts +0 -99
  254. package/src/model-compiler.ts +0 -277
  255. package/src/project-compiler.ts +0 -1629
  256. package/src/route-extractor.ts +0 -333
  257. package/src/testing/index.ts +0 -32
  258. package/src/testing/snapshot.ts +0 -252
  259. package/src/testing/test-utils.ts +0 -226
  260. package/src/types.ts +0 -68
  261. package/src/vite/index.ts +0 -288
  262. package/test-compile.mjs +0 -142
  263. package/tsconfig.json +0 -25
  264. package/tsup.config.ts +0 -23
  265. 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
- });