@react-email/preview-server 1.0.0-canary.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.
Files changed (265) hide show
  1. package/.next/BUILD_ID +1 -0
  2. package/.next/app-build-manifest.json +44 -0
  3. package/.next/app-path-routes-manifest.json +6 -0
  4. package/.next/build-manifest.json +33 -0
  5. package/.next/diagnostics/build-diagnostics.json +6 -0
  6. package/.next/diagnostics/framework.json +1 -0
  7. package/.next/export-marker.json +6 -0
  8. package/.next/images-manifest.json +57 -0
  9. package/.next/next-minimal-server.js.nft.json +1 -0
  10. package/.next/next-server.js.nft.json +1 -0
  11. package/.next/package.json +1 -0
  12. package/.next/prerender-manifest.json +41 -0
  13. package/.next/react-loadable-manifest.json +1 -0
  14. package/.next/required-server-files.json +311 -0
  15. package/.next/routes-manifest.json +64 -0
  16. package/.next/server/app/_not-found/page.js +1 -0
  17. package/.next/server/app/_not-found/page.js.nft.json +1 -0
  18. package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -0
  19. package/.next/server/app/favicon.ico/route.js +1 -0
  20. package/.next/server/app/favicon.ico/route.js.nft.json +1 -0
  21. package/.next/server/app/favicon.ico.body +0 -0
  22. package/.next/server/app/favicon.ico.meta +1 -0
  23. package/.next/server/app/page.js +1 -0
  24. package/.next/server/app/page.js.nft.json +1 -0
  25. package/.next/server/app/page_client-reference-manifest.js +1 -0
  26. package/.next/server/app/preview/[...slug]/page.js +321 -0
  27. package/.next/server/app/preview/[...slug]/page.js.nft.json +1 -0
  28. package/.next/server/app/preview/[...slug]/page_client-reference-manifest.js +1 -0
  29. package/.next/server/app-paths-manifest.json +6 -0
  30. package/.next/server/chunks/134.js +6 -0
  31. package/.next/server/chunks/235.js +15 -0
  32. package/.next/server/chunks/278.js +1 -0
  33. package/.next/server/chunks/343.js +20 -0
  34. package/.next/server/chunks/428.js +14 -0
  35. package/.next/server/chunks/963.js +1 -0
  36. package/.next/server/chunks/999.js +1 -0
  37. package/.next/server/functions-config-manifest.json +4 -0
  38. package/.next/server/interception-route-rewrite-manifest.js +1 -0
  39. package/.next/server/middleware-build-manifest.js +1 -0
  40. package/.next/server/middleware-manifest.json +6 -0
  41. package/.next/server/middleware-react-loadable-manifest.js +1 -0
  42. package/.next/server/next-font-manifest.js +1 -0
  43. package/.next/server/next-font-manifest.json +1 -0
  44. package/.next/server/pages/500.html +1 -0
  45. package/.next/server/pages/_app.js +1 -0
  46. package/.next/server/pages/_app.js.nft.json +1 -0
  47. package/.next/server/pages/_document.js +1 -0
  48. package/.next/server/pages/_document.js.nft.json +1 -0
  49. package/.next/server/pages/_error.js +1 -0
  50. package/.next/server/pages/_error.js.nft.json +1 -0
  51. package/.next/server/pages-manifest.json +5 -0
  52. package/.next/server/server-reference-manifest.js +1 -0
  53. package/.next/server/server-reference-manifest.json +1 -0
  54. package/.next/server/webpack-runtime.js +1 -0
  55. package/.next/static/VkyJa9K30jCrKBesOgrQT/_buildManifest.js +1 -0
  56. package/.next/static/VkyJa9K30jCrKBesOgrQT/_ssgManifest.js +1 -0
  57. package/.next/static/chunks/107-3043079e7cb8bcae.js +1 -0
  58. package/.next/static/chunks/293-67391ef0e44ffa4f.js +1 -0
  59. package/.next/static/chunks/3bd82e28-cda2c00a924937c5.js +1 -0
  60. package/.next/static/chunks/45-1021fac82f766268.js +1 -0
  61. package/.next/static/chunks/484-1969fe871b27074e.js +1 -0
  62. package/.next/static/chunks/484-e02de792ff5f9ea5.js +1 -0
  63. package/.next/static/chunks/589-817d8691661d370e.js +1 -0
  64. package/.next/static/chunks/902-c34acb56733e0ce1.js +1 -0
  65. package/.next/static/chunks/app/_not-found/page-4cbc7dce3ad33336.js +1 -0
  66. package/.next/static/chunks/app/layout-74628781c0b7e7bf.js +1 -0
  67. package/.next/static/chunks/app/layout-daeba68330ab58bb.js +1 -0
  68. package/.next/static/chunks/app/page-55cf199b7ca71958.js +1 -0
  69. package/.next/static/chunks/app/preview/[...slug]/page-07dd9a701d0b3e56.js +1 -0
  70. package/.next/static/chunks/app/preview/[...slug]/page-61b0ea70a8d72916.js +1 -0
  71. package/.next/static/chunks/f33a14d2-ec7c5f0b91818561.js +6 -0
  72. package/.next/static/chunks/framework-b887e9fc751a9906.js +1 -0
  73. package/.next/static/chunks/main-9a03e7ba8acb1900.js +1 -0
  74. package/.next/static/chunks/main-app-5bc2d814f500db60.js +1 -0
  75. package/.next/static/chunks/pages/_app-542a93a5a214e1c0.js +1 -0
  76. package/.next/static/chunks/pages/_error-d5fe1b1612642f76.js +1 -0
  77. package/.next/static/chunks/polyfills-42372ed130431b0a.js +1 -0
  78. package/.next/static/chunks/webpack-31c45daa2bd82a7b.js +1 -0
  79. package/.next/static/css/35e8811589f0962b.css +3 -0
  80. package/.next/static/j4oDiQwPKPQgK5jAoiUTk/_buildManifest.js +1 -0
  81. package/.next/static/j4oDiQwPKPQgK5jAoiUTk/_ssgManifest.js +1 -0
  82. package/.next/static/media/05613964ce6c782e-s.p.otf +0 -0
  83. package/.next/static/media/11c6126b9369e85e-s.p.otf +0 -0
  84. package/.next/static/media/26a46d62cd723877-s.woff2 +0 -0
  85. package/.next/static/media/26cb97734d8cb717-s.p.otf +0 -0
  86. package/.next/static/media/55c55f0601d81cf3-s.woff2 +0 -0
  87. package/.next/static/media/581909926a08bbc8-s.woff2 +0 -0
  88. package/.next/static/media/6d93bde91c0c2823-s.woff2 +0 -0
  89. package/.next/static/media/97e0cb1ae144a2a9-s.woff2 +0 -0
  90. package/.next/static/media/a34f9d1faa5f3315-s.p.woff2 +0 -0
  91. package/.next/static/media/bb6462617151f6b7-s.p.otf +0 -0
  92. package/.next/static/media/cf6daef822ab0142-s.p.otf +0 -0
  93. package/.next/static/media/df0a9ae256c0569c-s.woff2 +0 -0
  94. package/.next/static/media/e4051546b3043204-s.p.otf +0 -0
  95. package/.next/static/media/logo.2ce2a759.png +0 -0
  96. package/.next/trace +27 -0
  97. package/.next/types/app/layout.ts +84 -0
  98. package/.next/types/app/page.ts +84 -0
  99. package/.next/types/app/preview/[...slug]/page.ts +84 -0
  100. package/.next/types/cache-life.d.ts +141 -0
  101. package/.next/types/package.json +1 -0
  102. package/_index.js +4 -0
  103. package/license.md +7 -0
  104. package/module-punycode.d.ts +3 -0
  105. package/next-env.d.ts +5 -0
  106. package/next.config.js +22 -0
  107. package/package.json +82 -0
  108. package/postcss.config.js +8 -0
  109. package/scripts/build-preview-server.mjs +29 -0
  110. package/scripts/fill-caniemail-data.mjs +36 -0
  111. package/src/actions/email-validation/caniemail-data.ts +85993 -0
  112. package/src/actions/email-validation/check-compatibility.ts +333 -0
  113. package/src/actions/email-validation/check-images.spec.tsx +100 -0
  114. package/src/actions/email-validation/check-images.ts +160 -0
  115. package/src/actions/email-validation/check-links.spec.tsx +113 -0
  116. package/src/actions/email-validation/check-links.ts +113 -0
  117. package/src/actions/email-validation/get-code-location-from-ast-element.ts +18 -0
  118. package/src/actions/email-validation/quick-fetch.ts +14 -0
  119. package/src/actions/get-email-path-from-slug.ts +32 -0
  120. package/src/actions/get-emails-directory-metadata-action.ts +19 -0
  121. package/src/actions/render-email-by-path.tsx +121 -0
  122. package/src/animated-icons-data/help.json +1082 -0
  123. package/src/animated-icons-data/link.json +1309 -0
  124. package/src/animated-icons-data/load.json +443 -0
  125. package/src/animated-icons-data/mail.json +1320 -0
  126. package/src/app/env.ts +15 -0
  127. package/src/app/favicon.ico +0 -0
  128. package/src/app/fonts/SFMono/SFMonoBold.otf +0 -0
  129. package/src/app/fonts/SFMono/SFMonoBoldItalic.otf +0 -0
  130. package/src/app/fonts/SFMono/SFMonoHeavy.otf +0 -0
  131. package/src/app/fonts/SFMono/SFMonoHeavyItalic.otf +0 -0
  132. package/src/app/fonts/SFMono/SFMonoLight.otf +0 -0
  133. package/src/app/fonts/SFMono/SFMonoLightItalic.otf +0 -0
  134. package/src/app/fonts/SFMono/SFMonoMedium.otf +0 -0
  135. package/src/app/fonts/SFMono/SFMonoMediumItalic.otf +0 -0
  136. package/src/app/fonts/SFMono/SFMonoRegular.otf +0 -0
  137. package/src/app/fonts/SFMono/SFMonoRegularItalic.otf +0 -0
  138. package/src/app/fonts/SFMono/SFMonoSemibold.otf +0 -0
  139. package/src/app/fonts/SFMono/SFMonoSemiboldItalic.otf +0 -0
  140. package/src/app/fonts.ts +39 -0
  141. package/src/app/globals.css +15 -0
  142. package/src/app/layout.tsx +43 -0
  143. package/src/app/logo.png +0 -0
  144. package/src/app/page.tsx +46 -0
  145. package/src/app/preview/[...slug]/page.tsx +157 -0
  146. package/src/app/preview/[...slug]/preview.tsx +216 -0
  147. package/src/app/preview/[...slug]/rendering-error.tsx +40 -0
  148. package/src/components/button.tsx +101 -0
  149. package/src/components/code-container.tsx +164 -0
  150. package/src/components/code-snippet.tsx +9 -0
  151. package/src/components/code.tsx +184 -0
  152. package/src/components/heading.tsx +113 -0
  153. package/src/components/icons/icon-arrow-down.tsx +16 -0
  154. package/src/components/icons/icon-base.tsx +26 -0
  155. package/src/components/icons/icon-bug.tsx +19 -0
  156. package/src/components/icons/icon-button.tsx +23 -0
  157. package/src/components/icons/icon-check.tsx +19 -0
  158. package/src/components/icons/icon-clipboard.tsx +40 -0
  159. package/src/components/icons/icon-download.tsx +19 -0
  160. package/src/components/icons/icon-email.tsx +18 -0
  161. package/src/components/icons/icon-file.tsx +19 -0
  162. package/src/components/icons/icon-folder-open.tsx +19 -0
  163. package/src/components/icons/icon-folder.tsx +18 -0
  164. package/src/components/icons/icon-hide-sidebar.tsx +23 -0
  165. package/src/components/icons/icon-image.tsx +19 -0
  166. package/src/components/icons/icon-info.tsx +18 -0
  167. package/src/components/icons/icon-link.tsx +14 -0
  168. package/src/components/icons/icon-monitor.tsx +19 -0
  169. package/src/components/icons/icon-phone.tsx +26 -0
  170. package/src/components/icons/icon-reload.tsx +18 -0
  171. package/src/components/icons/icon-source.tsx +19 -0
  172. package/src/components/icons/icon-stamp.tsx +14 -0
  173. package/src/components/icons/icon-warning.tsx +31 -0
  174. package/src/components/index.ts +7 -0
  175. package/src/components/logo.tsx +63 -0
  176. package/src/components/resizable-wrapper.tsx +173 -0
  177. package/src/components/send.tsx +134 -0
  178. package/src/components/shell.tsx +92 -0
  179. package/src/components/sidebar/file-tree-directory-children.tsx +139 -0
  180. package/src/components/sidebar/file-tree-directory.tsx +92 -0
  181. package/src/components/sidebar/file-tree.tsx +31 -0
  182. package/src/components/sidebar/index.ts +1 -0
  183. package/src/components/sidebar/sidebar.tsx +43 -0
  184. package/src/components/text.tsx +99 -0
  185. package/src/components/toolbar/checking-results.tsx +150 -0
  186. package/src/components/toolbar/code-preview-line-link.tsx +40 -0
  187. package/src/components/toolbar/compatibility.tsx +113 -0
  188. package/src/components/toolbar/linter.tsx +278 -0
  189. package/src/components/toolbar/results-table.tsx +0 -0
  190. package/src/components/toolbar/results.tsx +51 -0
  191. package/src/components/toolbar/spam-assassin.tsx +155 -0
  192. package/src/components/toolbar/toolbar-button.tsx +52 -0
  193. package/src/components/toolbar/use-cached-state.ts +33 -0
  194. package/src/components/toolbar.tsx +349 -0
  195. package/src/components/tooltip-content.tsx +31 -0
  196. package/src/components/tooltip.tsx +19 -0
  197. package/src/components/topbar/active-view-toggle-group.tsx +86 -0
  198. package/src/components/topbar/view-size-controls.tsx +247 -0
  199. package/src/components/topbar.tsx +59 -0
  200. package/src/contexts/emails.tsx +59 -0
  201. package/src/contexts/fragment-identifier.tsx +48 -0
  202. package/src/contexts/preview.tsx +79 -0
  203. package/src/hooks/use-clamped-state.ts +24 -0
  204. package/src/hooks/use-email-rendering-result.ts +58 -0
  205. package/src/hooks/use-fragment-identifier.ts +14 -0
  206. package/src/hooks/use-hot-reload.ts +31 -0
  207. package/src/hooks/use-icon-animation.ts +41 -0
  208. package/src/hooks/use-rendering-metadata.ts +36 -0
  209. package/src/utils/__snapshots__/get-email-component.spec.ts.snap +3 -0
  210. package/src/utils/caniemail/all-css-properties.ts +358 -0
  211. package/src/utils/caniemail/ast/__snapshots__/get-object-variables.spec.ts.snap +74 -0
  212. package/src/utils/caniemail/ast/__snapshots__/get-used-style-properties.spec.ts.snap +24 -0
  213. package/src/utils/caniemail/ast/get-object-variables.spec.ts +19 -0
  214. package/src/utils/caniemail/ast/get-object-variables.ts +61 -0
  215. package/src/utils/caniemail/ast/get-used-style-properties.spec.ts +23 -0
  216. package/src/utils/caniemail/ast/get-used-style-properties.ts +91 -0
  217. package/src/utils/caniemail/get-compatibility-stats-for-entry.ts +118 -0
  218. package/src/utils/caniemail/get-css-functions.ts +25 -0
  219. package/src/utils/caniemail/get-css-property-names.ts +32 -0
  220. package/src/utils/caniemail/get-css-property-with-value.ts +14 -0
  221. package/src/utils/caniemail/get-css-unit.ts +3 -0
  222. package/src/utils/caniemail/get-element-attributes.ts +7 -0
  223. package/src/utils/caniemail/get-element-names.ts +20 -0
  224. package/src/utils/caniemail/tailwind/generate-tailwind-rules.ts +30 -0
  225. package/src/utils/caniemail/tailwind/get-tailwind-config.ts +187 -0
  226. package/src/utils/caniemail/tailwind/get-tailwind-metadata.spec.ts +25 -0
  227. package/src/utils/caniemail/tailwind/get-tailwind-metadata.ts +45 -0
  228. package/src/utils/caniemail/tailwind/setup-tailwind-context.ts +15 -0
  229. package/src/utils/cn.ts +6 -0
  230. package/src/utils/constants.ts +6 -0
  231. package/src/utils/contains-email-template.spec.ts +124 -0
  232. package/src/utils/contains-email-template.ts +33 -0
  233. package/src/utils/copy-text-to-clipboard.ts +7 -0
  234. package/src/utils/esbuild/escape-string-for-regex.ts +3 -0
  235. package/src/utils/esbuild/renderring-utilities-exporter.ts +63 -0
  236. package/src/utils/get-email-component.spec.ts +41 -0
  237. package/src/utils/get-email-component.ts +134 -0
  238. package/src/utils/get-emails-directory-metadata.spec.ts +82 -0
  239. package/src/utils/get-emails-directory-metadata.ts +141 -0
  240. package/src/utils/get-line-and-column-from-offset.spec.ts +11 -0
  241. package/src/utils/get-line-and-column-from-offset.ts +11 -0
  242. package/src/utils/improve-error-with-sourcemap.ts +85 -0
  243. package/src/utils/index.ts +6 -0
  244. package/src/utils/js-email-detection.spec.ts +24 -0
  245. package/src/utils/language-map.ts +7 -0
  246. package/src/utils/linting.ts +60 -0
  247. package/src/utils/load-stream.ts +15 -0
  248. package/src/utils/register-spinner-autostopping.ts +28 -0
  249. package/src/utils/result.ts +49 -0
  250. package/src/utils/run-bundled-code.ts +64 -0
  251. package/src/utils/sanitize.ts +6 -0
  252. package/src/utils/static-node-modules-for-vm.ts +93 -0
  253. package/src/utils/testing/js-email-export-default.js +17 -0
  254. package/src/utils/testing/js-email-test.js +18 -0
  255. package/src/utils/testing/mdx-email-test.js +128 -0
  256. package/src/utils/testing/request-response-email.tsx +9 -0
  257. package/src/utils/types/as.ts +26 -0
  258. package/src/utils/types/email-template.ts +8 -0
  259. package/src/utils/types/error-object.ts +11 -0
  260. package/src/utils/types/hot-reload-change.ts +13 -0
  261. package/src/utils/unreachable.ts +8 -0
  262. package/tailwind-internals.d.ts +133 -0
  263. package/tailwind.config.ts +99 -0
  264. package/tsconfig.json +45 -0
  265. package/vitest.config.ts +15 -0
@@ -0,0 +1,333 @@
1
+ 'use server';
2
+ import { parse } from '@babel/parser';
3
+ import traverse from '@babel/traverse';
4
+ import {
5
+ convertLocationIntoObject,
6
+ getObjectVariables,
7
+ type SourceLocation,
8
+ } from '../../utils/caniemail/ast/get-object-variables';
9
+ import type { StylePropertyUsage } from '../../utils/caniemail/ast/get-used-style-properties';
10
+ import {
11
+ doesPropertyHaveLocation,
12
+ getUsedStyleProperties,
13
+ } from '../../utils/caniemail/ast/get-used-style-properties';
14
+ import type {
15
+ CompatibilityStats,
16
+ SupportStatus,
17
+ } from '../../utils/caniemail/get-compatibility-stats-for-entry';
18
+ import { getCompatibilityStatsForEntry } from '../../utils/caniemail/get-compatibility-stats-for-entry';
19
+ import { getCssFunctions } from '../../utils/caniemail/get-css-functions';
20
+ import { getCssPropertyNames } from '../../utils/caniemail/get-css-property-names';
21
+ import { getCssPropertyWithValue } from '../../utils/caniemail/get-css-property-with-value';
22
+ import { getCssUnit } from '../../utils/caniemail/get-css-unit';
23
+ import { getElementAttributes } from '../../utils/caniemail/get-element-attributes';
24
+ import { getElementNames } from '../../utils/caniemail/get-element-names';
25
+ import { supportEntries } from './caniemail-data';
26
+
27
+ export interface CompatibilityCheckingResult {
28
+ location: SourceLocation;
29
+ source: string;
30
+ entry: SupportEntry;
31
+ status: SupportStatus;
32
+ statsPerEmailClient: CompatibilityStats['perEmailClient'];
33
+ }
34
+
35
+ export type EmailClient =
36
+ | 'gmail'
37
+ | 'outlook'
38
+ | 'yahoo'
39
+ | 'apple-mail'
40
+ | 'aol'
41
+ | 'thunderbird'
42
+ | 'microsoft'
43
+ | 'samsung-email'
44
+ | 'sfr'
45
+ | 'orange'
46
+ | 'protonmail'
47
+ | 'hey'
48
+ | 'mail-ru'
49
+ | 'fastmail'
50
+ | 'laposte'
51
+ | 't-online-de'
52
+ | 'free-fr'
53
+ | 'gmx'
54
+ | 'web-de'
55
+ | 'ionos-1and1'
56
+ | 'rainloop'
57
+ | 'wp-pl';
58
+
59
+ export type Platform =
60
+ | 'desktop-app'
61
+ | 'desktop-webmail'
62
+ | 'mobile-webmail'
63
+ | 'webmail'
64
+ | 'ios'
65
+ | 'android'
66
+ | 'windows'
67
+ | 'macos'
68
+ | 'windows-mail'
69
+ | 'outlook-com';
70
+
71
+ export type SupportEntryCategroy = 'html' | 'css' | 'image' | 'others';
72
+
73
+ export interface SupportEntry {
74
+ slug: string;
75
+ title: string;
76
+ description: string | null;
77
+ url: string;
78
+ category: SupportEntryCategroy;
79
+ tags: string[];
80
+ keywords: string | null;
81
+ last_test_date: string;
82
+ test_url: string;
83
+ test_results_url: string | null;
84
+ stats: Partial<
85
+ Record<
86
+ EmailClient,
87
+ Partial<
88
+ Record<
89
+ Platform,
90
+ /*
91
+ This last Record<string, string> has only one key, as the
92
+ ordered version of caniemail's data is meant to be something like:
93
+
94
+ [
95
+ { "1.0": "u" },
96
+ { "2.0": "y" },
97
+ { "3.0": "p #1" },
98
+ ]
99
+
100
+ So only one key for each object inside of this array, TypeScript can't really
101
+ describe this though AFAIK.
102
+ */
103
+ Record</* version */ string, string>[]
104
+ >
105
+ >
106
+ >
107
+ >;
108
+ notes: string | null;
109
+ notes_by_num: Record<number, string> | null;
110
+ }
111
+
112
+ const relevantEmailClients: EmailClient[] = [
113
+ 'gmail',
114
+ 'apple-mail',
115
+ 'outlook',
116
+ 'yahoo',
117
+ ];
118
+
119
+ export const checkCompatibility = async (
120
+ reactCode: string,
121
+ emailPath: string,
122
+ ) => {
123
+ const ast = parse(reactCode, {
124
+ strictMode: false,
125
+ errorRecovery: true,
126
+ sourceType: 'unambiguous',
127
+ plugins: ['jsx', 'typescript', 'decorators'],
128
+ });
129
+
130
+ const getSourceCodeAt = (location: SourceLocation) => {
131
+ const codeLines = reactCode.split(/\n|\r|\r\n/);
132
+ const source = codeLines
133
+ .slice(
134
+ Math.max(location.start.line - 2, 0),
135
+ Math.min(location.end.line + 2, codeLines.length),
136
+ )
137
+ .join('\n');
138
+ return source;
139
+ };
140
+
141
+ const objectVariables = getObjectVariables(ast);
142
+ const usedStyleProperties = await getUsedStyleProperties(
143
+ ast,
144
+ reactCode,
145
+ emailPath,
146
+ objectVariables,
147
+ );
148
+ const readableStream = new ReadableStream<CompatibilityCheckingResult>({
149
+ async start(controller) {
150
+ for (const entry of supportEntries) {
151
+ const compatibilityStats = getCompatibilityStatsForEntry(
152
+ entry,
153
+ relevantEmailClients,
154
+ );
155
+ if (Object.keys(compatibilityStats.perEmailClient).length === 0)
156
+ continue;
157
+ if (
158
+ compatibilityStats.status === 'success' ||
159
+ compatibilityStats.status === 'warning'
160
+ )
161
+ continue;
162
+
163
+ if (entry.category === 'html') {
164
+ const entryElements = getElementNames(entry.title, entry.keywords);
165
+ const entryAttributes = getElementAttributes(entry.title);
166
+ const htmlEntryType = (() => {
167
+ if (entryElements.length > 0) {
168
+ return 'element';
169
+ }
170
+
171
+ if (entryAttributes.length > 0) {
172
+ return 'attribute';
173
+ }
174
+ })();
175
+
176
+ if (!htmlEntryType) continue;
177
+
178
+ let addedInsight = false;
179
+ if (htmlEntryType === 'element') {
180
+ traverse(ast, {
181
+ JSXOpeningElement(path) {
182
+ if (path.node.name.type === 'JSXIdentifier' && !addedInsight) {
183
+ const elementName = path.node.name.name;
184
+ if (
185
+ entryElements.includes(elementName) &&
186
+ path.node.name.loc
187
+ ) {
188
+ addedInsight = true;
189
+ controller.enqueue({
190
+ entry,
191
+ source: getSourceCodeAt(path.node.name.loc),
192
+ location: convertLocationIntoObject(path.node.name.loc),
193
+ statsPerEmailClient: compatibilityStats.perEmailClient,
194
+ status: compatibilityStats.status,
195
+ });
196
+ }
197
+ }
198
+ },
199
+ });
200
+ } else {
201
+ traverse(ast, {
202
+ JSXAttribute(path) {
203
+ if (path.node.name.type === 'JSXIdentifier' && !addedInsight) {
204
+ const attributeName = path.node.name.name;
205
+ if (
206
+ entryAttributes.includes(attributeName) &&
207
+ path.node.name.loc
208
+ ) {
209
+ addedInsight = true;
210
+ controller.enqueue({
211
+ entry,
212
+ source: getSourceCodeAt(path.node.name.loc),
213
+ location: convertLocationIntoObject(path.node.name.loc),
214
+ statsPerEmailClient: compatibilityStats.perEmailClient,
215
+ status: compatibilityStats.status,
216
+ });
217
+ }
218
+ }
219
+ },
220
+ });
221
+ }
222
+ }
223
+
224
+ if (entry.category === 'css') {
225
+ const entryFullProperty = getCssPropertyWithValue(entry.title);
226
+ const entryProperties = getCssPropertyNames(
227
+ entry.title,
228
+ entry.keywords,
229
+ );
230
+ const entryUnit = getCssUnit(entry.title);
231
+ const entryFunctions = getCssFunctions(entry.title);
232
+
233
+ const cssEntryType = (() => {
234
+ if (entryFullProperty?.name && entryFullProperty.value) {
235
+ return 'full property';
236
+ }
237
+
238
+ if (entryFunctions.length > 0) {
239
+ return 'function';
240
+ }
241
+
242
+ if (entryUnit) {
243
+ return 'unit';
244
+ }
245
+
246
+ if (entryProperties.length > 0) {
247
+ return 'property name';
248
+ }
249
+ })();
250
+
251
+ if (!cssEntryType) continue;
252
+ const addToInsights = (
253
+ property: StylePropertyUsage & { location: SourceLocation },
254
+ ) => {
255
+ controller.enqueue({
256
+ entry,
257
+ location: convertLocationIntoObject(property.location),
258
+ source: getSourceCodeAt(property.location),
259
+ statsPerEmailClient: compatibilityStats.perEmailClient,
260
+ status: compatibilityStats.status,
261
+ });
262
+ };
263
+
264
+ for (const property of usedStyleProperties) {
265
+ if (!doesPropertyHaveLocation(property)) {
266
+ throw new Error(
267
+ "One of the properties' node did not contain the proper location for it on the source code. This must be an issue because we always need access to the source.",
268
+ {
269
+ cause: {
270
+ property,
271
+ entry,
272
+ reactCode,
273
+ ast,
274
+ },
275
+ },
276
+ );
277
+ }
278
+
279
+ if (cssEntryType === 'full property') {
280
+ if (
281
+ snakeToCamel(property.name) ===
282
+ snakeToCamel(entryFullProperty!.name) &&
283
+ property.value === entryFullProperty!.value
284
+ ) {
285
+ addToInsights(property);
286
+ break;
287
+ }
288
+ } else if (cssEntryType === 'function') {
289
+ const functionRegex =
290
+ /(?<functionName>[a-zA-Z_][a-zA-Z0-9_-]*)\s*\(/g;
291
+ const functionName = functionRegex.exec(property.value)?.groups
292
+ ?.functionName;
293
+ if (functionName !== undefined) {
294
+ if (entryFunctions.includes(functionName)) {
295
+ addToInsights(property);
296
+ break;
297
+ }
298
+ }
299
+ } else if (cssEntryType === 'unit') {
300
+ const match = property.value.match(/[0-9](?<unit>[a-zA-Z%]+)$/g);
301
+ if (match) {
302
+ const unit = match.groups?.unit;
303
+ if (entryUnit && unit && entryUnit === unit) {
304
+ addToInsights(property);
305
+ break;
306
+ }
307
+ }
308
+ } else if (
309
+ entryProperties.some(
310
+ (propertyName) =>
311
+ snakeToCamel(property.name) === snakeToCamel(propertyName),
312
+ )
313
+ ) {
314
+ addToInsights(property);
315
+ break;
316
+ }
317
+ }
318
+ }
319
+ }
320
+ controller.close();
321
+ },
322
+ });
323
+
324
+ return readableStream;
325
+ };
326
+
327
+ const snakeToCamel = (snakeStr: string) => {
328
+ return snakeStr
329
+ .toLowerCase()
330
+ .replace(/-+([a-z])/g, (_match, letter) => letter.toUpperCase());
331
+ };
332
+
333
+ export type AST = ReturnType<typeof parse>;
@@ -0,0 +1,100 @@
1
+ import { checkImages, type ImageCheckingResult } from './check-images';
2
+
3
+ test('checkImages()', async () => {
4
+ const results: ImageCheckingResult[] = [];
5
+ const html = `<div>
6
+ <img src="https://resend.com/static/brand/resend-icon-white.png" />,
7
+ <img src="/static/codepen-challengers.png" alt="codepen challenges" />,
8
+ </div>`;
9
+ const stream = await checkImages(html, 'https://demo.react.email');
10
+ const reader = stream.getReader();
11
+ while (true) {
12
+ const { done, value } = await reader.read();
13
+ if (value) {
14
+ results.push(value);
15
+ }
16
+ if (done) {
17
+ break;
18
+ }
19
+ }
20
+ expect(results).toEqual([
21
+ {
22
+ source: 'https://resend.com/static/brand/resend-icon-white.png',
23
+ codeLocation: {
24
+ line: 2,
25
+ column: 3,
26
+ },
27
+ checks: [
28
+ {
29
+ passed: false,
30
+ type: 'accessibility',
31
+ metadata: {
32
+ alt: undefined,
33
+ },
34
+ },
35
+ {
36
+ passed: true,
37
+ type: 'syntax',
38
+ },
39
+ {
40
+ passed: true,
41
+ type: 'security',
42
+ },
43
+ {
44
+ passed: true,
45
+ type: 'fetch_attempt',
46
+ metadata: {
47
+ fetchStatusCode: 200,
48
+ },
49
+ },
50
+ {
51
+ passed: true,
52
+ type: 'image_size',
53
+ metadata: {
54
+ byteCount: 23_138,
55
+ },
56
+ },
57
+ ],
58
+ status: 'warning',
59
+ },
60
+ {
61
+ codeLocation: {
62
+ line: 3,
63
+ column: 3,
64
+ },
65
+ checks: [
66
+ {
67
+ metadata: {
68
+ alt: 'codepen challenges',
69
+ },
70
+ passed: true,
71
+ type: 'accessibility',
72
+ },
73
+ {
74
+ passed: true,
75
+ type: 'syntax',
76
+ },
77
+ {
78
+ passed: true,
79
+ type: 'security',
80
+ },
81
+ {
82
+ metadata: {
83
+ fetchStatusCode: 200,
84
+ },
85
+ passed: true,
86
+ type: 'fetch_attempt',
87
+ },
88
+ {
89
+ metadata: {
90
+ byteCount: 111_922,
91
+ },
92
+ passed: true,
93
+ type: 'image_size',
94
+ },
95
+ ],
96
+ source: '/static/codepen-challengers.png',
97
+ status: 'success',
98
+ },
99
+ ] satisfies ImageCheckingResult[]);
100
+ });
@@ -0,0 +1,160 @@
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 ImageCheck = { passed: boolean } & (
12
+ | {
13
+ type: 'accessibility';
14
+ metadata: {
15
+ alt: string | undefined;
16
+ };
17
+ }
18
+ | {
19
+ type: 'fetch_attempt';
20
+ metadata: {
21
+ fetchStatusCode: number | undefined;
22
+ };
23
+ }
24
+ | {
25
+ type: 'image_size';
26
+ metadata: {
27
+ byteCount: number | undefined;
28
+ };
29
+ }
30
+ | {
31
+ type: 'syntax';
32
+ }
33
+ | {
34
+ type: 'security';
35
+ }
36
+ );
37
+
38
+ export interface ImageCheckingResult {
39
+ status: 'success' | 'warning' | 'error';
40
+ source: string;
41
+ codeLocation: CodeLocation;
42
+ checks: ImageCheck[];
43
+ }
44
+
45
+ const getResponseSizeInBytes = async (res: IncomingMessage) => {
46
+ let totalBytes = 0;
47
+ for await (const chunk of res) {
48
+ totalBytes += chunk.byteLength;
49
+ }
50
+ return totalBytes;
51
+ };
52
+
53
+ export const checkImages = async (code: string, base: string) => {
54
+ const ast = parse(code);
55
+
56
+ const readableStream = new ReadableStream<ImageCheckingResult>({
57
+ async start(controller) {
58
+ const images = ast.querySelectorAll('img');
59
+ for await (const image of images) {
60
+ const rawSource = image.attributes.src;
61
+ if (!rawSource) continue;
62
+
63
+ const source = rawSource?.startsWith('/')
64
+ ? `${base}${rawSource}`
65
+ : rawSource;
66
+
67
+ const result: ImageCheckingResult = {
68
+ source: rawSource,
69
+ codeLocation: getCodeLocationFromAstElement(image, code),
70
+ status: 'success',
71
+ checks: [],
72
+ };
73
+
74
+ const alt = image.attributes.alt;
75
+ result.checks.push({
76
+ passed: alt !== undefined,
77
+ type: 'accessibility',
78
+ metadata: {
79
+ alt,
80
+ },
81
+ });
82
+ if (alt === undefined) {
83
+ result.status = 'warning';
84
+ }
85
+
86
+ try {
87
+ const url = new URL(source);
88
+ result.checks.push({
89
+ passed: true,
90
+ type: 'syntax',
91
+ });
92
+
93
+ if (rawSource.startsWith('http://')) {
94
+ result.checks.push({
95
+ passed: false,
96
+ type: 'security',
97
+ });
98
+ result.status = 'warning';
99
+ } else {
100
+ result.checks.push({
101
+ passed: true,
102
+ type: 'security',
103
+ });
104
+ }
105
+
106
+ let res: IncomingMessage | undefined = undefined;
107
+ try {
108
+ res = await quickFetch(url);
109
+ const hasSucceeded =
110
+ res.statusCode?.toString().startsWith('2') ?? false;
111
+ result.checks.push({
112
+ type: 'fetch_attempt',
113
+ passed: hasSucceeded,
114
+ metadata: {
115
+ fetchStatusCode: res.statusCode,
116
+ },
117
+ });
118
+ if (!hasSucceeded) {
119
+ result.status = res.statusCode?.toString().startsWith('3')
120
+ ? 'warning'
121
+ : 'error';
122
+ }
123
+
124
+ const responseSizeBytes = await getResponseSizeInBytes(res);
125
+ result.checks.push({
126
+ type: 'image_size',
127
+ passed: responseSizeBytes < 1_048_576, // 1024 x 1024 bytes
128
+ metadata: {
129
+ byteCount: responseSizeBytes,
130
+ },
131
+ });
132
+ if (responseSizeBytes > 1_048_576 && result.status !== 'error') {
133
+ result.status = 'warning';
134
+ }
135
+ } catch (exception) {
136
+ result.checks.push({
137
+ type: 'fetch_attempt',
138
+ passed: false,
139
+ metadata: {
140
+ fetchStatusCode: undefined,
141
+ },
142
+ });
143
+ result.status = 'error';
144
+ }
145
+ } catch (exception) {
146
+ result.checks.push({
147
+ passed: false,
148
+ type: 'syntax',
149
+ });
150
+ result.status = 'error';
151
+ }
152
+
153
+ controller.enqueue(result);
154
+ }
155
+ controller.close();
156
+ },
157
+ });
158
+
159
+ return readableStream;
160
+ };
@@ -0,0 +1,113 @@
1
+ import { checkLinks, type LinkCheckingResult } from './check-links';
2
+
3
+ test('checkLinks()', async () => {
4
+ const results: LinkCheckingResult[] = [];
5
+ const html = `<div>
6
+ <a href="/">Root</a>
7
+ <a href="https://resend.com">Resend</a>
8
+ <a href="https://notion.so">Notion</a>
9
+ <a href="http://react.email">React Email unsafe</a>
10
+ </div>`;
11
+ const stream = await checkLinks(html);
12
+ const reader = stream.getReader();
13
+ while (true) {
14
+ const { done, value } = await reader.read();
15
+ if (value) {
16
+ results.push(value);
17
+ }
18
+ if (done) {
19
+ break;
20
+ }
21
+ }
22
+ expect(results).toEqual([
23
+ {
24
+ status: 'error',
25
+ codeLocation: {
26
+ line: 2,
27
+ column: 3,
28
+ },
29
+ checks: [
30
+ {
31
+ type: 'syntax',
32
+ passed: false,
33
+ },
34
+ ],
35
+ link: '/',
36
+ },
37
+ {
38
+ status: 'success',
39
+ codeLocation: {
40
+ line: 3,
41
+ column: 3,
42
+ },
43
+ checks: [
44
+ {
45
+ type: 'syntax',
46
+ passed: true,
47
+ },
48
+ {
49
+ type: 'security',
50
+ passed: true,
51
+ },
52
+ {
53
+ type: 'fetch_attempt',
54
+ passed: true,
55
+ metadata: {
56
+ fetchStatusCode: 200,
57
+ },
58
+ },
59
+ ],
60
+ link: 'https://resend.com',
61
+ },
62
+ {
63
+ status: 'warning',
64
+ codeLocation: {
65
+ line: 4,
66
+ column: 3,
67
+ },
68
+ checks: [
69
+ {
70
+ type: 'syntax',
71
+ passed: true,
72
+ },
73
+ {
74
+ type: 'security',
75
+ passed: true,
76
+ },
77
+ {
78
+ type: 'fetch_attempt',
79
+ metadata: {
80
+ fetchStatusCode: 301,
81
+ },
82
+ passed: false,
83
+ },
84
+ ],
85
+ link: 'https://notion.so',
86
+ },
87
+ {
88
+ status: 'warning',
89
+ codeLocation: {
90
+ line: 5,
91
+ column: 3,
92
+ },
93
+ checks: [
94
+ {
95
+ type: 'syntax',
96
+ passed: true,
97
+ },
98
+ {
99
+ type: 'security',
100
+ passed: false,
101
+ },
102
+ {
103
+ type: 'fetch_attempt',
104
+ metadata: {
105
+ fetchStatusCode: 308,
106
+ },
107
+ passed: false,
108
+ },
109
+ ],
110
+ link: 'http://react.email',
111
+ },
112
+ ] satisfies LinkCheckingResult[]);
113
+ });