@emailens/engine 0.3.0 → 0.5.0
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/README.md +255 -246
- package/dist/chunk-LW2IMTBA.js +326 -0
- package/dist/chunk-LW2IMTBA.js.map +1 -0
- package/dist/chunk-PFONR3YC.js +56 -0
- package/dist/chunk-PFONR3YC.js.map +1 -0
- package/dist/chunk-SZ5O5PDZ.js +78 -0
- package/dist/chunk-SZ5O5PDZ.js.map +1 -0
- package/dist/chunk-W4SPWESS.js +64 -0
- package/dist/chunk-W4SPWESS.js.map +1 -0
- package/dist/compile/index.cjs +585 -0
- package/dist/compile/index.cjs.map +1 -0
- package/dist/compile/index.d.cts +47 -0
- package/dist/compile/index.d.ts +47 -0
- package/dist/compile/index.js +59 -0
- package/dist/compile/index.js.map +1 -0
- package/dist/index.cjs +5485 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +273 -0
- package/dist/index.d.ts +86 -227
- package/dist/index.js +1571 -942
- package/dist/index.js.map +1 -1
- package/dist/maizzle-YDSYDVSM.js +8 -0
- package/dist/maizzle-YDSYDVSM.js.map +1 -0
- package/dist/mjml-IYGC6AOM.js +8 -0
- package/dist/mjml-IYGC6AOM.js.map +1 -0
- package/dist/react-email-AN62KVVF.js +8 -0
- package/dist/react-email-AN62KVVF.js.map +1 -0
- package/dist/react-email-BQljgXbo.d.cts +289 -0
- package/dist/react-email-BQljgXbo.d.ts +289 -0
- package/package.json +97 -60
|
@@ -0,0 +1,585 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __esm = (fn, res) => function __init() {
|
|
9
|
+
return fn && (res = (0, fn[__getOwnPropNames(fn)[0]])(fn = 0)), res;
|
|
10
|
+
};
|
|
11
|
+
var __export = (target, all) => {
|
|
12
|
+
for (var name in all)
|
|
13
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
14
|
+
};
|
|
15
|
+
var __copyProps = (to, from, except, desc) => {
|
|
16
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
17
|
+
for (let key of __getOwnPropNames(from))
|
|
18
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
19
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
20
|
+
}
|
|
21
|
+
return to;
|
|
22
|
+
};
|
|
23
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
24
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
25
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
26
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
27
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
28
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
29
|
+
mod
|
|
30
|
+
));
|
|
31
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
32
|
+
|
|
33
|
+
// src/compile/errors.ts
|
|
34
|
+
var CompileError;
|
|
35
|
+
var init_errors = __esm({
|
|
36
|
+
"src/compile/errors.ts"() {
|
|
37
|
+
"use strict";
|
|
38
|
+
CompileError = class extends Error {
|
|
39
|
+
constructor(message, format, phase) {
|
|
40
|
+
super(message);
|
|
41
|
+
this.name = "CompileError";
|
|
42
|
+
this.format = format;
|
|
43
|
+
this.phase = phase;
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
// src/compile/react-email.ts
|
|
50
|
+
var react_email_exports = {};
|
|
51
|
+
__export(react_email_exports, {
|
|
52
|
+
compileReactEmail: () => compileReactEmail
|
|
53
|
+
});
|
|
54
|
+
async function compileReactEmail(source, options) {
|
|
55
|
+
var _a, _b;
|
|
56
|
+
const strategy = (_a = options == null ? void 0 : options.sandbox) != null ? _a : "isolated-vm";
|
|
57
|
+
if (!source || !source.trim()) {
|
|
58
|
+
throw new CompileError("JSX source must not be empty.", "jsx", "validation");
|
|
59
|
+
}
|
|
60
|
+
if (source.length > MAX_SOURCE_SIZE) {
|
|
61
|
+
throw new CompileError(
|
|
62
|
+
`JSX source exceeds ${MAX_SOURCE_SIZE / 1e3}KB limit.`,
|
|
63
|
+
"jsx",
|
|
64
|
+
"validation"
|
|
65
|
+
);
|
|
66
|
+
}
|
|
67
|
+
let transform;
|
|
68
|
+
let React;
|
|
69
|
+
let ReactEmailComponents;
|
|
70
|
+
let render;
|
|
71
|
+
try {
|
|
72
|
+
({ transform } = await import("sucrase"));
|
|
73
|
+
} catch (e) {
|
|
74
|
+
throw new CompileError(
|
|
75
|
+
'JSX compilation requires "sucrase". Install it:\n npm install sucrase',
|
|
76
|
+
"jsx",
|
|
77
|
+
"transpile"
|
|
78
|
+
);
|
|
79
|
+
}
|
|
80
|
+
try {
|
|
81
|
+
React = await import("react");
|
|
82
|
+
} catch (e) {
|
|
83
|
+
throw new CompileError(
|
|
84
|
+
'JSX compilation requires "react". Install it:\n npm install react',
|
|
85
|
+
"jsx",
|
|
86
|
+
"transpile"
|
|
87
|
+
);
|
|
88
|
+
}
|
|
89
|
+
try {
|
|
90
|
+
ReactEmailComponents = await import("@react-email/components");
|
|
91
|
+
} catch (e) {
|
|
92
|
+
throw new CompileError(
|
|
93
|
+
'JSX compilation requires "@react-email/components". Install it:\n npm install @react-email/components',
|
|
94
|
+
"jsx",
|
|
95
|
+
"transpile"
|
|
96
|
+
);
|
|
97
|
+
}
|
|
98
|
+
try {
|
|
99
|
+
({ render } = await import("@react-email/render"));
|
|
100
|
+
} catch (e) {
|
|
101
|
+
throw new CompileError(
|
|
102
|
+
'JSX compilation requires "@react-email/render". Install it:\n npm install @react-email/render',
|
|
103
|
+
"jsx",
|
|
104
|
+
"transpile"
|
|
105
|
+
);
|
|
106
|
+
}
|
|
107
|
+
let transpiledCode;
|
|
108
|
+
try {
|
|
109
|
+
const result = transform(source, {
|
|
110
|
+
transforms: ["typescript", "jsx", "imports"],
|
|
111
|
+
jsxRuntime: "classic",
|
|
112
|
+
production: true
|
|
113
|
+
});
|
|
114
|
+
transpiledCode = result.code;
|
|
115
|
+
} catch (err) {
|
|
116
|
+
const message = err instanceof Error ? err.message : "Unknown transpilation error";
|
|
117
|
+
throw new CompileError(`JSX syntax error: ${message}`, "jsx", "transpile");
|
|
118
|
+
}
|
|
119
|
+
let moduleExports;
|
|
120
|
+
switch (strategy) {
|
|
121
|
+
case "isolated-vm":
|
|
122
|
+
moduleExports = await executeInIsolatedVm(transpiledCode, React, ReactEmailComponents);
|
|
123
|
+
break;
|
|
124
|
+
case "quickjs":
|
|
125
|
+
moduleExports = await executeInQuickJs(transpiledCode, React, ReactEmailComponents);
|
|
126
|
+
break;
|
|
127
|
+
case "vm":
|
|
128
|
+
moduleExports = executeInVm(transpiledCode, React, ReactEmailComponents);
|
|
129
|
+
break;
|
|
130
|
+
default:
|
|
131
|
+
throw new CompileError(
|
|
132
|
+
`Unknown sandbox strategy: "${strategy}". Use "vm", "isolated-vm", or "quickjs".`,
|
|
133
|
+
"jsx",
|
|
134
|
+
"execution"
|
|
135
|
+
);
|
|
136
|
+
}
|
|
137
|
+
let Component = (_b = moduleExports.default) != null ? _b : moduleExports;
|
|
138
|
+
if (typeof Component !== "function" && typeof Component === "object" && Component !== null) {
|
|
139
|
+
const values = Object.values(Component);
|
|
140
|
+
const fn = values.find((v) => typeof v === "function");
|
|
141
|
+
if (fn) Component = fn;
|
|
142
|
+
}
|
|
143
|
+
if (typeof Component !== "function") {
|
|
144
|
+
throw new CompileError(
|
|
145
|
+
'The JSX source must export a React component function. Use "export default function Email() { ... }" or "export function Email() { ... }".',
|
|
146
|
+
"jsx",
|
|
147
|
+
"execution"
|
|
148
|
+
);
|
|
149
|
+
}
|
|
150
|
+
try {
|
|
151
|
+
const element = React.createElement(Component);
|
|
152
|
+
const html = await render(element);
|
|
153
|
+
return html;
|
|
154
|
+
} catch (err) {
|
|
155
|
+
const message = err instanceof Error ? err.message : "Unknown rendering error";
|
|
156
|
+
throw new CompileError(`React rendering error: ${message}`, "jsx", "render");
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
function executeInVm(code, React, ReactEmailComponents) {
|
|
160
|
+
const { createContext, Script } = require("vm");
|
|
161
|
+
const ALLOWED_MODULES = {
|
|
162
|
+
react: React,
|
|
163
|
+
"@react-email/components": ReactEmailComponents
|
|
164
|
+
};
|
|
165
|
+
const moduleExports = {};
|
|
166
|
+
const moduleObj = { exports: moduleExports };
|
|
167
|
+
const mockRequire = (moduleName) => {
|
|
168
|
+
if (moduleName in ALLOWED_MODULES) {
|
|
169
|
+
return ALLOWED_MODULES[moduleName];
|
|
170
|
+
}
|
|
171
|
+
throw new Error(
|
|
172
|
+
`Import of "${moduleName}" is not allowed. Only "react" and "@react-email/components" can be imported.`
|
|
173
|
+
);
|
|
174
|
+
};
|
|
175
|
+
const sandbox = {
|
|
176
|
+
module: moduleObj,
|
|
177
|
+
exports: moduleExports,
|
|
178
|
+
require: mockRequire,
|
|
179
|
+
React,
|
|
180
|
+
Object,
|
|
181
|
+
Array,
|
|
182
|
+
String,
|
|
183
|
+
Number,
|
|
184
|
+
Boolean,
|
|
185
|
+
Map,
|
|
186
|
+
Set,
|
|
187
|
+
WeakMap,
|
|
188
|
+
WeakSet,
|
|
189
|
+
JSON,
|
|
190
|
+
Math,
|
|
191
|
+
Date,
|
|
192
|
+
RegExp,
|
|
193
|
+
Error,
|
|
194
|
+
TypeError,
|
|
195
|
+
RangeError,
|
|
196
|
+
ReferenceError,
|
|
197
|
+
SyntaxError,
|
|
198
|
+
URIError,
|
|
199
|
+
Promise,
|
|
200
|
+
Symbol,
|
|
201
|
+
Proxy: void 0,
|
|
202
|
+
Reflect: void 0,
|
|
203
|
+
parseInt,
|
|
204
|
+
parseFloat,
|
|
205
|
+
isNaN,
|
|
206
|
+
isFinite,
|
|
207
|
+
encodeURIComponent,
|
|
208
|
+
decodeURIComponent,
|
|
209
|
+
encodeURI,
|
|
210
|
+
decodeURI,
|
|
211
|
+
undefined: void 0,
|
|
212
|
+
NaN: NaN,
|
|
213
|
+
Infinity: Infinity,
|
|
214
|
+
console: { log: () => {
|
|
215
|
+
}, warn: () => {
|
|
216
|
+
}, error: () => {
|
|
217
|
+
}, info: () => {
|
|
218
|
+
}, debug: () => {
|
|
219
|
+
} },
|
|
220
|
+
setTimeout: void 0,
|
|
221
|
+
setInterval: void 0,
|
|
222
|
+
setImmediate: void 0,
|
|
223
|
+
queueMicrotask: void 0,
|
|
224
|
+
process: void 0,
|
|
225
|
+
globalThis: void 0,
|
|
226
|
+
global: void 0,
|
|
227
|
+
Buffer: void 0,
|
|
228
|
+
__dirname: void 0,
|
|
229
|
+
__filename: void 0
|
|
230
|
+
};
|
|
231
|
+
const context = createContext(sandbox, {
|
|
232
|
+
codeGeneration: { strings: false, wasm: false }
|
|
233
|
+
});
|
|
234
|
+
try {
|
|
235
|
+
const script = new Script(code, { filename: "user-email-component.tsx" });
|
|
236
|
+
script.runInContext(context, { timeout: EXECUTION_TIMEOUT_MS, displayErrors: true });
|
|
237
|
+
} catch (err) {
|
|
238
|
+
const message = err instanceof Error ? err.message : "Unknown execution error";
|
|
239
|
+
if (message.includes("Script execution timed out")) {
|
|
240
|
+
throw new CompileError("JSX execution timed out (possible infinite loop).", "jsx", "execution");
|
|
241
|
+
}
|
|
242
|
+
throw new CompileError(`JSX execution error: ${message}`, "jsx", "execution");
|
|
243
|
+
}
|
|
244
|
+
return moduleObj.exports;
|
|
245
|
+
}
|
|
246
|
+
async function executeInIsolatedVm(code, React, ReactEmailComponents) {
|
|
247
|
+
let ivm;
|
|
248
|
+
try {
|
|
249
|
+
ivm = await import("isolated-vm");
|
|
250
|
+
} catch (e) {
|
|
251
|
+
throw new CompileError(
|
|
252
|
+
'Sandbox strategy "isolated-vm" requires the "isolated-vm" package. Install it:\n npm install isolated-vm\nOr use sandbox: "vm" for a lighter (but less secure) alternative.',
|
|
253
|
+
"jsx",
|
|
254
|
+
"execution"
|
|
255
|
+
);
|
|
256
|
+
}
|
|
257
|
+
const isolate = new ivm.Isolate({ memoryLimit: 128 });
|
|
258
|
+
try {
|
|
259
|
+
const ivmContext = await isolate.createContext();
|
|
260
|
+
const jail = ivmContext.global;
|
|
261
|
+
const ALLOWED_MODULES = {
|
|
262
|
+
react: React,
|
|
263
|
+
"@react-email/components": ReactEmailComponents
|
|
264
|
+
};
|
|
265
|
+
const hostRequire = new ivm.Reference(function(name) {
|
|
266
|
+
if (name in ALLOWED_MODULES) {
|
|
267
|
+
return ALLOWED_MODULES[name];
|
|
268
|
+
}
|
|
269
|
+
throw new Error(
|
|
270
|
+
`Import of "${name}" is not allowed. Only "react" and "@react-email/components" can be imported.`
|
|
271
|
+
);
|
|
272
|
+
});
|
|
273
|
+
await jail.set("__hostRequire", hostRequire);
|
|
274
|
+
await jail.set("__hostReact", new ivm.Reference(React));
|
|
275
|
+
const wrapperCode = `
|
|
276
|
+
(function() {
|
|
277
|
+
const module = { exports: {} };
|
|
278
|
+
const exports = module.exports;
|
|
279
|
+
|
|
280
|
+
const React = __hostReact.derefInto();
|
|
281
|
+
function require(name) {
|
|
282
|
+
return __hostRequire.applySync(undefined, [name], { result: { copy: true }, arguments: { copy: true } });
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
${code}
|
|
286
|
+
|
|
287
|
+
return module.exports;
|
|
288
|
+
})()
|
|
289
|
+
`;
|
|
290
|
+
try {
|
|
291
|
+
const result = await ivmContext.evalClosureSync(wrapperCode, [], {
|
|
292
|
+
timeout: EXECUTION_TIMEOUT_MS
|
|
293
|
+
});
|
|
294
|
+
return typeof result === "object" && result !== null ? result : { default: result };
|
|
295
|
+
} catch (err) {
|
|
296
|
+
const message = err instanceof Error ? err.message : "Unknown execution error";
|
|
297
|
+
if (message.includes("timed out") || message.includes("Timeout")) {
|
|
298
|
+
throw new CompileError("JSX execution timed out (possible infinite loop).", "jsx", "execution");
|
|
299
|
+
}
|
|
300
|
+
throw new CompileError(`JSX execution error: ${message}`, "jsx", "execution");
|
|
301
|
+
}
|
|
302
|
+
} finally {
|
|
303
|
+
isolate.dispose();
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
async function executeInQuickJs(code, React, ReactEmailComponents) {
|
|
307
|
+
var _a;
|
|
308
|
+
let getQuickJS;
|
|
309
|
+
try {
|
|
310
|
+
({ getQuickJS } = await import("quickjs-emscripten"));
|
|
311
|
+
} catch (e) {
|
|
312
|
+
throw new CompileError(
|
|
313
|
+
'Sandbox strategy "quickjs" requires the "quickjs-emscripten" package. Install it:\n npm install quickjs-emscripten\nOr use sandbox: "vm" for a lighter alternative.',
|
|
314
|
+
"jsx",
|
|
315
|
+
"execution"
|
|
316
|
+
);
|
|
317
|
+
}
|
|
318
|
+
const QuickJS = await getQuickJS();
|
|
319
|
+
const vm = QuickJS.newContext();
|
|
320
|
+
try {
|
|
321
|
+
const validationCode = `
|
|
322
|
+
(function() {
|
|
323
|
+
var module = { exports: {} };
|
|
324
|
+
var exports = module.exports;
|
|
325
|
+
var React = { createElement: function() { return {}; } };
|
|
326
|
+
function require(name) {
|
|
327
|
+
if (name === "react" || name === "@react-email/components") {
|
|
328
|
+
return React;
|
|
329
|
+
}
|
|
330
|
+
throw new Error('Import of "' + name + '" is not allowed.');
|
|
331
|
+
}
|
|
332
|
+
try {
|
|
333
|
+
${code}
|
|
334
|
+
return JSON.stringify({ ok: true });
|
|
335
|
+
} catch(e) {
|
|
336
|
+
return JSON.stringify({ ok: false, error: e.message || "Unknown error" });
|
|
337
|
+
}
|
|
338
|
+
})()
|
|
339
|
+
`;
|
|
340
|
+
const result = vm.evalCode(validationCode);
|
|
341
|
+
if (result.error) {
|
|
342
|
+
const errorVal = vm.dump(result.error);
|
|
343
|
+
result.error.dispose();
|
|
344
|
+
throw new CompileError(
|
|
345
|
+
`JSX execution error: ${typeof errorVal === "string" ? errorVal : "QuickJS execution failed"}`,
|
|
346
|
+
"jsx",
|
|
347
|
+
"execution"
|
|
348
|
+
);
|
|
349
|
+
}
|
|
350
|
+
const resultStr = vm.dump(result.value);
|
|
351
|
+
result.value.dispose();
|
|
352
|
+
if (typeof resultStr === "string") {
|
|
353
|
+
const parsed = JSON.parse(resultStr);
|
|
354
|
+
if (!parsed.ok) {
|
|
355
|
+
throw new CompileError(
|
|
356
|
+
`JSX execution error: ${(_a = parsed.error) != null ? _a : "Unknown error"}`,
|
|
357
|
+
"jsx",
|
|
358
|
+
"execution"
|
|
359
|
+
);
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
return executeInVm(code, React, ReactEmailComponents);
|
|
363
|
+
} finally {
|
|
364
|
+
vm.dispose();
|
|
365
|
+
}
|
|
366
|
+
}
|
|
367
|
+
var MAX_SOURCE_SIZE, EXECUTION_TIMEOUT_MS;
|
|
368
|
+
var init_react_email = __esm({
|
|
369
|
+
"src/compile/react-email.ts"() {
|
|
370
|
+
"use strict";
|
|
371
|
+
init_errors();
|
|
372
|
+
MAX_SOURCE_SIZE = 256e3;
|
|
373
|
+
EXECUTION_TIMEOUT_MS = 5e3;
|
|
374
|
+
}
|
|
375
|
+
});
|
|
376
|
+
|
|
377
|
+
// src/compile/mjml.ts
|
|
378
|
+
var mjml_exports = {};
|
|
379
|
+
__export(mjml_exports, {
|
|
380
|
+
compileMjml: () => compileMjml
|
|
381
|
+
});
|
|
382
|
+
async function compileMjml(source) {
|
|
383
|
+
var _a;
|
|
384
|
+
if (!source || !source.trim()) {
|
|
385
|
+
throw new CompileError("MJML source must not be empty.", "mjml", "validation");
|
|
386
|
+
}
|
|
387
|
+
if (source.length > MAX_SOURCE_SIZE2) {
|
|
388
|
+
throw new CompileError(
|
|
389
|
+
`MJML source exceeds ${MAX_SOURCE_SIZE2 / 1e3}KB limit.`,
|
|
390
|
+
"mjml",
|
|
391
|
+
"validation"
|
|
392
|
+
);
|
|
393
|
+
}
|
|
394
|
+
if (!/<mjml[\s>]/i.test(source)) {
|
|
395
|
+
throw new CompileError(
|
|
396
|
+
"MJML source must contain a root <mjml> element. Example: <mjml><mj-body><mj-section><mj-column><mj-text>Hello</mj-text></mj-column></mj-section></mj-body></mjml>",
|
|
397
|
+
"mjml",
|
|
398
|
+
"validation"
|
|
399
|
+
);
|
|
400
|
+
}
|
|
401
|
+
let mjml2html;
|
|
402
|
+
try {
|
|
403
|
+
const mjmlModule = await import("mjml");
|
|
404
|
+
mjml2html = (_a = mjmlModule.default) != null ? _a : mjmlModule;
|
|
405
|
+
} catch (e) {
|
|
406
|
+
throw new CompileError(
|
|
407
|
+
'MJML compilation requires "mjml". Install it:\n npm install mjml',
|
|
408
|
+
"mjml",
|
|
409
|
+
"compile"
|
|
410
|
+
);
|
|
411
|
+
}
|
|
412
|
+
try {
|
|
413
|
+
const result = await mjml2html(source, {
|
|
414
|
+
validationLevel: "soft",
|
|
415
|
+
keepComments: false
|
|
416
|
+
});
|
|
417
|
+
if (result.errors && result.errors.length > 0 && !result.html) {
|
|
418
|
+
const errorMessages = result.errors.map((e) => `Line ${e.line}: ${e.message} (${e.tagName})`).join("; ");
|
|
419
|
+
throw new CompileError(
|
|
420
|
+
`MJML compilation errors: ${errorMessages}`,
|
|
421
|
+
"mjml",
|
|
422
|
+
"compile"
|
|
423
|
+
);
|
|
424
|
+
}
|
|
425
|
+
if (!result.html) {
|
|
426
|
+
throw new CompileError("MJML compilation produced empty output.", "mjml", "compile");
|
|
427
|
+
}
|
|
428
|
+
return result.html;
|
|
429
|
+
} catch (err) {
|
|
430
|
+
if (err instanceof CompileError) throw err;
|
|
431
|
+
const message = err instanceof Error ? err.message : "Unknown MJML compilation error";
|
|
432
|
+
throw new CompileError(`MJML compilation failed: ${message}`, "mjml", "compile");
|
|
433
|
+
}
|
|
434
|
+
}
|
|
435
|
+
var MAX_SOURCE_SIZE2;
|
|
436
|
+
var init_mjml = __esm({
|
|
437
|
+
"src/compile/mjml.ts"() {
|
|
438
|
+
"use strict";
|
|
439
|
+
init_errors();
|
|
440
|
+
MAX_SOURCE_SIZE2 = 512e3;
|
|
441
|
+
}
|
|
442
|
+
});
|
|
443
|
+
|
|
444
|
+
// src/compile/maizzle.ts
|
|
445
|
+
var maizzle_exports = {};
|
|
446
|
+
__export(maizzle_exports, {
|
|
447
|
+
compileMaizzle: () => compileMaizzle
|
|
448
|
+
});
|
|
449
|
+
async function compileMaizzle(source) {
|
|
450
|
+
if (!source || !source.trim()) {
|
|
451
|
+
throw new CompileError("Maizzle source must not be empty.", "maizzle", "validation");
|
|
452
|
+
}
|
|
453
|
+
if (source.length > MAX_SOURCE_SIZE3) {
|
|
454
|
+
throw new CompileError(
|
|
455
|
+
`Maizzle source exceeds ${MAX_SOURCE_SIZE3 / 1e3}KB limit.`,
|
|
456
|
+
"maizzle",
|
|
457
|
+
"validation"
|
|
458
|
+
);
|
|
459
|
+
}
|
|
460
|
+
if (DANGEROUS_DIRECTIVE_RE.test(source)) {
|
|
461
|
+
throw new CompileError(
|
|
462
|
+
"Maizzle templates may not use <extends>, <component>, <fetch>, <include>, <module>, <slot>, <fill>, <raw>, <block>, or <yield> directives. These directives access the server file system at compile time. Use inline HTML and Tailwind utility classes instead.",
|
|
463
|
+
"maizzle",
|
|
464
|
+
"validation"
|
|
465
|
+
);
|
|
466
|
+
}
|
|
467
|
+
let maizzleRender;
|
|
468
|
+
try {
|
|
469
|
+
const maizzle = await import("@maizzle/framework");
|
|
470
|
+
maizzleRender = maizzle.render;
|
|
471
|
+
} catch (e) {
|
|
472
|
+
throw new CompileError(
|
|
473
|
+
'Maizzle compilation requires "@maizzle/framework". Install it:\n npm install @maizzle/framework',
|
|
474
|
+
"maizzle",
|
|
475
|
+
"compile"
|
|
476
|
+
);
|
|
477
|
+
}
|
|
478
|
+
const compilePromise = maizzleRender(source, {
|
|
479
|
+
css: {
|
|
480
|
+
inline: {
|
|
481
|
+
removeInlinedSelectors: true,
|
|
482
|
+
applyWidthAttributes: true,
|
|
483
|
+
applyHeightAttributes: true
|
|
484
|
+
},
|
|
485
|
+
shorthand: true,
|
|
486
|
+
sixHex: true
|
|
487
|
+
},
|
|
488
|
+
locals: {}
|
|
489
|
+
});
|
|
490
|
+
const timeoutPromise = new Promise((_, reject) => {
|
|
491
|
+
const t = setTimeout(() => {
|
|
492
|
+
reject(
|
|
493
|
+
new CompileError(
|
|
494
|
+
`Maizzle compilation timed out after ${COMPILE_TIMEOUT_MS / 1e3}s.`,
|
|
495
|
+
"maizzle",
|
|
496
|
+
"compile"
|
|
497
|
+
)
|
|
498
|
+
);
|
|
499
|
+
}, COMPILE_TIMEOUT_MS);
|
|
500
|
+
if (typeof t.unref === "function") t.unref();
|
|
501
|
+
});
|
|
502
|
+
try {
|
|
503
|
+
const { html } = await Promise.race([compilePromise, timeoutPromise]);
|
|
504
|
+
if (!html) {
|
|
505
|
+
throw new CompileError("Maizzle compilation produced empty output.", "maizzle", "compile");
|
|
506
|
+
}
|
|
507
|
+
return html;
|
|
508
|
+
} catch (err) {
|
|
509
|
+
if (err instanceof CompileError) throw err;
|
|
510
|
+
const message = err instanceof Error ? err.message : "Unknown Maizzle compilation error";
|
|
511
|
+
throw new CompileError(`Maizzle compilation failed: ${message}`, "maizzle", "compile");
|
|
512
|
+
}
|
|
513
|
+
}
|
|
514
|
+
var MAX_SOURCE_SIZE3, COMPILE_TIMEOUT_MS, DANGEROUS_DIRECTIVE_RE;
|
|
515
|
+
var init_maizzle = __esm({
|
|
516
|
+
"src/compile/maizzle.ts"() {
|
|
517
|
+
"use strict";
|
|
518
|
+
init_errors();
|
|
519
|
+
MAX_SOURCE_SIZE3 = 512e3;
|
|
520
|
+
COMPILE_TIMEOUT_MS = 15e3;
|
|
521
|
+
DANGEROUS_DIRECTIVE_RE = /<\s*(?:extends|component|fetch|include|module|slot|fill|raw|block|yield)\b/i;
|
|
522
|
+
}
|
|
523
|
+
});
|
|
524
|
+
|
|
525
|
+
// src/compile/index.ts
|
|
526
|
+
var compile_exports = {};
|
|
527
|
+
__export(compile_exports, {
|
|
528
|
+
CompileError: () => CompileError,
|
|
529
|
+
compile: () => compile,
|
|
530
|
+
compileMaizzle: () => compileMaizzle,
|
|
531
|
+
compileMjml: () => compileMjml,
|
|
532
|
+
compileReactEmail: () => compileReactEmail,
|
|
533
|
+
detectFormat: () => detectFormat
|
|
534
|
+
});
|
|
535
|
+
module.exports = __toCommonJS(compile_exports);
|
|
536
|
+
var import_node_path = require("path");
|
|
537
|
+
init_errors();
|
|
538
|
+
init_react_email();
|
|
539
|
+
init_mjml();
|
|
540
|
+
init_maizzle();
|
|
541
|
+
async function compile(source, format, filePath) {
|
|
542
|
+
switch (format) {
|
|
543
|
+
case "html":
|
|
544
|
+
return source;
|
|
545
|
+
case "jsx": {
|
|
546
|
+
const { compileReactEmail: compileReactEmail2 } = await Promise.resolve().then(() => (init_react_email(), react_email_exports));
|
|
547
|
+
return compileReactEmail2(source);
|
|
548
|
+
}
|
|
549
|
+
case "mjml": {
|
|
550
|
+
const { compileMjml: compileMjml2 } = await Promise.resolve().then(() => (init_mjml(), mjml_exports));
|
|
551
|
+
return compileMjml2(source);
|
|
552
|
+
}
|
|
553
|
+
case "maizzle": {
|
|
554
|
+
const { compileMaizzle: compileMaizzle2 } = await Promise.resolve().then(() => (init_maizzle(), maizzle_exports));
|
|
555
|
+
return compileMaizzle2(source);
|
|
556
|
+
}
|
|
557
|
+
default:
|
|
558
|
+
throw new Error(`Unknown format: "${format}". Use html, jsx, mjml, or maizzle.`);
|
|
559
|
+
}
|
|
560
|
+
}
|
|
561
|
+
function detectFormat(filePath) {
|
|
562
|
+
const ext = (0, import_node_path.extname)(filePath).toLowerCase();
|
|
563
|
+
switch (ext) {
|
|
564
|
+
case ".tsx":
|
|
565
|
+
case ".jsx":
|
|
566
|
+
return "jsx";
|
|
567
|
+
case ".mjml":
|
|
568
|
+
return "mjml";
|
|
569
|
+
case ".html":
|
|
570
|
+
case ".htm":
|
|
571
|
+
return "html";
|
|
572
|
+
default:
|
|
573
|
+
return "html";
|
|
574
|
+
}
|
|
575
|
+
}
|
|
576
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
577
|
+
0 && (module.exports = {
|
|
578
|
+
CompileError,
|
|
579
|
+
compile,
|
|
580
|
+
compileMaizzle,
|
|
581
|
+
compileMjml,
|
|
582
|
+
compileReactEmail,
|
|
583
|
+
detectFormat
|
|
584
|
+
});
|
|
585
|
+
//# sourceMappingURL=index.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/compile/errors.ts","../../src/compile/react-email.ts","../../src/compile/mjml.ts","../../src/compile/maizzle.ts","../../src/compile/index.ts"],"sourcesContent":["import type { InputFormat } from \"../types.js\";\r\n\r\n/**\r\n * Unified error class for all email compilation failures.\r\n *\r\n * Replaces the per-format error classes (ReactEmailCompileError,\r\n * MjmlCompileError, MaizzleCompileError) with a single class that\r\n * carries the source format and failure phase.\r\n */\r\nexport class CompileError extends Error {\r\n override name = \"CompileError\";\r\n readonly format: Exclude<InputFormat, \"html\">;\r\n readonly phase: \"validation\" | \"transpile\" | \"execution\" | \"render\" | \"compile\";\r\n\r\n constructor(\r\n message: string,\r\n format: CompileError[\"format\"],\r\n phase: CompileError[\"phase\"],\r\n ) {\r\n super(message);\r\n this.format = format;\r\n this.phase = phase;\r\n }\r\n}\r\n","import { CompileError } from \"./errors.js\";\r\n\r\n/** Maximum source code size: 256KB */\r\nconst MAX_SOURCE_SIZE = 256_000;\r\n\r\n/** Execution timeout: 5 seconds */\r\nconst EXECUTION_TIMEOUT_MS = 5_000;\r\n\r\n/**\r\n * Sandbox strategy for JSX execution.\r\n *\r\n * - `\"vm\"` — `node:vm` with hardened globals. Fast, zero-dependency,\r\n * but NOT a true security boundary (prototype-chain escapes are possible).\r\n * Suitable for CLI / local use where users run their own code.\r\n *\r\n * - `\"isolated-vm\"` (default) — Separate V8 isolate via the `isolated-vm`\r\n * npm package. True heap isolation; escapes require a V8 engine bug.\r\n * Requires `isolated-vm` to be installed (native addon).\r\n *\r\n * - `\"quickjs\"` — Validates code structure in a QuickJS WASM sandbox, then\r\n * executes in `node:vm` for React rendering. Security is equivalent to\r\n * `node:vm` — the QuickJS phase validates import restrictions only.\r\n * No native addons needed, but only supports ES2020 and is slower.\r\n * For true isolation on servers, use `isolated-vm`.\r\n */\r\nexport type SandboxStrategy = \"vm\" | \"isolated-vm\" | \"quickjs\";\r\n\r\nexport interface CompileReactEmailOptions {\r\n /**\r\n * Sandbox strategy to use for executing user JSX code.\r\n * @default \"isolated-vm\"\r\n */\r\n sandbox?: SandboxStrategy;\r\n}\r\n\r\n/**\r\n * Compile a React Email JSX/TSX source string into an HTML email string.\r\n *\r\n * Pipeline:\r\n * 1. Validate input (size, basic checks)\r\n * 2. Transpile JSX/TSX → CommonJS JS using sucrase\r\n * 3. Execute inside a sandbox (configurable strategy)\r\n * 4. Render the exported component to a full HTML email string\r\n *\r\n * Requires peer dependencies: sucrase, react, @react-email/components,\r\n * @react-email/render. Additionally:\r\n * - sandbox \"isolated-vm\" requires `isolated-vm`\r\n * - sandbox \"quickjs\" requires `quickjs-emscripten`\r\n */\r\nexport async function compileReactEmail(\r\n source: string,\r\n options?: CompileReactEmailOptions,\r\n): Promise<string> {\r\n const strategy = options?.sandbox ?? \"isolated-vm\";\r\n\r\n // ── 1. Validate ──────────────────────────────────────────────────────\r\n if (!source || !source.trim()) {\r\n throw new CompileError(\"JSX source must not be empty.\", \"jsx\", \"validation\");\r\n }\r\n\r\n if (source.length > MAX_SOURCE_SIZE) {\r\n throw new CompileError(\r\n `JSX source exceeds ${MAX_SOURCE_SIZE / 1000}KB limit.`,\r\n \"jsx\",\r\n \"validation\",\r\n );\r\n }\r\n\r\n // ── 2. Load peer dependencies ────────────────────────────────────────\r\n let transform: typeof import(\"sucrase\").transform;\r\n let React: typeof import(\"react\");\r\n let ReactEmailComponents: typeof import(\"@react-email/components\");\r\n let render: typeof import(\"@react-email/render\").render;\r\n\r\n try {\r\n ({ transform } = await import(\"sucrase\"));\r\n } catch {\r\n throw new CompileError(\r\n 'JSX compilation requires \"sucrase\". Install it:\\n npm install sucrase',\r\n \"jsx\",\r\n \"transpile\",\r\n );\r\n }\r\n\r\n try {\r\n React = await import(\"react\");\r\n } catch {\r\n throw new CompileError(\r\n 'JSX compilation requires \"react\". Install it:\\n npm install react',\r\n \"jsx\",\r\n \"transpile\",\r\n );\r\n }\r\n\r\n try {\r\n ReactEmailComponents = await import(\"@react-email/components\");\r\n } catch {\r\n throw new CompileError(\r\n 'JSX compilation requires \"@react-email/components\". Install it:\\n npm install @react-email/components',\r\n \"jsx\",\r\n \"transpile\",\r\n );\r\n }\r\n\r\n try {\r\n ({ render } = await import(\"@react-email/render\"));\r\n } catch {\r\n throw new CompileError(\r\n 'JSX compilation requires \"@react-email/render\". Install it:\\n npm install @react-email/render',\r\n \"jsx\",\r\n \"transpile\",\r\n );\r\n }\r\n\r\n // ── 3. Transpile JSX/TSX → CommonJS ──────────────────────────────────\r\n let transpiledCode: string;\r\n try {\r\n const result = transform(source, {\r\n transforms: [\"typescript\", \"jsx\", \"imports\"],\r\n jsxRuntime: \"classic\",\r\n production: true,\r\n });\r\n transpiledCode = result.code;\r\n } catch (err: unknown) {\r\n const message = err instanceof Error ? err.message : \"Unknown transpilation error\";\r\n throw new CompileError(`JSX syntax error: ${message}`, \"jsx\", \"transpile\");\r\n }\r\n\r\n // ── 4. Execute in sandbox ────────────────────────────────────────────\r\n let moduleExports: Record<string, unknown>;\r\n\r\n switch (strategy) {\r\n case \"isolated-vm\":\r\n moduleExports = await executeInIsolatedVm(transpiledCode, React, ReactEmailComponents);\r\n break;\r\n case \"quickjs\":\r\n moduleExports = await executeInQuickJs(transpiledCode, React, ReactEmailComponents);\r\n break;\r\n case \"vm\":\r\n moduleExports = executeInVm(transpiledCode, React, ReactEmailComponents);\r\n break;\r\n default:\r\n throw new CompileError(\r\n `Unknown sandbox strategy: \"${strategy}\". Use \"vm\", \"isolated-vm\", or \"quickjs\".`,\r\n \"jsx\",\r\n \"execution\",\r\n );\r\n }\r\n\r\n // ── 5. Extract component and render ──────────────────────────────────\r\n let Component: unknown = moduleExports.default ?? moduleExports;\r\n\r\n if (typeof Component !== \"function\" && typeof Component === \"object\" && Component !== null) {\r\n const values = Object.values(Component as Record<string, unknown>);\r\n const fn = values.find((v) => typeof v === \"function\");\r\n if (fn) Component = fn;\r\n }\r\n\r\n if (typeof Component !== \"function\") {\r\n throw new CompileError(\r\n 'The JSX source must export a React component function. ' +\r\n 'Use \"export default function Email() { ... }\" or ' +\r\n '\"export function Email() { ... }\".',\r\n \"jsx\",\r\n \"execution\",\r\n );\r\n }\r\n\r\n try {\r\n const element = React.createElement(Component as React.FC);\r\n const html = await render(element);\r\n return html;\r\n } catch (err: unknown) {\r\n const message = err instanceof Error ? err.message : \"Unknown rendering error\";\r\n throw new CompileError(`React rendering error: ${message}`, \"jsx\", \"render\");\r\n }\r\n}\r\n\r\n// ─── Sandbox: node:vm ──────────────────────────────────────────────────────\r\n\r\n/**\r\n * Execute transpiled code in a node:vm context with hardened globals.\r\n *\r\n * NOT a security boundary — see node:vm documentation. Suitable for CLI\r\n * use where the user runs their own code. For server use, prefer\r\n * \"isolated-vm\" or \"quickjs\".\r\n */\r\nfunction executeInVm(\r\n code: string,\r\n React: typeof import(\"react\"),\r\n ReactEmailComponents: typeof import(\"@react-email/components\"),\r\n): Record<string, unknown> {\r\n const { createContext, Script } = require(\"node:vm\") as typeof import(\"node:vm\");\r\n\r\n const ALLOWED_MODULES: Record<string, unknown> = {\r\n react: React,\r\n \"@react-email/components\": ReactEmailComponents,\r\n };\r\n\r\n const moduleExports: Record<string, unknown> = {};\r\n const moduleObj = { exports: moduleExports };\r\n\r\n const mockRequire = (moduleName: string): unknown => {\r\n if (moduleName in ALLOWED_MODULES) {\r\n return ALLOWED_MODULES[moduleName];\r\n }\r\n throw new Error(\r\n `Import of \"${moduleName}\" is not allowed. ` +\r\n `Only \"react\" and \"@react-email/components\" can be imported.`,\r\n );\r\n };\r\n\r\n const sandbox: Record<string, unknown> = {\r\n module: moduleObj,\r\n exports: moduleExports,\r\n require: mockRequire,\r\n React,\r\n Object, Array, String, Number, Boolean,\r\n Map, Set, WeakMap, WeakSet,\r\n JSON, Math, Date, RegExp,\r\n Error, TypeError, RangeError, ReferenceError, SyntaxError, URIError,\r\n Promise, Symbol,\r\n Proxy: undefined, Reflect: undefined,\r\n parseInt, parseFloat, isNaN, isFinite,\r\n encodeURIComponent, decodeURIComponent, encodeURI, decodeURI,\r\n undefined, NaN, Infinity,\r\n console: { log: () => {}, warn: () => {}, error: () => {}, info: () => {}, debug: () => {} },\r\n setTimeout: undefined, setInterval: undefined, setImmediate: undefined, queueMicrotask: undefined,\r\n process: undefined, globalThis: undefined, global: undefined, Buffer: undefined,\r\n __dirname: undefined, __filename: undefined,\r\n };\r\n\r\n const context = createContext(sandbox, {\r\n codeGeneration: { strings: false, wasm: false },\r\n });\r\n\r\n try {\r\n const script = new Script(code, { filename: \"user-email-component.tsx\" });\r\n script.runInContext(context, { timeout: EXECUTION_TIMEOUT_MS, displayErrors: true });\r\n } catch (err: unknown) {\r\n const message = err instanceof Error ? err.message : \"Unknown execution error\";\r\n if (message.includes(\"Script execution timed out\")) {\r\n throw new CompileError(\"JSX execution timed out (possible infinite loop).\", \"jsx\", \"execution\");\r\n }\r\n throw new CompileError(`JSX execution error: ${message}`, \"jsx\", \"execution\");\r\n }\r\n\r\n return moduleObj.exports as Record<string, unknown>;\r\n}\r\n\r\n// ─── Sandbox: isolated-vm ──────────────────────────────────────────────────\r\n\r\n/**\r\n * Execute transpiled code in a separate V8 isolate via `isolated-vm`.\r\n *\r\n * True heap isolation — the user code runs in a distinct V8 isolate with\r\n * no access to the host process. Module requires and globals must be\r\n * explicitly transferred. Escape requires exploiting a V8 engine bug.\r\n *\r\n * Implementation note: React Email components create complex object graphs\r\n * (React elements, component refs) that cannot be serialized across isolate\r\n * boundaries. We use a callback-based approach where the isolate calls back\r\n * into the host for module resolution, and the code runs with a strict\r\n * timeout and memory limit.\r\n */\r\nasync function executeInIsolatedVm(\r\n code: string,\r\n React: typeof import(\"react\"),\r\n ReactEmailComponents: typeof import(\"@react-email/components\"),\r\n): Promise<Record<string, unknown>> {\r\n let ivm: typeof import(\"isolated-vm\");\r\n try {\r\n ivm = await import(\"isolated-vm\");\r\n } catch {\r\n throw new CompileError(\r\n 'Sandbox strategy \"isolated-vm\" requires the \"isolated-vm\" package. Install it:\\n' +\r\n \" npm install isolated-vm\\n\" +\r\n 'Or use sandbox: \"vm\" for a lighter (but less secure) alternative.',\r\n \"jsx\",\r\n \"execution\",\r\n );\r\n }\r\n\r\n const isolate = new ivm.Isolate({ memoryLimit: 128 });\r\n try {\r\n const ivmContext = await isolate.createContext();\r\n const jail = ivmContext.global;\r\n\r\n const ALLOWED_MODULES: Record<string, unknown> = {\r\n react: React,\r\n \"@react-email/components\": ReactEmailComponents,\r\n };\r\n\r\n // Set up host callbacks for module resolution\r\n const hostRequire = new ivm.Reference(function (name: string) {\r\n if (name in ALLOWED_MODULES) {\r\n return ALLOWED_MODULES[name];\r\n }\r\n throw new Error(\r\n `Import of \"${name}\" is not allowed. Only \"react\" and \"@react-email/components\" can be imported.`,\r\n );\r\n });\r\n\r\n await jail.set(\"__hostRequire\", hostRequire);\r\n await jail.set(\"__hostReact\", new ivm.Reference(React));\r\n\r\n // Run code in the isolate with host callbacks for module resolution\r\n const wrapperCode = `\r\n (function() {\r\n const module = { exports: {} };\r\n const exports = module.exports;\r\n\r\n const React = __hostReact.derefInto();\r\n function require(name) {\r\n return __hostRequire.applySync(undefined, [name], { result: { copy: true }, arguments: { copy: true } });\r\n }\r\n\r\n ${code}\r\n\r\n return module.exports;\r\n })()\r\n `;\r\n\r\n try {\r\n const result = await ivmContext.evalClosureSync(wrapperCode, [], {\r\n timeout: EXECUTION_TIMEOUT_MS,\r\n });\r\n\r\n return typeof result === \"object\" && result !== null\r\n ? (result as Record<string, unknown>)\r\n : { default: result };\r\n } catch (err: unknown) {\r\n const message = err instanceof Error ? err.message : \"Unknown execution error\";\r\n if (message.includes(\"timed out\") || message.includes(\"Timeout\")) {\r\n throw new CompileError(\"JSX execution timed out (possible infinite loop).\", \"jsx\", \"execution\");\r\n }\r\n throw new CompileError(`JSX execution error: ${message}`, \"jsx\", \"execution\");\r\n }\r\n } finally {\r\n isolate.dispose();\r\n }\r\n}\r\n\r\n// ─── Sandbox: QuickJS (WASM) ──────────────────────────────────────────────\r\n\r\n/**\r\n * Validate code structure in QuickJS WASM, then execute in `node:vm`.\r\n *\r\n * Two-phase approach:\r\n * 1. Validate that the code doesn't access disallowed modules by running it\r\n * in a QuickJS WASM sandbox with stub implementations.\r\n * 2. Execute in `node:vm` for actual React rendering (React objects can't\r\n * cross the WASM boundary).\r\n *\r\n * **Security note:** The actual execution happens in `node:vm`, so runtime\r\n * security is equivalent to the `\"vm\"` strategy. The QuickJS phase only\r\n * validates import restrictions. For true isolation on servers, use\r\n * `\"isolated-vm\"`.\r\n */\r\nasync function executeInQuickJs(\r\n code: string,\r\n React: typeof import(\"react\"),\r\n ReactEmailComponents: typeof import(\"@react-email/components\"),\r\n): Promise<Record<string, unknown>> {\r\n let getQuickJS: typeof import(\"quickjs-emscripten\").getQuickJS;\r\n try {\r\n ({ getQuickJS } = await import(\"quickjs-emscripten\"));\r\n } catch {\r\n throw new CompileError(\r\n 'Sandbox strategy \"quickjs\" requires the \"quickjs-emscripten\" package. Install it:\\n' +\r\n \" npm install quickjs-emscripten\\n\" +\r\n 'Or use sandbox: \"vm\" for a lighter alternative.',\r\n \"jsx\",\r\n \"execution\",\r\n );\r\n }\r\n\r\n const QuickJS = await getQuickJS();\r\n const vm = QuickJS.newContext();\r\n\r\n try {\r\n // Phase 1: Validate code safety in the WASM sandbox.\r\n // We provide stub implementations of React and the module system so\r\n // the code can execute without errors, but we only care that it\r\n // doesn't try to access anything dangerous.\r\n const validationCode = `\r\n (function() {\r\n var module = { exports: {} };\r\n var exports = module.exports;\r\n var React = { createElement: function() { return {}; } };\r\n function require(name) {\r\n if (name === \"react\" || name === \"@react-email/components\") {\r\n return React;\r\n }\r\n throw new Error('Import of \"' + name + '\" is not allowed.');\r\n }\r\n try {\r\n ${code}\r\n return JSON.stringify({ ok: true });\r\n } catch(e) {\r\n return JSON.stringify({ ok: false, error: e.message || \"Unknown error\" });\r\n }\r\n })()\r\n `;\r\n\r\n const result = vm.evalCode(validationCode);\r\n if (result.error) {\r\n const errorVal = vm.dump(result.error);\r\n result.error.dispose();\r\n throw new CompileError(\r\n `JSX execution error: ${typeof errorVal === \"string\" ? errorVal : \"QuickJS execution failed\"}`,\r\n \"jsx\",\r\n \"execution\",\r\n );\r\n }\r\n\r\n const resultStr = vm.dump(result.value);\r\n result.value.dispose();\r\n\r\n if (typeof resultStr === \"string\") {\r\n const parsed = JSON.parse(resultStr) as { ok: boolean; error?: string };\r\n if (!parsed.ok) {\r\n throw new CompileError(\r\n `JSX execution error: ${parsed.error ?? \"Unknown error\"}`,\r\n \"jsx\",\r\n \"execution\",\r\n );\r\n }\r\n }\r\n\r\n // Phase 2: Code validated as safe — execute in node:vm for actual\r\n // React rendering (React objects can't cross the WASM boundary)\r\n return executeInVm(code, React, ReactEmailComponents);\r\n } finally {\r\n vm.dispose();\r\n }\r\n}\r\n","import { CompileError } from \"./errors.js\";\r\n\r\n/** Maximum MJML source size: 512KB */\r\nconst MAX_SOURCE_SIZE = 512_000;\r\n\r\n/**\r\n * Compile an MJML source string into an HTML email string.\r\n *\r\n * MJML is a declarative markup language with no code execution capability\r\n * (unlike JSX). The mjml library parses XML and generates HTML; there is\r\n * no sandboxing concern.\r\n *\r\n * Requires peer dependency: mjml\r\n */\r\nexport async function compileMjml(source: string): Promise<string> {\r\n // ── 1. Validate ──────────────────────────────────────────────────────\r\n if (!source || !source.trim()) {\r\n throw new CompileError(\"MJML source must not be empty.\", \"mjml\", \"validation\");\r\n }\r\n\r\n if (source.length > MAX_SOURCE_SIZE) {\r\n throw new CompileError(\r\n `MJML source exceeds ${MAX_SOURCE_SIZE / 1000}KB limit.`,\r\n \"mjml\",\r\n \"validation\",\r\n );\r\n }\r\n\r\n if (!/<mjml[\\s>]/i.test(source)) {\r\n throw new CompileError(\r\n \"MJML source must contain a root <mjml> element. \" +\r\n \"Example: <mjml><mj-body><mj-section><mj-column><mj-text>Hello</mj-text></mj-column></mj-section></mj-body></mjml>\",\r\n \"mjml\",\r\n \"validation\",\r\n );\r\n }\r\n\r\n // ── 2. Load peer dependency ──────────────────────────────────────────\r\n type MjmlResult = {\r\n html: string;\r\n errors: Array<{ line: number; message: string; tagName: string; formattedMessage: string }>;\r\n };\r\n let mjml2html: (\r\n input: string,\r\n options?: Record<string, unknown>,\r\n ) => MjmlResult | Promise<MjmlResult>;\r\n\r\n try {\r\n const mjmlModule = await import(\"mjml\");\r\n mjml2html = mjmlModule.default ?? mjmlModule;\r\n } catch {\r\n throw new CompileError(\r\n 'MJML compilation requires \"mjml\". Install it:\\n npm install mjml',\r\n \"mjml\",\r\n \"compile\",\r\n );\r\n }\r\n\r\n // ── 3. Compile ───────────────────────────────────────────────────────\r\n try {\r\n // mjml v5+ returns a Promise; v4 returns synchronously.\r\n // Await handles both cases.\r\n const result = await mjml2html(source, {\r\n validationLevel: \"soft\",\r\n keepComments: false,\r\n });\r\n\r\n if (result.errors && result.errors.length > 0 && !result.html) {\r\n const errorMessages = result.errors\r\n .map((e) => `Line ${e.line}: ${e.message} (${e.tagName})`)\r\n .join(\"; \");\r\n throw new CompileError(\r\n `MJML compilation errors: ${errorMessages}`,\r\n \"mjml\",\r\n \"compile\",\r\n );\r\n }\r\n\r\n if (!result.html) {\r\n throw new CompileError(\"MJML compilation produced empty output.\", \"mjml\", \"compile\");\r\n }\r\n\r\n return result.html;\r\n } catch (err: unknown) {\r\n if (err instanceof CompileError) throw err;\r\n const message = err instanceof Error ? err.message : \"Unknown MJML compilation error\";\r\n throw new CompileError(`MJML compilation failed: ${message}`, \"mjml\", \"compile\");\r\n }\r\n}\r\n","import { CompileError } from \"./errors.js\";\r\n\r\n/** Maximum Maizzle source size: 512KB */\r\nconst MAX_SOURCE_SIZE = 512_000;\r\n\r\n/** Compilation timeout: 15 seconds */\r\nconst COMPILE_TIMEOUT_MS = 15_000;\r\n\r\n/**\r\n * PostHTML directives that perform file-system reads or network fetches.\r\n *\r\n * Maizzle's PostHTML pipeline resolves these at compile time: a template\r\n * like `<extends src=\"/etc/passwd\">` causes the server to read that path\r\n * and include the content in rendered output (server-side file read). We\r\n * reject any input containing these directives rather than stripping them,\r\n * because stripping is error-prone with nested or malformed markup.\r\n *\r\n * Affected plugins: posthtml-extend, posthtml-fetch, posthtml-components,\r\n * posthtml-include, posthtml-modules.\r\n */\r\nconst DANGEROUS_DIRECTIVE_RE =\r\n /<\\s*(?:extends|component|fetch|include|module|slot|fill|raw|block|yield)\\b/i;\r\n\r\n/**\r\n * Compile a Maizzle template string into an HTML email string.\r\n *\r\n * Pipeline:\r\n * 1. Validate input (size, basic structure check)\r\n * 2. Reject inputs containing PostHTML file-access directives\r\n * 3. Compile using @maizzle/framework\r\n *\r\n * Security:\r\n * - PostHTML file-system directives are rejected at validation time\r\n * to prevent server-side file reads and SSRF.\r\n * - Template expressions ({{ expr }}) are evaluated by posthtml-expressions\r\n * with empty `locals`, so unknown identifiers like `process` or `require`\r\n * return the literal string '{local}' rather than accessing Node.js globals.\r\n * - A hard timeout prevents pathological PostCSS/Tailwind inputs from\r\n * hanging indefinitely.\r\n *\r\n * Requires peer dependency: @maizzle/framework\r\n */\r\nexport async function compileMaizzle(source: string): Promise<string> {\r\n // ── 1. Validate ──────────────────────────────────────────────────────\r\n if (!source || !source.trim()) {\r\n throw new CompileError(\"Maizzle source must not be empty.\", \"maizzle\", \"validation\");\r\n }\r\n\r\n if (source.length > MAX_SOURCE_SIZE) {\r\n throw new CompileError(\r\n `Maizzle source exceeds ${MAX_SOURCE_SIZE / 1000}KB limit.`,\r\n \"maizzle\",\r\n \"validation\",\r\n );\r\n }\r\n\r\n // ── 2. Block file-system and network PostHTML directives ─────────────\r\n if (DANGEROUS_DIRECTIVE_RE.test(source)) {\r\n throw new CompileError(\r\n \"Maizzle templates may not use <extends>, <component>, <fetch>, <include>, \" +\r\n \"<module>, <slot>, <fill>, <raw>, <block>, or <yield> directives. These directives \" +\r\n \"access the server file system at compile time. Use inline HTML and Tailwind utility classes instead.\",\r\n \"maizzle\",\r\n \"validation\",\r\n );\r\n }\r\n\r\n // ── 3. Load peer dependency ──────────────────────────────────────────\r\n let maizzleRender: (\r\n input: string,\r\n options: Record<string, unknown>,\r\n ) => Promise<{ html: string }>;\r\n\r\n try {\r\n const maizzle = await import(\"@maizzle/framework\");\r\n maizzleRender = maizzle.render;\r\n } catch {\r\n throw new CompileError(\r\n 'Maizzle compilation requires \"@maizzle/framework\". Install it:\\n npm install @maizzle/framework',\r\n \"maizzle\",\r\n \"compile\",\r\n );\r\n }\r\n\r\n // ── 4. Compile with timeout ──────────────────────────────────────────\r\n const compilePromise = maizzleRender(source, {\r\n css: {\r\n inline: {\r\n removeInlinedSelectors: true,\r\n applyWidthAttributes: true,\r\n applyHeightAttributes: true,\r\n },\r\n shorthand: true,\r\n sixHex: true,\r\n },\r\n locals: {},\r\n });\r\n\r\n const timeoutPromise = new Promise<never>((_, reject) => {\r\n const t = setTimeout(() => {\r\n reject(\r\n new CompileError(\r\n `Maizzle compilation timed out after ${COMPILE_TIMEOUT_MS / 1000}s.`,\r\n \"maizzle\",\r\n \"compile\",\r\n ),\r\n );\r\n }, COMPILE_TIMEOUT_MS);\r\n if (typeof t.unref === \"function\") t.unref();\r\n });\r\n\r\n try {\r\n const { html } = await Promise.race([compilePromise, timeoutPromise]);\r\n\r\n if (!html) {\r\n throw new CompileError(\"Maizzle compilation produced empty output.\", \"maizzle\", \"compile\");\r\n }\r\n\r\n return html;\r\n } catch (err: unknown) {\r\n if (err instanceof CompileError) throw err;\r\n const message = err instanceof Error ? err.message : \"Unknown Maizzle compilation error\";\r\n throw new CompileError(`Maizzle compilation failed: ${message}`, \"maizzle\", \"compile\");\r\n }\r\n}\r\n","import { extname } from \"node:path\";\r\nimport type { InputFormat } from \"../types.js\";\r\n\r\nexport { CompileError } from \"./errors.js\";\r\nexport { compileReactEmail } from \"./react-email.js\";\r\nexport type { SandboxStrategy, CompileReactEmailOptions } from \"./react-email.js\";\r\nexport { compileMjml } from \"./mjml.js\";\r\nexport { compileMaizzle } from \"./maizzle.js\";\r\n\r\n/**\r\n * Compile source to HTML based on format.\r\n * Returns the HTML unchanged if format is \"html\".\r\n * Lazily imports per-format compilers to avoid loading unnecessary deps.\r\n */\r\nexport async function compile(\r\n source: string,\r\n format: InputFormat,\r\n filePath?: string,\r\n): Promise<string> {\r\n switch (format) {\r\n case \"html\":\r\n return source;\r\n\r\n case \"jsx\": {\r\n const { compileReactEmail } = await import(\"./react-email.js\");\r\n return compileReactEmail(source);\r\n }\r\n\r\n case \"mjml\": {\r\n const { compileMjml } = await import(\"./mjml.js\");\r\n return compileMjml(source);\r\n }\r\n\r\n case \"maizzle\": {\r\n const { compileMaizzle } = await import(\"./maizzle.js\");\r\n return compileMaizzle(source);\r\n }\r\n\r\n default:\r\n throw new Error(`Unknown format: \"${format}\". Use html, jsx, mjml, or maizzle.`);\r\n }\r\n}\r\n\r\n/**\r\n * Auto-detect input format from file extension.\r\n */\r\nexport function detectFormat(filePath: string): InputFormat {\r\n const ext = extname(filePath).toLowerCase();\r\n\r\n switch (ext) {\r\n case \".tsx\":\r\n case \".jsx\":\r\n return \"jsx\";\r\n case \".mjml\":\r\n return \"mjml\";\r\n case \".html\":\r\n case \".htm\":\r\n return \"html\";\r\n default:\r\n return \"html\";\r\n }\r\n}\r\n"],"mappings":"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","names":["MAX_SOURCE_SIZE","MAX_SOURCE_SIZE","compileReactEmail","compileMjml","compileMaizzle"]}
|