@mmapp/react-compiler 0.1.0-alpha.1

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 (314) hide show
  1. package/README.md +107 -0
  2. package/compile-blueprint-chat.mjs +99 -0
  3. package/compile-blueprint-glass-console.mjs +98 -0
  4. package/compile-chat-defs.mjs +92 -0
  5. package/dist/babel/index.d.mts +3 -0
  6. package/dist/babel/index.d.ts +3 -0
  7. package/dist/babel/index.js +4851 -0
  8. package/dist/babel/index.mjs +7 -0
  9. package/dist/chunk-26U577GB.mjs +3465 -0
  10. package/dist/chunk-2FBDFAX6.mjs +2362 -0
  11. package/dist/chunk-2L4QSMXG.mjs +175 -0
  12. package/dist/chunk-2REDFOER.mjs +931 -0
  13. package/dist/chunk-46YKSHQR.mjs +175 -0
  14. package/dist/chunk-4XHK6FWL.mjs +2058 -0
  15. package/dist/chunk-5M7DKKBC.mjs +215 -0
  16. package/dist/chunk-5VNJ7C6N.mjs +154 -0
  17. package/dist/chunk-6CQOAAMV.mjs +1803 -0
  18. package/dist/chunk-6SEVAAVT.mjs +3516 -0
  19. package/dist/chunk-6YLR5ZDA.mjs +2829 -0
  20. package/dist/chunk-AOGY2GK6.mjs +3292 -0
  21. package/dist/chunk-AXXUXRNA.mjs +1434 -0
  22. package/dist/chunk-CHLVKMQW.mjs +175 -0
  23. package/dist/chunk-CKGOZAB7.mjs +939 -0
  24. package/dist/chunk-D34RAZUX.mjs +2223 -0
  25. package/dist/chunk-EQGA6A6D.mjs +121 -0
  26. package/dist/chunk-EY2CSXYA.mjs +822 -0
  27. package/dist/chunk-FIQ65CDR.mjs +925 -0
  28. package/dist/chunk-FOZXJFAR.mjs +186 -0
  29. package/dist/chunk-FX6URXWN.mjs +186 -0
  30. package/dist/chunk-G7SMOWOL.mjs +828 -0
  31. package/dist/chunk-GGB4G5YY.mjs +175 -0
  32. package/dist/chunk-HLRGCCIL.mjs +4839 -0
  33. package/dist/chunk-HOIUP6IF.mjs +690 -0
  34. package/dist/chunk-I3AU7GRD.mjs +120 -0
  35. package/dist/chunk-ILFGMUVD.mjs +1933 -0
  36. package/dist/chunk-IPTX5MJU.mjs +3223 -0
  37. package/dist/chunk-ITGUSH2Z.mjs +2783 -0
  38. package/dist/chunk-IXHBCAMF.mjs +3306 -0
  39. package/dist/chunk-J7TWJ3TM.mjs +2784 -0
  40. package/dist/chunk-JDPLDGVF.mjs +4810 -0
  41. package/dist/chunk-K53XP2DL.mjs +148 -0
  42. package/dist/chunk-K5HX2SVL.mjs +1902 -0
  43. package/dist/chunk-KFGYOOVS.mjs +214 -0
  44. package/dist/chunk-KFVVOS5N.mjs +925 -0
  45. package/dist/chunk-L2OZ4CDV.mjs +113 -0
  46. package/dist/chunk-MIZV3TAN.mjs +3293 -0
  47. package/dist/chunk-NKKLQE5V.mjs +148 -0
  48. package/dist/chunk-NOW23XFZ.mjs +186 -0
  49. package/dist/chunk-NRXQKQ74.mjs +148 -0
  50. package/dist/chunk-OWI6XWCD.mjs +3375 -0
  51. package/dist/chunk-PRUMNNDI.mjs +3192 -0
  52. package/dist/chunk-QTBD5B3F.mjs +148 -0
  53. package/dist/chunk-SKSDPPNT.mjs +3788 -0
  54. package/dist/chunk-SP2YUS33.mjs +186 -0
  55. package/dist/chunk-SU4E6E7B.mjs +3153 -0
  56. package/dist/chunk-SYUUKW5A.mjs +3379 -0
  57. package/dist/chunk-UL2XZEMA.mjs +3128 -0
  58. package/dist/chunk-XMWUHQVV.mjs +939 -0
  59. package/dist/chunk-XZNEDRGN.mjs +3876 -0
  60. package/dist/chunk-Y6FXYEAI.mjs +10 -0
  61. package/dist/chunk-YFS6JMYO.mjs +3342 -0
  62. package/dist/chunk-Z6AIQ4KL.mjs +113 -0
  63. package/dist/cli/index.d.mts +1 -0
  64. package/dist/cli/index.d.ts +1 -0
  65. package/dist/cli/index.js +11585 -0
  66. package/dist/cli/index.mjs +701 -0
  67. package/dist/codemod/cli.d.mts +1 -0
  68. package/dist/codemod/cli.d.ts +1 -0
  69. package/dist/codemod/cli.js +1104 -0
  70. package/dist/codemod/cli.mjs +157 -0
  71. package/dist/codemod/index.d.mts +148 -0
  72. package/dist/codemod/index.d.ts +148 -0
  73. package/dist/codemod/index.js +981 -0
  74. package/dist/codemod/index.mjs +25 -0
  75. package/dist/dev-server-Bs_sz2DG.d.mts +111 -0
  76. package/dist/dev-server-Bs_sz2DG.d.ts +111 -0
  77. package/dist/dev-server-CjoufJ-u.d.mts +109 -0
  78. package/dist/dev-server-CjoufJ-u.d.ts +109 -0
  79. package/dist/dev-server.d.mts +3 -0
  80. package/dist/dev-server.d.ts +3 -0
  81. package/dist/dev-server.js +7603 -0
  82. package/dist/dev-server.mjs +11 -0
  83. package/dist/envelope-DD7v0v6E.d.mts +265 -0
  84. package/dist/envelope-DD7v0v6E.d.ts +265 -0
  85. package/dist/envelope-vCVjrHlo.d.mts +265 -0
  86. package/dist/envelope-vCVjrHlo.d.ts +265 -0
  87. package/dist/envelope.d.mts +2 -0
  88. package/dist/envelope.d.ts +2 -0
  89. package/dist/envelope.js +5184 -0
  90. package/dist/envelope.mjs +9 -0
  91. package/dist/index-B5gSgvnd.d.mts +44 -0
  92. package/dist/index-B5gSgvnd.d.ts +44 -0
  93. package/dist/index-Bs0MnR54.d.mts +103 -0
  94. package/dist/index-Bs0MnR54.d.ts +103 -0
  95. package/dist/index-DR0nNc_f.d.mts +101 -0
  96. package/dist/index-DR0nNc_f.d.ts +101 -0
  97. package/dist/index-revho_gS.d.mts +104 -0
  98. package/dist/index-revho_gS.d.ts +104 -0
  99. package/dist/index.d.mts +1099 -0
  100. package/dist/index.d.ts +1099 -0
  101. package/dist/index.js +10162 -0
  102. package/dist/index.mjs +372 -0
  103. package/dist/init-IXEE2RCF.mjs +340 -0
  104. package/dist/project-compiler-EGJUTAJU.mjs +10 -0
  105. package/dist/project-compiler-VFR6CSDX.mjs +10 -0
  106. package/dist/project-decompiler-5GY2KSG4.mjs +7 -0
  107. package/dist/pull-A2QUHW4K.mjs +109 -0
  108. package/dist/pull-JBEQWVPE.mjs +109 -0
  109. package/dist/testing/index.d.mts +211 -0
  110. package/dist/testing/index.d.ts +211 -0
  111. package/dist/testing/index.js +5106 -0
  112. package/dist/testing/index.mjs +247 -0
  113. package/dist/vite/index.d.mts +59 -0
  114. package/dist/vite/index.d.ts +59 -0
  115. package/dist/vite/index.js +5023 -0
  116. package/dist/vite/index.mjs +8 -0
  117. package/examples/README.md +72 -0
  118. package/examples/authentication/main.workflow.tsx +139 -0
  119. package/examples/authentication/mm.config.ts +22 -0
  120. package/examples/authentication/models/auth.ts +45 -0
  121. package/examples/authentication/pages/LoginPage.tsx +79 -0
  122. package/examples/authentication/pages/SignupPage.tsx +87 -0
  123. package/examples/counter.workflow.tsx +65 -0
  124. package/examples/dashboard.workflow.tsx +419 -0
  125. package/examples/invoice-approval/actions/invoice.server.ts +72 -0
  126. package/examples/invoice-approval/main.workflow.tsx +168 -0
  127. package/examples/invoice-approval/mm.config.ts +18 -0
  128. package/examples/invoice-approval/models/invoice.ts +46 -0
  129. package/examples/invoice-approval/pages/InvoiceDetailPage.tsx +175 -0
  130. package/examples/invoice-approval/pages/InvoiceFormPage.tsx +198 -0
  131. package/examples/invoice-approval/pages/InvoiceListPage.tsx +141 -0
  132. package/examples/todo-app.workflow.tsx +131 -0
  133. package/examples/uber-app/actions/matching.server.ts +177 -0
  134. package/examples/uber-app/actions/notifications.server.ts +176 -0
  135. package/examples/uber-app/actions/payments.server.ts +184 -0
  136. package/examples/uber-app/actions/pricing.server.ts +176 -0
  137. package/examples/uber-app/app/admin/analytics.tsx +102 -0
  138. package/examples/uber-app/app/admin/fleet.tsx +102 -0
  139. package/examples/uber-app/app/admin/surge-pricing.tsx +95 -0
  140. package/examples/uber-app/app/driver/dashboard.tsx +87 -0
  141. package/examples/uber-app/app/driver/earnings.tsx +101 -0
  142. package/examples/uber-app/app/driver/navigation.tsx +94 -0
  143. package/examples/uber-app/app/driver/ride-acceptance.tsx +103 -0
  144. package/examples/uber-app/app/rider/home.tsx +109 -0
  145. package/examples/uber-app/app/rider/payment-methods.tsx +134 -0
  146. package/examples/uber-app/app/rider/ride-history.tsx +90 -0
  147. package/examples/uber-app/app/rider/ride-tracking.tsx +108 -0
  148. package/examples/uber-app/components/DriverCard.tsx +176 -0
  149. package/examples/uber-app/components/MapView.tsx +216 -0
  150. package/examples/uber-app/components/RatingStars.tsx +227 -0
  151. package/examples/uber-app/components/RideCard.tsx +167 -0
  152. package/examples/uber-app/mm.config.ts +30 -0
  153. package/examples/uber-app/models/location.model.ts +70 -0
  154. package/examples/uber-app/models/payment.model.ts +87 -0
  155. package/examples/uber-app/models/rating.model.ts +54 -0
  156. package/examples/uber-app/models/ride.model.ts +118 -0
  157. package/examples/uber-app/models/user.model.ts +66 -0
  158. package/examples/uber-app/models/vehicle.model.ts +63 -0
  159. package/examples/uber-app/tests/payment.test.tsx +129 -0
  160. package/examples/uber-app/tests/ride-flow.test.tsx +123 -0
  161. package/examples/uber-app/workflows/dispute-resolution.workflow.tsx +205 -0
  162. package/examples/uber-app/workflows/driver-onboarding.workflow.tsx +227 -0
  163. package/examples/uber-app/workflows/payment-processing.workflow.tsx +223 -0
  164. package/examples/uber-app/workflows/ride-request.workflow.tsx +194 -0
  165. package/package.json +77 -0
  166. package/package.json.backup +86 -0
  167. package/scripts/decompile.ts +226 -0
  168. package/scripts/seed-auth.ts +267 -0
  169. package/scripts/seed-uber.ts +248 -0
  170. package/scripts/validate-uber.ts +119 -0
  171. package/seed-blueprint-chat.mjs +444 -0
  172. package/seed-blueprint-glass-console.mjs +445 -0
  173. package/seed-compiled.mjs +318 -0
  174. package/src/RoundTripValidator.ts +400 -0
  175. package/src/__tests__/atom-rendering-coverage.test.ts +680 -0
  176. package/src/__tests__/auth-module-compilation.test.ts +247 -0
  177. package/src/__tests__/auth-template-compilation.test.ts +589 -0
  178. package/src/__tests__/change-extractor.test.ts +142 -0
  179. package/src/__tests__/cli-pull.test.ts +73 -0
  180. package/src/__tests__/cli-test.test.ts +72 -0
  181. package/src/__tests__/component-extractor.test.ts +331 -0
  182. package/src/__tests__/context-extractor.test.ts +145 -0
  183. package/src/__tests__/decompiler.test.ts +718 -0
  184. package/src/__tests__/define-blueprint.test.ts +133 -0
  185. package/src/__tests__/definition-validator.test.ts +519 -0
  186. package/src/__tests__/during-extractor.test.ts +152 -0
  187. package/src/__tests__/effect-extractor.test.ts +107 -0
  188. package/src/__tests__/event-emission.test.ts +127 -0
  189. package/src/__tests__/examples.test.ts +236 -0
  190. package/src/__tests__/full-blueprint-coverage.test.ts +1221 -0
  191. package/src/__tests__/golden-suite.test.ts +403 -0
  192. package/src/__tests__/grammar-island-extractor.test.ts +289 -0
  193. package/src/__tests__/instance-key.test.ts +82 -0
  194. package/src/__tests__/ir-migration.test.ts +255 -0
  195. package/src/__tests__/lock-file.test.ts +117 -0
  196. package/src/__tests__/model-extractor.test.ts +195 -0
  197. package/src/__tests__/model-field-acl.test.ts +237 -0
  198. package/src/__tests__/model-hooks.test.ts +130 -0
  199. package/src/__tests__/model-ref-resolution.test.ts +268 -0
  200. package/src/__tests__/model-roundtrip.test.ts +502 -0
  201. package/src/__tests__/model-runtime.test.ts +112 -0
  202. package/src/__tests__/model-transitions.test.ts +183 -0
  203. package/src/__tests__/nrt-action-trace.test.ts +391 -0
  204. package/src/__tests__/pipeline-hardening.test.ts +413 -0
  205. package/src/__tests__/project-compiler.test.ts +546 -0
  206. package/src/__tests__/project-decompiler.test.ts +343 -0
  207. package/src/__tests__/query-compilation.test.ts +145 -0
  208. package/src/__tests__/round-trip/PLAN.md +158 -0
  209. package/src/__tests__/round-trip/README.md +52 -0
  210. package/src/__tests__/round-trip/RESULTS.md +86 -0
  211. package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +55 -0
  212. package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +11 -0
  213. package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +54 -0
  214. package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +79 -0
  215. package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +12 -0
  216. package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +50 -0
  217. package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +25 -0
  218. package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +11 -0
  219. package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +32 -0
  220. package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +79 -0
  221. package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +10 -0
  222. package/src/__tests__/round-trip/round-trip.test.ts +2598 -0
  223. package/src/__tests__/round-trip-ir.test.ts +300 -0
  224. package/src/__tests__/round-trip.test.ts +1212 -0
  225. package/src/__tests__/route-merging.test.ts +372 -0
  226. package/src/__tests__/router-composition.test.ts +489 -0
  227. package/src/__tests__/router-extractor.test.ts +176 -0
  228. package/src/__tests__/server-action-extractor.test.ts +128 -0
  229. package/src/__tests__/smart-type-inference.test.ts +365 -0
  230. package/src/__tests__/source-envelope.test.ts +284 -0
  231. package/src/__tests__/source-fidelity.test.ts +516 -0
  232. package/src/__tests__/state-extractor.test.ts +115 -0
  233. package/src/__tests__/strict-mode.test.ts +227 -0
  234. package/src/__tests__/transition-effect-extractor.test.ts +119 -0
  235. package/src/__tests__/transition-extractor.test.ts +68 -0
  236. package/src/__tests__/ts-to-expression.test.ts +462 -0
  237. package/src/__tests__/type-generator.test.ts +201 -0
  238. package/src/__tests__/uber-validation.test.ts +502 -0
  239. package/src/action-compiler.ts +361 -0
  240. package/src/babel/emitters/experience-transform.ts +199 -0
  241. package/src/babel/emitters/ir-to-tsx-emitter.ts +110 -0
  242. package/src/babel/emitters/pure-form-emitter.ts +1023 -0
  243. package/src/babel/emitters/runtime-glue-emitter.ts +39 -0
  244. package/src/babel/extractors/change-extractor.ts +199 -0
  245. package/src/babel/extractors/component-extractor.ts +907 -0
  246. package/src/babel/extractors/computed-extractor.ts +262 -0
  247. package/src/babel/extractors/context-extractor.ts +277 -0
  248. package/src/babel/extractors/during-extractor.ts +295 -0
  249. package/src/babel/extractors/effect-extractor.ts +340 -0
  250. package/src/babel/extractors/event-extractor.ts +235 -0
  251. package/src/babel/extractors/grammar-island-extractor.ts +302 -0
  252. package/src/babel/extractors/model-extractor.ts +1018 -0
  253. package/src/babel/extractors/router-extractor.ts +303 -0
  254. package/src/babel/extractors/server-action-extractor.ts +173 -0
  255. package/src/babel/extractors/server-action-hook-extractor.ts +72 -0
  256. package/src/babel/extractors/server-state-extractor.ts +88 -0
  257. package/src/babel/extractors/state-extractor.ts +214 -0
  258. package/src/babel/extractors/transition-effect-extractor.ts +176 -0
  259. package/src/babel/extractors/transition-extractor.ts +143 -0
  260. package/src/babel/index.ts +24 -0
  261. package/src/babel/transpilers/ts-to-expression.ts +674 -0
  262. package/src/babel/visitor.ts +807 -0
  263. package/src/cli/auth.ts +255 -0
  264. package/src/cli/build.ts +288 -0
  265. package/src/cli/deploy.ts +206 -0
  266. package/src/cli/index.ts +328 -0
  267. package/src/cli/init.ts +388 -0
  268. package/src/cli/installer.ts +261 -0
  269. package/src/cli/lock-file.ts +94 -0
  270. package/src/cli/mmrc.ts +22 -0
  271. package/src/cli/pull.ts +172 -0
  272. package/src/cli/registry-client.ts +175 -0
  273. package/src/cli/test.ts +397 -0
  274. package/src/cli/type-generator.ts +243 -0
  275. package/src/codemod/__tests__/forward.test.ts +239 -0
  276. package/src/codemod/__tests__/reverse.test.ts +145 -0
  277. package/src/codemod/__tests__/round-trip.test.ts +137 -0
  278. package/src/codemod/annotation.ts +97 -0
  279. package/src/codemod/classify.ts +197 -0
  280. package/src/codemod/cli.ts +207 -0
  281. package/src/codemod/control-flow.ts +409 -0
  282. package/src/codemod/forward.ts +244 -0
  283. package/src/codemod/import-manager.ts +171 -0
  284. package/src/codemod/index.ts +120 -0
  285. package/src/codemod/reverse.ts +197 -0
  286. package/src/codemod/rules.ts +174 -0
  287. package/src/codemod/state-transform.ts +126 -0
  288. package/src/decompiler/ast-builder.ts +538 -0
  289. package/src/decompiler/config-generator.ts +151 -0
  290. package/src/decompiler/index.ts +315 -0
  291. package/src/decompiler/project-decompiler.ts +1776 -0
  292. package/src/decompiler/project.ts +862 -0
  293. package/src/decompiler/split-strategy.ts +140 -0
  294. package/src/decompiler/state-emitter.ts +1053 -0
  295. package/src/decompiler/sx-emitter.ts +318 -0
  296. package/src/decompiler/workspace-hydrator.ts +189 -0
  297. package/src/dev-server.ts +238 -0
  298. package/src/envelope/fs-tree.ts +217 -0
  299. package/src/envelope/source-envelope.ts +264 -0
  300. package/src/envelope.ts +315 -0
  301. package/src/incremental-compiler.ts +401 -0
  302. package/src/index.ts +99 -0
  303. package/src/model-compiler.ts +277 -0
  304. package/src/project-compiler.ts +1629 -0
  305. package/src/route-extractor.ts +333 -0
  306. package/src/testing/index.ts +32 -0
  307. package/src/testing/snapshot.ts +252 -0
  308. package/src/testing/test-utils.ts +226 -0
  309. package/src/types.ts +68 -0
  310. package/src/vite/index.ts +288 -0
  311. package/test-compile.mjs +142 -0
  312. package/tsconfig.json +25 -0
  313. package/tsup.config.ts +23 -0
  314. package/vitest.config.ts +9 -0
@@ -0,0 +1,445 @@
1
+ /**
2
+ * Blueprint Glass Console — 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-glass-console.mjs → full (compile + seed)
9
+ * node seed-blueprint-glass-console.mjs --seed-only → seed only (skip compile)
10
+ * node seed-blueprint-glass-console.mjs --compile-only → compile only (skip seed)
11
+ * node seed-blueprint-glass-console.mjs --watch → watch mode (auto-recompile + reseed)
12
+ * node seed-blueprint-glass-console.mjs --silent → suppress non-error output
13
+ */
14
+ import { readFileSync, readdirSync, watchFile, unwatchFile, existsSync } from 'fs';
15
+ import { join, dirname } from 'path';
16
+ import { fileURLToPath } from 'url';
17
+ import { execSync } from 'child_process';
18
+
19
+ const __filename = fileURLToPath(import.meta.url);
20
+ const __dirname = dirname(__filename);
21
+
22
+ // ---------------------------------------------------------------------------
23
+ // CLI argument parsing
24
+ // ---------------------------------------------------------------------------
25
+ const args = process.argv.slice(2);
26
+ const SEED_ONLY = args.includes('--seed-only');
27
+ const COMPILE_ONLY = args.includes('--compile-only');
28
+ const WATCH = args.includes('--watch');
29
+ const SILENT = args.includes('--silent');
30
+
31
+ function log(...msg) {
32
+ if (!SILENT) console.log(...msg);
33
+ }
34
+
35
+ const compiledDir = join(__dirname, '../frontend/src/workflows/compiled/blueprint-glass-console');
36
+ const sourceDir = join(__dirname, '../blueprint-glass-console/app');
37
+ const API = 'http://178.156.197.133:3010/api/v1';
38
+
39
+ // ---------------------------------------------------------------------------
40
+ // Blueprint Manifest — single source of truth
41
+ // ---------------------------------------------------------------------------
42
+ const BLUEPRINT_MANIFEST = {
43
+ workflows: [{ slug: 'mm-glass-console', role: 'primary' }],
44
+ experience_id: 'mm-glass-console',
45
+ routes: [
46
+ {
47
+ path: '/mm-glass-console',
48
+ node: 'mm-glass-console',
49
+ label: 'Glass Console',
50
+ entityType: 'mm-glass-console',
51
+ entityIdSource: 'user',
52
+ },
53
+ ],
54
+ config: {
55
+ full_bleed: true,
56
+ default_mode: 'read',
57
+ auth_required: true,
58
+ },
59
+ };
60
+
61
+ // ---------------------------------------------------------------------------
62
+ // Compiled view loading
63
+ // ---------------------------------------------------------------------------
64
+ function loadCompiledView(slug) {
65
+ const data = JSON.parse(readFileSync(join(compiledDir, `${slug}.json`), 'utf-8'));
66
+ return data.ir.views?.default || data.ir.metadata?.experience;
67
+ }
68
+
69
+ function loadCompiledMeta(slug) {
70
+ const data = JSON.parse(readFileSync(join(compiledDir, `${slug}.json`), 'utf-8'));
71
+ return data.ir.metadata || {};
72
+ }
73
+
74
+ // ---------------------------------------------------------------------------
75
+ // Slot → Component mapping
76
+ // ---------------------------------------------------------------------------
77
+ const SLOT_TO_COMPONENT = {
78
+ artifact: 'GcArtifactViewer',
79
+ timeline: 'GcRunTimeline',
80
+ };
81
+
82
+ // ---------------------------------------------------------------------------
83
+ // Tree composition
84
+ // ---------------------------------------------------------------------------
85
+ function composeTree(node, childrenComponent) {
86
+ if (!node) return node;
87
+
88
+ // Replace Slot nodes with component references
89
+ if (node.component === 'Slot' && node.config?.name) {
90
+ const componentId = SLOT_TO_COMPONENT[node.config.name];
91
+ if (componentId) {
92
+ const replacement = {
93
+ id: `composed-${node.config.name}`,
94
+ component: componentId,
95
+ };
96
+ if (node.bindings?.props) {
97
+ const propBindings = parseSlotProps(node.bindings.props);
98
+ if (Object.keys(propBindings).length > 0) {
99
+ replacement.bindings = propBindings;
100
+ }
101
+ }
102
+ return replacement;
103
+ }
104
+ }
105
+
106
+ // Replace $instance.children with the appropriate page component
107
+ if (node.component === 'Text' && node.bindings?.value === '$instance.children') {
108
+ if (childrenComponent) {
109
+ return { id: 'composed-children', component: childrenComponent, config: { flex: 1, overflow: 'hidden' } };
110
+ }
111
+ }
112
+
113
+ if (node.children) {
114
+ node.children = node.children.map((child) => composeTree(child, childrenComponent));
115
+ }
116
+
117
+ return node;
118
+ }
119
+
120
+ // ---------------------------------------------------------------------------
121
+ // Slot props parser
122
+ // ---------------------------------------------------------------------------
123
+ function parseSlotProps(propsStr) {
124
+ const bindings = {};
125
+ let inner = propsStr.trim();
126
+ if (inner.startsWith('{') && inner.endsWith('}')) {
127
+ inner = inner.slice(1, -1).trim();
128
+ }
129
+
130
+ let key = '';
131
+ let value = '';
132
+ let inValue = false;
133
+ let depth = 0;
134
+ let inString = false;
135
+ let stringChar = '';
136
+
137
+ for (let i = 0; i < inner.length; i++) {
138
+ const ch = inner[i];
139
+
140
+ if (inString) {
141
+ value += ch;
142
+ if (ch === stringChar && inner[i - 1] !== '\\') inString = false;
143
+ continue;
144
+ }
145
+
146
+ if (ch === '"' || ch === "'" || ch === '`') {
147
+ inString = true;
148
+ stringChar = ch;
149
+ if (inValue) value += ch;
150
+ else key += ch;
151
+ continue;
152
+ }
153
+
154
+ if (ch === '(' || ch === '{' || ch === '[') depth++;
155
+ if (ch === ')' || ch === '}' || ch === ']') depth--;
156
+
157
+ if (!inValue && ch === ':' && depth === 0) {
158
+ inValue = true;
159
+ continue;
160
+ }
161
+
162
+ if (inValue && ch === ',' && depth === 0) {
163
+ const k = key.trim();
164
+ const v = value.trim();
165
+ if (k && v) bindings[k] = v;
166
+ key = '';
167
+ value = '';
168
+ inValue = false;
169
+ continue;
170
+ }
171
+
172
+ if (inValue) value += ch;
173
+ else key += ch;
174
+ }
175
+
176
+ const k = key.trim();
177
+ const v = value.trim();
178
+ if (k && v) bindings[k] = v;
179
+
180
+ return bindings;
181
+ }
182
+
183
+ // ---------------------------------------------------------------------------
184
+ // Compile step — runs the Babel compiler
185
+ // ---------------------------------------------------------------------------
186
+ function runCompile() {
187
+ log('\n--- Compiling blueprint-glass-console views ---');
188
+ try {
189
+ execSync('node compile-blueprint-glass-console.mjs', {
190
+ cwd: __dirname,
191
+ stdio: SILENT ? 'pipe' : 'inherit',
192
+ });
193
+ log('Compile complete.\n');
194
+ return true;
195
+ } catch (err) {
196
+ console.error('Compile failed:', err.message);
197
+ return false;
198
+ }
199
+ }
200
+
201
+ // ---------------------------------------------------------------------------
202
+ // Seed step — compose tree + PATCH to DB (view + manifest)
203
+ // ---------------------------------------------------------------------------
204
+ async function runSeed() {
205
+ log('\n--- Seeding blueprint-glass-console to DB ---');
206
+
207
+ // Login
208
+ const loginRes = await fetch(API + '/auth/login', {
209
+ method: 'POST',
210
+ headers: { 'Content-Type': 'application/json' },
211
+ body: JSON.stringify({ email: 'admin@mindmatrix.com', password: 'Admin123!' }),
212
+ });
213
+ const { token } = await loginRes.json();
214
+ log('Logged in');
215
+
216
+ // Load all compiled views
217
+ const files = readdirSync(compiledDir).filter((f) => f.endsWith('.json'));
218
+ log(`Found ${files.length} compiled views`);
219
+
220
+ const views = {};
221
+ const metas = {};
222
+ for (const f of files) {
223
+ const slug = f.replace('.json', '');
224
+ views[slug] = loadCompiledView(slug);
225
+ metas[slug] = loadCompiledMeta(slug);
226
+ }
227
+
228
+ // Compose root tree
229
+ const rootView = JSON.parse(JSON.stringify(views['gc-layout']));
230
+ if (!rootView) {
231
+ console.error('gc-layout view tree not found!');
232
+ return false;
233
+ }
234
+
235
+ const rootDs = metas['gc-layout'].dataSources || [];
236
+ if (rootDs.length > 0) rootView.dataSources = rootDs;
237
+
238
+ const composedRoot = composeTree(rootView, 'GcPage');
239
+
240
+ // Log tree structure
241
+ let slotCount = 0;
242
+ let childrenCount = 0;
243
+ function countReplacements(node) {
244
+ if (node.id?.startsWith('composed-')) {
245
+ if (node.id === 'composed-children') childrenCount++;
246
+ else slotCount++;
247
+ }
248
+ (node.children || []).forEach(countReplacements);
249
+ }
250
+ countReplacements(composedRoot);
251
+ log(` Replaced ${slotCount} slots, ${childrenCount} children placeholders`);
252
+
253
+ if (!SILENT) {
254
+ function logTree(node, depth = 0) {
255
+ const ds = node.dataSources?.length || 0;
256
+ const bind = Object.keys(node.bindings || {}).length;
257
+ const extra = [];
258
+ if (ds > 0) extra.push(`ds=${ds}`);
259
+ if (bind > 0) extra.push(`bind=${bind}`);
260
+ if (node.config?.localDefaults) extra.push('localDefaults');
261
+ const suffix = extra.length > 0 ? ` [${extra.join(', ')}]` : '';
262
+ console.log(`${' '.repeat(depth)}${node.component} (${node.id})${suffix}`);
263
+ (node.children || []).forEach((c) => logTree(c, depth + 1));
264
+ }
265
+ console.log('\nComposed tree:');
266
+ logTree(composedRoot);
267
+ }
268
+
269
+ log(`\nTree size: ${JSON.stringify(composedRoot).length} bytes`);
270
+
271
+ // Get or create mm-glass-console definition
272
+ let def;
273
+ const defRes = await fetch(API + '/definitions/mm-glass-console', {
274
+ headers: { Authorization: 'Bearer ' + token },
275
+ });
276
+
277
+ if (defRes.ok) {
278
+ def = await defRes.json();
279
+ log(`Definition ID: ${def.id}`);
280
+ } else {
281
+ // Create the definition if it doesn't exist
282
+ log('Definition not found — creating...');
283
+ const createRes = await fetch(API + '/definitions', {
284
+ method: 'POST',
285
+ headers: {
286
+ 'Content-Type': 'application/json',
287
+ Authorization: 'Bearer ' + token,
288
+ },
289
+ body: JSON.stringify({
290
+ slug: 'mm-glass-console',
291
+ name: 'Glass Console',
292
+ description: 'Intent-driven data agent — calm interface for living reports',
293
+ category: ['blueprint', 'glass-console'],
294
+ version: '1.0.0',
295
+ view: composedRoot,
296
+ metadata: { blueprint_manifest: BLUEPRINT_MANIFEST },
297
+ }),
298
+ });
299
+
300
+ if (!createRes.ok) {
301
+ console.error('Create failed:', createRes.status, await createRes.text());
302
+ return false;
303
+ }
304
+
305
+ def = await createRes.json();
306
+ log(`Created definition: ${def.id}`);
307
+ }
308
+
309
+ // Merge manifest into existing metadata
310
+ const existingMetadata = def.metadata || {};
311
+ const updatedMetadata = {
312
+ ...existingMetadata,
313
+ blueprint_manifest: BLUEPRINT_MANIFEST,
314
+ };
315
+
316
+ // PATCH with BOTH view tree AND manifest metadata
317
+ const patchRes = await fetch(API + '/definitions/' + def.id, {
318
+ method: 'PATCH',
319
+ headers: {
320
+ 'Content-Type': 'application/json',
321
+ Authorization: 'Bearer ' + token,
322
+ },
323
+ body: JSON.stringify({
324
+ view: composedRoot,
325
+ metadata: updatedMetadata,
326
+ }),
327
+ });
328
+
329
+ if (!patchRes.ok) {
330
+ console.error('PATCH failed:', patchRes.status, await patchRes.text());
331
+ return false;
332
+ }
333
+ log('View tree + manifest updated');
334
+
335
+ // Verify
336
+ const verifyRes = await fetch(API + '/definitions/mm-glass-console', {
337
+ headers: { Authorization: 'Bearer ' + token },
338
+ });
339
+ const verified = await verifyRes.json();
340
+ const hasManifest = !!verified.metadata?.blueprint_manifest;
341
+ const hasFullBleed = verified.metadata?.blueprint_manifest?.config?.full_bleed === true;
342
+ const hasRoutes = verified.metadata?.blueprint_manifest?.routes?.length > 0;
343
+ log(`\nVerified:`);
344
+ log(` view: ${JSON.stringify(verified.view).length} bytes, root=${verified.view?.component}`);
345
+ log(` manifest: ${hasManifest ? 'present' : 'MISSING'}`);
346
+ log(` full_bleed: ${hasFullBleed}`);
347
+ log(` routes: ${hasRoutes ? verified.metadata.blueprint_manifest.routes.length : 0}`);
348
+
349
+ if (!hasManifest || !hasFullBleed) {
350
+ console.error('Manifest verification failed!');
351
+ return false;
352
+ }
353
+
354
+ log('\nSeed complete.');
355
+ return true;
356
+ }
357
+
358
+ // ---------------------------------------------------------------------------
359
+ // Watch mode
360
+ // ---------------------------------------------------------------------------
361
+ function startWatch() {
362
+ log('\n--- Watch mode: monitoring blueprint-glass-console sources ---');
363
+ log(` Source: ${sourceDir}`);
364
+ log(` Compiled: ${compiledDir}`);
365
+ log(' Press Ctrl+C to stop.\n');
366
+
367
+ function findFiles(dir) {
368
+ const entries = readdirSync(dir, { withFileTypes: true });
369
+ const files = [];
370
+ for (const entry of entries) {
371
+ const full = join(dir, entry.name);
372
+ if (entry.isDirectory()) files.push(...findFiles(full));
373
+ else if (entry.name.endsWith('.tsx')) files.push(full);
374
+ }
375
+ return files;
376
+ }
377
+
378
+ const sourceFiles = existsSync(sourceDir) ? findFiles(sourceDir) : [];
379
+ log(`Watching ${sourceFiles.length} source files...`);
380
+
381
+ let debounceTimer = null;
382
+ let isRunning = false;
383
+
384
+ async function onChange(filename) {
385
+ if (isRunning) return;
386
+ if (debounceTimer) clearTimeout(debounceTimer);
387
+
388
+ debounceTimer = setTimeout(async () => {
389
+ isRunning = true;
390
+ const ts = new Date().toLocaleTimeString();
391
+ log(`\n[${ts}] Change detected: ${filename}`);
392
+
393
+ const compiled = runCompile();
394
+ if (compiled) {
395
+ await runSeed();
396
+ }
397
+
398
+ isRunning = false;
399
+ log(`\n[${new Date().toLocaleTimeString()}] Waiting for changes...`);
400
+ }, 500);
401
+ }
402
+
403
+ for (const file of sourceFiles) {
404
+ watchFile(file, { interval: 1000 }, () => onChange(file));
405
+ }
406
+
407
+ process.on('SIGINT', () => {
408
+ log('\nStopping watch...');
409
+ for (const file of sourceFiles) unwatchFile(file);
410
+ process.exit(0);
411
+ });
412
+ }
413
+
414
+ // ---------------------------------------------------------------------------
415
+ // Main
416
+ // ---------------------------------------------------------------------------
417
+ async function main() {
418
+ log('Blueprint Glass Console Pipeline');
419
+ log(` Mode: ${WATCH ? 'watch' : COMPILE_ONLY ? 'compile-only' : SEED_ONLY ? 'seed-only' : 'full'}`);
420
+
421
+ if (WATCH) {
422
+ const compiled = runCompile();
423
+ if (compiled) await runSeed();
424
+ startWatch();
425
+ return;
426
+ }
427
+
428
+ if (COMPILE_ONLY) {
429
+ runCompile();
430
+ return;
431
+ }
432
+
433
+ if (SEED_ONLY) {
434
+ await runSeed();
435
+ return;
436
+ }
437
+
438
+ // Full mode: compile + seed
439
+ const compiled = runCompile();
440
+ if (compiled) {
441
+ await runSeed();
442
+ }
443
+ }
444
+
445
+ main().catch((e) => console.error(e));