@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
package/seed-blueprint-chat.mjs
DELETED
|
@@ -1,444 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Blueprint Chat — Compile + Seed Pipeline
|
|
3
|
-
*
|
|
4
|
-
* Compiles React views to JSON, composes the view tree, and seeds
|
|
5
|
-
* the database with both the view tree and the blueprint manifest.
|
|
6
|
-
*
|
|
7
|
-
* Modes:
|
|
8
|
-
* node seed-blueprint-chat.mjs → full (compile + seed)
|
|
9
|
-
* node seed-blueprint-chat.mjs --seed-only → seed only (skip compile)
|
|
10
|
-
* node seed-blueprint-chat.mjs --compile-only → compile only (skip seed)
|
|
11
|
-
* node seed-blueprint-chat.mjs --watch → watch mode (auto-recompile + reseed on file change)
|
|
12
|
-
* node seed-blueprint-chat.mjs --silent → suppress non-error output
|
|
13
|
-
*
|
|
14
|
-
* The manifest is stored in metadata.blueprint_manifest on the definition,
|
|
15
|
-
* which the frontend reads to detect full_bleed mode, routes, etc.
|
|
16
|
-
* No hardcoded blueprint registrations needed in the frontend codebase.
|
|
17
|
-
*/
|
|
18
|
-
import { readFileSync, readdirSync, watchFile, unwatchFile, existsSync } from 'fs';
|
|
19
|
-
import { join, dirname } from 'path';
|
|
20
|
-
import { fileURLToPath } from 'url';
|
|
21
|
-
import { execSync } from 'child_process';
|
|
22
|
-
|
|
23
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
24
|
-
const __dirname = dirname(__filename);
|
|
25
|
-
|
|
26
|
-
// ---------------------------------------------------------------------------
|
|
27
|
-
// CLI argument parsing
|
|
28
|
-
// ---------------------------------------------------------------------------
|
|
29
|
-
const args = process.argv.slice(2);
|
|
30
|
-
const SEED_ONLY = args.includes('--seed-only');
|
|
31
|
-
const COMPILE_ONLY = args.includes('--compile-only');
|
|
32
|
-
const WATCH = args.includes('--watch');
|
|
33
|
-
const SILENT = args.includes('--silent');
|
|
34
|
-
|
|
35
|
-
function log(...msg) {
|
|
36
|
-
if (!SILENT) console.log(...msg);
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
const compiledDir = join(__dirname, '../frontend/src/workflows/compiled/blueprint-chat');
|
|
40
|
-
const sourceDir = join(__dirname, '../blueprint-chat/app');
|
|
41
|
-
const API = 'http://178.156.197.133:3010/api/v1';
|
|
42
|
-
|
|
43
|
-
// ---------------------------------------------------------------------------
|
|
44
|
-
// Blueprint Manifest — single source of truth
|
|
45
|
-
// ---------------------------------------------------------------------------
|
|
46
|
-
// This is what gets written to metadata.blueprint_manifest in the DB.
|
|
47
|
-
// The frontend reads this to determine routing, full_bleed, auth, etc.
|
|
48
|
-
const BLUEPRINT_MANIFEST = {
|
|
49
|
-
workflows: [{ slug: 'mm-chat-app', role: 'primary' }],
|
|
50
|
-
experience_id: 'mm-chat-app',
|
|
51
|
-
routes: [
|
|
52
|
-
{
|
|
53
|
-
path: '/mm-chat-app',
|
|
54
|
-
node: 'mm-chat-app',
|
|
55
|
-
label: 'Chat',
|
|
56
|
-
entityType: 'mm-chat-app',
|
|
57
|
-
entityIdSource: 'user',
|
|
58
|
-
},
|
|
59
|
-
],
|
|
60
|
-
config: {
|
|
61
|
-
full_bleed: true,
|
|
62
|
-
default_mode: 'read',
|
|
63
|
-
auth_required: true,
|
|
64
|
-
},
|
|
65
|
-
};
|
|
66
|
-
|
|
67
|
-
// ---------------------------------------------------------------------------
|
|
68
|
-
// Compiled view loading
|
|
69
|
-
// ---------------------------------------------------------------------------
|
|
70
|
-
function loadCompiledView(slug) {
|
|
71
|
-
const data = JSON.parse(readFileSync(join(compiledDir, `${slug}.json`), 'utf-8'));
|
|
72
|
-
return data.ir.views?.default || data.ir.metadata?.experience;
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
function loadCompiledMeta(slug) {
|
|
76
|
-
const data = JSON.parse(readFileSync(join(compiledDir, `${slug}.json`), 'utf-8'));
|
|
77
|
-
return data.ir.metadata || {};
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
// ---------------------------------------------------------------------------
|
|
81
|
-
// Slot → Component mapping
|
|
82
|
-
// ---------------------------------------------------------------------------
|
|
83
|
-
const SLOT_TO_COMPONENT = {
|
|
84
|
-
sidebar: 'ChatSidebar',
|
|
85
|
-
search: 'ChatSearch',
|
|
86
|
-
thread: 'ChatThreadPanel',
|
|
87
|
-
call: 'ChatCallPanel',
|
|
88
|
-
'user-profile': 'ChatUserProfile',
|
|
89
|
-
'channel-info': 'ChatChannelInfo',
|
|
90
|
-
};
|
|
91
|
-
|
|
92
|
-
// ---------------------------------------------------------------------------
|
|
93
|
-
// Tree composition
|
|
94
|
-
// ---------------------------------------------------------------------------
|
|
95
|
-
function composeTree(node, childrenComponent) {
|
|
96
|
-
if (!node) return node;
|
|
97
|
-
|
|
98
|
-
// Replace Slot nodes with component references
|
|
99
|
-
if (node.component === 'Slot' && node.config?.name) {
|
|
100
|
-
const componentId = SLOT_TO_COMPONENT[node.config.name];
|
|
101
|
-
if (componentId) {
|
|
102
|
-
const replacement = {
|
|
103
|
-
id: `composed-${node.config.name}`,
|
|
104
|
-
component: componentId,
|
|
105
|
-
};
|
|
106
|
-
if (node.bindings?.props) {
|
|
107
|
-
const propBindings = parseSlotProps(node.bindings.props);
|
|
108
|
-
if (Object.keys(propBindings).length > 0) {
|
|
109
|
-
replacement.bindings = propBindings;
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
return replacement;
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
// Replace $instance.children with the appropriate page component
|
|
117
|
-
// Inherit parent's flex/overflow config so the component fills available space.
|
|
118
|
-
if (node.component === 'Text' && node.bindings?.value === '$instance.children') {
|
|
119
|
-
if (childrenComponent) {
|
|
120
|
-
return { id: 'composed-children', component: childrenComponent, config: { flex: 1, overflow: 'hidden' } };
|
|
121
|
-
}
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
// For Stack/Row with $instance.children in text binding
|
|
125
|
-
if (node.bindings?.text === '$instance.children') {
|
|
126
|
-
if (childrenComponent) {
|
|
127
|
-
return {
|
|
128
|
-
id: node.id,
|
|
129
|
-
component: 'Stack',
|
|
130
|
-
config: { ...node.config, gap: 0 },
|
|
131
|
-
layout: 'stack',
|
|
132
|
-
children: [
|
|
133
|
-
{ id: 'composed-message-thread', component: 'ChatMessageThread' },
|
|
134
|
-
{ id: 'composed-message-input', component: 'ChatMessageInput' },
|
|
135
|
-
],
|
|
136
|
-
};
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
if (node.children) {
|
|
141
|
-
node.children = node.children.map((child) => composeTree(child, childrenComponent));
|
|
142
|
-
}
|
|
143
|
-
|
|
144
|
-
return node;
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
// ---------------------------------------------------------------------------
|
|
148
|
-
// Slot props parser
|
|
149
|
-
// ---------------------------------------------------------------------------
|
|
150
|
-
function parseSlotProps(propsStr) {
|
|
151
|
-
const bindings = {};
|
|
152
|
-
let inner = propsStr.trim();
|
|
153
|
-
if (inner.startsWith('{') && inner.endsWith('}')) {
|
|
154
|
-
inner = inner.slice(1, -1).trim();
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
let key = '';
|
|
158
|
-
let value = '';
|
|
159
|
-
let inValue = false;
|
|
160
|
-
let depth = 0;
|
|
161
|
-
let inString = false;
|
|
162
|
-
let stringChar = '';
|
|
163
|
-
|
|
164
|
-
for (let i = 0; i < inner.length; i++) {
|
|
165
|
-
const ch = inner[i];
|
|
166
|
-
|
|
167
|
-
if (inString) {
|
|
168
|
-
value += ch;
|
|
169
|
-
if (ch === stringChar && inner[i - 1] !== '\\') inString = false;
|
|
170
|
-
continue;
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
if (ch === '"' || ch === "'" || ch === '`') {
|
|
174
|
-
inString = true;
|
|
175
|
-
stringChar = ch;
|
|
176
|
-
if (inValue) value += ch;
|
|
177
|
-
else key += ch;
|
|
178
|
-
continue;
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
if (ch === '(' || ch === '{' || ch === '[') depth++;
|
|
182
|
-
if (ch === ')' || ch === '}' || ch === ']') depth--;
|
|
183
|
-
|
|
184
|
-
if (!inValue && ch === ':' && depth === 0) {
|
|
185
|
-
inValue = true;
|
|
186
|
-
continue;
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
if (inValue && ch === ',' && depth === 0) {
|
|
190
|
-
const k = key.trim();
|
|
191
|
-
const v = value.trim();
|
|
192
|
-
if (k && v) bindings[k] = v;
|
|
193
|
-
key = '';
|
|
194
|
-
value = '';
|
|
195
|
-
inValue = false;
|
|
196
|
-
continue;
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
if (inValue) value += ch;
|
|
200
|
-
else key += ch;
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
const k = key.trim();
|
|
204
|
-
const v = value.trim();
|
|
205
|
-
if (k && v) bindings[k] = v;
|
|
206
|
-
|
|
207
|
-
return bindings;
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
// ---------------------------------------------------------------------------
|
|
211
|
-
// Compile step — runs the Babel compiler
|
|
212
|
-
// ---------------------------------------------------------------------------
|
|
213
|
-
function runCompile() {
|
|
214
|
-
log('\n--- Compiling blueprint-chat views ---');
|
|
215
|
-
try {
|
|
216
|
-
execSync('node compile-blueprint-chat.mjs', {
|
|
217
|
-
cwd: __dirname,
|
|
218
|
-
stdio: SILENT ? 'pipe' : 'inherit',
|
|
219
|
-
});
|
|
220
|
-
log('Compile complete.\n');
|
|
221
|
-
return true;
|
|
222
|
-
} catch (err) {
|
|
223
|
-
console.error('Compile failed:', err.message);
|
|
224
|
-
return false;
|
|
225
|
-
}
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
// ---------------------------------------------------------------------------
|
|
229
|
-
// Seed step — compose tree + PATCH to DB (view + manifest)
|
|
230
|
-
// ---------------------------------------------------------------------------
|
|
231
|
-
async function runSeed() {
|
|
232
|
-
log('\n--- Seeding blueprint-chat to DB ---');
|
|
233
|
-
|
|
234
|
-
// Login
|
|
235
|
-
const loginRes = await fetch(API + '/auth/login', {
|
|
236
|
-
method: 'POST',
|
|
237
|
-
headers: { 'Content-Type': 'application/json' },
|
|
238
|
-
body: JSON.stringify({ email: 'admin@mindmatrix.com', password: 'Admin123!' }),
|
|
239
|
-
});
|
|
240
|
-
const { token } = await loginRes.json();
|
|
241
|
-
log('Logged in');
|
|
242
|
-
|
|
243
|
-
// Load all compiled views
|
|
244
|
-
const files = readdirSync(compiledDir).filter((f) => f.endsWith('.json'));
|
|
245
|
-
log(`Found ${files.length} compiled views`);
|
|
246
|
-
|
|
247
|
-
const views = {};
|
|
248
|
-
const metas = {};
|
|
249
|
-
for (const f of files) {
|
|
250
|
-
const slug = f.replace('.json', '');
|
|
251
|
-
views[slug] = loadCompiledView(slug);
|
|
252
|
-
metas[slug] = loadCompiledMeta(slug);
|
|
253
|
-
}
|
|
254
|
-
|
|
255
|
-
// Compose root tree
|
|
256
|
-
const rootView = JSON.parse(JSON.stringify(views['chat-layout']));
|
|
257
|
-
if (!rootView) {
|
|
258
|
-
console.error('chat-layout view tree not found!');
|
|
259
|
-
return false;
|
|
260
|
-
}
|
|
261
|
-
|
|
262
|
-
const rootDs = metas['chat-layout'].dataSources || [];
|
|
263
|
-
if (rootDs.length > 0) rootView.dataSources = rootDs;
|
|
264
|
-
|
|
265
|
-
const composedRoot = composeTree(rootView, 'ChatChannelLayout');
|
|
266
|
-
|
|
267
|
-
// Log tree structure
|
|
268
|
-
let slotCount = 0;
|
|
269
|
-
let childrenCount = 0;
|
|
270
|
-
function countReplacements(node) {
|
|
271
|
-
if (node.id?.startsWith('composed-')) {
|
|
272
|
-
if (node.id === 'composed-children') childrenCount++;
|
|
273
|
-
else slotCount++;
|
|
274
|
-
}
|
|
275
|
-
(node.children || []).forEach(countReplacements);
|
|
276
|
-
}
|
|
277
|
-
countReplacements(composedRoot);
|
|
278
|
-
log(` Replaced ${slotCount} slots, ${childrenCount} children placeholders`);
|
|
279
|
-
|
|
280
|
-
if (!SILENT) {
|
|
281
|
-
function logTree(node, depth = 0) {
|
|
282
|
-
const ds = node.dataSources?.length || 0;
|
|
283
|
-
const bind = Object.keys(node.bindings || {}).length;
|
|
284
|
-
const extra = [];
|
|
285
|
-
if (ds > 0) extra.push(`ds=${ds}`);
|
|
286
|
-
if (bind > 0) extra.push(`bind=${bind}`);
|
|
287
|
-
if (node.config?.localDefaults) extra.push('localDefaults');
|
|
288
|
-
const suffix = extra.length > 0 ? ` [${extra.join(', ')}]` : '';
|
|
289
|
-
console.log(`${' '.repeat(depth)}${node.component} (${node.id})${suffix}`);
|
|
290
|
-
(node.children || []).forEach((c) => logTree(c, depth + 1));
|
|
291
|
-
}
|
|
292
|
-
console.log('\nComposed tree:');
|
|
293
|
-
logTree(composedRoot);
|
|
294
|
-
}
|
|
295
|
-
|
|
296
|
-
log(`\nTree size: ${JSON.stringify(composedRoot).length} bytes`);
|
|
297
|
-
|
|
298
|
-
// Get mm-chat-app definition
|
|
299
|
-
const defRes = await fetch(API + '/definitions/mm-chat-app', {
|
|
300
|
-
headers: { Authorization: 'Bearer ' + token },
|
|
301
|
-
});
|
|
302
|
-
const def = await defRes.json();
|
|
303
|
-
log(`Definition ID: ${def.id}`);
|
|
304
|
-
|
|
305
|
-
// Merge manifest into existing metadata (don't clobber other metadata keys)
|
|
306
|
-
const existingMetadata = def.metadata || {};
|
|
307
|
-
const updatedMetadata = {
|
|
308
|
-
...existingMetadata,
|
|
309
|
-
blueprint_manifest: BLUEPRINT_MANIFEST,
|
|
310
|
-
};
|
|
311
|
-
|
|
312
|
-
// PATCH with BOTH view tree AND manifest metadata
|
|
313
|
-
const patchRes = await fetch(API + '/definitions/' + def.id, {
|
|
314
|
-
method: 'PATCH',
|
|
315
|
-
headers: {
|
|
316
|
-
'Content-Type': 'application/json',
|
|
317
|
-
Authorization: 'Bearer ' + token,
|
|
318
|
-
},
|
|
319
|
-
body: JSON.stringify({
|
|
320
|
-
view: composedRoot,
|
|
321
|
-
metadata: updatedMetadata,
|
|
322
|
-
}),
|
|
323
|
-
});
|
|
324
|
-
|
|
325
|
-
if (!patchRes.ok) {
|
|
326
|
-
console.error('PATCH failed:', patchRes.status, await patchRes.text());
|
|
327
|
-
return false;
|
|
328
|
-
}
|
|
329
|
-
log('View tree + manifest updated');
|
|
330
|
-
|
|
331
|
-
// Verify
|
|
332
|
-
const verifyRes = await fetch(API + '/definitions/mm-chat-app', {
|
|
333
|
-
headers: { Authorization: 'Bearer ' + token },
|
|
334
|
-
});
|
|
335
|
-
const verified = await verifyRes.json();
|
|
336
|
-
const hasManifest = !!verified.metadata?.blueprint_manifest;
|
|
337
|
-
const hasFullBleed = verified.metadata?.blueprint_manifest?.config?.full_bleed === true;
|
|
338
|
-
const hasRoutes = verified.metadata?.blueprint_manifest?.routes?.length > 0;
|
|
339
|
-
log(`\nVerified:`);
|
|
340
|
-
log(` view: ${JSON.stringify(verified.view).length} bytes, root=${verified.view?.component}`);
|
|
341
|
-
log(` manifest: ${hasManifest ? 'present' : 'MISSING'}`);
|
|
342
|
-
log(` full_bleed: ${hasFullBleed}`);
|
|
343
|
-
log(` routes: ${hasRoutes ? verified.metadata.blueprint_manifest.routes.length : 0}`);
|
|
344
|
-
|
|
345
|
-
if (!hasManifest || !hasFullBleed) {
|
|
346
|
-
console.error('Manifest verification failed!');
|
|
347
|
-
return false;
|
|
348
|
-
}
|
|
349
|
-
|
|
350
|
-
log('\nSeed complete.');
|
|
351
|
-
return true;
|
|
352
|
-
}
|
|
353
|
-
|
|
354
|
-
// ---------------------------------------------------------------------------
|
|
355
|
-
// Watch mode
|
|
356
|
-
// ---------------------------------------------------------------------------
|
|
357
|
-
function startWatch() {
|
|
358
|
-
log('\n--- Watch mode: monitoring blueprint-chat sources ---');
|
|
359
|
-
log(` Source: ${sourceDir}`);
|
|
360
|
-
log(` Compiled: ${compiledDir}`);
|
|
361
|
-
log(' Press Ctrl+C to stop.\n');
|
|
362
|
-
|
|
363
|
-
// Find all source files to watch
|
|
364
|
-
function findFiles(dir) {
|
|
365
|
-
const entries = readdirSync(dir, { withFileTypes: true });
|
|
366
|
-
const files = [];
|
|
367
|
-
for (const entry of entries) {
|
|
368
|
-
const full = join(dir, entry.name);
|
|
369
|
-
if (entry.isDirectory()) files.push(...findFiles(full));
|
|
370
|
-
else if (entry.name.endsWith('.tsx')) files.push(full);
|
|
371
|
-
}
|
|
372
|
-
return files;
|
|
373
|
-
}
|
|
374
|
-
|
|
375
|
-
const sourceFiles = existsSync(sourceDir) ? findFiles(sourceDir) : [];
|
|
376
|
-
log(`Watching ${sourceFiles.length} source files...`);
|
|
377
|
-
|
|
378
|
-
let debounceTimer = null;
|
|
379
|
-
let isRunning = false;
|
|
380
|
-
|
|
381
|
-
async function onChange(filename) {
|
|
382
|
-
if (isRunning) return;
|
|
383
|
-
if (debounceTimer) clearTimeout(debounceTimer);
|
|
384
|
-
|
|
385
|
-
debounceTimer = setTimeout(async () => {
|
|
386
|
-
isRunning = true;
|
|
387
|
-
const ts = new Date().toLocaleTimeString();
|
|
388
|
-
log(`\n[${ts}] Change detected: ${filename}`);
|
|
389
|
-
|
|
390
|
-
const compiled = runCompile();
|
|
391
|
-
if (compiled) {
|
|
392
|
-
await runSeed();
|
|
393
|
-
}
|
|
394
|
-
|
|
395
|
-
isRunning = false;
|
|
396
|
-
log(`\n[${new Date().toLocaleTimeString()}] Waiting for changes...`);
|
|
397
|
-
}, 500); // 500ms debounce
|
|
398
|
-
}
|
|
399
|
-
|
|
400
|
-
for (const file of sourceFiles) {
|
|
401
|
-
watchFile(file, { interval: 1000 }, () => onChange(file));
|
|
402
|
-
}
|
|
403
|
-
|
|
404
|
-
// Handle shutdown
|
|
405
|
-
process.on('SIGINT', () => {
|
|
406
|
-
log('\nStopping watch...');
|
|
407
|
-
for (const file of sourceFiles) unwatchFile(file);
|
|
408
|
-
process.exit(0);
|
|
409
|
-
});
|
|
410
|
-
}
|
|
411
|
-
|
|
412
|
-
// ---------------------------------------------------------------------------
|
|
413
|
-
// Main
|
|
414
|
-
// ---------------------------------------------------------------------------
|
|
415
|
-
async function main() {
|
|
416
|
-
log('Blueprint Chat Pipeline');
|
|
417
|
-
log(` Mode: ${WATCH ? 'watch' : COMPILE_ONLY ? 'compile-only' : SEED_ONLY ? 'seed-only' : 'full'}`);
|
|
418
|
-
|
|
419
|
-
if (WATCH) {
|
|
420
|
-
// Initial full run, then watch
|
|
421
|
-
const compiled = runCompile();
|
|
422
|
-
if (compiled) await runSeed();
|
|
423
|
-
startWatch();
|
|
424
|
-
return; // watch mode keeps process alive
|
|
425
|
-
}
|
|
426
|
-
|
|
427
|
-
if (COMPILE_ONLY) {
|
|
428
|
-
runCompile();
|
|
429
|
-
return;
|
|
430
|
-
}
|
|
431
|
-
|
|
432
|
-
if (SEED_ONLY) {
|
|
433
|
-
await runSeed();
|
|
434
|
-
return;
|
|
435
|
-
}
|
|
436
|
-
|
|
437
|
-
// Full mode: compile + seed
|
|
438
|
-
const compiled = runCompile();
|
|
439
|
-
if (compiled) {
|
|
440
|
-
await runSeed();
|
|
441
|
-
}
|
|
442
|
-
}
|
|
443
|
-
|
|
444
|
-
main().catch((e) => console.error(e));
|