├── .env.example ├── .gitignore ├── LICENSE ├── README.md ├── app ├── api │ └── send │ │ └── route.ts ├── email-template.tsx ├── layout.tsx └── page.tsx ├── e2e └── app.spec.ts ├── package-lock.json ├── package.json ├── playwright.config.ts ├── public ├── favicon.ico └── vercel.svg ├── styles ├── Home.module.css └── globals.css └── tsconfig.json /.env.example: -------------------------------------------------------------------------------- 1 | RESEND_API_KEY="" -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | /node_modules 5 | /.pnp 6 | .pnp.* 7 | .yarn/* 8 | !.yarn/patches 9 | !.yarn/plugins 10 | !.yarn/releases 11 | !.yarn/versions 12 | 13 | # testing 14 | /blob-report/ 15 | /coverage 16 | /playwright/.cache/ 17 | /playwright-report/ 18 | /test-results/ 19 | 20 | # next.js 21 | /.next/ 22 | /out/ 23 | 24 | # production 25 | /build 26 | 27 | # misc 28 | .DS_Store 29 | *.pem 30 | 31 | # debug 32 | npm-debug.log* 33 | yarn-debug.log* 34 | yarn-error.log* 35 | 36 | # local env files 37 | .env*.local 38 | 39 | # vercel 40 | .vercel 41 | 42 | # typescript 43 | *.tsbuildinfo 44 | next-env.d.ts 45 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2023 Plus Five Five, Inc. 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Resend with Next.js and Playwright 2 | 3 | This example shows how to do E2E testing with Resend, [Next.js](https://nextjs.org), and [Playwright](https://playwright.dev). 4 | 5 | ## Deploy your own 6 | 7 | Deploy the example using [Vercel](https://vercel.com): 8 | 9 | [![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/clone?repository-url=https://github.com/resend/resend-nextjs-playwright-example&project-name=resend-nextjs-playwright-example&repository-name=resend-nextjs-playwright-example&env=RESEND_API_KEY) 10 | 11 | ## Instructions 12 | 13 | 1. Define environment variables in `.env` file. 14 | 15 | ```sh 16 | cp .env.example .env 17 | ``` 18 | 19 | 2. Install dependencies: 20 | 21 | ```sh 22 | npm install 23 | ``` 24 | 25 | 3. Run Next.js locally: 26 | 27 | ```sh 28 | npm run dev 29 | ``` 30 | 31 | 4. Make a curl request 32 | 33 | ```sh 34 | curl -X http://localhost:3000/api/send 35 | ``` 36 | 37 | ## Running tests 38 | 39 | 1. Install Playwright: 40 | 41 | ```sh 42 | npx playwright install 43 | ``` 44 | 45 | 2. Run Playwright: 46 | 47 | ```sh 48 | npx playwright test 49 | ``` 50 | 51 | ## License 52 | 53 | MIT License -------------------------------------------------------------------------------- /app/api/send/route.ts: -------------------------------------------------------------------------------- 1 | import { EmailTemplate } from '../../email-template'; 2 | import { Resend } from 'resend'; 3 | import * as React from 'react'; 4 | 5 | const resend = new Resend(process.env.RESEND_API_KEY); 6 | 7 | export async function GET() { 8 | try { 9 | const { data, error } = await resend.emails.send({ 10 | from: 'Acme ', 11 | to: ['delivered@resend.dev'], 12 | subject: "Hello world", 13 | react: EmailTemplate({ firstName: "John" }) as React.ReactElement, 14 | }); 15 | 16 | if (error) { 17 | return Response.json({ error }, { status: 500 }); 18 | } 19 | 20 | return Response.json({ data }); 21 | } catch (error) { 22 | return Response.json({ error }, { status: 500 }); 23 | } 24 | } 25 | -------------------------------------------------------------------------------- /app/email-template.tsx: -------------------------------------------------------------------------------- 1 | import * as React from "react"; 2 | 3 | interface EmailTemplateProps { 4 | firstName: string; 5 | } 6 | 7 | export const EmailTemplate: React.FC> = ({ 8 | firstName, 9 | }) => ( 10 |
11 |

Welcome, {firstName}!

12 |
13 | ); 14 | 15 | export default EmailTemplate; 16 | -------------------------------------------------------------------------------- /app/layout.tsx: -------------------------------------------------------------------------------- 1 | export default function RootLayout({ 2 | children, 3 | }: { 4 | children: React.ReactNode; 5 | }) { 6 | return ( 7 | 8 | {children} 9 | 10 | ); 11 | } 12 | -------------------------------------------------------------------------------- /app/page.tsx: -------------------------------------------------------------------------------- 1 | import Link from "next/link"; 2 | 3 | export default function Page() { 4 | return ( 5 |
6 | Send email 7 |
8 | ); 9 | } 10 | -------------------------------------------------------------------------------- /e2e/app.spec.ts: -------------------------------------------------------------------------------- 1 | import { test, expect } from "@playwright/test"; 2 | 3 | test("mocks the response and doesn't call the Resend API", async ({ page }) => { 4 | // Sample response from Resend 5 | const body = JSON.stringify({ 6 | "data": { 7 | "id":"621f3ecf-f4d2-453a-9f82-21332409b4d2" 8 | } 9 | }); 10 | 11 | // Mock the api call before navigating 12 | await page.route('*/**/api/send', async route => { 13 | await route.fulfill({ 14 | body, 15 | contentType: 'application/json', 16 | status: 200, 17 | }); 18 | }); 19 | 20 | // Go to the page 21 | await page.goto("http://localhost:3000/api/send"); 22 | 23 | // Assert that the response is visible 24 | await expect(page.getByText(body)).toBeVisible(); 25 | }); 26 | 27 | test("does not mock the response and calls the Resend API", async ({ page }) => { 28 | // Go to the page 29 | await page.goto("http://localhost:3000/api/send"); 30 | 31 | // Assert that the response is visible 32 | await expect(page.getByText('id')).toBeVisible(); 33 | }); -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "resend-nextjs-playwright-example", 3 | "lockfileVersion": 3, 4 | "requires": true, 5 | "packages": { 6 | "": { 7 | "dependencies": { 8 | "next": "latest", 9 | "react": "^18.2.0", 10 | "react-dom": "^18.2.0", 11 | "resend": "^4.0.1" 12 | }, 13 | "devDependencies": { 14 | "@playwright/test": "^1.40.1", 15 | "@types/node": "20.10.4", 16 | "@types/react": "18.2.45", 17 | "typescript": "5.3.3" 18 | } 19 | }, 20 | "node_modules/@emnapi/runtime": { 21 | "version": "1.3.1", 22 | "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.3.1.tgz", 23 | "integrity": "sha512-kEBmG8KyqtxJZv+ygbEim+KCGtIq1fC22Ms3S4ziXmYKm8uyoLX0MHONVKwp+9opg390VaKRNt4a7A9NwmpNhw==", 24 | "optional": true, 25 | "dependencies": { 26 | "tslib": "^2.4.0" 27 | } 28 | }, 29 | "node_modules/@img/sharp-darwin-arm64": { 30 | "version": "0.33.5", 31 | "resolved": "https://registry.npmjs.org/@img/sharp-darwin-arm64/-/sharp-darwin-arm64-0.33.5.tgz", 32 | "integrity": "sha512-UT4p+iz/2H4twwAoLCqfA9UH5pI6DggwKEGuaPy7nCVQ8ZsiY5PIcrRvD1DzuY3qYL07NtIQcWnBSY/heikIFQ==", 33 | "cpu": [ 34 | "arm64" 35 | ], 36 | "optional": true, 37 | "os": [ 38 | "darwin" 39 | ], 40 | "engines": { 41 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 42 | }, 43 | "funding": { 44 | "url": "https://opencollective.com/libvips" 45 | }, 46 | "optionalDependencies": { 47 | "@img/sharp-libvips-darwin-arm64": "1.0.4" 48 | } 49 | }, 50 | "node_modules/@img/sharp-darwin-x64": { 51 | "version": "0.33.5", 52 | "resolved": "https://registry.npmjs.org/@img/sharp-darwin-x64/-/sharp-darwin-x64-0.33.5.tgz", 53 | "integrity": "sha512-fyHac4jIc1ANYGRDxtiqelIbdWkIuQaI84Mv45KvGRRxSAa7o7d1ZKAOBaYbnepLC1WqxfpimdeWfvqqSGwR2Q==", 54 | "cpu": [ 55 | "x64" 56 | ], 57 | "optional": true, 58 | "os": [ 59 | "darwin" 60 | ], 61 | "engines": { 62 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 63 | }, 64 | "funding": { 65 | "url": "https://opencollective.com/libvips" 66 | }, 67 | "optionalDependencies": { 68 | "@img/sharp-libvips-darwin-x64": "1.0.4" 69 | } 70 | }, 71 | "node_modules/@img/sharp-libvips-darwin-arm64": { 72 | "version": "1.0.4", 73 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-darwin-arm64/-/sharp-libvips-darwin-arm64-1.0.4.tgz", 74 | "integrity": "sha512-XblONe153h0O2zuFfTAbQYAX2JhYmDHeWikp1LM9Hul9gVPjFY427k6dFEcOL72O01QxQsWi761svJ/ev9xEDg==", 75 | "cpu": [ 76 | "arm64" 77 | ], 78 | "optional": true, 79 | "os": [ 80 | "darwin" 81 | ], 82 | "funding": { 83 | "url": "https://opencollective.com/libvips" 84 | } 85 | }, 86 | "node_modules/@img/sharp-libvips-darwin-x64": { 87 | "version": "1.0.4", 88 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-darwin-x64/-/sharp-libvips-darwin-x64-1.0.4.tgz", 89 | "integrity": "sha512-xnGR8YuZYfJGmWPvmlunFaWJsb9T/AO2ykoP3Fz/0X5XV2aoYBPkX6xqCQvUTKKiLddarLaxpzNe+b1hjeWHAQ==", 90 | "cpu": [ 91 | "x64" 92 | ], 93 | "optional": true, 94 | "os": [ 95 | "darwin" 96 | ], 97 | "funding": { 98 | "url": "https://opencollective.com/libvips" 99 | } 100 | }, 101 | "node_modules/@img/sharp-libvips-linux-arm": { 102 | "version": "1.0.5", 103 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-arm/-/sharp-libvips-linux-arm-1.0.5.tgz", 104 | "integrity": "sha512-gvcC4ACAOPRNATg/ov8/MnbxFDJqf/pDePbBnuBDcjsI8PssmjoKMAz4LtLaVi+OnSb5FK/yIOamqDwGmXW32g==", 105 | "cpu": [ 106 | "arm" 107 | ], 108 | "optional": true, 109 | "os": [ 110 | "linux" 111 | ], 112 | "funding": { 113 | "url": "https://opencollective.com/libvips" 114 | } 115 | }, 116 | "node_modules/@img/sharp-libvips-linux-arm64": { 117 | "version": "1.0.4", 118 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-arm64/-/sharp-libvips-linux-arm64-1.0.4.tgz", 119 | "integrity": "sha512-9B+taZ8DlyyqzZQnoeIvDVR/2F4EbMepXMc/NdVbkzsJbzkUjhXv/70GQJ7tdLA4YJgNP25zukcxpX2/SueNrA==", 120 | "cpu": [ 121 | "arm64" 122 | ], 123 | "optional": true, 124 | "os": [ 125 | "linux" 126 | ], 127 | "funding": { 128 | "url": "https://opencollective.com/libvips" 129 | } 130 | }, 131 | "node_modules/@img/sharp-libvips-linux-s390x": { 132 | "version": "1.0.4", 133 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-s390x/-/sharp-libvips-linux-s390x-1.0.4.tgz", 134 | "integrity": "sha512-u7Wz6ntiSSgGSGcjZ55im6uvTrOxSIS8/dgoVMoiGE9I6JAfU50yH5BoDlYA1tcuGS7g/QNtetJnxA6QEsCVTA==", 135 | "cpu": [ 136 | "s390x" 137 | ], 138 | "optional": true, 139 | "os": [ 140 | "linux" 141 | ], 142 | "funding": { 143 | "url": "https://opencollective.com/libvips" 144 | } 145 | }, 146 | "node_modules/@img/sharp-libvips-linux-x64": { 147 | "version": "1.0.4", 148 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-x64/-/sharp-libvips-linux-x64-1.0.4.tgz", 149 | "integrity": "sha512-MmWmQ3iPFZr0Iev+BAgVMb3ZyC4KeFc3jFxnNbEPas60e1cIfevbtuyf9nDGIzOaW9PdnDciJm+wFFaTlj5xYw==", 150 | "cpu": [ 151 | "x64" 152 | ], 153 | "optional": true, 154 | "os": [ 155 | "linux" 156 | ], 157 | "funding": { 158 | "url": "https://opencollective.com/libvips" 159 | } 160 | }, 161 | "node_modules/@img/sharp-libvips-linuxmusl-arm64": { 162 | "version": "1.0.4", 163 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linuxmusl-arm64/-/sharp-libvips-linuxmusl-arm64-1.0.4.tgz", 164 | "integrity": "sha512-9Ti+BbTYDcsbp4wfYib8Ctm1ilkugkA/uscUn6UXK1ldpC1JjiXbLfFZtRlBhjPZ5o1NCLiDbg8fhUPKStHoTA==", 165 | "cpu": [ 166 | "arm64" 167 | ], 168 | "optional": true, 169 | "os": [ 170 | "linux" 171 | ], 172 | "funding": { 173 | "url": "https://opencollective.com/libvips" 174 | } 175 | }, 176 | "node_modules/@img/sharp-libvips-linuxmusl-x64": { 177 | "version": "1.0.4", 178 | "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linuxmusl-x64/-/sharp-libvips-linuxmusl-x64-1.0.4.tgz", 179 | "integrity": "sha512-viYN1KX9m+/hGkJtvYYp+CCLgnJXwiQB39damAO7WMdKWlIhmYTfHjwSbQeUK/20vY154mwezd9HflVFM1wVSw==", 180 | "cpu": [ 181 | "x64" 182 | ], 183 | "optional": true, 184 | "os": [ 185 | "linux" 186 | ], 187 | "funding": { 188 | "url": "https://opencollective.com/libvips" 189 | } 190 | }, 191 | "node_modules/@img/sharp-linux-arm": { 192 | "version": "0.33.5", 193 | "resolved": "https://registry.npmjs.org/@img/sharp-linux-arm/-/sharp-linux-arm-0.33.5.tgz", 194 | "integrity": "sha512-JTS1eldqZbJxjvKaAkxhZmBqPRGmxgu+qFKSInv8moZ2AmT5Yib3EQ1c6gp493HvrvV8QgdOXdyaIBrhvFhBMQ==", 195 | "cpu": [ 196 | "arm" 197 | ], 198 | "optional": true, 199 | "os": [ 200 | "linux" 201 | ], 202 | "engines": { 203 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 204 | }, 205 | "funding": { 206 | "url": "https://opencollective.com/libvips" 207 | }, 208 | "optionalDependencies": { 209 | "@img/sharp-libvips-linux-arm": "1.0.5" 210 | } 211 | }, 212 | "node_modules/@img/sharp-linux-arm64": { 213 | "version": "0.33.5", 214 | "resolved": "https://registry.npmjs.org/@img/sharp-linux-arm64/-/sharp-linux-arm64-0.33.5.tgz", 215 | "integrity": "sha512-JMVv+AMRyGOHtO1RFBiJy/MBsgz0x4AWrT6QoEVVTyh1E39TrCUpTRI7mx9VksGX4awWASxqCYLCV4wBZHAYxA==", 216 | "cpu": [ 217 | "arm64" 218 | ], 219 | "optional": true, 220 | "os": [ 221 | "linux" 222 | ], 223 | "engines": { 224 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 225 | }, 226 | "funding": { 227 | "url": "https://opencollective.com/libvips" 228 | }, 229 | "optionalDependencies": { 230 | "@img/sharp-libvips-linux-arm64": "1.0.4" 231 | } 232 | }, 233 | "node_modules/@img/sharp-linux-s390x": { 234 | "version": "0.33.5", 235 | "resolved": "https://registry.npmjs.org/@img/sharp-linux-s390x/-/sharp-linux-s390x-0.33.5.tgz", 236 | "integrity": "sha512-y/5PCd+mP4CA/sPDKl2961b+C9d+vPAveS33s6Z3zfASk2j5upL6fXVPZi7ztePZ5CuH+1kW8JtvxgbuXHRa4Q==", 237 | "cpu": [ 238 | "s390x" 239 | ], 240 | "optional": true, 241 | "os": [ 242 | "linux" 243 | ], 244 | "engines": { 245 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 246 | }, 247 | "funding": { 248 | "url": "https://opencollective.com/libvips" 249 | }, 250 | "optionalDependencies": { 251 | "@img/sharp-libvips-linux-s390x": "1.0.4" 252 | } 253 | }, 254 | "node_modules/@img/sharp-linux-x64": { 255 | "version": "0.33.5", 256 | "resolved": "https://registry.npmjs.org/@img/sharp-linux-x64/-/sharp-linux-x64-0.33.5.tgz", 257 | "integrity": "sha512-opC+Ok5pRNAzuvq1AG0ar+1owsu842/Ab+4qvU879ippJBHvyY5n2mxF1izXqkPYlGuP/M556uh53jRLJmzTWA==", 258 | "cpu": [ 259 | "x64" 260 | ], 261 | "optional": true, 262 | "os": [ 263 | "linux" 264 | ], 265 | "engines": { 266 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 267 | }, 268 | "funding": { 269 | "url": "https://opencollective.com/libvips" 270 | }, 271 | "optionalDependencies": { 272 | "@img/sharp-libvips-linux-x64": "1.0.4" 273 | } 274 | }, 275 | "node_modules/@img/sharp-linuxmusl-arm64": { 276 | "version": "0.33.5", 277 | "resolved": "https://registry.npmjs.org/@img/sharp-linuxmusl-arm64/-/sharp-linuxmusl-arm64-0.33.5.tgz", 278 | "integrity": "sha512-XrHMZwGQGvJg2V/oRSUfSAfjfPxO+4DkiRh6p2AFjLQztWUuY/o8Mq0eMQVIY7HJ1CDQUJlxGGZRw1a5bqmd1g==", 279 | "cpu": [ 280 | "arm64" 281 | ], 282 | "optional": true, 283 | "os": [ 284 | "linux" 285 | ], 286 | "engines": { 287 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 288 | }, 289 | "funding": { 290 | "url": "https://opencollective.com/libvips" 291 | }, 292 | "optionalDependencies": { 293 | "@img/sharp-libvips-linuxmusl-arm64": "1.0.4" 294 | } 295 | }, 296 | "node_modules/@img/sharp-linuxmusl-x64": { 297 | "version": "0.33.5", 298 | "resolved": "https://registry.npmjs.org/@img/sharp-linuxmusl-x64/-/sharp-linuxmusl-x64-0.33.5.tgz", 299 | "integrity": "sha512-WT+d/cgqKkkKySYmqoZ8y3pxx7lx9vVejxW/W4DOFMYVSkErR+w7mf2u8m/y4+xHe7yY9DAXQMWQhpnMuFfScw==", 300 | "cpu": [ 301 | "x64" 302 | ], 303 | "optional": true, 304 | "os": [ 305 | "linux" 306 | ], 307 | "engines": { 308 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 309 | }, 310 | "funding": { 311 | "url": "https://opencollective.com/libvips" 312 | }, 313 | "optionalDependencies": { 314 | "@img/sharp-libvips-linuxmusl-x64": "1.0.4" 315 | } 316 | }, 317 | "node_modules/@img/sharp-wasm32": { 318 | "version": "0.33.5", 319 | "resolved": "https://registry.npmjs.org/@img/sharp-wasm32/-/sharp-wasm32-0.33.5.tgz", 320 | "integrity": "sha512-ykUW4LVGaMcU9lu9thv85CbRMAwfeadCJHRsg2GmeRa/cJxsVY9Rbd57JcMxBkKHag5U/x7TSBpScF4U8ElVzg==", 321 | "cpu": [ 322 | "wasm32" 323 | ], 324 | "optional": true, 325 | "dependencies": { 326 | "@emnapi/runtime": "^1.2.0" 327 | }, 328 | "engines": { 329 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 330 | }, 331 | "funding": { 332 | "url": "https://opencollective.com/libvips" 333 | } 334 | }, 335 | "node_modules/@img/sharp-win32-ia32": { 336 | "version": "0.33.5", 337 | "resolved": "https://registry.npmjs.org/@img/sharp-win32-ia32/-/sharp-win32-ia32-0.33.5.tgz", 338 | "integrity": "sha512-T36PblLaTwuVJ/zw/LaH0PdZkRz5rd3SmMHX8GSmR7vtNSP5Z6bQkExdSK7xGWyxLw4sUknBuugTelgw2faBbQ==", 339 | "cpu": [ 340 | "ia32" 341 | ], 342 | "optional": true, 343 | "os": [ 344 | "win32" 345 | ], 346 | "engines": { 347 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 348 | }, 349 | "funding": { 350 | "url": "https://opencollective.com/libvips" 351 | } 352 | }, 353 | "node_modules/@img/sharp-win32-x64": { 354 | "version": "0.33.5", 355 | "resolved": "https://registry.npmjs.org/@img/sharp-win32-x64/-/sharp-win32-x64-0.33.5.tgz", 356 | "integrity": "sha512-MpY/o8/8kj+EcnxwvrP4aTJSWw/aZ7JIGR4aBeZkZw5B7/Jn+tY9/VNwtcoGmdT7GfggGIU4kygOMSbYnOrAbg==", 357 | "cpu": [ 358 | "x64" 359 | ], 360 | "optional": true, 361 | "os": [ 362 | "win32" 363 | ], 364 | "engines": { 365 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 366 | }, 367 | "funding": { 368 | "url": "https://opencollective.com/libvips" 369 | } 370 | }, 371 | "node_modules/@isaacs/cliui": { 372 | "version": "8.0.2", 373 | "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", 374 | "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", 375 | "dependencies": { 376 | "string-width": "^5.1.2", 377 | "string-width-cjs": "npm:string-width@^4.2.0", 378 | "strip-ansi": "^7.0.1", 379 | "strip-ansi-cjs": "npm:strip-ansi@^6.0.1", 380 | "wrap-ansi": "^8.1.0", 381 | "wrap-ansi-cjs": "npm:wrap-ansi@^7.0.0" 382 | }, 383 | "engines": { 384 | "node": ">=12" 385 | } 386 | }, 387 | "node_modules/@next/env": { 388 | "version": "15.1.2", 389 | "resolved": "https://registry.npmjs.org/@next/env/-/env-15.1.2.tgz", 390 | "integrity": "sha512-Hm3jIGsoUl6RLB1vzY+dZeqb+/kWPZ+h34yiWxW0dV87l8Im/eMOwpOA+a0L78U0HM04syEjXuRlCozqpwuojQ==" 391 | }, 392 | "node_modules/@next/swc-darwin-arm64": { 393 | "version": "15.1.2", 394 | "resolved": "https://registry.npmjs.org/@next/swc-darwin-arm64/-/swc-darwin-arm64-15.1.2.tgz", 395 | "integrity": "sha512-b9TN7q+j5/7+rGLhFAVZiKJGIASuo8tWvInGfAd8wsULjB1uNGRCj1z1WZwwPWzVQbIKWFYqc+9L7W09qwt52w==", 396 | "cpu": [ 397 | "arm64" 398 | ], 399 | "optional": true, 400 | "os": [ 401 | "darwin" 402 | ], 403 | "engines": { 404 | "node": ">= 10" 405 | } 406 | }, 407 | "node_modules/@next/swc-darwin-x64": { 408 | "version": "15.1.2", 409 | "resolved": "https://registry.npmjs.org/@next/swc-darwin-x64/-/swc-darwin-x64-15.1.2.tgz", 410 | "integrity": "sha512-caR62jNDUCU+qobStO6YJ05p9E+LR0EoXh1EEmyU69cYydsAy7drMcOlUlRtQihM6K6QfvNwJuLhsHcCzNpqtA==", 411 | "cpu": [ 412 | "x64" 413 | ], 414 | "optional": true, 415 | "os": [ 416 | "darwin" 417 | ], 418 | "engines": { 419 | "node": ">= 10" 420 | } 421 | }, 422 | "node_modules/@next/swc-linux-arm64-gnu": { 423 | "version": "15.1.2", 424 | "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-gnu/-/swc-linux-arm64-gnu-15.1.2.tgz", 425 | "integrity": "sha512-fHHXBusURjBmN6VBUtu6/5s7cCeEkuGAb/ZZiGHBLVBXMBy4D5QpM8P33Or8JD1nlOjm/ZT9sEE5HouQ0F+hUA==", 426 | "cpu": [ 427 | "arm64" 428 | ], 429 | "optional": true, 430 | "os": [ 431 | "linux" 432 | ], 433 | "engines": { 434 | "node": ">= 10" 435 | } 436 | }, 437 | "node_modules/@next/swc-linux-arm64-musl": { 438 | "version": "15.1.2", 439 | "resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-musl/-/swc-linux-arm64-musl-15.1.2.tgz", 440 | "integrity": "sha512-9CF1Pnivij7+M3G74lxr+e9h6o2YNIe7QtExWq1KUK4hsOLTBv6FJikEwCaC3NeYTflzrm69E5UfwEAbV2U9/g==", 441 | "cpu": [ 442 | "arm64" 443 | ], 444 | "optional": true, 445 | "os": [ 446 | "linux" 447 | ], 448 | "engines": { 449 | "node": ">= 10" 450 | } 451 | }, 452 | "node_modules/@next/swc-linux-x64-gnu": { 453 | "version": "15.1.2", 454 | "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-gnu/-/swc-linux-x64-gnu-15.1.2.tgz", 455 | "integrity": "sha512-tINV7WmcTUf4oM/eN3Yuu/f8jQ5C6AkueZPKeALs/qfdfX57eNv4Ij7rt0SA6iZ8+fMobVfcFVv664Op0caCCg==", 456 | "cpu": [ 457 | "x64" 458 | ], 459 | "optional": true, 460 | "os": [ 461 | "linux" 462 | ], 463 | "engines": { 464 | "node": ">= 10" 465 | } 466 | }, 467 | "node_modules/@next/swc-linux-x64-musl": { 468 | "version": "15.1.2", 469 | "resolved": "https://registry.npmjs.org/@next/swc-linux-x64-musl/-/swc-linux-x64-musl-15.1.2.tgz", 470 | "integrity": "sha512-jf2IseC4WRsGkzeUw/cK3wci9pxR53GlLAt30+y+B+2qAQxMw6WAC3QrANIKxkcoPU3JFh/10uFfmoMDF9JXKg==", 471 | "cpu": [ 472 | "x64" 473 | ], 474 | "optional": true, 475 | "os": [ 476 | "linux" 477 | ], 478 | "engines": { 479 | "node": ">= 10" 480 | } 481 | }, 482 | "node_modules/@next/swc-win32-arm64-msvc": { 483 | "version": "15.1.2", 484 | "resolved": "https://registry.npmjs.org/@next/swc-win32-arm64-msvc/-/swc-win32-arm64-msvc-15.1.2.tgz", 485 | "integrity": "sha512-wvg7MlfnaociP7k8lxLX4s2iBJm4BrNiNFhVUY+Yur5yhAJHfkS8qPPeDEUH8rQiY0PX3u/P7Q/wcg6Mv6GSAA==", 486 | "cpu": [ 487 | "arm64" 488 | ], 489 | "optional": true, 490 | "os": [ 491 | "win32" 492 | ], 493 | "engines": { 494 | "node": ">= 10" 495 | } 496 | }, 497 | "node_modules/@next/swc-win32-x64-msvc": { 498 | "version": "15.1.2", 499 | "resolved": "https://registry.npmjs.org/@next/swc-win32-x64-msvc/-/swc-win32-x64-msvc-15.1.2.tgz", 500 | "integrity": "sha512-D3cNA8NoT3aWISWmo7HF5Eyko/0OdOO+VagkoJuiTk7pyX3P/b+n8XA/MYvyR+xSVcbKn68B1rY9fgqjNISqzQ==", 501 | "cpu": [ 502 | "x64" 503 | ], 504 | "optional": true, 505 | "os": [ 506 | "win32" 507 | ], 508 | "engines": { 509 | "node": ">= 10" 510 | } 511 | }, 512 | "node_modules/@one-ini/wasm": { 513 | "version": "0.1.1", 514 | "resolved": "https://registry.npmjs.org/@one-ini/wasm/-/wasm-0.1.1.tgz", 515 | "integrity": "sha512-XuySG1E38YScSJoMlqovLru4KTUNSjgVTIjyh7qMX6aNN5HY5Ct5LhRJdxO79JtTzKfzV/bnWpz+zquYrISsvw==" 516 | }, 517 | "node_modules/@pkgjs/parseargs": { 518 | "version": "0.11.0", 519 | "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", 520 | "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", 521 | "optional": true, 522 | "engines": { 523 | "node": ">=14" 524 | } 525 | }, 526 | "node_modules/@playwright/test": { 527 | "version": "1.49.1", 528 | "resolved": "https://registry.npmjs.org/@playwright/test/-/test-1.49.1.tgz", 529 | "integrity": "sha512-Ky+BVzPz8pL6PQxHqNRW1k3mIyv933LML7HktS8uik0bUXNCdPhoS/kLihiO1tMf/egaJb4IutXd7UywvXEW+g==", 530 | "devOptional": true, 531 | "dependencies": { 532 | "playwright": "1.49.1" 533 | }, 534 | "bin": { 535 | "playwright": "cli.js" 536 | }, 537 | "engines": { 538 | "node": ">=18" 539 | } 540 | }, 541 | "node_modules/@react-email/render": { 542 | "version": "1.0.1", 543 | "resolved": "https://registry.npmjs.org/@react-email/render/-/render-1.0.1.tgz", 544 | "integrity": "sha512-W3gTrcmLOVYnG80QuUp22ReIT/xfLsVJ+n7ghSlG2BITB8evNABn1AO2rGQoXuK84zKtDAlxCdm3hRyIpZdGSA==", 545 | "dependencies": { 546 | "html-to-text": "9.0.5", 547 | "js-beautify": "^1.14.11", 548 | "react-promise-suspense": "0.3.4" 549 | }, 550 | "engines": { 551 | "node": ">=18.0.0" 552 | }, 553 | "peerDependencies": { 554 | "react": "^18.0 || ^19.0 || ^19.0.0-rc", 555 | "react-dom": "^18.0 || ^19.0 || ^19.0.0-rc" 556 | } 557 | }, 558 | "node_modules/@selderee/plugin-htmlparser2": { 559 | "version": "0.11.0", 560 | "resolved": "https://registry.npmjs.org/@selderee/plugin-htmlparser2/-/plugin-htmlparser2-0.11.0.tgz", 561 | "integrity": "sha512-P33hHGdldxGabLFjPPpaTxVolMrzrcegejx+0GxjrIb9Zv48D8yAIA/QTDR2dFl7Uz7urX8aX6+5bCZslr+gWQ==", 562 | "dependencies": { 563 | "domhandler": "^5.0.3", 564 | "selderee": "^0.11.0" 565 | }, 566 | "funding": { 567 | "url": "https://ko-fi.com/killymxi" 568 | } 569 | }, 570 | "node_modules/@swc/counter": { 571 | "version": "0.1.3", 572 | "resolved": "https://registry.npmjs.org/@swc/counter/-/counter-0.1.3.tgz", 573 | "integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==" 574 | }, 575 | "node_modules/@swc/helpers": { 576 | "version": "0.5.15", 577 | "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.15.tgz", 578 | "integrity": "sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g==", 579 | "dependencies": { 580 | "tslib": "^2.8.0" 581 | } 582 | }, 583 | "node_modules/@types/node": { 584 | "version": "20.10.4", 585 | "resolved": "https://registry.npmjs.org/@types/node/-/node-20.10.4.tgz", 586 | "integrity": "sha512-D08YG6rr8X90YB56tSIuBaddy/UXAA9RKJoFvrsnogAum/0pmjkgi4+2nx96A330FmioegBWmEYQ+syqCFaveg==", 587 | "dev": true, 588 | "dependencies": { 589 | "undici-types": "~5.26.4" 590 | } 591 | }, 592 | "node_modules/@types/prop-types": { 593 | "version": "15.7.14", 594 | "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.14.tgz", 595 | "integrity": "sha512-gNMvNH49DJ7OJYv+KAKn0Xp45p8PLl6zo2YnvDIbTd4J6MER2BmWN49TG7n9LvkyihINxeKW8+3bfS2yDC9dzQ==", 596 | "dev": true 597 | }, 598 | "node_modules/@types/react": { 599 | "version": "18.2.45", 600 | "resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.45.tgz", 601 | "integrity": "sha512-TtAxCNrlrBp8GoeEp1npd5g+d/OejJHFxS3OWmrPBMFaVQMSN0OFySozJio5BHxTuTeug00AVXVAjfDSfk+lUg==", 602 | "dev": true, 603 | "dependencies": { 604 | "@types/prop-types": "*", 605 | "@types/scheduler": "*", 606 | "csstype": "^3.0.2" 607 | } 608 | }, 609 | "node_modules/@types/scheduler": { 610 | "version": "0.23.0", 611 | "resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.23.0.tgz", 612 | "integrity": "sha512-YIoDCTH3Af6XM5VuwGG/QL/CJqga1Zm3NkU3HZ4ZHK2fRMPYP1VczsTUqtsf43PH/iJNVlPHAo2oWX7BSdB2Hw==", 613 | "dev": true 614 | }, 615 | "node_modules/abbrev": { 616 | "version": "2.0.0", 617 | "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-2.0.0.tgz", 618 | "integrity": "sha512-6/mh1E2u2YgEsCHdY0Yx5oW+61gZU+1vXaoiHHrpKeuRNNgFvS+/jrwHiQhB5apAf5oB7UB7E19ol2R2LKH8hQ==", 619 | "engines": { 620 | "node": "^14.17.0 || ^16.13.0 || >=18.0.0" 621 | } 622 | }, 623 | "node_modules/ansi-regex": { 624 | "version": "6.1.0", 625 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz", 626 | "integrity": "sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA==", 627 | "engines": { 628 | "node": ">=12" 629 | }, 630 | "funding": { 631 | "url": "https://github.com/chalk/ansi-regex?sponsor=1" 632 | } 633 | }, 634 | "node_modules/ansi-styles": { 635 | "version": "6.2.1", 636 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz", 637 | "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==", 638 | "engines": { 639 | "node": ">=12" 640 | }, 641 | "funding": { 642 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 643 | } 644 | }, 645 | "node_modules/balanced-match": { 646 | "version": "1.0.2", 647 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 648 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" 649 | }, 650 | "node_modules/brace-expansion": { 651 | "version": "2.0.1", 652 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", 653 | "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", 654 | "dependencies": { 655 | "balanced-match": "^1.0.0" 656 | } 657 | }, 658 | "node_modules/busboy": { 659 | "version": "1.6.0", 660 | "resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz", 661 | "integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==", 662 | "dependencies": { 663 | "streamsearch": "^1.1.0" 664 | }, 665 | "engines": { 666 | "node": ">=10.16.0" 667 | } 668 | }, 669 | "node_modules/caniuse-lite": { 670 | "version": "1.0.30001690", 671 | "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001690.tgz", 672 | "integrity": "sha512-5ExiE3qQN6oF8Clf8ifIDcMRCRE/dMGcETG/XGMD8/XiXm6HXQgQTh1yZYLXXpSOsEUlJm1Xr7kGULZTuGtP/w==", 673 | "funding": [ 674 | { 675 | "type": "opencollective", 676 | "url": "https://opencollective.com/browserslist" 677 | }, 678 | { 679 | "type": "tidelift", 680 | "url": "https://tidelift.com/funding/github/npm/caniuse-lite" 681 | }, 682 | { 683 | "type": "github", 684 | "url": "https://github.com/sponsors/ai" 685 | } 686 | ] 687 | }, 688 | "node_modules/client-only": { 689 | "version": "0.0.1", 690 | "resolved": "https://registry.npmjs.org/client-only/-/client-only-0.0.1.tgz", 691 | "integrity": "sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==" 692 | }, 693 | "node_modules/color": { 694 | "version": "4.2.3", 695 | "resolved": "https://registry.npmjs.org/color/-/color-4.2.3.tgz", 696 | "integrity": "sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A==", 697 | "optional": true, 698 | "dependencies": { 699 | "color-convert": "^2.0.1", 700 | "color-string": "^1.9.0" 701 | }, 702 | "engines": { 703 | "node": ">=12.5.0" 704 | } 705 | }, 706 | "node_modules/color-convert": { 707 | "version": "2.0.1", 708 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 709 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 710 | "dependencies": { 711 | "color-name": "~1.1.4" 712 | }, 713 | "engines": { 714 | "node": ">=7.0.0" 715 | } 716 | }, 717 | "node_modules/color-name": { 718 | "version": "1.1.4", 719 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 720 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" 721 | }, 722 | "node_modules/color-string": { 723 | "version": "1.9.1", 724 | "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.9.1.tgz", 725 | "integrity": "sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==", 726 | "optional": true, 727 | "dependencies": { 728 | "color-name": "^1.0.0", 729 | "simple-swizzle": "^0.2.2" 730 | } 731 | }, 732 | "node_modules/commander": { 733 | "version": "10.0.1", 734 | "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.1.tgz", 735 | "integrity": "sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==", 736 | "engines": { 737 | "node": ">=14" 738 | } 739 | }, 740 | "node_modules/config-chain": { 741 | "version": "1.1.13", 742 | "resolved": "https://registry.npmjs.org/config-chain/-/config-chain-1.1.13.tgz", 743 | "integrity": "sha512-qj+f8APARXHrM0hraqXYb2/bOVSV4PvJQlNZ/DVj0QrmNM2q2euizkeuVckQ57J+W0mRH6Hvi+k50M4Jul2VRQ==", 744 | "dependencies": { 745 | "ini": "^1.3.4", 746 | "proto-list": "~1.2.1" 747 | } 748 | }, 749 | "node_modules/cross-spawn": { 750 | "version": "7.0.6", 751 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", 752 | "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", 753 | "dependencies": { 754 | "path-key": "^3.1.0", 755 | "shebang-command": "^2.0.0", 756 | "which": "^2.0.1" 757 | }, 758 | "engines": { 759 | "node": ">= 8" 760 | } 761 | }, 762 | "node_modules/csstype": { 763 | "version": "3.1.3", 764 | "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.3.tgz", 765 | "integrity": "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw==", 766 | "dev": true 767 | }, 768 | "node_modules/deepmerge": { 769 | "version": "4.3.1", 770 | "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", 771 | "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", 772 | "engines": { 773 | "node": ">=0.10.0" 774 | } 775 | }, 776 | "node_modules/detect-libc": { 777 | "version": "2.0.3", 778 | "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz", 779 | "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==", 780 | "optional": true, 781 | "engines": { 782 | "node": ">=8" 783 | } 784 | }, 785 | "node_modules/dom-serializer": { 786 | "version": "2.0.0", 787 | "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-2.0.0.tgz", 788 | "integrity": "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==", 789 | "dependencies": { 790 | "domelementtype": "^2.3.0", 791 | "domhandler": "^5.0.2", 792 | "entities": "^4.2.0" 793 | }, 794 | "funding": { 795 | "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" 796 | } 797 | }, 798 | "node_modules/domelementtype": { 799 | "version": "2.3.0", 800 | "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz", 801 | "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", 802 | "funding": [ 803 | { 804 | "type": "github", 805 | "url": "https://github.com/sponsors/fb55" 806 | } 807 | ] 808 | }, 809 | "node_modules/domhandler": { 810 | "version": "5.0.3", 811 | "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-5.0.3.tgz", 812 | "integrity": "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==", 813 | "dependencies": { 814 | "domelementtype": "^2.3.0" 815 | }, 816 | "engines": { 817 | "node": ">= 4" 818 | }, 819 | "funding": { 820 | "url": "https://github.com/fb55/domhandler?sponsor=1" 821 | } 822 | }, 823 | "node_modules/domutils": { 824 | "version": "3.2.1", 825 | "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.2.1.tgz", 826 | "integrity": "sha512-xWXmuRnN9OMP6ptPd2+H0cCbcYBULa5YDTbMm/2lvkWvNA3O4wcW+GvzooqBuNM8yy6pl3VIAeJTUUWUbfI5Fw==", 827 | "dependencies": { 828 | "dom-serializer": "^2.0.0", 829 | "domelementtype": "^2.3.0", 830 | "domhandler": "^5.0.3" 831 | }, 832 | "funding": { 833 | "url": "https://github.com/fb55/domutils?sponsor=1" 834 | } 835 | }, 836 | "node_modules/eastasianwidth": { 837 | "version": "0.2.0", 838 | "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", 839 | "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==" 840 | }, 841 | "node_modules/editorconfig": { 842 | "version": "1.0.4", 843 | "resolved": "https://registry.npmjs.org/editorconfig/-/editorconfig-1.0.4.tgz", 844 | "integrity": "sha512-L9Qe08KWTlqYMVvMcTIvMAdl1cDUubzRNYL+WfA4bLDMHe4nemKkpmYzkznE1FwLKu0EEmy6obgQKzMJrg4x9Q==", 845 | "dependencies": { 846 | "@one-ini/wasm": "0.1.1", 847 | "commander": "^10.0.0", 848 | "minimatch": "9.0.1", 849 | "semver": "^7.5.3" 850 | }, 851 | "bin": { 852 | "editorconfig": "bin/editorconfig" 853 | }, 854 | "engines": { 855 | "node": ">=14" 856 | } 857 | }, 858 | "node_modules/emoji-regex": { 859 | "version": "9.2.2", 860 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", 861 | "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==" 862 | }, 863 | "node_modules/entities": { 864 | "version": "4.5.0", 865 | "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", 866 | "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", 867 | "engines": { 868 | "node": ">=0.12" 869 | }, 870 | "funding": { 871 | "url": "https://github.com/fb55/entities?sponsor=1" 872 | } 873 | }, 874 | "node_modules/fast-deep-equal": { 875 | "version": "2.0.1", 876 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", 877 | "integrity": "sha512-bCK/2Z4zLidyB4ReuIsvALH6w31YfAQDmXMqMx6FyfHqvBxtjC0eRumeSu4Bs3XtXwpyIywtSTrVT99BxY1f9w==" 878 | }, 879 | "node_modules/foreground-child": { 880 | "version": "3.3.0", 881 | "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.0.tgz", 882 | "integrity": "sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg==", 883 | "dependencies": { 884 | "cross-spawn": "^7.0.0", 885 | "signal-exit": "^4.0.1" 886 | }, 887 | "engines": { 888 | "node": ">=14" 889 | }, 890 | "funding": { 891 | "url": "https://github.com/sponsors/isaacs" 892 | } 893 | }, 894 | "node_modules/fsevents": { 895 | "version": "2.3.2", 896 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 897 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 898 | "dev": true, 899 | "hasInstallScript": true, 900 | "optional": true, 901 | "os": [ 902 | "darwin" 903 | ], 904 | "engines": { 905 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 906 | } 907 | }, 908 | "node_modules/glob": { 909 | "version": "10.4.5", 910 | "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz", 911 | "integrity": "sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==", 912 | "dependencies": { 913 | "foreground-child": "^3.1.0", 914 | "jackspeak": "^3.1.2", 915 | "minimatch": "^9.0.4", 916 | "minipass": "^7.1.2", 917 | "package-json-from-dist": "^1.0.0", 918 | "path-scurry": "^1.11.1" 919 | }, 920 | "bin": { 921 | "glob": "dist/esm/bin.mjs" 922 | }, 923 | "funding": { 924 | "url": "https://github.com/sponsors/isaacs" 925 | } 926 | }, 927 | "node_modules/glob/node_modules/minimatch": { 928 | "version": "9.0.5", 929 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", 930 | "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", 931 | "dependencies": { 932 | "brace-expansion": "^2.0.1" 933 | }, 934 | "engines": { 935 | "node": ">=16 || 14 >=14.17" 936 | }, 937 | "funding": { 938 | "url": "https://github.com/sponsors/isaacs" 939 | } 940 | }, 941 | "node_modules/html-to-text": { 942 | "version": "9.0.5", 943 | "resolved": "https://registry.npmjs.org/html-to-text/-/html-to-text-9.0.5.tgz", 944 | "integrity": "sha512-qY60FjREgVZL03vJU6IfMV4GDjGBIoOyvuFdpBDIX9yTlDw0TjxVBQp+P8NvpdIXNJvfWBTNul7fsAQJq2FNpg==", 945 | "dependencies": { 946 | "@selderee/plugin-htmlparser2": "^0.11.0", 947 | "deepmerge": "^4.3.1", 948 | "dom-serializer": "^2.0.0", 949 | "htmlparser2": "^8.0.2", 950 | "selderee": "^0.11.0" 951 | }, 952 | "engines": { 953 | "node": ">=14" 954 | } 955 | }, 956 | "node_modules/htmlparser2": { 957 | "version": "8.0.2", 958 | "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.2.tgz", 959 | "integrity": "sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==", 960 | "funding": [ 961 | "https://github.com/fb55/htmlparser2?sponsor=1", 962 | { 963 | "type": "github", 964 | "url": "https://github.com/sponsors/fb55" 965 | } 966 | ], 967 | "dependencies": { 968 | "domelementtype": "^2.3.0", 969 | "domhandler": "^5.0.3", 970 | "domutils": "^3.0.1", 971 | "entities": "^4.4.0" 972 | } 973 | }, 974 | "node_modules/ini": { 975 | "version": "1.3.8", 976 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", 977 | "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==" 978 | }, 979 | "node_modules/is-arrayish": { 980 | "version": "0.3.2", 981 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", 982 | "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==", 983 | "optional": true 984 | }, 985 | "node_modules/is-fullwidth-code-point": { 986 | "version": "3.0.0", 987 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 988 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 989 | "engines": { 990 | "node": ">=8" 991 | } 992 | }, 993 | "node_modules/isexe": { 994 | "version": "2.0.0", 995 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", 996 | "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==" 997 | }, 998 | "node_modules/jackspeak": { 999 | "version": "3.4.3", 1000 | "resolved": "https://registry.npmjs.org/jackspeak/-/jackspeak-3.4.3.tgz", 1001 | "integrity": "sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==", 1002 | "dependencies": { 1003 | "@isaacs/cliui": "^8.0.2" 1004 | }, 1005 | "funding": { 1006 | "url": "https://github.com/sponsors/isaacs" 1007 | }, 1008 | "optionalDependencies": { 1009 | "@pkgjs/parseargs": "^0.11.0" 1010 | } 1011 | }, 1012 | "node_modules/js-beautify": { 1013 | "version": "1.15.1", 1014 | "resolved": "https://registry.npmjs.org/js-beautify/-/js-beautify-1.15.1.tgz", 1015 | "integrity": "sha512-ESjNzSlt/sWE8sciZH8kBF8BPlwXPwhR6pWKAw8bw4Bwj+iZcnKW6ONWUutJ7eObuBZQpiIb8S7OYspWrKt7rA==", 1016 | "dependencies": { 1017 | "config-chain": "^1.1.13", 1018 | "editorconfig": "^1.0.4", 1019 | "glob": "^10.3.3", 1020 | "js-cookie": "^3.0.5", 1021 | "nopt": "^7.2.0" 1022 | }, 1023 | "bin": { 1024 | "css-beautify": "js/bin/css-beautify.js", 1025 | "html-beautify": "js/bin/html-beautify.js", 1026 | "js-beautify": "js/bin/js-beautify.js" 1027 | }, 1028 | "engines": { 1029 | "node": ">=14" 1030 | } 1031 | }, 1032 | "node_modules/js-cookie": { 1033 | "version": "3.0.5", 1034 | "resolved": "https://registry.npmjs.org/js-cookie/-/js-cookie-3.0.5.tgz", 1035 | "integrity": "sha512-cEiJEAEoIbWfCZYKWhVwFuvPX1gETRYPw6LlaTKoxD3s2AkXzkCjnp6h0V77ozyqj0jakteJ4YqDJT830+lVGw==", 1036 | "engines": { 1037 | "node": ">=14" 1038 | } 1039 | }, 1040 | "node_modules/js-tokens": { 1041 | "version": "4.0.0", 1042 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 1043 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" 1044 | }, 1045 | "node_modules/leac": { 1046 | "version": "0.6.0", 1047 | "resolved": "https://registry.npmjs.org/leac/-/leac-0.6.0.tgz", 1048 | "integrity": "sha512-y+SqErxb8h7nE/fiEX07jsbuhrpO9lL8eca7/Y1nuWV2moNlXhyd59iDGcRf6moVyDMbmTNzL40SUyrFU/yDpg==", 1049 | "funding": { 1050 | "url": "https://ko-fi.com/killymxi" 1051 | } 1052 | }, 1053 | "node_modules/loose-envify": { 1054 | "version": "1.4.0", 1055 | "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", 1056 | "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", 1057 | "dependencies": { 1058 | "js-tokens": "^3.0.0 || ^4.0.0" 1059 | }, 1060 | "bin": { 1061 | "loose-envify": "cli.js" 1062 | } 1063 | }, 1064 | "node_modules/lru-cache": { 1065 | "version": "10.4.3", 1066 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", 1067 | "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==" 1068 | }, 1069 | "node_modules/minimatch": { 1070 | "version": "9.0.1", 1071 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.1.tgz", 1072 | "integrity": "sha512-0jWhJpD/MdhPXwPuiRkCbfYfSKp2qnn2eOc279qI7f+osl/l+prKSrvhg157zSYvx/1nmgn2NqdT6k2Z7zSH9w==", 1073 | "dependencies": { 1074 | "brace-expansion": "^2.0.1" 1075 | }, 1076 | "engines": { 1077 | "node": ">=16 || 14 >=14.17" 1078 | }, 1079 | "funding": { 1080 | "url": "https://github.com/sponsors/isaacs" 1081 | } 1082 | }, 1083 | "node_modules/minipass": { 1084 | "version": "7.1.2", 1085 | "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", 1086 | "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", 1087 | "engines": { 1088 | "node": ">=16 || 14 >=14.17" 1089 | } 1090 | }, 1091 | "node_modules/nanoid": { 1092 | "version": "3.3.8", 1093 | "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz", 1094 | "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==", 1095 | "funding": [ 1096 | { 1097 | "type": "github", 1098 | "url": "https://github.com/sponsors/ai" 1099 | } 1100 | ], 1101 | "bin": { 1102 | "nanoid": "bin/nanoid.cjs" 1103 | }, 1104 | "engines": { 1105 | "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" 1106 | } 1107 | }, 1108 | "node_modules/next": { 1109 | "version": "15.1.2", 1110 | "resolved": "https://registry.npmjs.org/next/-/next-15.1.2.tgz", 1111 | "integrity": "sha512-nLJDV7peNy+0oHlmY2JZjzMfJ8Aj0/dd3jCwSZS8ZiO5nkQfcZRqDrRN3U5rJtqVTQneIOGZzb6LCNrk7trMCQ==", 1112 | "dependencies": { 1113 | "@next/env": "15.1.2", 1114 | "@swc/counter": "0.1.3", 1115 | "@swc/helpers": "0.5.15", 1116 | "busboy": "1.6.0", 1117 | "caniuse-lite": "^1.0.30001579", 1118 | "postcss": "8.4.31", 1119 | "styled-jsx": "5.1.6" 1120 | }, 1121 | "bin": { 1122 | "next": "dist/bin/next" 1123 | }, 1124 | "engines": { 1125 | "node": "^18.18.0 || ^19.8.0 || >= 20.0.0" 1126 | }, 1127 | "optionalDependencies": { 1128 | "@next/swc-darwin-arm64": "15.1.2", 1129 | "@next/swc-darwin-x64": "15.1.2", 1130 | "@next/swc-linux-arm64-gnu": "15.1.2", 1131 | "@next/swc-linux-arm64-musl": "15.1.2", 1132 | "@next/swc-linux-x64-gnu": "15.1.2", 1133 | "@next/swc-linux-x64-musl": "15.1.2", 1134 | "@next/swc-win32-arm64-msvc": "15.1.2", 1135 | "@next/swc-win32-x64-msvc": "15.1.2", 1136 | "sharp": "^0.33.5" 1137 | }, 1138 | "peerDependencies": { 1139 | "@opentelemetry/api": "^1.1.0", 1140 | "@playwright/test": "^1.41.2", 1141 | "babel-plugin-react-compiler": "*", 1142 | "react": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", 1143 | "react-dom": "^18.2.0 || 19.0.0-rc-de68d2f4-20241204 || ^19.0.0", 1144 | "sass": "^1.3.0" 1145 | }, 1146 | "peerDependenciesMeta": { 1147 | "@opentelemetry/api": { 1148 | "optional": true 1149 | }, 1150 | "@playwright/test": { 1151 | "optional": true 1152 | }, 1153 | "babel-plugin-react-compiler": { 1154 | "optional": true 1155 | }, 1156 | "sass": { 1157 | "optional": true 1158 | } 1159 | } 1160 | }, 1161 | "node_modules/nopt": { 1162 | "version": "7.2.1", 1163 | "resolved": "https://registry.npmjs.org/nopt/-/nopt-7.2.1.tgz", 1164 | "integrity": "sha512-taM24ViiimT/XntxbPyJQzCG+p4EKOpgD3mxFwW38mGjVUrfERQOeY4EDHjdnptttfHuHQXFx+lTP08Q+mLa/w==", 1165 | "dependencies": { 1166 | "abbrev": "^2.0.0" 1167 | }, 1168 | "bin": { 1169 | "nopt": "bin/nopt.js" 1170 | }, 1171 | "engines": { 1172 | "node": "^14.17.0 || ^16.13.0 || >=18.0.0" 1173 | } 1174 | }, 1175 | "node_modules/package-json-from-dist": { 1176 | "version": "1.0.1", 1177 | "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", 1178 | "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==" 1179 | }, 1180 | "node_modules/parseley": { 1181 | "version": "0.12.1", 1182 | "resolved": "https://registry.npmjs.org/parseley/-/parseley-0.12.1.tgz", 1183 | "integrity": "sha512-e6qHKe3a9HWr0oMRVDTRhKce+bRO8VGQR3NyVwcjwrbhMmFCX9KszEV35+rn4AdilFAq9VPxP/Fe1wC9Qjd2lw==", 1184 | "dependencies": { 1185 | "leac": "^0.6.0", 1186 | "peberminta": "^0.9.0" 1187 | }, 1188 | "funding": { 1189 | "url": "https://ko-fi.com/killymxi" 1190 | } 1191 | }, 1192 | "node_modules/path-key": { 1193 | "version": "3.1.1", 1194 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", 1195 | "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", 1196 | "engines": { 1197 | "node": ">=8" 1198 | } 1199 | }, 1200 | "node_modules/path-scurry": { 1201 | "version": "1.11.1", 1202 | "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", 1203 | "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", 1204 | "dependencies": { 1205 | "lru-cache": "^10.2.0", 1206 | "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0" 1207 | }, 1208 | "engines": { 1209 | "node": ">=16 || 14 >=14.18" 1210 | }, 1211 | "funding": { 1212 | "url": "https://github.com/sponsors/isaacs" 1213 | } 1214 | }, 1215 | "node_modules/peberminta": { 1216 | "version": "0.9.0", 1217 | "resolved": "https://registry.npmjs.org/peberminta/-/peberminta-0.9.0.tgz", 1218 | "integrity": "sha512-XIxfHpEuSJbITd1H3EeQwpcZbTLHc+VVr8ANI9t5sit565tsI4/xK3KWTUFE2e6QiangUkh3B0jihzmGnNrRsQ==", 1219 | "funding": { 1220 | "url": "https://ko-fi.com/killymxi" 1221 | } 1222 | }, 1223 | "node_modules/picocolors": { 1224 | "version": "1.1.1", 1225 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", 1226 | "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==" 1227 | }, 1228 | "node_modules/playwright": { 1229 | "version": "1.49.1", 1230 | "resolved": "https://registry.npmjs.org/playwright/-/playwright-1.49.1.tgz", 1231 | "integrity": "sha512-VYL8zLoNTBxVOrJBbDuRgDWa3i+mfQgDTrL8Ah9QXZ7ax4Dsj0MSq5bYgytRnDVVe+njoKnfsYkH3HzqVj5UZA==", 1232 | "devOptional": true, 1233 | "dependencies": { 1234 | "playwright-core": "1.49.1" 1235 | }, 1236 | "bin": { 1237 | "playwright": "cli.js" 1238 | }, 1239 | "engines": { 1240 | "node": ">=18" 1241 | }, 1242 | "optionalDependencies": { 1243 | "fsevents": "2.3.2" 1244 | } 1245 | }, 1246 | "node_modules/playwright-core": { 1247 | "version": "1.49.1", 1248 | "resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.49.1.tgz", 1249 | "integrity": "sha512-BzmpVcs4kE2CH15rWfzpjzVGhWERJfmnXmniSyKeRZUs9Ws65m+RGIi7mjJK/euCegfn3i7jvqWeWyHe9y3Vgg==", 1250 | "devOptional": true, 1251 | "bin": { 1252 | "playwright-core": "cli.js" 1253 | }, 1254 | "engines": { 1255 | "node": ">=18" 1256 | } 1257 | }, 1258 | "node_modules/postcss": { 1259 | "version": "8.4.31", 1260 | "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz", 1261 | "integrity": "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==", 1262 | "funding": [ 1263 | { 1264 | "type": "opencollective", 1265 | "url": "https://opencollective.com/postcss/" 1266 | }, 1267 | { 1268 | "type": "tidelift", 1269 | "url": "https://tidelift.com/funding/github/npm/postcss" 1270 | }, 1271 | { 1272 | "type": "github", 1273 | "url": "https://github.com/sponsors/ai" 1274 | } 1275 | ], 1276 | "dependencies": { 1277 | "nanoid": "^3.3.6", 1278 | "picocolors": "^1.0.0", 1279 | "source-map-js": "^1.0.2" 1280 | }, 1281 | "engines": { 1282 | "node": "^10 || ^12 || >=14" 1283 | } 1284 | }, 1285 | "node_modules/proto-list": { 1286 | "version": "1.2.4", 1287 | "resolved": "https://registry.npmjs.org/proto-list/-/proto-list-1.2.4.tgz", 1288 | "integrity": "sha512-vtK/94akxsTMhe0/cbfpR+syPuszcuwhqVjJq26CuNDgFGj682oRBXOP5MJpv2r7JtE8MsiepGIqvvOTBwn2vA==" 1289 | }, 1290 | "node_modules/react": { 1291 | "version": "18.3.1", 1292 | "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz", 1293 | "integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==", 1294 | "dependencies": { 1295 | "loose-envify": "^1.1.0" 1296 | }, 1297 | "engines": { 1298 | "node": ">=0.10.0" 1299 | } 1300 | }, 1301 | "node_modules/react-dom": { 1302 | "version": "18.3.1", 1303 | "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz", 1304 | "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==", 1305 | "dependencies": { 1306 | "loose-envify": "^1.1.0", 1307 | "scheduler": "^0.23.2" 1308 | }, 1309 | "peerDependencies": { 1310 | "react": "^18.3.1" 1311 | } 1312 | }, 1313 | "node_modules/react-promise-suspense": { 1314 | "version": "0.3.4", 1315 | "resolved": "https://registry.npmjs.org/react-promise-suspense/-/react-promise-suspense-0.3.4.tgz", 1316 | "integrity": "sha512-I42jl7L3Ze6kZaq+7zXWSunBa3b1on5yfvUW6Eo/3fFOj6dZ5Bqmcd264nJbTK/gn1HjjILAjSwnZbV4RpSaNQ==", 1317 | "dependencies": { 1318 | "fast-deep-equal": "^2.0.1" 1319 | } 1320 | }, 1321 | "node_modules/resend": { 1322 | "version": "4.0.1", 1323 | "resolved": "https://registry.npmjs.org/resend/-/resend-4.0.1.tgz", 1324 | "integrity": "sha512-EkCRfzKw9JX7N75L+0BC8oXohDBLhlhl4w7AgrkEW2TAsOMBsVcbQHPe8cRWP6Ea7KDhD158TsNjbCBcohed5A==", 1325 | "dependencies": { 1326 | "@react-email/render": "1.0.1" 1327 | }, 1328 | "engines": { 1329 | "node": ">=18" 1330 | } 1331 | }, 1332 | "node_modules/scheduler": { 1333 | "version": "0.23.2", 1334 | "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz", 1335 | "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==", 1336 | "dependencies": { 1337 | "loose-envify": "^1.1.0" 1338 | } 1339 | }, 1340 | "node_modules/selderee": { 1341 | "version": "0.11.0", 1342 | "resolved": "https://registry.npmjs.org/selderee/-/selderee-0.11.0.tgz", 1343 | "integrity": "sha512-5TF+l7p4+OsnP8BCCvSyZiSPc4x4//p5uPwK8TCnVPJYRmU2aYKMpOXvw8zM5a5JvuuCGN1jmsMwuU2W02ukfA==", 1344 | "dependencies": { 1345 | "parseley": "^0.12.0" 1346 | }, 1347 | "funding": { 1348 | "url": "https://ko-fi.com/killymxi" 1349 | } 1350 | }, 1351 | "node_modules/semver": { 1352 | "version": "7.6.3", 1353 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", 1354 | "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", 1355 | "bin": { 1356 | "semver": "bin/semver.js" 1357 | }, 1358 | "engines": { 1359 | "node": ">=10" 1360 | } 1361 | }, 1362 | "node_modules/sharp": { 1363 | "version": "0.33.5", 1364 | "resolved": "https://registry.npmjs.org/sharp/-/sharp-0.33.5.tgz", 1365 | "integrity": "sha512-haPVm1EkS9pgvHrQ/F3Xy+hgcuMV0Wm9vfIBSiwZ05k+xgb0PkBQpGsAA/oWdDobNaZTH5ppvHtzCFbnSEwHVw==", 1366 | "hasInstallScript": true, 1367 | "optional": true, 1368 | "dependencies": { 1369 | "color": "^4.2.3", 1370 | "detect-libc": "^2.0.3", 1371 | "semver": "^7.6.3" 1372 | }, 1373 | "engines": { 1374 | "node": "^18.17.0 || ^20.3.0 || >=21.0.0" 1375 | }, 1376 | "funding": { 1377 | "url": "https://opencollective.com/libvips" 1378 | }, 1379 | "optionalDependencies": { 1380 | "@img/sharp-darwin-arm64": "0.33.5", 1381 | "@img/sharp-darwin-x64": "0.33.5", 1382 | "@img/sharp-libvips-darwin-arm64": "1.0.4", 1383 | "@img/sharp-libvips-darwin-x64": "1.0.4", 1384 | "@img/sharp-libvips-linux-arm": "1.0.5", 1385 | "@img/sharp-libvips-linux-arm64": "1.0.4", 1386 | "@img/sharp-libvips-linux-s390x": "1.0.4", 1387 | "@img/sharp-libvips-linux-x64": "1.0.4", 1388 | "@img/sharp-libvips-linuxmusl-arm64": "1.0.4", 1389 | "@img/sharp-libvips-linuxmusl-x64": "1.0.4", 1390 | "@img/sharp-linux-arm": "0.33.5", 1391 | "@img/sharp-linux-arm64": "0.33.5", 1392 | "@img/sharp-linux-s390x": "0.33.5", 1393 | "@img/sharp-linux-x64": "0.33.5", 1394 | "@img/sharp-linuxmusl-arm64": "0.33.5", 1395 | "@img/sharp-linuxmusl-x64": "0.33.5", 1396 | "@img/sharp-wasm32": "0.33.5", 1397 | "@img/sharp-win32-ia32": "0.33.5", 1398 | "@img/sharp-win32-x64": "0.33.5" 1399 | } 1400 | }, 1401 | "node_modules/shebang-command": { 1402 | "version": "2.0.0", 1403 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", 1404 | "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", 1405 | "dependencies": { 1406 | "shebang-regex": "^3.0.0" 1407 | }, 1408 | "engines": { 1409 | "node": ">=8" 1410 | } 1411 | }, 1412 | "node_modules/shebang-regex": { 1413 | "version": "3.0.0", 1414 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", 1415 | "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", 1416 | "engines": { 1417 | "node": ">=8" 1418 | } 1419 | }, 1420 | "node_modules/signal-exit": { 1421 | "version": "4.1.0", 1422 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", 1423 | "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", 1424 | "engines": { 1425 | "node": ">=14" 1426 | }, 1427 | "funding": { 1428 | "url": "https://github.com/sponsors/isaacs" 1429 | } 1430 | }, 1431 | "node_modules/simple-swizzle": { 1432 | "version": "0.2.2", 1433 | "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", 1434 | "integrity": "sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==", 1435 | "optional": true, 1436 | "dependencies": { 1437 | "is-arrayish": "^0.3.1" 1438 | } 1439 | }, 1440 | "node_modules/source-map-js": { 1441 | "version": "1.2.1", 1442 | "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", 1443 | "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==", 1444 | "engines": { 1445 | "node": ">=0.10.0" 1446 | } 1447 | }, 1448 | "node_modules/streamsearch": { 1449 | "version": "1.1.0", 1450 | "resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz", 1451 | "integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==", 1452 | "engines": { 1453 | "node": ">=10.0.0" 1454 | } 1455 | }, 1456 | "node_modules/string-width": { 1457 | "version": "5.1.2", 1458 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", 1459 | "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", 1460 | "dependencies": { 1461 | "eastasianwidth": "^0.2.0", 1462 | "emoji-regex": "^9.2.2", 1463 | "strip-ansi": "^7.0.1" 1464 | }, 1465 | "engines": { 1466 | "node": ">=12" 1467 | }, 1468 | "funding": { 1469 | "url": "https://github.com/sponsors/sindresorhus" 1470 | } 1471 | }, 1472 | "node_modules/string-width-cjs": { 1473 | "name": "string-width", 1474 | "version": "4.2.3", 1475 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 1476 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 1477 | "dependencies": { 1478 | "emoji-regex": "^8.0.0", 1479 | "is-fullwidth-code-point": "^3.0.0", 1480 | "strip-ansi": "^6.0.1" 1481 | }, 1482 | "engines": { 1483 | "node": ">=8" 1484 | } 1485 | }, 1486 | "node_modules/string-width-cjs/node_modules/ansi-regex": { 1487 | "version": "5.0.1", 1488 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 1489 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 1490 | "engines": { 1491 | "node": ">=8" 1492 | } 1493 | }, 1494 | "node_modules/string-width-cjs/node_modules/emoji-regex": { 1495 | "version": "8.0.0", 1496 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1497 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" 1498 | }, 1499 | "node_modules/string-width-cjs/node_modules/strip-ansi": { 1500 | "version": "6.0.1", 1501 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 1502 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 1503 | "dependencies": { 1504 | "ansi-regex": "^5.0.1" 1505 | }, 1506 | "engines": { 1507 | "node": ">=8" 1508 | } 1509 | }, 1510 | "node_modules/strip-ansi": { 1511 | "version": "7.1.0", 1512 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", 1513 | "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", 1514 | "dependencies": { 1515 | "ansi-regex": "^6.0.1" 1516 | }, 1517 | "engines": { 1518 | "node": ">=12" 1519 | }, 1520 | "funding": { 1521 | "url": "https://github.com/chalk/strip-ansi?sponsor=1" 1522 | } 1523 | }, 1524 | "node_modules/strip-ansi-cjs": { 1525 | "name": "strip-ansi", 1526 | "version": "6.0.1", 1527 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 1528 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 1529 | "dependencies": { 1530 | "ansi-regex": "^5.0.1" 1531 | }, 1532 | "engines": { 1533 | "node": ">=8" 1534 | } 1535 | }, 1536 | "node_modules/strip-ansi-cjs/node_modules/ansi-regex": { 1537 | "version": "5.0.1", 1538 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 1539 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 1540 | "engines": { 1541 | "node": ">=8" 1542 | } 1543 | }, 1544 | "node_modules/styled-jsx": { 1545 | "version": "5.1.6", 1546 | "resolved": "https://registry.npmjs.org/styled-jsx/-/styled-jsx-5.1.6.tgz", 1547 | "integrity": "sha512-qSVyDTeMotdvQYoHWLNGwRFJHC+i+ZvdBRYosOFgC+Wg1vx4frN2/RG/NA7SYqqvKNLf39P2LSRA2pu6n0XYZA==", 1548 | "dependencies": { 1549 | "client-only": "0.0.1" 1550 | }, 1551 | "engines": { 1552 | "node": ">= 12.0.0" 1553 | }, 1554 | "peerDependencies": { 1555 | "react": ">= 16.8.0 || 17.x.x || ^18.0.0-0 || ^19.0.0-0" 1556 | }, 1557 | "peerDependenciesMeta": { 1558 | "@babel/core": { 1559 | "optional": true 1560 | }, 1561 | "babel-plugin-macros": { 1562 | "optional": true 1563 | } 1564 | } 1565 | }, 1566 | "node_modules/tslib": { 1567 | "version": "2.8.1", 1568 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", 1569 | "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==" 1570 | }, 1571 | "node_modules/typescript": { 1572 | "version": "5.3.3", 1573 | "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.3.3.tgz", 1574 | "integrity": "sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==", 1575 | "dev": true, 1576 | "bin": { 1577 | "tsc": "bin/tsc", 1578 | "tsserver": "bin/tsserver" 1579 | }, 1580 | "engines": { 1581 | "node": ">=14.17" 1582 | } 1583 | }, 1584 | "node_modules/undici-types": { 1585 | "version": "5.26.5", 1586 | "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", 1587 | "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", 1588 | "dev": true 1589 | }, 1590 | "node_modules/which": { 1591 | "version": "2.0.2", 1592 | "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", 1593 | "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", 1594 | "dependencies": { 1595 | "isexe": "^2.0.0" 1596 | }, 1597 | "bin": { 1598 | "node-which": "bin/node-which" 1599 | }, 1600 | "engines": { 1601 | "node": ">= 8" 1602 | } 1603 | }, 1604 | "node_modules/wrap-ansi": { 1605 | "version": "8.1.0", 1606 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", 1607 | "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", 1608 | "dependencies": { 1609 | "ansi-styles": "^6.1.0", 1610 | "string-width": "^5.0.1", 1611 | "strip-ansi": "^7.0.1" 1612 | }, 1613 | "engines": { 1614 | "node": ">=12" 1615 | }, 1616 | "funding": { 1617 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1" 1618 | } 1619 | }, 1620 | "node_modules/wrap-ansi-cjs": { 1621 | "name": "wrap-ansi", 1622 | "version": "7.0.0", 1623 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 1624 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 1625 | "dependencies": { 1626 | "ansi-styles": "^4.0.0", 1627 | "string-width": "^4.1.0", 1628 | "strip-ansi": "^6.0.0" 1629 | }, 1630 | "engines": { 1631 | "node": ">=10" 1632 | }, 1633 | "funding": { 1634 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1" 1635 | } 1636 | }, 1637 | "node_modules/wrap-ansi-cjs/node_modules/ansi-regex": { 1638 | "version": "5.0.1", 1639 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 1640 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 1641 | "engines": { 1642 | "node": ">=8" 1643 | } 1644 | }, 1645 | "node_modules/wrap-ansi-cjs/node_modules/ansi-styles": { 1646 | "version": "4.3.0", 1647 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 1648 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 1649 | "dependencies": { 1650 | "color-convert": "^2.0.1" 1651 | }, 1652 | "engines": { 1653 | "node": ">=8" 1654 | }, 1655 | "funding": { 1656 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 1657 | } 1658 | }, 1659 | "node_modules/wrap-ansi-cjs/node_modules/emoji-regex": { 1660 | "version": "8.0.0", 1661 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 1662 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" 1663 | }, 1664 | "node_modules/wrap-ansi-cjs/node_modules/string-width": { 1665 | "version": "4.2.3", 1666 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 1667 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 1668 | "dependencies": { 1669 | "emoji-regex": "^8.0.0", 1670 | "is-fullwidth-code-point": "^3.0.0", 1671 | "strip-ansi": "^6.0.1" 1672 | }, 1673 | "engines": { 1674 | "node": ">=8" 1675 | } 1676 | }, 1677 | "node_modules/wrap-ansi-cjs/node_modules/strip-ansi": { 1678 | "version": "6.0.1", 1679 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 1680 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 1681 | "dependencies": { 1682 | "ansi-regex": "^5.0.1" 1683 | }, 1684 | "engines": { 1685 | "node": ">=8" 1686 | } 1687 | } 1688 | } 1689 | } 1690 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "private": true, 3 | "scripts": { 4 | "dev": "next dev", 5 | "build": "next build", 6 | "start": "next start", 7 | "test:e2e": "playwright test" 8 | }, 9 | "dependencies": { 10 | "next": "latest", 11 | "react": "^18.2.0", 12 | "react-dom": "^18.2.0", 13 | "resend": "^4.0.1" 14 | }, 15 | "devDependencies": { 16 | "@playwright/test": "^1.40.1", 17 | "@types/node": "20.10.4", 18 | "@types/react": "18.2.45", 19 | "typescript": "5.3.3" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /playwright.config.ts: -------------------------------------------------------------------------------- 1 | import { defineConfig, devices } from "@playwright/test"; 2 | import path from "path"; 3 | 4 | // Use process.env.PORT by default and fallback to port 3000 5 | const PORT = process.env.PORT || 3000; 6 | 7 | // Set webServer.url and use.baseURL with the location of the WebServer respecting the correct set port 8 | const baseURL = `http://localhost:${PORT}`; 9 | 10 | // Reference: https://playwright.dev/docs/test-configuration 11 | export default defineConfig({ 12 | // Timeout per test 13 | timeout: 30 * 1000, 14 | // Test directory 15 | testDir: path.join(__dirname, "e2e"), 16 | // If a test fails, retry it additional 2 times 17 | retries: 2, 18 | // Artifacts folder where screenshots, videos, and traces are stored. 19 | outputDir: "test-results/", 20 | 21 | // Run your local dev server before starting the tests: 22 | // https://playwright.dev/docs/test-advanced#launching-a-development-web-server-during-the-tests 23 | webServer: { 24 | command: "npm run dev", 25 | url: baseURL, 26 | timeout: 120 * 1000, 27 | reuseExistingServer: !process.env.CI, 28 | }, 29 | 30 | use: { 31 | // Use baseURL so to make navigations relative. 32 | // More information: https://playwright.dev/docs/api/class-testoptions#test-options-base-url 33 | baseURL, 34 | 35 | // Retry a test if its failing with enabled tracing. This allows you to analyze the DOM, console logs, network traffic etc. 36 | // More information: https://playwright.dev/docs/trace-viewer 37 | trace: "retry-with-trace", 38 | 39 | // All available context options: https://playwright.dev/docs/api/class-browser#browser-new-context 40 | // contextOptions: { 41 | // ignoreHTTPSErrors: true, 42 | // }, 43 | }, 44 | 45 | projects: [ 46 | { 47 | name: "Desktop Chrome", 48 | use: { 49 | ...devices["Desktop Chrome"], 50 | }, 51 | }, 52 | // { 53 | // name: 'Desktop Firefox', 54 | // use: { 55 | // ...devices['Desktop Firefox'], 56 | // }, 57 | // }, 58 | // { 59 | // name: 'Desktop Safari', 60 | // use: { 61 | // ...devices['Desktop Safari'], 62 | // }, 63 | // }, 64 | // Test against mobile viewports. 65 | { 66 | name: "Mobile Chrome", 67 | use: { 68 | ...devices["Pixel 5"], 69 | }, 70 | }, 71 | { 72 | name: "Mobile Safari", 73 | use: devices["iPhone 12"], 74 | }, 75 | ], 76 | }); 77 | -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/resend/resend-nextjs-playwright-example/519d1d5945c2fdc5aa37eb6e8f5142f83f12c9ac/public/favicon.ico -------------------------------------------------------------------------------- /public/vercel.svg: -------------------------------------------------------------------------------- 1 | 3 | 4 | -------------------------------------------------------------------------------- /styles/Home.module.css: -------------------------------------------------------------------------------- 1 | .container { 2 | min-height: 100vh; 3 | padding: 0 0.5rem; 4 | display: flex; 5 | flex-direction: column; 6 | justify-content: center; 7 | align-items: center; 8 | } 9 | 10 | .main { 11 | padding: 5rem 0; 12 | flex: 1; 13 | display: flex; 14 | flex-direction: column; 15 | justify-content: center; 16 | align-items: center; 17 | } 18 | 19 | .footer { 20 | width: 100%; 21 | height: 100px; 22 | border-top: 1px solid #eaeaea; 23 | display: flex; 24 | justify-content: center; 25 | align-items: center; 26 | } 27 | 28 | .footer a { 29 | display: flex; 30 | justify-content: center; 31 | align-items: center; 32 | flex-grow: 1; 33 | } 34 | 35 | .title a { 36 | color: #0070f3; 37 | text-decoration: none; 38 | } 39 | 40 | .title a:hover, 41 | .title a:focus, 42 | .title a:active { 43 | text-decoration: underline; 44 | } 45 | 46 | .title { 47 | margin: 0; 48 | line-height: 1.15; 49 | font-size: 4rem; 50 | } 51 | 52 | .title, 53 | .description { 54 | text-align: center; 55 | } 56 | 57 | .description { 58 | line-height: 1.5; 59 | font-size: 1.5rem; 60 | } 61 | 62 | .code { 63 | background: #fafafa; 64 | border-radius: 5px; 65 | padding: 0.75rem; 66 | font-size: 1.1rem; 67 | font-family: 68 | Menlo, 69 | Monaco, 70 | Lucida Console, 71 | Liberation Mono, 72 | DejaVu Sans Mono, 73 | Bitstream Vera Sans Mono, 74 | Courier New, 75 | monospace; 76 | } 77 | 78 | .grid { 79 | display: flex; 80 | align-items: center; 81 | justify-content: center; 82 | flex-wrap: wrap; 83 | max-width: 800px; 84 | margin-top: 3rem; 85 | } 86 | 87 | .card { 88 | margin: 1rem; 89 | padding: 1.5rem; 90 | text-align: left; 91 | color: inherit; 92 | text-decoration: none; 93 | border: 1px solid #eaeaea; 94 | border-radius: 10px; 95 | transition: 96 | color 0.15s ease, 97 | border-color 0.15s ease; 98 | width: 45%; 99 | } 100 | 101 | .card:hover, 102 | .card:focus, 103 | .card:active { 104 | color: #0070f3; 105 | border-color: #0070f3; 106 | } 107 | 108 | .card h2 { 109 | margin: 0 0 1rem 0; 110 | font-size: 1.5rem; 111 | } 112 | 113 | .card p { 114 | margin: 0; 115 | font-size: 1.25rem; 116 | line-height: 1.5; 117 | } 118 | 119 | .logo { 120 | height: 1em; 121 | margin-left: 0.5rem; 122 | } 123 | 124 | @media (max-width: 600px) { 125 | .grid { 126 | width: 100%; 127 | flex-direction: column; 128 | } 129 | } 130 | -------------------------------------------------------------------------------- /styles/globals.css: -------------------------------------------------------------------------------- 1 | html, 2 | body { 3 | padding: 0; 4 | margin: 0; 5 | font-family: 6 | -apple-system, 7 | BlinkMacSystemFont, 8 | Segoe UI, 9 | Roboto, 10 | Oxygen, 11 | Ubuntu, 12 | Cantarell, 13 | Fira Sans, 14 | Droid Sans, 15 | Helvetica Neue, 16 | sans-serif; 17 | } 18 | 19 | a { 20 | color: inherit; 21 | text-decoration: none; 22 | } 23 | 24 | * { 25 | box-sizing: border-box; 26 | } 27 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "lib": [ 4 | "dom", 5 | "dom.iterable", 6 | "esnext" 7 | ], 8 | "allowJs": true, 9 | "skipLibCheck": true, 10 | "strict": false, 11 | "noEmit": true, 12 | "incremental": true, 13 | "esModuleInterop": true, 14 | "module": "esnext", 15 | "moduleResolution": "bundler", 16 | "resolveJsonModule": true, 17 | "isolatedModules": true, 18 | "jsx": "preserve", 19 | "plugins": [ 20 | { 21 | "name": "next" 22 | } 23 | ], 24 | "strictNullChecks": true, 25 | "target": "ES2017" 26 | }, 27 | "include": [ 28 | "next-env.d.ts", 29 | ".next/types/**/*.ts", 30 | "**/*.ts", 31 | "**/*.tsx" 32 | ], 33 | "exclude": [ 34 | "node_modules" 35 | ] 36 | } 37 | --------------------------------------------------------------------------------