@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.
- package/ATOM-PIPELINE.md +144 -0
- package/README.md +88 -40
- package/dist/babel/index.js +113 -6
- package/dist/babel/index.mjs +2 -2
- package/dist/chunk-3USIFFE4.mjs +2190 -0
- package/dist/chunk-45YMGEVT.mjs +186 -0
- package/dist/chunk-4FN2AISW.mjs +148 -0
- package/dist/chunk-4OPI5L7G.mjs +2593 -0
- package/dist/chunk-4RYTKOOJ.mjs +186 -0
- package/dist/chunk-5RKTOVR5.mjs +244 -0
- package/dist/chunk-5YDMOO4X.mjs +214 -0
- package/dist/chunk-64ZWEMLJ.mjs +148 -0
- package/dist/chunk-6XP4KSWQ.mjs +2190 -0
- package/dist/chunk-72QWL54I.mjs +175 -0
- package/dist/chunk-7B4TRI7C.mjs +4835 -0
- package/dist/chunk-7ZKGHTNB.mjs +4952 -0
- package/dist/chunk-CIESM3BP.mjs +33 -0
- package/dist/chunk-DE3ZGQAC.mjs +148 -0
- package/dist/chunk-DMCY3BBG.mjs +1933 -0
- package/dist/chunk-DPIK3PJS.mjs +244 -0
- package/dist/chunk-E5IVH4RE.mjs +186 -0
- package/dist/chunk-E6FZNUR5.mjs +4953 -0
- package/dist/chunk-EJRBDQDP.mjs +2607 -0
- package/dist/chunk-ELO4TXJL.mjs +186 -0
- package/dist/chunk-FKRO52XH.mjs +3446 -0
- package/dist/chunk-FL4YAKU6.mjs +4941 -0
- package/dist/chunk-FYT47UBU.mjs +5076 -0
- package/dist/chunk-GCLGPOJZ.mjs +148 -0
- package/dist/chunk-GXB4JOP7.mjs +5072 -0
- package/dist/chunk-HFXOUMTD.mjs +175 -0
- package/dist/chunk-HWIZ47US.mjs +214 -0
- package/dist/chunk-IB7MNPQL.mjs +4953 -0
- package/dist/chunk-ICSIHQCG.mjs +148 -0
- package/dist/chunk-JLA5VNQ3.mjs +186 -0
- package/dist/chunk-JQLWFCTM.mjs +214 -0
- package/dist/chunk-KFJJCQAL.mjs +148 -0
- package/dist/chunk-KJUIIEQE.mjs +186 -0
- package/dist/chunk-KNWTHRVQ.mjs +175 -0
- package/dist/chunk-KSG4XSZF.mjs +175 -0
- package/dist/chunk-LF5N6DOU.mjs +175 -0
- package/dist/chunk-LJQCM2IM.mjs +214 -0
- package/dist/chunk-NW6555WJ.mjs +186 -0
- package/dist/chunk-OMZE6VLQ.mjs +214 -0
- package/dist/chunk-P4BR7WVO.mjs +2190 -0
- package/dist/chunk-QQHVYH2X.mjs +244 -0
- package/dist/chunk-S5QLWLLT.mjs +186 -0
- package/dist/chunk-SCWGT2FY.mjs +2190 -0
- package/dist/chunk-SMKJUSB3.mjs +2190 -0
- package/dist/chunk-VCAY2KGM.mjs +175 -0
- package/dist/chunk-WECAV6QB.mjs +148 -0
- package/dist/chunk-WMKBXUCE.mjs +3228 -0
- package/dist/chunk-XAJ5BKKL.mjs +4947 -0
- package/dist/chunk-XG2X7AEA.mjs +175 -0
- package/dist/chunk-XG7Z23NQ.mjs +148 -0
- package/dist/chunk-XWZAOCQ7.mjs +2607 -0
- package/dist/chunk-Y6MA7ULW.mjs +148 -0
- package/dist/chunk-YMS7Q7LG.mjs +214 -0
- package/dist/chunk-ZA37XTGA.mjs +175 -0
- package/dist/cli/index.js +1616 -366
- package/dist/cli/index.mjs +8 -8
- package/dist/codemod/cli.mjs +1 -1
- package/dist/codemod/index.mjs +1 -1
- package/dist/dev-server-RmGHIntF.d.mts +113 -0
- package/dist/dev-server-RmGHIntF.d.ts +113 -0
- package/dist/dev-server.d.mts +1 -1
- package/dist/dev-server.d.ts +1 -1
- package/dist/dev-server.js +982 -53
- package/dist/dev-server.mjs +5 -5
- package/dist/envelope.js +113 -6
- package/dist/envelope.mjs +3 -3
- package/dist/index.d.mts +5 -1
- package/dist/index.d.ts +5 -1
- package/dist/index.js +992 -63
- package/dist/index.mjs +8 -8
- package/{src/cli/init.ts → dist/init-7JQMAAXS.mjs} +70 -95
- package/dist/init-EHO4VQ22.mjs +369 -0
- package/dist/init-UC3FWPIW.mjs +367 -0
- package/dist/init-UNSMVKIK.mjs +366 -0
- package/dist/init-UNV5XIDE.mjs +367 -0
- package/dist/project-compiler-2P4N4DR7.mjs +10 -0
- package/dist/project-compiler-D2LCC27O.mjs +10 -0
- package/dist/project-compiler-EJ3GANJE.mjs +10 -0
- package/dist/project-compiler-LOQKVRZJ.mjs +10 -0
- package/dist/project-compiler-RQ6OQKRM.mjs +10 -0
- package/dist/project-compiler-VWNNCHGO.mjs +10 -0
- package/dist/project-compiler-XVAAU4C5.mjs +10 -0
- package/dist/project-compiler-YES5FGMD.mjs +10 -0
- package/dist/project-compiler-ZKMQDLGU.mjs +10 -0
- package/dist/project-decompiler-FLXCEJHS.mjs +7 -0
- package/dist/project-decompiler-VLPR22QF.mjs +7 -0
- package/dist/pull-FUS5QYZS.mjs +109 -0
- package/dist/pull-LD5ENLGY.mjs +109 -0
- package/dist/testing/index.js +113 -6
- package/dist/testing/index.mjs +2 -2
- package/dist/vite/index.js +113 -6
- package/dist/vite/index.mjs +3 -3
- package/examples/uber-app/app/admin/fleet.tsx +19 -19
- package/package.json +4 -3
- package/compile-blueprint-chat.mjs +0 -99
- package/compile-blueprint-glass-console.mjs +0 -98
- package/compile-chat-defs.mjs +0 -92
- package/examples/uber-app/tests/payment.test.tsx +0 -129
- package/examples/uber-app/tests/ride-flow.test.tsx +0 -123
- package/package.json.backup +0 -86
- package/scripts/decompile.ts +0 -226
- package/scripts/seed-auth.ts +0 -267
- package/scripts/seed-uber.ts +0 -248
- package/scripts/validate-uber.ts +0 -119
- package/seed-blueprint-chat.mjs +0 -444
- package/seed-blueprint-glass-console.mjs +0 -445
- package/seed-compiled.mjs +0 -318
- package/src/RoundTripValidator.ts +0 -400
- package/src/__tests__/atom-rendering-coverage.test.ts +0 -680
- package/src/__tests__/auth-module-compilation.test.ts +0 -247
- package/src/__tests__/auth-template-compilation.test.ts +0 -589
- package/src/__tests__/change-extractor.test.ts +0 -142
- package/src/__tests__/cli-pull.test.ts +0 -73
- package/src/__tests__/cli-test.test.ts +0 -72
- package/src/__tests__/component-extractor.test.ts +0 -331
- package/src/__tests__/context-extractor.test.ts +0 -145
- package/src/__tests__/decompiler.test.ts +0 -718
- package/src/__tests__/define-blueprint.test.ts +0 -133
- package/src/__tests__/definition-validator.test.ts +0 -519
- package/src/__tests__/during-extractor.test.ts +0 -152
- package/src/__tests__/effect-extractor.test.ts +0 -107
- package/src/__tests__/event-emission.test.ts +0 -127
- package/src/__tests__/examples.test.ts +0 -236
- package/src/__tests__/full-blueprint-coverage.test.ts +0 -1221
- package/src/__tests__/golden-suite.test.ts +0 -403
- package/src/__tests__/grammar-island-extractor.test.ts +0 -289
- package/src/__tests__/instance-key.test.ts +0 -82
- package/src/__tests__/ir-migration.test.ts +0 -255
- package/src/__tests__/lock-file.test.ts +0 -117
- package/src/__tests__/model-extractor.test.ts +0 -195
- package/src/__tests__/model-field-acl.test.ts +0 -237
- package/src/__tests__/model-hooks.test.ts +0 -130
- package/src/__tests__/model-ref-resolution.test.ts +0 -268
- package/src/__tests__/model-roundtrip.test.ts +0 -502
- package/src/__tests__/model-runtime.test.ts +0 -112
- package/src/__tests__/model-transitions.test.ts +0 -183
- package/src/__tests__/nrt-action-trace.test.ts +0 -391
- package/src/__tests__/pipeline-hardening.test.ts +0 -413
- package/src/__tests__/project-compiler.test.ts +0 -546
- package/src/__tests__/project-decompiler.test.ts +0 -343
- package/src/__tests__/query-compilation.test.ts +0 -145
- package/src/__tests__/round-trip/PLAN.md +0 -158
- package/src/__tests__/round-trip/README.md +0 -52
- package/src/__tests__/round-trip/RESULTS.md +0 -86
- package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +0 -55
- package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +0 -11
- package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +0 -54
- package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +0 -79
- package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +0 -12
- package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +0 -50
- package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +0 -25
- package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +0 -11
- package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +0 -32
- package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +0 -79
- package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +0 -10
- package/src/__tests__/round-trip/round-trip.test.ts +0 -2598
- package/src/__tests__/round-trip-ir.test.ts +0 -300
- package/src/__tests__/round-trip.test.ts +0 -1212
- package/src/__tests__/route-merging.test.ts +0 -372
- package/src/__tests__/router-composition.test.ts +0 -489
- package/src/__tests__/router-extractor.test.ts +0 -176
- package/src/__tests__/server-action-extractor.test.ts +0 -128
- package/src/__tests__/smart-type-inference.test.ts +0 -365
- package/src/__tests__/source-envelope.test.ts +0 -284
- package/src/__tests__/source-fidelity.test.ts +0 -516
- package/src/__tests__/state-extractor.test.ts +0 -115
- package/src/__tests__/strict-mode.test.ts +0 -227
- package/src/__tests__/transition-effect-extractor.test.ts +0 -119
- package/src/__tests__/transition-extractor.test.ts +0 -68
- package/src/__tests__/ts-to-expression.test.ts +0 -462
- package/src/__tests__/type-generator.test.ts +0 -201
- package/src/__tests__/uber-validation.test.ts +0 -502
- package/src/action-compiler.ts +0 -361
- package/src/babel/emitters/experience-transform.ts +0 -199
- package/src/babel/emitters/ir-to-tsx-emitter.ts +0 -110
- package/src/babel/emitters/pure-form-emitter.ts +0 -1023
- package/src/babel/emitters/runtime-glue-emitter.ts +0 -39
- package/src/babel/extractors/change-extractor.ts +0 -199
- package/src/babel/extractors/component-extractor.ts +0 -907
- package/src/babel/extractors/computed-extractor.ts +0 -262
- package/src/babel/extractors/context-extractor.ts +0 -277
- package/src/babel/extractors/during-extractor.ts +0 -295
- package/src/babel/extractors/effect-extractor.ts +0 -340
- package/src/babel/extractors/event-extractor.ts +0 -235
- package/src/babel/extractors/grammar-island-extractor.ts +0 -302
- package/src/babel/extractors/model-extractor.ts +0 -1018
- package/src/babel/extractors/router-extractor.ts +0 -303
- package/src/babel/extractors/server-action-extractor.ts +0 -173
- package/src/babel/extractors/server-action-hook-extractor.ts +0 -72
- package/src/babel/extractors/server-state-extractor.ts +0 -88
- package/src/babel/extractors/state-extractor.ts +0 -214
- package/src/babel/extractors/transition-effect-extractor.ts +0 -176
- package/src/babel/extractors/transition-extractor.ts +0 -143
- package/src/babel/index.ts +0 -24
- package/src/babel/transpilers/ts-to-expression.ts +0 -674
- package/src/babel/visitor.ts +0 -807
- package/src/cli/auth.ts +0 -255
- package/src/cli/build.ts +0 -288
- package/src/cli/deploy.ts +0 -206
- package/src/cli/index.ts +0 -328
- package/src/cli/installer.ts +0 -261
- package/src/cli/lock-file.ts +0 -94
- package/src/cli/mmrc.ts +0 -22
- package/src/cli/pull.ts +0 -172
- package/src/cli/registry-client.ts +0 -175
- package/src/cli/test.ts +0 -397
- package/src/cli/type-generator.ts +0 -243
- package/src/codemod/__tests__/forward.test.ts +0 -239
- package/src/codemod/__tests__/reverse.test.ts +0 -145
- package/src/codemod/__tests__/round-trip.test.ts +0 -137
- package/src/codemod/annotation.ts +0 -97
- package/src/codemod/classify.ts +0 -197
- package/src/codemod/cli.ts +0 -207
- package/src/codemod/control-flow.ts +0 -409
- package/src/codemod/forward.ts +0 -244
- package/src/codemod/import-manager.ts +0 -171
- package/src/codemod/index.ts +0 -120
- package/src/codemod/reverse.ts +0 -197
- package/src/codemod/rules.ts +0 -174
- package/src/codemod/state-transform.ts +0 -126
- package/src/decompiler/ast-builder.ts +0 -538
- package/src/decompiler/config-generator.ts +0 -151
- package/src/decompiler/index.ts +0 -315
- package/src/decompiler/project-decompiler.ts +0 -1776
- package/src/decompiler/project.ts +0 -862
- package/src/decompiler/split-strategy.ts +0 -140
- package/src/decompiler/state-emitter.ts +0 -1053
- package/src/decompiler/sx-emitter.ts +0 -318
- package/src/decompiler/workspace-hydrator.ts +0 -189
- package/src/dev-server.ts +0 -238
- package/src/envelope/fs-tree.ts +0 -217
- package/src/envelope/source-envelope.ts +0 -264
- package/src/envelope.ts +0 -315
- package/src/incremental-compiler.ts +0 -401
- package/src/index.ts +0 -99
- package/src/model-compiler.ts +0 -277
- package/src/project-compiler.ts +0 -1629
- package/src/route-extractor.ts +0 -333
- package/src/testing/index.ts +0 -32
- package/src/testing/snapshot.ts +0 -252
- package/src/testing/test-utils.ts +0 -226
- package/src/types.ts +0 -68
- package/src/vite/index.ts +0 -288
- package/test-compile.mjs +0 -142
- package/tsconfig.json +0 -25
- package/tsup.config.ts +0 -23
- 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
|
-
});
|