react-email 4.0.16 → 4.1.0-canary.10

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 (470) hide show
  1. package/CHANGELOG.md +65 -0
  2. package/dev/index.js +40 -0
  3. package/dev/package.json +13 -0
  4. package/dist/index.js +1 -1
  5. package/package.json +13 -46
  6. package/readme.md +16 -0
  7. package/src/commands/build.ts +306 -0
  8. package/src/commands/dev.ts +27 -0
  9. package/src/commands/export.ts +204 -0
  10. package/src/commands/start.ts +38 -0
  11. package/src/index.ts +55 -0
  12. package/src/utils/__snapshots__/tree.spec.ts.snap +27 -0
  13. package/src/utils/esbuild/renderring-utilities-exporter.ts +1 -1
  14. package/src/utils/get-emails-directory-metadata.spec.ts +1 -1
  15. package/src/utils/get-preview-server-location.ts +51 -0
  16. package/src/utils/index.ts +2 -6
  17. package/src/utils/packageJson.ts +4 -0
  18. package/src/utils/preview/get-env-variables-for-preview-app.ts +14 -0
  19. package/src/utils/preview/hot-reloading/create-dependency-graph.spec.ts +284 -0
  20. package/src/utils/preview/hot-reloading/create-dependency-graph.ts +321 -0
  21. package/src/utils/preview/hot-reloading/get-imported-modules.spec.ts +151 -0
  22. package/src/utils/preview/hot-reloading/get-imported-modules.ts +49 -0
  23. package/src/utils/preview/hot-reloading/resolve-path-aliases.spec.ts +11 -0
  24. package/src/utils/preview/hot-reloading/resolve-path-aliases.ts +32 -0
  25. package/src/utils/preview/hot-reloading/setup-hot-reloading.ts +121 -0
  26. package/src/utils/preview/hot-reloading/test/tsconfig.json +8 -0
  27. package/src/utils/preview/index.ts +2 -0
  28. package/src/utils/preview/serve-static-file.ts +51 -0
  29. package/src/utils/preview/start-dev-server.ts +234 -0
  30. package/src/utils/tree.spec.ts +5 -0
  31. package/src/utils/tree.ts +76 -0
  32. package/src/utils/types/hot-reload-change.ts +1 -1
  33. package/src/utils/types/hot-reload-event.ts +1 -1
  34. package/tsconfig.json +4 -10
  35. package/dist/cli/index.d.mts +0 -1
  36. package/dist/cli/index.d.ts +0 -1
  37. package/dist/cli/index.js +0 -2785
  38. package/dist/cli/index.mjs +0 -1361
  39. package/dist/index.d.mts +0 -20
  40. package/dist/index.d.ts +0 -20
  41. package/dist/index.mjs +0 -21
  42. package/dist/package/index.d.mts +0 -33
  43. package/dist/package/index.d.ts +0 -33
  44. package/dist/package/index.js +0 -62
  45. package/dist/package/index.mjs +0 -7
  46. package/dist/preview/.next/BUILD_ID +0 -1
  47. package/dist/preview/.next/app-build-manifest.json +0 -44
  48. package/dist/preview/.next/app-path-routes-manifest.json +0 -6
  49. package/dist/preview/.next/build-manifest.json +0 -33
  50. package/dist/preview/.next/diagnostics/build-diagnostics.json +0 -6
  51. package/dist/preview/.next/diagnostics/framework.json +0 -1
  52. package/dist/preview/.next/export-marker.json +0 -6
  53. package/dist/preview/.next/images-manifest.json +0 -57
  54. package/dist/preview/.next/next-minimal-server.js.nft.json +0 -1
  55. package/dist/preview/.next/next-server.js.nft.json +0 -1
  56. package/dist/preview/.next/package.json +0 -1
  57. package/dist/preview/.next/prerender-manifest.json +0 -41
  58. package/dist/preview/.next/react-loadable-manifest.json +0 -1
  59. package/dist/preview/.next/required-server-files.json +0 -311
  60. package/dist/preview/.next/routes-manifest.json +0 -64
  61. package/dist/preview/.next/server/app/_not-found/page.js +0 -1
  62. package/dist/preview/.next/server/app/_not-found/page.js.nft.json +0 -1
  63. package/dist/preview/.next/server/app/_not-found/page_client-reference-manifest.js +0 -1
  64. package/dist/preview/.next/server/app/favicon.ico/route.js +0 -1
  65. package/dist/preview/.next/server/app/favicon.ico/route.js.nft.json +0 -1
  66. package/dist/preview/.next/server/app/favicon.ico.body +0 -0
  67. package/dist/preview/.next/server/app/favicon.ico.meta +0 -1
  68. package/dist/preview/.next/server/app/page.js +0 -1
  69. package/dist/preview/.next/server/app/page.js.nft.json +0 -1
  70. package/dist/preview/.next/server/app/page_client-reference-manifest.js +0 -1
  71. package/dist/preview/.next/server/app/preview/[...slug]/page.js +0 -321
  72. package/dist/preview/.next/server/app/preview/[...slug]/page.js.nft.json +0 -1
  73. package/dist/preview/.next/server/app/preview/[...slug]/page_client-reference-manifest.js +0 -1
  74. package/dist/preview/.next/server/app-paths-manifest.json +0 -6
  75. package/dist/preview/.next/server/chunks/134.js +0 -6
  76. package/dist/preview/.next/server/chunks/235.js +0 -15
  77. package/dist/preview/.next/server/chunks/275.js +0 -1
  78. package/dist/preview/.next/server/chunks/343.js +0 -20
  79. package/dist/preview/.next/server/chunks/428.js +0 -14
  80. package/dist/preview/.next/server/chunks/963.js +0 -1
  81. package/dist/preview/.next/server/functions-config-manifest.json +0 -4
  82. package/dist/preview/.next/server/interception-route-rewrite-manifest.js +0 -1
  83. package/dist/preview/.next/server/middleware-build-manifest.js +0 -1
  84. package/dist/preview/.next/server/middleware-manifest.json +0 -6
  85. package/dist/preview/.next/server/middleware-react-loadable-manifest.js +0 -1
  86. package/dist/preview/.next/server/next-font-manifest.js +0 -1
  87. package/dist/preview/.next/server/next-font-manifest.json +0 -1
  88. package/dist/preview/.next/server/pages/500.html +0 -1
  89. package/dist/preview/.next/server/pages/_app.js +0 -1
  90. package/dist/preview/.next/server/pages/_app.js.nft.json +0 -1
  91. package/dist/preview/.next/server/pages/_document.js +0 -1
  92. package/dist/preview/.next/server/pages/_document.js.nft.json +0 -1
  93. package/dist/preview/.next/server/pages/_error.js +0 -1
  94. package/dist/preview/.next/server/pages/_error.js.nft.json +0 -1
  95. package/dist/preview/.next/server/pages-manifest.json +0 -5
  96. package/dist/preview/.next/server/server-reference-manifest.js +0 -1
  97. package/dist/preview/.next/server/server-reference-manifest.json +0 -1
  98. package/dist/preview/.next/server/webpack-runtime.js +0 -1
  99. package/dist/preview/.next/static/chunks/107-3043079e7cb8bcae.js +0 -1
  100. package/dist/preview/.next/static/chunks/293-297b1eb2241f9a70.js +0 -1
  101. package/dist/preview/.next/static/chunks/3bd82e28-cda2c00a924937c5.js +0 -1
  102. package/dist/preview/.next/static/chunks/45-1021fac82f766268.js +0 -1
  103. package/dist/preview/.next/static/chunks/484-e38a627386aae911.js +0 -1
  104. package/dist/preview/.next/static/chunks/589-817d8691661d370e.js +0 -1
  105. package/dist/preview/.next/static/chunks/902-c34acb56733e0ce1.js +0 -1
  106. package/dist/preview/.next/static/chunks/app/_not-found/page-4cbc7dce3ad33336.js +0 -1
  107. package/dist/preview/.next/static/chunks/app/layout-269b5cbd8f4cd2e3.js +0 -1
  108. package/dist/preview/.next/static/chunks/app/page-2dbfb5b2dc4b1191.js +0 -1
  109. package/dist/preview/.next/static/chunks/app/preview/[...slug]/page-79e0c9b528a62f8b.js +0 -1
  110. package/dist/preview/.next/static/chunks/f33a14d2-ec7c5f0b91818561.js +0 -6
  111. package/dist/preview/.next/static/chunks/framework-b887e9fc751a9906.js +0 -1
  112. package/dist/preview/.next/static/chunks/main-9a03e7ba8acb1900.js +0 -1
  113. package/dist/preview/.next/static/chunks/main-app-976577a424e11c75.js +0 -1
  114. package/dist/preview/.next/static/chunks/pages/_app-542a93a5a214e1c0.js +0 -1
  115. package/dist/preview/.next/static/chunks/pages/_error-d5fe1b1612642f76.js +0 -1
  116. package/dist/preview/.next/static/chunks/polyfills-42372ed130431b0a.js +0 -1
  117. package/dist/preview/.next/static/chunks/webpack-31c45daa2bd82a7b.js +0 -1
  118. package/dist/preview/.next/static/css/ac3decd5d6736fbe.css +0 -3
  119. package/dist/preview/.next/static/media/05613964ce6c782e-s.p.otf +0 -0
  120. package/dist/preview/.next/static/media/11c6126b9369e85e-s.p.otf +0 -0
  121. package/dist/preview/.next/static/media/26a46d62cd723877-s.woff2 +0 -0
  122. package/dist/preview/.next/static/media/26cb97734d8cb717-s.p.otf +0 -0
  123. package/dist/preview/.next/static/media/55c55f0601d81cf3-s.woff2 +0 -0
  124. package/dist/preview/.next/static/media/581909926a08bbc8-s.woff2 +0 -0
  125. package/dist/preview/.next/static/media/8e9860b6e62d6359-s.woff2 +0 -0
  126. package/dist/preview/.next/static/media/97e0cb1ae144a2a9-s.woff2 +0 -0
  127. package/dist/preview/.next/static/media/bb6462617151f6b7-s.p.otf +0 -0
  128. package/dist/preview/.next/static/media/cf6daef822ab0142-s.p.otf +0 -0
  129. package/dist/preview/.next/static/media/df0a9ae256c0569c-s.woff2 +0 -0
  130. package/dist/preview/.next/static/media/e4051546b3043204-s.p.otf +0 -0
  131. package/dist/preview/.next/static/media/e4af272ccee01ff0-s.p.woff2 +0 -0
  132. package/dist/preview/.next/static/media/logo.2ce2a759.png +0 -0
  133. package/dist/preview/.next/static/yxjkw7Y7HRLmRWkl3S43z/_buildManifest.js +0 -1
  134. package/dist/preview/.next/static/yxjkw7Y7HRLmRWkl3S43z/_ssgManifest.js +0 -1
  135. package/dist/preview/.next/trace +0 -27
  136. package/dist/preview/.next/types/app/layout.ts +0 -84
  137. package/dist/preview/.next/types/app/page.ts +0 -84
  138. package/dist/preview/.next/types/app/preview/[...slug]/page.ts +0 -84
  139. package/dist/preview/.next/types/cache-life.d.ts +0 -141
  140. package/dist/preview/.next/types/package.json +0 -1
  141. package/module-punycode.d.ts +0 -3
  142. package/next-env.d.ts +0 -5
  143. package/next.config.js +0 -22
  144. package/postcss.config.js +0 -8
  145. package/scripts/build-preview-server.mjs +0 -33
  146. package/scripts/fill-caniemail-data.mjs +0 -36
  147. package/src/actions/email-validation/caniemail-data.ts +0 -85993
  148. package/src/actions/email-validation/check-compatibility.ts +0 -333
  149. package/src/actions/email-validation/check-images.spec.tsx +0 -100
  150. package/src/actions/email-validation/check-images.ts +0 -160
  151. package/src/actions/email-validation/check-links.spec.tsx +0 -113
  152. package/src/actions/email-validation/check-links.ts +0 -113
  153. package/src/actions/email-validation/get-code-location-from-ast-element.ts +0 -18
  154. package/src/actions/email-validation/quick-fetch.ts +0 -14
  155. package/src/actions/get-email-path-from-slug.ts +0 -32
  156. package/src/actions/get-emails-directory-metadata-action.ts +0 -19
  157. package/src/actions/render-email-by-path.tsx +0 -121
  158. package/src/animated-icons-data/help.json +0 -1082
  159. package/src/animated-icons-data/link.json +0 -1309
  160. package/src/animated-icons-data/load.json +0 -443
  161. package/src/animated-icons-data/mail.json +0 -1320
  162. package/src/app/env.ts +0 -15
  163. package/src/app/favicon.ico +0 -0
  164. package/src/app/fonts/SFMono/SFMonoBold.otf +0 -0
  165. package/src/app/fonts/SFMono/SFMonoBoldItalic.otf +0 -0
  166. package/src/app/fonts/SFMono/SFMonoHeavy.otf +0 -0
  167. package/src/app/fonts/SFMono/SFMonoHeavyItalic.otf +0 -0
  168. package/src/app/fonts/SFMono/SFMonoLight.otf +0 -0
  169. package/src/app/fonts/SFMono/SFMonoLightItalic.otf +0 -0
  170. package/src/app/fonts/SFMono/SFMonoMedium.otf +0 -0
  171. package/src/app/fonts/SFMono/SFMonoMediumItalic.otf +0 -0
  172. package/src/app/fonts/SFMono/SFMonoRegular.otf +0 -0
  173. package/src/app/fonts/SFMono/SFMonoRegularItalic.otf +0 -0
  174. package/src/app/fonts/SFMono/SFMonoSemibold.otf +0 -0
  175. package/src/app/fonts/SFMono/SFMonoSemiboldItalic.otf +0 -0
  176. package/src/app/fonts.ts +0 -39
  177. package/src/app/globals.css +0 -15
  178. package/src/app/layout.tsx +0 -43
  179. package/src/app/logo.png +0 -0
  180. package/src/app/page.tsx +0 -46
  181. package/src/app/preview/[...slug]/page.tsx +0 -157
  182. package/src/app/preview/[...slug]/preview.tsx +0 -216
  183. package/src/app/preview/[...slug]/rendering-error.tsx +0 -40
  184. package/src/commands/testing/out/magic-links/aws-verify-email.html +0 -165
  185. package/src/commands/testing/out/magic-links/linear-login-code.html +0 -90
  186. package/src/commands/testing/out/magic-links/notion-magic-link.html +0 -76
  187. package/src/commands/testing/out/magic-links/plaid-verify-identity.html +0 -78
  188. package/src/commands/testing/out/magic-links/raycast-magic-link.html +0 -91
  189. package/src/commands/testing/out/magic-links/slack-confirm.html +0 -240
  190. package/src/commands/testing/out/newsletters/codepen-challengers.html +0 -548
  191. package/src/commands/testing/out/newsletters/google-play-policy-update.html +0 -339
  192. package/src/commands/testing/out/newsletters/stack-overflow-tips.html +0 -231
  193. package/src/commands/testing/out/notifications/github-access-token.html +0 -104
  194. package/src/commands/testing/out/notifications/papermark-year-in-review.html +0 -317
  195. package/src/commands/testing/out/notifications/vercel-invite-user.html +0 -171
  196. package/src/commands/testing/out/notifications/yelp-recent-login.html +0 -195
  197. package/src/commands/testing/out/receipts/apple-receipt.html +0 -677
  198. package/src/commands/testing/out/receipts/nike-receipt.html +0 -724
  199. package/src/commands/testing/out/reset-password/dropbox-reset-password.html +0 -98
  200. package/src/commands/testing/out/reset-password/twitch-reset-password.html +0 -220
  201. package/src/commands/testing/out/reviews/airbnb-review.html +0 -206
  202. package/src/commands/testing/out/reviews/amazon-review.html +0 -356
  203. package/src/commands/testing/out/static/airbnb-logo.png +0 -0
  204. package/src/commands/testing/out/static/airbnb-review-user.jpg +0 -0
  205. package/src/commands/testing/out/static/amazon-book.jpg +0 -0
  206. package/src/commands/testing/out/static/amazon-facebook.jpg +0 -0
  207. package/src/commands/testing/out/static/amazon-instagram.jpg +0 -0
  208. package/src/commands/testing/out/static/amazon-logo.png +0 -0
  209. package/src/commands/testing/out/static/amazon-prime-logo.png +0 -0
  210. package/src/commands/testing/out/static/amazon-rating.gif +0 -0
  211. package/src/commands/testing/out/static/amazon-twitter.jpg +0 -0
  212. package/src/commands/testing/out/static/apple-card-icon.png +0 -0
  213. package/src/commands/testing/out/static/apple-hbo-max-icon.jpeg +0 -0
  214. package/src/commands/testing/out/static/apple-logo.png +0 -0
  215. package/src/commands/testing/out/static/apple-wallet.png +0 -0
  216. package/src/commands/testing/out/static/aws-logo.png +0 -0
  217. package/src/commands/testing/out/static/codepen-challengers.png +0 -0
  218. package/src/commands/testing/out/static/codepen-cube.png +0 -0
  219. package/src/commands/testing/out/static/codepen-pro.png +0 -0
  220. package/src/commands/testing/out/static/dropbox-logo.png +0 -0
  221. package/src/commands/testing/out/static/github.png +0 -0
  222. package/src/commands/testing/out/static/google-play-academy.png +0 -0
  223. package/src/commands/testing/out/static/google-play-chat.png +0 -0
  224. package/src/commands/testing/out/static/google-play-footer.png +0 -0
  225. package/src/commands/testing/out/static/google-play-header.png +0 -0
  226. package/src/commands/testing/out/static/google-play-icon.png +0 -0
  227. package/src/commands/testing/out/static/google-play-logo.png +0 -0
  228. package/src/commands/testing/out/static/google-play-pl.png +0 -0
  229. package/src/commands/testing/out/static/google-play.png +0 -0
  230. package/src/commands/testing/out/static/koala-logo.png +0 -0
  231. package/src/commands/testing/out/static/linear-logo.png +0 -0
  232. package/src/commands/testing/out/static/netlify-logo.png +0 -0
  233. package/src/commands/testing/out/static/nike-logo.png +0 -0
  234. package/src/commands/testing/out/static/nike-phone.png +0 -0
  235. package/src/commands/testing/out/static/nike-product.png +0 -0
  236. package/src/commands/testing/out/static/nike-recomendation-1.png +0 -0
  237. package/src/commands/testing/out/static/nike-recomendation-2.png +0 -0
  238. package/src/commands/testing/out/static/nike-recomendation-3.png +0 -0
  239. package/src/commands/testing/out/static/nike-recomendation-4.png +0 -0
  240. package/src/commands/testing/out/static/notion-logo.png +0 -0
  241. package/src/commands/testing/out/static/plaid-logo.png +0 -0
  242. package/src/commands/testing/out/static/raycast-bg.png +0 -0
  243. package/src/commands/testing/out/static/raycast-logo.png +0 -0
  244. package/src/commands/testing/out/static/slack-facebook.png +0 -0
  245. package/src/commands/testing/out/static/slack-linkedin.png +0 -0
  246. package/src/commands/testing/out/static/slack-logo.png +0 -0
  247. package/src/commands/testing/out/static/slack-twitter.png +0 -0
  248. package/src/commands/testing/out/static/stack-overflow-header.png +0 -0
  249. package/src/commands/testing/out/static/stack-overflow-logo-sm.png +0 -0
  250. package/src/commands/testing/out/static/stack-overflow-logo.png +0 -0
  251. package/src/commands/testing/out/static/stripe-logo.png +0 -0
  252. package/src/commands/testing/out/static/twitch-icon-facebook.png +0 -0
  253. package/src/commands/testing/out/static/twitch-icon-twitter.png +0 -0
  254. package/src/commands/testing/out/static/twitch-logo.png +0 -0
  255. package/src/commands/testing/out/static/vercel-arrow.png +0 -0
  256. package/src/commands/testing/out/static/vercel-logo.png +0 -0
  257. package/src/commands/testing/out/static/vercel-team.png +0 -0
  258. package/src/commands/testing/out/static/vercel-user.png +0 -0
  259. package/src/commands/testing/out/static/yelp-footer.png +0 -0
  260. package/src/commands/testing/out/static/yelp-header.png +0 -0
  261. package/src/commands/testing/out/static/yelp-logo.png +0 -0
  262. package/src/commands/testing/out/welcome/koala-welcome.html +0 -90
  263. package/src/commands/testing/out/welcome/netlify-welcome.html +0 -199
  264. package/src/commands/testing/out/welcome/stripe-welcome.html +0 -153
  265. package/src/components/button.tsx +0 -101
  266. package/src/components/code-container.tsx +0 -164
  267. package/src/components/code-snippet.tsx +0 -9
  268. package/src/components/code.tsx +0 -184
  269. package/src/components/heading.tsx +0 -113
  270. package/src/components/icons/icon-arrow-down.tsx +0 -16
  271. package/src/components/icons/icon-base.tsx +0 -26
  272. package/src/components/icons/icon-bug.tsx +0 -19
  273. package/src/components/icons/icon-button.tsx +0 -23
  274. package/src/components/icons/icon-check.tsx +0 -19
  275. package/src/components/icons/icon-clipboard.tsx +0 -40
  276. package/src/components/icons/icon-download.tsx +0 -19
  277. package/src/components/icons/icon-email.tsx +0 -18
  278. package/src/components/icons/icon-file.tsx +0 -19
  279. package/src/components/icons/icon-folder-open.tsx +0 -19
  280. package/src/components/icons/icon-folder.tsx +0 -18
  281. package/src/components/icons/icon-hide-sidebar.tsx +0 -23
  282. package/src/components/icons/icon-image.tsx +0 -19
  283. package/src/components/icons/icon-info.tsx +0 -18
  284. package/src/components/icons/icon-link.tsx +0 -14
  285. package/src/components/icons/icon-monitor.tsx +0 -19
  286. package/src/components/icons/icon-phone.tsx +0 -26
  287. package/src/components/icons/icon-reload.tsx +0 -18
  288. package/src/components/icons/icon-source.tsx +0 -19
  289. package/src/components/icons/icon-stamp.tsx +0 -14
  290. package/src/components/icons/icon-warning.tsx +0 -31
  291. package/src/components/index.ts +0 -7
  292. package/src/components/logo.tsx +0 -63
  293. package/src/components/resizable-wrapper.tsx +0 -173
  294. package/src/components/send.tsx +0 -134
  295. package/src/components/shell.tsx +0 -95
  296. package/src/components/sidebar/file-tree-directory-children.tsx +0 -134
  297. package/src/components/sidebar/file-tree-directory.tsx +0 -92
  298. package/src/components/sidebar/file-tree.tsx +0 -31
  299. package/src/components/sidebar/index.ts +0 -1
  300. package/src/components/sidebar/sidebar.tsx +0 -43
  301. package/src/components/text.tsx +0 -99
  302. package/src/components/toolbar/checking-results.tsx +0 -150
  303. package/src/components/toolbar/code-preview-line-link.tsx +0 -40
  304. package/src/components/toolbar/compatibility.tsx +0 -113
  305. package/src/components/toolbar/linter.tsx +0 -278
  306. package/src/components/toolbar/results.tsx +0 -51
  307. package/src/components/toolbar/spam-assassin.tsx +0 -155
  308. package/src/components/toolbar/toolbar-button.tsx +0 -52
  309. package/src/components/toolbar/use-cached-state.ts +0 -33
  310. package/src/components/toolbar.tsx +0 -349
  311. package/src/components/tooltip-content.tsx +0 -31
  312. package/src/components/tooltip.tsx +0 -19
  313. package/src/components/topbar/active-view-toggle-group.tsx +0 -86
  314. package/src/components/topbar/view-size-controls.tsx +0 -247
  315. package/src/components/topbar.tsx +0 -59
  316. package/src/contexts/emails.tsx +0 -59
  317. package/src/contexts/fragment-identifier.tsx +0 -48
  318. package/src/contexts/preview.tsx +0 -79
  319. package/src/hooks/use-clamped-state.ts +0 -24
  320. package/src/hooks/use-email-rendering-result.ts +0 -58
  321. package/src/hooks/use-fragment-identifier.ts +0 -14
  322. package/src/hooks/use-hot-reload.ts +0 -31
  323. package/src/hooks/use-icon-animation.ts +0 -41
  324. package/src/hooks/use-rendering-metadata.ts +0 -36
  325. package/src/package/body/dist/index.d.mts +0 -6
  326. package/src/package/body/dist/index.d.ts +0 -6
  327. package/src/package/body/dist/index.js +0 -79
  328. package/src/package/body/dist/index.mjs +0 -45
  329. package/src/package/button/dist/index.d.mts +0 -6
  330. package/src/package/button/dist/index.d.ts +0 -6
  331. package/src/package/button/dist/index.js +0 -252
  332. package/src/package/button/dist/index.mjs +0 -218
  333. package/src/package/code-block/dist/index.d.mts +0 -4906
  334. package/src/package/code-block/dist/index.d.ts +0 -4906
  335. package/src/package/code-block/dist/index.js +0 -18205
  336. package/src/package/code-block/dist/index.mjs +0 -18133
  337. package/src/package/code-inline/dist/index.d.mts +0 -11
  338. package/src/package/code-inline/dist/index.d.ts +0 -11
  339. package/src/package/code-inline/dist/index.js +0 -106
  340. package/src/package/code-inline/dist/index.mjs +0 -72
  341. package/src/package/column/dist/index.d.mts +0 -6
  342. package/src/package/column/dist/index.d.ts +0 -6
  343. package/src/package/column/dist/index.js +0 -79
  344. package/src/package/column/dist/index.mjs +0 -45
  345. package/src/package/components/dist/index.d.mts +0 -20
  346. package/src/package/components/dist/index.d.ts +0 -20
  347. package/src/package/components/dist/index.js +0 -62
  348. package/src/package/components/dist/index.mjs +0 -21
  349. package/src/package/container/dist/index.d.mts +0 -6
  350. package/src/package/container/dist/index.d.ts +0 -6
  351. package/src/package/container/dist/index.js +0 -93
  352. package/src/package/container/dist/index.mjs +0 -59
  353. package/src/package/font/dist/index.d.mts +0 -25
  354. package/src/package/font/dist/index.d.ts +0 -25
  355. package/src/package/font/dist/index.js +0 -55
  356. package/src/package/font/dist/index.mjs +0 -28
  357. package/src/package/head/dist/index.d.mts +0 -6
  358. package/src/package/head/dist/index.d.ts +0 -6
  359. package/src/package/head/dist/index.js +0 -83
  360. package/src/package/head/dist/index.mjs +0 -49
  361. package/src/package/heading/dist/index.d.mts +0 -43
  362. package/src/package/heading/dist/index.d.ts +0 -43
  363. package/src/package/heading/dist/index.js +0 -113
  364. package/src/package/heading/dist/index.mjs +0 -79
  365. package/src/package/hr/dist/index.d.mts +0 -6
  366. package/src/package/hr/dist/index.d.ts +0 -6
  367. package/src/package/hr/dist/index.js +0 -89
  368. package/src/package/hr/dist/index.mjs +0 -55
  369. package/src/package/html/dist/index.d.mts +0 -6
  370. package/src/package/html/dist/index.d.ts +0 -6
  371. package/src/package/html/dist/index.js +0 -79
  372. package/src/package/html/dist/index.mjs +0 -45
  373. package/src/package/img/dist/index.d.mts +0 -6
  374. package/src/package/img/dist/index.d.ts +0 -6
  375. package/src/package/img/dist/index.js +0 -94
  376. package/src/package/img/dist/index.mjs +0 -60
  377. package/src/package/link/dist/index.d.mts +0 -6
  378. package/src/package/link/dist/index.d.ts +0 -6
  379. package/src/package/link/dist/index.js +0 -90
  380. package/src/package/link/dist/index.mjs +0 -56
  381. package/src/package/markdown/dist/index.d.mts +0 -15
  382. package/src/package/markdown/dist/index.d.ts +0 -15
  383. package/src/package/markdown/dist/index.js +0 -92
  384. package/src/package/markdown/dist/index.mjs +0 -58
  385. package/src/package/preview/dist/index.d.mts +0 -12
  386. package/src/package/preview/dist/index.d.ts +0 -12
  387. package/src/package/preview/dist/index.js +0 -108
  388. package/src/package/preview/dist/index.mjs +0 -73
  389. package/src/package/render/dist/browser/index.d.mts +0 -24
  390. package/src/package/render/dist/browser/index.d.ts +0 -24
  391. package/src/package/render/dist/browser/index.js +0 -250
  392. package/src/package/render/dist/browser/index.mjs +0 -214
  393. package/src/package/render/dist/index.d.mts +0 -23
  394. package/src/package/render/dist/index.d.ts +0 -23
  395. package/src/package/render/dist/index.js +0 -768
  396. package/src/package/render/dist/index.mjs +0 -733
  397. package/src/package/render/dist/node/index.d.mts +0 -27
  398. package/src/package/render/dist/node/index.d.ts +0 -27
  399. package/src/package/render/dist/node/index.js +0 -212
  400. package/src/package/render/dist/node/index.mjs +0 -176
  401. package/src/package/row/dist/index.d.mts +0 -10
  402. package/src/package/row/dist/index.d.ts +0 -10
  403. package/src/package/row/dist/index.js +0 -93
  404. package/src/package/row/dist/index.mjs +0 -59
  405. package/src/package/section/dist/index.d.mts +0 -6
  406. package/src/package/section/dist/index.d.ts +0 -6
  407. package/src/package/section/dist/index.js +0 -93
  408. package/src/package/section/dist/index.mjs +0 -59
  409. package/src/package/tailwind/dist/index.d.ts +0 -19
  410. package/src/package/tailwind/dist/index.js +0 -48
  411. package/src/package/tailwind/dist/index.mjs +0 -17167
  412. package/src/package/tailwind/dist/tailwindcss/config.d.ts +0 -376
  413. package/src/package/tailwind/dist/tailwindcss/generated/.gitkeep +0 -0
  414. package/src/package/tailwind/dist/tailwindcss/generated/colors.d.ts +0 -298
  415. package/src/package/tailwind/dist/tailwindcss/generated/corePluginList.d.ts +0 -1
  416. package/src/package/tailwind/dist/tailwindcss/generated/default-theme.d.ts +0 -397
  417. package/src/package/tailwind/dist/tailwindcss/index.d.ts +0 -11
  418. package/src/package/text/dist/index.d.mts +0 -6
  419. package/src/package/text/dist/index.d.ts +0 -6
  420. package/src/package/text/dist/index.js +0 -89
  421. package/src/package/text/dist/index.mjs +0 -55
  422. package/src/utils/__snapshots__/get-email-component.spec.ts.snap +0 -3
  423. package/src/utils/caniemail/all-css-properties.ts +0 -358
  424. package/src/utils/caniemail/ast/__snapshots__/get-object-variables.spec.ts.snap +0 -74
  425. package/src/utils/caniemail/ast/__snapshots__/get-used-style-properties.spec.ts.snap +0 -24
  426. package/src/utils/caniemail/ast/get-object-variables.spec.ts +0 -19
  427. package/src/utils/caniemail/ast/get-object-variables.ts +0 -61
  428. package/src/utils/caniemail/ast/get-used-style-properties.spec.ts +0 -23
  429. package/src/utils/caniemail/ast/get-used-style-properties.ts +0 -91
  430. package/src/utils/caniemail/get-compatibility-stats-for-entry.ts +0 -118
  431. package/src/utils/caniemail/get-css-functions.ts +0 -25
  432. package/src/utils/caniemail/get-css-property-names.ts +0 -32
  433. package/src/utils/caniemail/get-css-property-with-value.ts +0 -14
  434. package/src/utils/caniemail/get-css-unit.ts +0 -3
  435. package/src/utils/caniemail/get-element-attributes.ts +0 -7
  436. package/src/utils/caniemail/get-element-names.ts +0 -20
  437. package/src/utils/caniemail/tailwind/generate-tailwind-rules.ts +0 -30
  438. package/src/utils/caniemail/tailwind/get-tailwind-config.ts +0 -187
  439. package/src/utils/caniemail/tailwind/get-tailwind-metadata.spec.ts +0 -25
  440. package/src/utils/caniemail/tailwind/get-tailwind-metadata.ts +0 -45
  441. package/src/utils/caniemail/tailwind/setup-tailwind-context.ts +0 -15
  442. package/src/utils/cn.ts +0 -6
  443. package/src/utils/constants.ts +0 -6
  444. package/src/utils/contains-email-template.spec.ts +0 -124
  445. package/src/utils/contains-email-template.ts +0 -33
  446. package/src/utils/copy-text-to-clipboard.ts +0 -7
  447. package/src/utils/get-email-component.spec.ts +0 -41
  448. package/src/utils/get-email-component.ts +0 -134
  449. package/src/utils/get-line-and-column-from-offset.spec.ts +0 -11
  450. package/src/utils/get-line-and-column-from-offset.ts +0 -11
  451. package/src/utils/improve-error-with-sourcemap.ts +0 -86
  452. package/src/utils/js-email-detection.spec.ts +0 -24
  453. package/src/utils/language-map.ts +0 -7
  454. package/src/utils/linting.ts +0 -60
  455. package/src/utils/load-stream.ts +0 -15
  456. package/src/utils/result.ts +0 -49
  457. package/src/utils/run-bundled-code.ts +0 -64
  458. package/src/utils/sanitize.ts +0 -6
  459. package/src/utils/static-node-modules-for-vm.ts +0 -93
  460. package/src/utils/testing/js-email-export-default.js +0 -17
  461. package/src/utils/testing/js-email-test.js +0 -18
  462. package/src/utils/testing/mdx-email-test.js +0 -128
  463. package/src/utils/testing/request-response-email.tsx +0 -9
  464. package/src/utils/types/as.ts +0 -26
  465. package/src/utils/types/email-template.ts +0 -8
  466. package/src/utils/types/error-object.ts +0 -11
  467. package/src/utils/unreachable.ts +0 -8
  468. package/tailwind-internals.d.ts +0 -133
  469. package/tailwind.config.ts +0 -99
  470. /package/src/{components/toolbar/results-table.tsx → utils/preview/hot-reloading/test/some-file.ts} +0 -0
@@ -1,333 +0,0 @@
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>;
@@ -1,100 +0,0 @@
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
- });
@@ -1,160 +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 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
- };
@@ -1,113 +0,0 @@
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
- });