0
Fork 0
mirror of https://github.com/withastro/astro.git synced 2025-03-31 23:31:30 -05:00

Revert "Replace internal cssScopeTo implementation to vite.cssScopeTo" (#13420)

* Revert "Replace internal cssScopeTo implementation to vite.cssScopeTo (#13347)"

This reverts commit d83f92a204.

* changeset

* Apply suggestions from code review
This commit is contained in:
Emanuele Stoppa 2025-03-13 16:19:47 +01:00 committed by GitHub
parent be866a1d1d
commit 2f039b927a
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
14 changed files with 120 additions and 66 deletions

View file

@ -0,0 +1,5 @@
---
'astro': patch
---
It fixes an issue that caused some regressions in how styles are bundled.

View file

@ -1,11 +1,12 @@
import type { GetModuleInfo } from 'rollup';
import type { BuildOptions, ResolvedConfig, Plugin as VitePlugin } from 'vite';
import type { BuildOptions, ResolvedConfig, Rollup, Plugin as VitePlugin } from 'vite';
import { isBuildableCSSRequest } from '../../../vite-plugin-astro-server/util.js';
import type { BuildInternals } from '../internal.js';
import type { AstroBuildPlugin, BuildTarget } from '../plugin.js';
import type { PageBuildData, StaticBuildOptions, StylesheetAsset } from '../types.js';
import { hasAssetPropagationFlag } from '../../../content/index.js';
import type { AstroPluginCssMetadata } from '../../../vite-plugin-astro/index.js';
import * as assetName from '../css-asset-name.js';
import {
getParentExtendedModuleInfos,
@ -155,6 +156,32 @@ function rollupPluginAstroBuildCSS(options: PluginOptions): VitePlugin[] {
},
};
/**
* This plugin is a port of https://github.com/vitejs/vite/pull/16058. It enables removing unused
* scoped CSS from the bundle if the scoped target (e.g. Astro files) were not bundled.
* Once/If that PR is merged, we can refactor this away, renaming `meta.astroCss` to `meta.vite`.
*/
const cssScopeToPlugin: VitePlugin = {
name: 'astro:rollup-plugin-css-scope-to',
renderChunk(_, chunk, __, meta) {
for (const id in chunk.modules) {
// If this CSS is scoped to its importers exports, check if those importers exports
// are rendered in the chunks. If they are not, we can skip bundling this CSS.
const modMeta = this.getModuleInfo(id)?.meta as AstroPluginCssMetadata | undefined;
const cssScopeTo = modMeta?.astroCss?.cssScopeTo;
if (cssScopeTo && !isCssScopeToRendered(cssScopeTo, Object.values(meta.chunks))) {
// If this CSS is not used, delete it from the chunk modules so that Vite is unable
// to trace that it's used
delete chunk.modules[id];
const moduleIdsIndex = chunk.moduleIds.indexOf(id);
if (moduleIdsIndex > -1) {
chunk.moduleIds.splice(moduleIdsIndex, 1);
}
}
}
},
};
const singleCssPlugin: VitePlugin = {
name: 'astro:rollup-plugin-single-css',
enforce: 'post',
@ -246,7 +273,7 @@ function rollupPluginAstroBuildCSS(options: PluginOptions): VitePlugin[] {
},
};
return [cssBuildPlugin, singleCssPlugin, inlineStylesheetsPlugin];
return [cssBuildPlugin, cssScopeToPlugin, singleCssPlugin, inlineStylesheetsPlugin];
}
/***** UTILITY FUNCTIONS *****/
@ -294,3 +321,25 @@ function appendCSSToPage(
}
}
}
/**
* `cssScopeTo` is a map of `importer`s to its `export`s. This function iterate each `cssScopeTo` entries
* and check if the `importer` and its `export`s exists in the final chunks. If at least one matches,
* `cssScopeTo` is considered "rendered" by Rollup and we return true.
*/
function isCssScopeToRendered(
cssScopeTo: Record<string, string[]>,
chunks: Rollup.RenderedChunk[],
) {
for (const moduleId in cssScopeTo) {
const exports = cssScopeTo[moduleId];
// Find the chunk that renders this `moduleId` and get the rendered module
const renderedModule = chunks.find((c) => c.moduleIds.includes(moduleId))?.modules[moduleId];
// Return true if `renderedModule` exists and one of its exports is rendered
if (renderedModule?.renderedExports.some((e) => exports.includes(e))) {
return true;
}
}
return false;
}

View file

@ -188,18 +188,7 @@ async function ssrBuild(
const encoded = encodeName(name);
return [prefix, encoded, suffix].join('');
},
assetFileNames(chunkInfo) {
const { names } = chunkInfo;
const name = names[0] ?? '';
// Sometimes chunks have the `@_@astro` suffix due to SSR logic. Remove it!
// TODO: refactor our build logic to avoid this
if (name.includes(ASTRO_PAGE_EXTENSION_POST_PATTERN)) {
const [sanitizedName] = name.split(ASTRO_PAGE_EXTENSION_POST_PATTERN);
return `${settings.config.build.assets}/${sanitizedName}.[hash][extname]`;
}
return `${settings.config.build.assets}/[name].[hash][extname]`;
},
assetFileNames: `${settings.config.build.assets}/[name].[hash][extname]`,
...viteConfig.build?.rollupOptions?.output,
entryFileNames(chunkInfo) {
if (chunkInfo.facadeModuleId?.startsWith(ASTRO_PAGE_RESOLVED_MODULE_ID)) {

View file

@ -2,7 +2,11 @@ import type { SourceDescription } from 'rollup';
import type * as vite from 'vite';
import type { Logger } from '../core/logger/core.js';
import type { AstroSettings } from '../types/astro.js';
import type { PluginMetadata as AstroPluginMetadata, CompileMetadata } from './types.js';
import type {
PluginCssMetadata as AstroPluginCssMetadata,
PluginMetadata as AstroPluginMetadata,
CompileMetadata,
} from './types.js';
import { defaultClientConditions, defaultServerConditions, normalizePath } from 'vite';
import type { AstroConfig } from '../types/public/config.js';
@ -12,7 +16,7 @@ import { handleHotUpdate } from './hmr.js';
import { parseAstroRequest } from './query.js';
import { loadId } from './utils.js';
export { getAstroMetadata } from './metadata.js';
export type { AstroPluginMetadata };
export type { AstroPluginMetadata, AstroPluginCssMetadata };
interface AstroPluginOptions {
settings: AstroSettings;
@ -134,9 +138,17 @@ export default function astro({ settings, logger }: AstroPluginOptions): vite.Pl
return {
code: result.code,
// `vite.cssScopeTo` is a Vite feature that allows this CSS to be treeshaken
// if the Astro component's default export is not used
meta: result.isGlobal ? undefined : { vite: { cssScopeTo: [filename, 'default'] } },
// This metadata is used by `cssScopeToPlugin` to remove this module from the bundle
// if the `filename` default export (the Astro component) is unused.
meta: result.isGlobal
? undefined
: ({
astroCss: {
cssScopeTo: {
[filename]: ['default'],
},
},
} satisfies AstroPluginCssMetadata),
};
}
case 'script': {

View file

@ -18,6 +18,27 @@ export interface PluginMetadata {
};
}
export interface PluginCssMetadata {
astroCss: {
/**
* For Astro CSS virtual modules, it can scope to the main Astro module's default export
* so that if those exports are treeshaken away, the CSS module will also be treeshaken.
*
* Example config if the CSS id is `/src/Foo.astro?astro&type=style&lang.css`:
* ```js
* cssScopeTo: {
* '/src/Foo.astro': ['default']
* }
* ```
*
* The above is the only config we use today, but we're exposing as a `Record` to follow the
* upstream Vite implementation: https://github.com/vitejs/vite/pull/16058. When/If that lands,
* we can also remove our custom implementation.
*/
cssScopeTo: Record<string, string[]>;
};
}
export interface CompileMetadata {
/** Used for HMR to compare code changes */
originalCode: string;

View file

@ -9,19 +9,10 @@ import { after, before, describe, it } from 'node:test';
import * as cheerio from 'cheerio';
import { loadFixture } from './test-utils.js';
async function getCssContent($, fixture) {
const contents = await Promise.all(
$('link[rel=stylesheet][href^=/_astro/]').map((_, el) =>
fixture.readFile(el.attribs.href.replace(/^\/?/, '/')),
),
);
return contents.join('').replace(/\s/g, '').replace('/n', '');
}
/** @type {import('./test-utils').Fixture} */
let fixture;
describe('CSS', function () {
/** @type {import('./test-utils').Fixture} */
let fixture;
before(async () => {
fixture = await loadFixture({ root: './fixtures/0-css/' });
});
@ -39,7 +30,10 @@ describe('CSS', function () {
// get bundled CSS (will be hashed, hence DOM query)
html = await fixture.readFile('/index.html');
$ = cheerio.load(html);
bundledCSS = await getCssContent($, fixture);
const bundledCSSHREF = $('link[rel=stylesheet][href^=/_astro/]').attr('href');
bundledCSS = (await fixture.readFile(bundledCSSHREF.replace(/^\/?/, '/')))
.replace(/\s/g, '')
.replace('/n', '');
},
{
timeout: 45000,
@ -105,7 +99,8 @@ describe('CSS', function () {
it('Styles through barrel files should only include used Astro scoped styles', async () => {
const barrelHtml = await fixture.readFile('/barrel-styles/index.html');
const barrel$ = cheerio.load(barrelHtml);
const style = await getCssContent(barrel$, fixture);
const barrelBundledCssHref = barrel$('link[rel=stylesheet][href^=/_astro/]').attr('href');
const style = await fixture.readFile(barrelBundledCssHref.replace(/^\/?/, '/'));
assert.match(style, /\.comp-a\[data-astro-cid/);
assert.match(style, /\.comp-c\{/);
assert.doesNotMatch(style, /\.comp-b/);

View file

@ -63,9 +63,9 @@ describe('CSS Bundling', function () {
}
});
it('there are 5 css files', async () => {
it('there are 4 css files', async () => {
const dir = await fixture.readdir('/_astro');
assert.equal(dir.length, 5);
assert.equal(dir.length, 4);
});
it('CSS includes hashes', async () => {
@ -96,9 +96,9 @@ describe('CSS Bundling', function () {
await fixture.build({ mode: 'production' });
});
it('there are 5 css files', async () => {
it('there are 4 css files', async () => {
const dir = await fixture.readdir('/assets');
assert.equal(dir.length, 5);
assert.equal(dir.length, 4);
});
it('CSS does not include hashes', async () => {

View file

@ -20,7 +20,7 @@ describe('Vite Config', async () => {
it('Allows overriding bundle naming options in the build', async () => {
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
assert.match($('link').attr('href'), /\/assets\/testing-.+\.css/);
assert.match($('link').attr('href'), /\/assets\/testing-[a-z\d]+\.css/);
});
});

View file

@ -113,7 +113,7 @@ describe('Setting inlineStylesheets to auto in static output', () => {
// the count of style/link tags depends on our css chunking logic
// this test should be updated if it changes
assert.equal($('style').length, 2);
assert.equal($('style').length, 3);
assert.equal($('link[rel=stylesheet]').length, 1);
});
@ -162,7 +162,7 @@ describe('Setting inlineStylesheets to auto in server output', () => {
// the count of style/link tags depends on our css chunking logic
// this test should be updated if it changes
assert.equal($('style').length, 2);
assert.equal($('style').length, 3);
assert.equal($('link[rel=stylesheet]').length, 1);
});

View file

@ -147,9 +147,9 @@ describe('CSS ordering - import order', () => {
const content = await Promise.all(
getLinks(html).map((href) => getLinkContent(href, fixture)),
);
let [link1, , link3] = content;
let [link1, link2] = content;
assert.ok(link1.css.includes('f0f8ff')); // aliceblue minified
assert.ok(link3.css.includes('ff0')); // yellow minified
assert.ok(link2.css.includes('ff0')); // yellow minified
});
});
});

View file

@ -90,7 +90,7 @@ describe('CSS production ordering', () => {
);
assert.ok(content.length, 3, 'there are 3 stylesheets');
const [, pageStyles, sharedStyles] = content;
const [, sharedStyles, pageStyles] = content;
assert.ok(/red/.exec(sharedStyles.css));
assert.ok(/#00f/.exec(pageStyles.css));

View file

@ -4,15 +4,6 @@ import * as cheerio from 'cheerio';
import eol from 'eol';
import { loadFixture } from './test-utils.js';
async function getCssContent($, fixture) {
const contents = await Promise.all(
$('link[rel=stylesheet][href^=/_astro/]').map((_, el) =>
fixture.readFile(el.attribs.href.replace(/^\/?/, '/')),
),
);
return contents.join('').replace(/\s/g, '').replace('/n', '');
}
describe('PostCSS', () => {
let fixture;
let bundledCSS;
@ -28,7 +19,10 @@ describe('PostCSS', () => {
// get bundled CSS (will be hashed, hence DOM query)
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
bundledCSS = await getCssContent($, fixture);
const bundledCSSHREF = $('link[rel=stylesheet][href^=/_astro/]').attr('href');
bundledCSS = (await fixture.readFile(bundledCSSHREF.replace(/^\/?/, '/')))
.replace(/\s/g, '')
.replace('/n', '');
},
{ timeout: 45000 },
);

View file

@ -3,15 +3,6 @@ import { before, describe, it } from 'node:test';
import * as cheerio from 'cheerio';
import { loadFixture } from './test-utils.js';
async function getCssContent($, fixture) {
const contents = await Promise.all(
$('link[rel=stylesheet][href^=/_astro/]').map((_, el) =>
fixture.readFile(el.attribs.href.replace(/^\/?/, '/')),
),
);
return contents.join('').replace(/\s/g, '').replace('/n', '');
}
describe('Remote CSS', () => {
let fixture;
@ -28,7 +19,8 @@ describe('Remote CSS', () => {
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
const css = await getCssContent($, fixture);
const relPath = $('link').attr('href');
const css = await fixture.readFile(relPath);
assert.match(css, /https:\/\/unpkg.com\/open-props/);
assert.match(css, /body/);

View file

@ -8,8 +8,5 @@ export default defineConfig({
configFile: fileURLToPath(new URL('./tailwind.config.js', import.meta.url)),
nesting: true
}),
],
build: {
inlineStylesheets: 'never'
}
]
});