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

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