react-email 4.1.0-canary.6 → 4.1.0-canary.8

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 (282) hide show
  1. package/CHANGELOG.md +24 -0
  2. package/dist/{cli/index.mjs → index.js} +437 -429
  3. package/package.json +10 -45
  4. package/src/commands/build.ts +306 -0
  5. package/src/commands/dev.ts +27 -0
  6. package/src/commands/export.ts +204 -0
  7. package/src/commands/start.ts +38 -0
  8. package/src/index.ts +55 -0
  9. package/src/utils/__snapshots__/tree.spec.ts.snap +27 -0
  10. package/src/utils/esbuild/renderring-utilities-exporter.ts +1 -1
  11. package/src/utils/get-emails-directory-metadata.spec.ts +1 -1
  12. package/src/utils/get-preview-server-location.ts +32 -0
  13. package/src/utils/index.ts +2 -6
  14. package/src/utils/packageJson.ts +4 -0
  15. package/src/utils/preview/get-env-variables-for-preview-app.ts +14 -0
  16. package/src/utils/preview/hot-reloading/create-dependency-graph.spec.ts +281 -0
  17. package/src/utils/preview/hot-reloading/create-dependency-graph.ts +321 -0
  18. package/src/utils/preview/hot-reloading/get-imported-modules.spec.ts +151 -0
  19. package/src/utils/preview/hot-reloading/get-imported-modules.ts +49 -0
  20. package/src/utils/preview/hot-reloading/resolve-path-aliases.spec.ts +11 -0
  21. package/src/utils/preview/hot-reloading/resolve-path-aliases.ts +32 -0
  22. package/src/utils/preview/hot-reloading/setup-hot-reloading.ts +121 -0
  23. package/src/utils/preview/hot-reloading/test/tsconfig.json +8 -0
  24. package/src/utils/preview/index.ts +2 -0
  25. package/src/utils/preview/serve-static-file.ts +51 -0
  26. package/src/utils/preview/start-dev-server.ts +234 -0
  27. package/src/utils/tree.spec.ts +5 -0
  28. package/src/utils/tree.ts +76 -0
  29. package/src/utils/types/hot-reload-change.ts +1 -1
  30. package/src/utils/types/hot-reload-event.ts +1 -1
  31. package/tsconfig.json +4 -10
  32. package/dist/preview/.next/BUILD_ID +0 -1
  33. package/dist/preview/.next/app-build-manifest.json +0 -44
  34. package/dist/preview/.next/app-path-routes-manifest.json +0 -6
  35. package/dist/preview/.next/build-manifest.json +0 -33
  36. package/dist/preview/.next/diagnostics/build-diagnostics.json +0 -6
  37. package/dist/preview/.next/diagnostics/framework.json +0 -1
  38. package/dist/preview/.next/export-marker.json +0 -6
  39. package/dist/preview/.next/images-manifest.json +0 -57
  40. package/dist/preview/.next/next-minimal-server.js.nft.json +0 -1
  41. package/dist/preview/.next/next-server.js.nft.json +0 -1
  42. package/dist/preview/.next/package.json +0 -1
  43. package/dist/preview/.next/prerender-manifest.json +0 -41
  44. package/dist/preview/.next/react-loadable-manifest.json +0 -1
  45. package/dist/preview/.next/required-server-files.json +0 -311
  46. package/dist/preview/.next/routes-manifest.json +0 -64
  47. package/dist/preview/.next/server/app/_not-found/page.js +0 -1
  48. package/dist/preview/.next/server/app/_not-found/page.js.nft.json +0 -1
  49. package/dist/preview/.next/server/app/_not-found/page_client-reference-manifest.js +0 -1
  50. package/dist/preview/.next/server/app/favicon.ico/route.js +0 -1
  51. package/dist/preview/.next/server/app/favicon.ico/route.js.nft.json +0 -1
  52. package/dist/preview/.next/server/app/favicon.ico.body +0 -0
  53. package/dist/preview/.next/server/app/favicon.ico.meta +0 -1
  54. package/dist/preview/.next/server/app/page.js +0 -1
  55. package/dist/preview/.next/server/app/page.js.nft.json +0 -1
  56. package/dist/preview/.next/server/app/page_client-reference-manifest.js +0 -1
  57. package/dist/preview/.next/server/app/preview/[...slug]/page.js +0 -321
  58. package/dist/preview/.next/server/app/preview/[...slug]/page.js.nft.json +0 -1
  59. package/dist/preview/.next/server/app/preview/[...slug]/page_client-reference-manifest.js +0 -1
  60. package/dist/preview/.next/server/app-paths-manifest.json +0 -6
  61. package/dist/preview/.next/server/chunks/134.js +0 -6
  62. package/dist/preview/.next/server/chunks/235.js +0 -15
  63. package/dist/preview/.next/server/chunks/315.js +0 -1
  64. package/dist/preview/.next/server/chunks/343.js +0 -20
  65. package/dist/preview/.next/server/chunks/428.js +0 -14
  66. package/dist/preview/.next/server/chunks/963.js +0 -1
  67. package/dist/preview/.next/server/functions-config-manifest.json +0 -4
  68. package/dist/preview/.next/server/interception-route-rewrite-manifest.js +0 -1
  69. package/dist/preview/.next/server/middleware-build-manifest.js +0 -1
  70. package/dist/preview/.next/server/middleware-manifest.json +0 -6
  71. package/dist/preview/.next/server/middleware-react-loadable-manifest.js +0 -1
  72. package/dist/preview/.next/server/next-font-manifest.js +0 -1
  73. package/dist/preview/.next/server/next-font-manifest.json +0 -1
  74. package/dist/preview/.next/server/pages/500.html +0 -1
  75. package/dist/preview/.next/server/pages/_app.js +0 -1
  76. package/dist/preview/.next/server/pages/_app.js.nft.json +0 -1
  77. package/dist/preview/.next/server/pages/_document.js +0 -1
  78. package/dist/preview/.next/server/pages/_document.js.nft.json +0 -1
  79. package/dist/preview/.next/server/pages/_error.js +0 -1
  80. package/dist/preview/.next/server/pages/_error.js.nft.json +0 -1
  81. package/dist/preview/.next/server/pages-manifest.json +0 -5
  82. package/dist/preview/.next/server/server-reference-manifest.js +0 -1
  83. package/dist/preview/.next/server/server-reference-manifest.json +0 -1
  84. package/dist/preview/.next/server/webpack-runtime.js +0 -1
  85. package/dist/preview/.next/static/EYH0WN4--LLC3GZrZIVN8/_buildManifest.js +0 -1
  86. package/dist/preview/.next/static/EYH0WN4--LLC3GZrZIVN8/_ssgManifest.js +0 -1
  87. package/dist/preview/.next/static/chunks/107-3043079e7cb8bcae.js +0 -1
  88. package/dist/preview/.next/static/chunks/293-297b1eb2241f9a70.js +0 -1
  89. package/dist/preview/.next/static/chunks/3bd82e28-cda2c00a924937c5.js +0 -1
  90. package/dist/preview/.next/static/chunks/45-1021fac82f766268.js +0 -1
  91. package/dist/preview/.next/static/chunks/484-a7b30a88a7939680.js +0 -1
  92. package/dist/preview/.next/static/chunks/589-817d8691661d370e.js +0 -1
  93. package/dist/preview/.next/static/chunks/902-c34acb56733e0ce1.js +0 -1
  94. package/dist/preview/.next/static/chunks/app/_not-found/page-4cbc7dce3ad33336.js +0 -1
  95. package/dist/preview/.next/static/chunks/app/layout-46a09d953364e102.js +0 -1
  96. package/dist/preview/.next/static/chunks/app/page-65fd67d48528e2ba.js +0 -1
  97. package/dist/preview/.next/static/chunks/app/preview/[...slug]/page-5e69ffe7506383a0.js +0 -1
  98. package/dist/preview/.next/static/chunks/f33a14d2-ec7c5f0b91818561.js +0 -6
  99. package/dist/preview/.next/static/chunks/framework-b887e9fc751a9906.js +0 -1
  100. package/dist/preview/.next/static/chunks/main-9a03e7ba8acb1900.js +0 -1
  101. package/dist/preview/.next/static/chunks/main-app-dbd8e1ec12eabb66.js +0 -1
  102. package/dist/preview/.next/static/chunks/pages/_app-542a93a5a214e1c0.js +0 -1
  103. package/dist/preview/.next/static/chunks/pages/_error-d5fe1b1612642f76.js +0 -1
  104. package/dist/preview/.next/static/chunks/polyfills-42372ed130431b0a.js +0 -1
  105. package/dist/preview/.next/static/chunks/webpack-31c45daa2bd82a7b.js +0 -1
  106. package/dist/preview/.next/static/css/6f42d128f111d7fa.css +0 -3
  107. package/dist/preview/.next/static/media/05613964ce6c782e-s.p.otf +0 -0
  108. package/dist/preview/.next/static/media/11c6126b9369e85e-s.p.otf +0 -0
  109. package/dist/preview/.next/static/media/26a46d62cd723877-s.woff2 +0 -0
  110. package/dist/preview/.next/static/media/26cb97734d8cb717-s.p.otf +0 -0
  111. package/dist/preview/.next/static/media/55c55f0601d81cf3-s.woff2 +0 -0
  112. package/dist/preview/.next/static/media/581909926a08bbc8-s.woff2 +0 -0
  113. package/dist/preview/.next/static/media/6d93bde91c0c2823-s.woff2 +0 -0
  114. package/dist/preview/.next/static/media/97e0cb1ae144a2a9-s.woff2 +0 -0
  115. package/dist/preview/.next/static/media/a34f9d1faa5f3315-s.p.woff2 +0 -0
  116. package/dist/preview/.next/static/media/bb6462617151f6b7-s.p.otf +0 -0
  117. package/dist/preview/.next/static/media/cf6daef822ab0142-s.p.otf +0 -0
  118. package/dist/preview/.next/static/media/df0a9ae256c0569c-s.woff2 +0 -0
  119. package/dist/preview/.next/static/media/e4051546b3043204-s.p.otf +0 -0
  120. package/dist/preview/.next/static/media/logo.2ce2a759.png +0 -0
  121. package/dist/preview/.next/trace +0 -28
  122. package/dist/preview/.next/types/app/layout.ts +0 -84
  123. package/dist/preview/.next/types/app/page.ts +0 -84
  124. package/dist/preview/.next/types/app/preview/[...slug]/page.ts +0 -84
  125. package/dist/preview/.next/types/cache-life.d.ts +0 -141
  126. package/dist/preview/.next/types/package.json +0 -1
  127. package/module-punycode.d.ts +0 -3
  128. package/next-env.d.ts +0 -5
  129. package/next.config.js +0 -22
  130. package/postcss.config.js +0 -8
  131. package/scripts/build-preview-server.mjs +0 -33
  132. package/scripts/fill-caniemail-data.mjs +0 -36
  133. package/src/actions/email-validation/caniemail-data.ts +0 -85993
  134. package/src/actions/email-validation/check-compatibility.ts +0 -333
  135. package/src/actions/email-validation/check-images.spec.tsx +0 -100
  136. package/src/actions/email-validation/check-images.ts +0 -160
  137. package/src/actions/email-validation/check-links.spec.tsx +0 -113
  138. package/src/actions/email-validation/check-links.ts +0 -113
  139. package/src/actions/email-validation/get-code-location-from-ast-element.ts +0 -18
  140. package/src/actions/email-validation/quick-fetch.ts +0 -14
  141. package/src/actions/get-email-path-from-slug.ts +0 -32
  142. package/src/actions/get-emails-directory-metadata-action.ts +0 -19
  143. package/src/actions/render-email-by-path.tsx +0 -121
  144. package/src/animated-icons-data/help.json +0 -1082
  145. package/src/animated-icons-data/link.json +0 -1309
  146. package/src/animated-icons-data/load.json +0 -443
  147. package/src/animated-icons-data/mail.json +0 -1320
  148. package/src/app/env.ts +0 -15
  149. package/src/app/favicon.ico +0 -0
  150. package/src/app/fonts/SFMono/SFMonoBold.otf +0 -0
  151. package/src/app/fonts/SFMono/SFMonoBoldItalic.otf +0 -0
  152. package/src/app/fonts/SFMono/SFMonoHeavy.otf +0 -0
  153. package/src/app/fonts/SFMono/SFMonoHeavyItalic.otf +0 -0
  154. package/src/app/fonts/SFMono/SFMonoLight.otf +0 -0
  155. package/src/app/fonts/SFMono/SFMonoLightItalic.otf +0 -0
  156. package/src/app/fonts/SFMono/SFMonoMedium.otf +0 -0
  157. package/src/app/fonts/SFMono/SFMonoMediumItalic.otf +0 -0
  158. package/src/app/fonts/SFMono/SFMonoRegular.otf +0 -0
  159. package/src/app/fonts/SFMono/SFMonoRegularItalic.otf +0 -0
  160. package/src/app/fonts/SFMono/SFMonoSemibold.otf +0 -0
  161. package/src/app/fonts/SFMono/SFMonoSemiboldItalic.otf +0 -0
  162. package/src/app/fonts.ts +0 -39
  163. package/src/app/globals.css +0 -15
  164. package/src/app/layout.tsx +0 -45
  165. package/src/app/logo.png +0 -0
  166. package/src/app/page.tsx +0 -46
  167. package/src/app/preview/[...slug]/page.tsx +0 -157
  168. package/src/app/preview/[...slug]/preview.tsx +0 -234
  169. package/src/app/preview/[...slug]/rendering-error.tsx +0 -40
  170. package/src/components/button.tsx +0 -101
  171. package/src/components/code-container.tsx +0 -164
  172. package/src/components/code-snippet.tsx +0 -9
  173. package/src/components/code.tsx +0 -184
  174. package/src/components/heading.tsx +0 -113
  175. package/src/components/icons/icon-arrow-down.tsx +0 -16
  176. package/src/components/icons/icon-base.tsx +0 -26
  177. package/src/components/icons/icon-bug.tsx +0 -19
  178. package/src/components/icons/icon-button.tsx +0 -23
  179. package/src/components/icons/icon-check.tsx +0 -19
  180. package/src/components/icons/icon-clipboard.tsx +0 -40
  181. package/src/components/icons/icon-download.tsx +0 -19
  182. package/src/components/icons/icon-email.tsx +0 -18
  183. package/src/components/icons/icon-file.tsx +0 -19
  184. package/src/components/icons/icon-folder-open.tsx +0 -19
  185. package/src/components/icons/icon-folder.tsx +0 -18
  186. package/src/components/icons/icon-hide-sidebar.tsx +0 -23
  187. package/src/components/icons/icon-image.tsx +0 -19
  188. package/src/components/icons/icon-info.tsx +0 -18
  189. package/src/components/icons/icon-link.tsx +0 -14
  190. package/src/components/icons/icon-monitor.tsx +0 -19
  191. package/src/components/icons/icon-moon.tsx +0 -16
  192. package/src/components/icons/icon-phone.tsx +0 -26
  193. package/src/components/icons/icon-reload.tsx +0 -18
  194. package/src/components/icons/icon-source.tsx +0 -19
  195. package/src/components/icons/icon-stamp.tsx +0 -14
  196. package/src/components/icons/icon-sun.tsx +0 -16
  197. package/src/components/icons/icon-warning.tsx +0 -31
  198. package/src/components/index.ts +0 -7
  199. package/src/components/logo.tsx +0 -63
  200. package/src/components/resizable-wrapper.tsx +0 -173
  201. package/src/components/send.tsx +0 -134
  202. package/src/components/shell.tsx +0 -92
  203. package/src/components/sidebar/file-tree-directory-children.tsx +0 -139
  204. package/src/components/sidebar/file-tree-directory.tsx +0 -92
  205. package/src/components/sidebar/file-tree.tsx +0 -31
  206. package/src/components/sidebar/index.ts +0 -1
  207. package/src/components/sidebar/sidebar.tsx +0 -43
  208. package/src/components/text.tsx +0 -99
  209. package/src/components/toolbar/checking-results.tsx +0 -150
  210. package/src/components/toolbar/code-preview-line-link.tsx +0 -40
  211. package/src/components/toolbar/compatibility.tsx +0 -113
  212. package/src/components/toolbar/linter.tsx +0 -278
  213. package/src/components/toolbar/results.tsx +0 -51
  214. package/src/components/toolbar/spam-assassin.tsx +0 -155
  215. package/src/components/toolbar/toolbar-button.tsx +0 -52
  216. package/src/components/toolbar/use-cached-state.ts +0 -33
  217. package/src/components/toolbar.tsx +0 -349
  218. package/src/components/tooltip-content.tsx +0 -31
  219. package/src/components/tooltip.tsx +0 -19
  220. package/src/components/topbar/active-view-toggle-group.tsx +0 -86
  221. package/src/components/topbar/theme-toggle-group.tsx +0 -87
  222. package/src/components/topbar/view-size-controls.tsx +0 -247
  223. package/src/components/topbar.tsx +0 -59
  224. package/src/contexts/emails.tsx +0 -59
  225. package/src/contexts/fragment-identifier.tsx +0 -48
  226. package/src/contexts/preview.tsx +0 -79
  227. package/src/hooks/use-clamped-state.ts +0 -24
  228. package/src/hooks/use-email-rendering-result.ts +0 -58
  229. package/src/hooks/use-fragment-identifier.ts +0 -14
  230. package/src/hooks/use-hot-reload.ts +0 -31
  231. package/src/hooks/use-icon-animation.ts +0 -41
  232. package/src/hooks/use-iframe-color-scheme.ts +0 -35
  233. package/src/hooks/use-rendering-metadata.ts +0 -36
  234. package/src/utils/__snapshots__/get-email-component.spec.ts.snap +0 -3
  235. package/src/utils/caniemail/all-css-properties.ts +0 -358
  236. package/src/utils/caniemail/ast/__snapshots__/get-object-variables.spec.ts.snap +0 -74
  237. package/src/utils/caniemail/ast/__snapshots__/get-used-style-properties.spec.ts.snap +0 -24
  238. package/src/utils/caniemail/ast/get-object-variables.spec.ts +0 -19
  239. package/src/utils/caniemail/ast/get-object-variables.ts +0 -61
  240. package/src/utils/caniemail/ast/get-used-style-properties.spec.ts +0 -23
  241. package/src/utils/caniemail/ast/get-used-style-properties.ts +0 -91
  242. package/src/utils/caniemail/get-compatibility-stats-for-entry.ts +0 -118
  243. package/src/utils/caniemail/get-css-functions.ts +0 -25
  244. package/src/utils/caniemail/get-css-property-names.ts +0 -32
  245. package/src/utils/caniemail/get-css-property-with-value.ts +0 -14
  246. package/src/utils/caniemail/get-css-unit.ts +0 -3
  247. package/src/utils/caniemail/get-element-attributes.ts +0 -7
  248. package/src/utils/caniemail/get-element-names.ts +0 -20
  249. package/src/utils/caniemail/tailwind/generate-tailwind-rules.ts +0 -30
  250. package/src/utils/caniemail/tailwind/get-tailwind-config.ts +0 -187
  251. package/src/utils/caniemail/tailwind/get-tailwind-metadata.spec.ts +0 -25
  252. package/src/utils/caniemail/tailwind/get-tailwind-metadata.ts +0 -45
  253. package/src/utils/caniemail/tailwind/setup-tailwind-context.ts +0 -15
  254. package/src/utils/cn.ts +0 -6
  255. package/src/utils/constants.ts +0 -6
  256. package/src/utils/contains-email-template.spec.ts +0 -107
  257. package/src/utils/contains-email-template.ts +0 -33
  258. package/src/utils/copy-text-to-clipboard.ts +0 -7
  259. package/src/utils/get-email-component.spec.ts +0 -41
  260. package/src/utils/get-email-component.ts +0 -134
  261. package/src/utils/get-line-and-column-from-offset.spec.ts +0 -11
  262. package/src/utils/get-line-and-column-from-offset.ts +0 -11
  263. package/src/utils/improve-error-with-sourcemap.ts +0 -86
  264. package/src/utils/js-email-detection.spec.ts +0 -24
  265. package/src/utils/language-map.ts +0 -7
  266. package/src/utils/linting.ts +0 -60
  267. package/src/utils/load-stream.ts +0 -15
  268. package/src/utils/result.ts +0 -49
  269. package/src/utils/run-bundled-code.ts +0 -64
  270. package/src/utils/sanitize.ts +0 -6
  271. package/src/utils/static-node-modules-for-vm.ts +0 -93
  272. package/src/utils/testing/js-email-export-default.js +0 -17
  273. package/src/utils/testing/js-email-test.js +0 -18
  274. package/src/utils/testing/mdx-email-test.js +0 -128
  275. package/src/utils/testing/request-response-email.tsx +0 -9
  276. package/src/utils/types/as.ts +0 -26
  277. package/src/utils/types/email-template.ts +0 -8
  278. package/src/utils/types/error-object.ts +0 -11
  279. package/src/utils/unreachable.ts +0 -8
  280. package/tailwind-internals.d.ts +0 -133
  281. package/tailwind.config.ts +0 -99
  282. /package/src/{components/toolbar/results-table.tsx → utils/preview/hot-reloading/test/some-file.ts} +0 -0
@@ -1,113 +0,0 @@
1
- 'use server';
2
-
3
- import type { IncomingMessage } from 'node:http';
4
- import { parse } from 'node-html-parser';
5
- import {
6
- type CodeLocation,
7
- getCodeLocationFromAstElement,
8
- } from './get-code-location-from-ast-element';
9
- import { quickFetch } from './quick-fetch';
10
-
11
- export type LinkCheck = { passed: boolean } & (
12
- | {
13
- type: 'fetch_attempt';
14
- metadata: {
15
- fetchStatusCode: number | undefined;
16
- };
17
- }
18
- | {
19
- type: 'syntax';
20
- }
21
- | {
22
- type: 'security';
23
- }
24
- );
25
-
26
- export interface LinkCheckingResult {
27
- status: 'success' | 'warning' | 'error';
28
- link: string;
29
- codeLocation: CodeLocation;
30
- checks: LinkCheck[];
31
- }
32
-
33
- export const checkLinks = async (code: string) => {
34
- const ast = parse(code);
35
-
36
- const readableStream = new ReadableStream<LinkCheckingResult>({
37
- async start(controller) {
38
- const anchors = ast.querySelectorAll('a');
39
- for await (const anchor of anchors) {
40
- const link = anchor.attributes.href;
41
- if (!link) continue;
42
- if (link.startsWith('mailto:')) continue;
43
-
44
- const result: LinkCheckingResult = {
45
- link,
46
- codeLocation: getCodeLocationFromAstElement(anchor, code),
47
- status: 'success',
48
- checks: [],
49
- };
50
-
51
- try {
52
- const url = new URL(link);
53
- result.checks.push({
54
- passed: true,
55
- type: 'syntax',
56
- });
57
-
58
- if (link.startsWith('http://')) {
59
- result.checks.push({
60
- passed: false,
61
- type: 'security',
62
- });
63
- result.status = 'warning';
64
- } else {
65
- result.checks.push({
66
- passed: true,
67
- type: 'security',
68
- });
69
- }
70
-
71
- let res: IncomingMessage | undefined = undefined;
72
- try {
73
- res = await quickFetch(url);
74
- const hasSucceeded =
75
- res.statusCode?.toString().startsWith('2') ?? false;
76
- result.checks.push({
77
- type: 'fetch_attempt',
78
- passed: hasSucceeded,
79
- metadata: {
80
- fetchStatusCode: res.statusCode,
81
- },
82
- });
83
- if (!hasSucceeded) {
84
- result.status = res.statusCode?.toString().startsWith('3')
85
- ? 'warning'
86
- : 'error';
87
- }
88
- } catch (exception) {
89
- result.checks.push({
90
- type: 'fetch_attempt',
91
- passed: false,
92
- metadata: {
93
- fetchStatusCode: undefined,
94
- },
95
- });
96
- result.status = 'error';
97
- }
98
- } catch (exception) {
99
- result.checks.push({
100
- passed: false,
101
- type: 'syntax',
102
- });
103
- result.status = 'error';
104
- }
105
-
106
- controller.enqueue(result);
107
- }
108
- controller.close();
109
- },
110
- });
111
-
112
- return readableStream;
113
- };
@@ -1,18 +0,0 @@
1
- import type { HTMLElement } from 'node-html-parser';
2
- import { getLineAndColumnFromOffset } from '../../utils/get-line-and-column-from-offset';
3
-
4
- export interface CodeLocation {
5
- line: number;
6
- column: number;
7
- }
8
-
9
- export const getCodeLocationFromAstElement = (
10
- ast: HTMLElement,
11
- html: string,
12
- ): CodeLocation => {
13
- const [line, column] = getLineAndColumnFromOffset(ast.range[0], html);
14
- return {
15
- line,
16
- column,
17
- };
18
- };
@@ -1,14 +0,0 @@
1
- import http from 'node:http';
2
- import type { IncomingMessage } from 'node:http';
3
- import https from 'node:https';
4
-
5
- export const quickFetch = (url: URL) => {
6
- return new Promise<IncomingMessage>((resolve, reject) => {
7
- const caller = url.protocol === 'https:' ? https : http;
8
- caller
9
- .get(url, (res) => {
10
- resolve(res);
11
- })
12
- .on('error', (error) => reject(error));
13
- });
14
- };
@@ -1,32 +0,0 @@
1
- 'use server';
2
- import fs from 'node:fs';
3
- import path from 'node:path';
4
- import { cache } from 'react';
5
- import { emailsDirectoryAbsolutePath } from '../app/env';
6
-
7
- // eslint-disable-next-line @typescript-eslint/require-await
8
- export const getEmailPathFromSlug = cache(async (slug: string) => {
9
- if (['.tsx', '.jsx', '.ts', '.js'].includes(path.extname(slug)))
10
- return path.join(emailsDirectoryAbsolutePath, slug);
11
-
12
- const pathWithoutExtension = path.join(emailsDirectoryAbsolutePath, slug);
13
-
14
- if (fs.existsSync(`${pathWithoutExtension}.tsx`)) {
15
- return `${pathWithoutExtension}.tsx`;
16
- }
17
- if (fs.existsSync(`${pathWithoutExtension}.jsx`)) {
18
- return `${pathWithoutExtension}.jsx`;
19
- }
20
- if (fs.existsSync(`${pathWithoutExtension}.ts`)) {
21
- return `${pathWithoutExtension}.ts`;
22
- }
23
- if (fs.existsSync(`${pathWithoutExtension}.js`)) {
24
- return `${pathWithoutExtension}.js`;
25
- }
26
-
27
- throw new Error(
28
- `Could not find your email file based on the slug (${slug}) by guessing the file extension. Tried .tsx, .jsx, .ts and .js.
29
-
30
- This is most likely not an issue with the preview server. It most likely is that the email doesn't exist.`,
31
- );
32
- });
@@ -1,19 +0,0 @@
1
- 'use server';
2
-
3
- import type { EmailsDirectory } from '../utils/get-emails-directory-metadata';
4
- import { getEmailsDirectoryMetadata } from '../utils/get-emails-directory-metadata';
5
-
6
- export const getEmailsDirectoryMetadataAction = async (
7
- absolutePathToEmailsDirectory: string,
8
- keepFileExtensions = false,
9
- isSubDirectory = false,
10
-
11
- baseDirectoryPath = absolutePathToEmailsDirectory,
12
- ): Promise<EmailsDirectory | undefined> => {
13
- return getEmailsDirectoryMetadata(
14
- absolutePathToEmailsDirectory,
15
- keepFileExtensions,
16
- isSubDirectory,
17
- baseDirectoryPath,
18
- );
19
- };
@@ -1,121 +0,0 @@
1
- 'use server';
2
- import fs from 'node:fs';
3
- import path from 'node:path';
4
- import chalk from 'chalk';
5
- import logSymbols from 'log-symbols';
6
- import ora, { type Ora } from 'ora';
7
- import { isBuilding, isPreviewDevelopment } from '../app/env';
8
- import { getEmailComponent } from '../utils/get-email-component';
9
- import { improveErrorWithSourceMap } from '../utils/improve-error-with-sourcemap';
10
- import { registerSpinnerAutostopping } from '../utils/register-spinner-autostopping';
11
- import type { ErrorObject } from '../utils/types/error-object';
12
-
13
- export interface RenderedEmailMetadata {
14
- markup: string;
15
- plainText: string;
16
- reactMarkup: string;
17
- }
18
-
19
- export type EmailRenderingResult =
20
- | RenderedEmailMetadata
21
- | {
22
- error: ErrorObject;
23
- };
24
-
25
- const cache = new Map<string, EmailRenderingResult>();
26
-
27
- export const renderEmailByPath = async (
28
- emailPath: string,
29
- invalidatingCache = false,
30
- ): Promise<EmailRenderingResult> => {
31
- if (invalidatingCache) cache.delete(emailPath);
32
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
33
- if (cache.has(emailPath)) return cache.get(emailPath)!;
34
-
35
- const timeBeforeEmailRendered = performance.now();
36
-
37
- const emailFilename = path.basename(emailPath);
38
- let spinner: Ora | undefined;
39
- if (!isBuilding && !isPreviewDevelopment) {
40
- spinner = ora({
41
- text: `Rendering email template ${emailFilename}\n`,
42
- prefixText: ' ',
43
- }).start();
44
- registerSpinnerAutostopping(spinner);
45
- }
46
-
47
- const componentResult = await getEmailComponent(emailPath);
48
-
49
- if ('error' in componentResult) {
50
- spinner?.stopAndPersist({
51
- symbol: logSymbols.error,
52
- text: `Failed while rendering ${emailFilename}`,
53
- });
54
- return { error: componentResult.error };
55
- }
56
-
57
- const {
58
- emailComponent: Email,
59
- createElement,
60
- render,
61
- sourceMapToOriginalFile,
62
- } = componentResult;
63
-
64
- const previewProps = Email.PreviewProps || {};
65
- const EmailComponent = Email as React.FC;
66
- try {
67
- const markup = await render(createElement(EmailComponent, previewProps), {
68
- pretty: true,
69
- });
70
- const plainText = await render(
71
- createElement(EmailComponent, previewProps),
72
- {
73
- plainText: true,
74
- },
75
- );
76
-
77
- const reactMarkup = await fs.promises.readFile(emailPath, 'utf-8');
78
-
79
- const millisecondsToRendered = performance.now() - timeBeforeEmailRendered;
80
- let timeForConsole = `${millisecondsToRendered.toFixed(0)}ms`;
81
- if (millisecondsToRendered <= 450) {
82
- timeForConsole = chalk.green(timeForConsole);
83
- } else if (millisecondsToRendered <= 1000) {
84
- timeForConsole = chalk.yellow(timeForConsole);
85
- } else {
86
- timeForConsole = chalk.red(timeForConsole);
87
- }
88
- spinner?.stopAndPersist({
89
- symbol: logSymbols.success,
90
- text: `Successfully rendered ${emailFilename} in ${timeForConsole}`,
91
- });
92
-
93
- const renderingResult = {
94
- // This ensures that no null byte character ends up in the rendered
95
- // markup making users suspect of any issues. These null byte characters
96
- // only seem to happen with React 18, as it has no similar incident with React 19.
97
- markup: markup.replaceAll('\0', ''),
98
- plainText,
99
- reactMarkup,
100
- };
101
-
102
- cache.set(emailPath, renderingResult);
103
-
104
- return renderingResult;
105
- } catch (exception) {
106
- const error = exception as Error;
107
-
108
- spinner?.stopAndPersist({
109
- symbol: logSymbols.error,
110
- text: `Failed while rendering ${emailFilename}`,
111
- });
112
-
113
- return {
114
- error: improveErrorWithSourceMap(
115
- error,
116
- emailPath,
117
- sourceMapToOriginalFile,
118
- ),
119
- };
120
- }
121
- };