diff --git a/packages/integrations/netlify/package.json b/packages/integrations/netlify/package.json index 9b2f197add6d..ff30831e46d8 100644 --- a/packages/integrations/netlify/package.json +++ b/packages/integrations/netlify/package.json @@ -30,7 +30,8 @@ "test": "npm run test-fn" }, "dependencies": { - "@astrojs/webapi": "^0.12.0" + "@astrojs/webapi": "^0.12.0", + "esbuild": "^0.14.42" }, "devDependencies": { "@netlify/edge-handler-types": "^0.34.1", diff --git a/packages/integrations/netlify/src/integration-edge-functions.ts b/packages/integrations/netlify/src/integration-edge-functions.ts index e43a7f482f86..3ef39eecbc40 100644 --- a/packages/integrations/netlify/src/integration-edge-functions.ts +++ b/packages/integrations/netlify/src/integration-edge-functions.ts @@ -1,6 +1,9 @@ -import type { AstroAdapter, AstroConfig, AstroIntegration, RouteData } from 'astro'; -import * as fs from 'fs'; +import type { AstroAdapter, AstroIntegration, AstroConfig, RouteData, BuildConfig } from 'astro'; import { createRedirects } from './shared.js'; +import esbuild from 'esbuild'; +import * as fs from 'fs'; +import { fileURLToPath } from 'url'; +import * as npath from 'path'; export function getAdapter(): AstroAdapter { return { @@ -62,9 +65,34 @@ async function createEdgeManifest(routes: RouteData[], entryFile: string, dir: U await fs.promises.writeFile(manifestURL, _manifest, 'utf-8'); } +async function bundleServerEntry(buildConfig: BuildConfig, vite: any) { + const entryUrl = new URL(buildConfig.serverEntry, buildConfig.server); + const pth = fileURLToPath(entryUrl); + await esbuild.build({ + target: 'es2020', + platform: 'browser', + entryPoints: [pth], + outfile: pth, + allowOverwrite: true, + format: 'esm', + bundle: true, + external: [ "@astrojs/markdown-remark"] + }); + + // Remove chunks, if they exist. Since we have bundled via esbuild these chunks are trash. + try { + const chunkFileNames = vite?.build?.rollupOptions?.output?.chunkFileNames ?? 'chunks/chunk.[hash].mjs'; + const chunkPath = npath.dirname(chunkFileNames); + const chunksDirUrl = new URL(chunkPath + '/', buildConfig.server); + await fs.promises.rm(chunksDirUrl, { recursive: true, force: true }); + } catch {} +} + export function netlifyEdgeFunctions({ dist }: NetlifyEdgeFunctionsOptions = {}): AstroIntegration { let _config: AstroConfig; let entryFile: string; + let _buildConfig: BuildConfig; + let _vite: any; return { name: '@astrojs/netlify/edge-functions', hooks: { @@ -80,6 +108,7 @@ export function netlifyEdgeFunctions({ dist }: NetlifyEdgeFunctionsOptions = {}) _config = config; }, 'astro:build:start': async ({ buildConfig }) => { + _buildConfig = buildConfig; entryFile = buildConfig.serverEntry.replace(/\.m?js/, ''); buildConfig.client = _config.outDir; buildConfig.server = new URL('./.netlify/edge-functions/', _config.root); @@ -87,6 +116,7 @@ export function netlifyEdgeFunctions({ dist }: NetlifyEdgeFunctionsOptions = {}) }, 'astro:build:setup': ({ vite, target }) => { if (target === 'server') { + _vite = vite; vite.resolve = vite.resolve || {}; vite.resolve.alias = vite.resolve.alias || {}; @@ -106,6 +136,7 @@ export function netlifyEdgeFunctions({ dist }: NetlifyEdgeFunctionsOptions = {}) } }, 'astro:build:done': async ({ routes, dir }) => { + await bundleServerEntry(_buildConfig, _vite); await createEdgeManifest(routes, entryFile, _config.root); await createRedirects(routes, dir, entryFile, true); }, diff --git a/packages/integrations/netlify/test/edge-functions/deps.ts b/packages/integrations/netlify/test/edge-functions/deps.ts index 1b23a94f743b..498b7e09e639 100644 --- a/packages/integrations/netlify/test/edge-functions/deps.ts +++ b/packages/integrations/netlify/test/edge-functions/deps.ts @@ -2,3 +2,4 @@ export { fromFileUrl } from 'https://deno.land/std@0.110.0/path/mod.ts'; export { assertEquals, assert } from 'https://deno.land/std@0.132.0/testing/asserts.ts'; export * from 'https://deno.land/x/deno_dom/deno-dom-wasm.ts'; +export * from 'https://deno.land/std@0.142.0/streams/conversion.ts'; diff --git a/packages/integrations/netlify/test/edge-functions/dynamic-import.test.js b/packages/integrations/netlify/test/edge-functions/dynamic-import.test.js new file mode 100644 index 000000000000..d4c61fb1f9cb --- /dev/null +++ b/packages/integrations/netlify/test/edge-functions/dynamic-import.test.js @@ -0,0 +1,27 @@ +// @ts-ignore +import { runBuild, runApp } from './test-utils.ts'; +// @ts-ignore +import { assertEquals, assert, DOMParser } from './deps.ts'; + +// @ts-ignore +Deno.test({ + name: 'Dynamic imports', + async fn() { + let close = await runBuild('./fixtures/dynimport/'); + let stop = await runApp('./fixtures/dynimport/prod.js'); + + try { + const response = await fetch('http://127.0.0.1:8085/'); + assertEquals(response.status, 200); + const html = await response.text(); + + assert(html, 'got some html'); + const doc = new DOMParser().parseFromString(html, `text/html`); + const div = doc.querySelector('#thing'); + assert(div, 'div exists') + } finally { + await close(); + await stop(); + } + }, +}); diff --git a/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/astro.config.mjs b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/astro.config.mjs new file mode 100644 index 000000000000..c55135e43b7a --- /dev/null +++ b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/astro.config.mjs @@ -0,0 +1,11 @@ +import { defineConfig } from 'astro/config'; +import { netlifyEdgeFunctions } from '@astrojs/netlify'; + +export default defineConfig({ + adapter: netlifyEdgeFunctions({ + dist: new URL('./dist/', import.meta.url), + }), + experimental: { + ssr: true + } +}) diff --git a/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/package.json b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/package.json new file mode 100644 index 000000000000..201a243d08bd --- /dev/null +++ b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/package.json @@ -0,0 +1,9 @@ +{ + "name": "@test/netlify-edge-astro-dynimport", + "version": "0.0.0", + "private": true, + "dependencies": { + "astro": "workspace:*", + "@astrojs/netlify": "workspace:*" + } +} diff --git a/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/prod.js b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/prod.js new file mode 100644 index 000000000000..3e7d6e64d448 --- /dev/null +++ b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/prod.js @@ -0,0 +1,11 @@ +import handler from './.netlify/edge-functions/entry.js'; +import { Server } from 'https://deno.land/std@0.132.0/http/server.ts'; + +const _server = new Server({ + port: 8085, + hostname: '0.0.0.0', + handler, +}); + +_server.listenAndServe(); +console.error(`Server running on port 8085`); diff --git a/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/src/components/Thing.astro b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/src/components/Thing.astro new file mode 100644 index 000000000000..8d8ef929a003 --- /dev/null +++ b/packages/integrations/netlify/test/edge-functions/fixtures/dynimport/src/components/Thing.astro @@ -0,0 +1,4 @@ +--- + +--- +