diff --git a/.changeset/fifty-mice-bow.md b/.changeset/fifty-mice-bow.md new file mode 100644 index 000000000..b36bc3b64 --- /dev/null +++ b/.changeset/fifty-mice-bow.md @@ -0,0 +1,5 @@ +--- +'astro': patch +--- + +Forces the correct mime type for CSS in HMR diff --git a/packages/astro/src/vite-plugin-astro-server/index.ts b/packages/astro/src/vite-plugin-astro-server/index.ts index f71e28ba0..ba5b54813 100644 --- a/packages/astro/src/vite-plugin-astro-server/index.ts +++ b/packages/astro/src/vite-plugin-astro-server/index.ts @@ -335,6 +335,31 @@ async function handleRequest( } } +/** + * Vite HMR sends requests for new CSS and those get returned as JS, but we want it to be CSS + * since they are inside of a link tag for Astro. + */ +const forceTextCSSForStylesMiddleware: vite.Connect.NextHandleFunction = function(req, res, next) { + if(req.url) { + // We are just using this to parse the URL to get the search params object + // so the second arg here doesn't matter + const url = new URL(req.url, 'https://astro.build'); + // lang.css is a search param that exists on Astro, Svelte, and Vue components. + // We only want to override for astro files. + if(url.searchParams.has('astro') && url.searchParams.has('lang.css')) { + // Override setHeader so we can set the correct content-type for this request. + const setHeader = res.setHeader; + res.setHeader = function(key, value) { + if(key.toLowerCase() === 'content-type') { + return setHeader.call(this, key, 'text/css'); + } + return setHeader.apply(this, [key, value]); + }; + } + } + next(); +}; + export default function createPlugin({ config, logging }: AstroPluginOptions): vite.Plugin { return { name: 'astro:server', @@ -354,6 +379,9 @@ export default function createPlugin({ config, logging }: AstroPluginOptions): v viteServer.watcher.on('change', rebuildManifest.bind(null, false)); return () => { removeViteHttpMiddleware(viteServer.middlewares); + + // Push this middleware to the front of the stack so that it can intercept responses. + viteServer.middlewares.stack.unshift({ route: '', handle: forceTextCSSForStylesMiddleware }); viteServer.middlewares.use(async (req, res) => { if (!req.url || !req.method) { throw new Error('Incomplete request'); diff --git a/packages/astro/test/fixtures/hmr-css/src/pages/index.astro b/packages/astro/test/fixtures/hmr-css/src/pages/index.astro new file mode 100644 index 000000000..840e60e01 --- /dev/null +++ b/packages/astro/test/fixtures/hmr-css/src/pages/index.astro @@ -0,0 +1,11 @@ + +
+