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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,186 @@
1
+ import {
2
+ createSourceEnvelope,
3
+ generateFsTree
4
+ } from "./chunk-5M7DKKBC.mjs";
5
+ import {
6
+ babelPlugin
7
+ } from "./chunk-OPJKP747.mjs";
8
+
9
+ // src/envelope.ts
10
+ import { createHash } from "crypto";
11
+ import { readFileSync } from "fs";
12
+ import { resolve, relative, basename } from "path";
13
+ import { glob } from "glob";
14
+ import { transformSync } from "@babel/core";
15
+ async function buildEnvelope(projectDir, options = {}) {
16
+ const root = resolve(projectDir);
17
+ const slug = options.slug ?? basename(root);
18
+ const version = options.version ?? "0.1.0";
19
+ const mode = options.mode ?? "infer";
20
+ const patterns = options.include ?? [
21
+ `${root}/**/*.workflow.{tsx,ts,jsx,js}`,
22
+ `${root}/**/models/**/*.{ts,tsx}`,
23
+ `${root}/**/*.server.{ts,tsx,js,jsx}`
24
+ ];
25
+ const discoveredFiles = [];
26
+ for (const pattern of patterns) {
27
+ const matches = await glob(pattern, { absolute: true });
28
+ discoveredFiles.push(...matches);
29
+ }
30
+ const uniqueFiles = [...new Set(discoveredFiles)];
31
+ const allSourceFiles = await glob(`${root}/**/*.{ts,tsx,js,jsx,css,json}`, {
32
+ absolute: true,
33
+ ignore: ["**/node_modules/**", "**/dist/**", "**/.git/**"]
34
+ });
35
+ const fsTree = generateFsTree(allSourceFiles, root);
36
+ const definitions = [];
37
+ const envelopeFiles = [];
38
+ const errors = [];
39
+ const sourceBindings = [];
40
+ for (const file of uniqueFiles) {
41
+ const relativePath = relative(root, file);
42
+ let fileContent;
43
+ try {
44
+ fileContent = readFileSync(file);
45
+ } catch {
46
+ errors.push({ file: relativePath, message: "Could not read file" });
47
+ continue;
48
+ }
49
+ const fileHash = createHash("sha256").update(fileContent).digest("hex");
50
+ const code = fileContent.toString("utf-8");
51
+ try {
52
+ const hasTsx = /\.(tsx|jsx)$/.test(file);
53
+ const parserPlugins = ["typescript"];
54
+ if (hasTsx) parserPlugins.push("jsx");
55
+ const result = transformSync(code, {
56
+ filename: file,
57
+ plugins: [[babelPlugin, { mode }]],
58
+ parserOpts: { plugins: parserPlugins, attachComment: true },
59
+ sourceMaps: true
60
+ });
61
+ const ir = result?.metadata?.mindmatrixIR;
62
+ if (ir) {
63
+ definitions.push(ir);
64
+ const bindings = extractSourceBindings(ir, relativePath);
65
+ sourceBindings.push(...bindings);
66
+ const irErrors = ir.metadata?.errors;
67
+ if (irErrors?.length) {
68
+ for (const e of irErrors) {
69
+ errors.push({ file: relativePath, message: e.message, line: e.line });
70
+ }
71
+ }
72
+ envelopeFiles.push({
73
+ path: relativePath,
74
+ hash: fileHash,
75
+ size: fileContent.length,
76
+ slug: ir.slug,
77
+ compiled: true
78
+ });
79
+ } else {
80
+ envelopeFiles.push({
81
+ path: relativePath,
82
+ hash: fileHash,
83
+ size: fileContent.length,
84
+ compiled: false
85
+ });
86
+ }
87
+ } catch (error) {
88
+ const msg = error instanceof Error ? error.message : String(error);
89
+ errors.push({ file: relativePath, message: msg });
90
+ envelopeFiles.push({
91
+ path: relativePath,
92
+ hash: fileHash,
93
+ size: fileContent.length,
94
+ compiled: false
95
+ });
96
+ }
97
+ }
98
+ const envelope = createSourceEnvelope({
99
+ blueprintSlug: slug,
100
+ blueprintVersion: version,
101
+ definitions,
102
+ fsTree,
103
+ sourceBindings,
104
+ environment: options.environment,
105
+ mode
106
+ });
107
+ const envelopeJson = JSON.stringify(envelope);
108
+ const manifestHash = createHash("sha256").update(envelopeJson).digest("hex");
109
+ const manifest = {
110
+ hash: manifestHash,
111
+ files: envelopeFiles,
112
+ version,
113
+ timestamp: envelope.createdAt,
114
+ definitionCount: definitions.length,
115
+ errorCount: errors.length
116
+ };
117
+ return {
118
+ envelope,
119
+ manifest,
120
+ files: envelopeFiles,
121
+ definitions,
122
+ errors
123
+ };
124
+ }
125
+ function extractSourceBindings(ir, filePath) {
126
+ const bindings = [];
127
+ const slug = ir.slug ?? "";
128
+ for (const field of ir.fields) {
129
+ const source = field.__source;
130
+ bindings.push({
131
+ kind: "field",
132
+ name: field.name,
133
+ workflowSlug: slug,
134
+ file: filePath,
135
+ startLine: source?.line ?? 0,
136
+ startColumn: source?.column ?? 0,
137
+ endLine: source?.endLine,
138
+ endColumn: source?.endColumn
139
+ });
140
+ }
141
+ for (const state of ir.states) {
142
+ const source = state.__source;
143
+ bindings.push({
144
+ kind: "state",
145
+ name: state.name,
146
+ workflowSlug: slug,
147
+ file: filePath,
148
+ startLine: source?.line ?? 0,
149
+ startColumn: source?.column ?? 0,
150
+ endLine: source?.endLine,
151
+ endColumn: source?.endColumn
152
+ });
153
+ }
154
+ for (const transition of ir.transitions) {
155
+ const source = transition.__source;
156
+ bindings.push({
157
+ kind: "transition",
158
+ name: transition.name,
159
+ workflowSlug: slug,
160
+ file: filePath,
161
+ startLine: source?.line ?? 0,
162
+ startColumn: source?.column ?? 0,
163
+ endLine: source?.endLine,
164
+ endColumn: source?.endColumn
165
+ });
166
+ }
167
+ if (ir.extensions) {
168
+ for (const [tag, islands] of Object.entries(ir.extensions)) {
169
+ for (const island of islands) {
170
+ bindings.push({
171
+ kind: "grammar-island",
172
+ name: `${tag}:${island.contextTag}`,
173
+ workflowSlug: slug,
174
+ file: filePath,
175
+ startLine: island.__source?.line ?? 0,
176
+ startColumn: island.__source?.column ?? 0
177
+ });
178
+ }
179
+ }
180
+ }
181
+ return bindings;
182
+ }
183
+
184
+ export {
185
+ buildEnvelope
186
+ };