@pathscale/rebuild-plugin-ui-css-purge 0.1.0 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.ts +2 -2
- package/dist/index.js +22 -137
- package/dist/postbuild-purge.d.ts +12 -0
- package/dist/postbuild-purge.js +362 -0
- package/dist/scan-consumer.d.ts +2 -1
- package/package.json +2 -6
- package/src/generate-manifest.ts +3 -1
- package/dist/rsbuild-plugin.d.ts +0 -28
package/dist/index.d.ts
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export {
|
|
2
|
-
export type {
|
|
1
|
+
export { extractUIImports, extractJSXUsages, buildSafelists, scanConsumerSource } from "./scan-consumer";
|
|
2
|
+
export type { PropUsage, PurgeManifest, ComponentManifest, Safelists } from "./scan-consumer";
|
package/dist/index.js
CHANGED
|
@@ -1,10 +1,8 @@
|
|
|
1
|
-
//
|
|
2
|
-
|
|
3
|
-
import postcss from "postcss";
|
|
1
|
+
// @bun
|
|
2
|
+
// src/scan-consumer.ts
|
|
4
3
|
import swc from "@swc/core";
|
|
4
|
+
var {Glob } = globalThis.Bun;
|
|
5
5
|
import path from "path";
|
|
6
|
-
import { readFile as nodeReadFile } from "node:fs/promises";
|
|
7
|
-
import { glob as fastGlob } from "fast-glob";
|
|
8
6
|
function walkAST(node, visitor) {
|
|
9
7
|
if (!node || typeof node !== "object")
|
|
10
8
|
return;
|
|
@@ -108,10 +106,12 @@ function buildSafelists(allUsages, manifest) {
|
|
|
108
106
|
}
|
|
109
107
|
for (const [entryName, entry] of Object.entries(manifest)) {
|
|
110
108
|
const matchingUsages = findMatchingUsages(entryName, componentUsages);
|
|
111
|
-
if (matchingUsages.length === 0)
|
|
109
|
+
if (matchingUsages.length === 0) {
|
|
112
110
|
continue;
|
|
113
|
-
|
|
111
|
+
}
|
|
112
|
+
for (const cls of entry.classes.always) {
|
|
114
113
|
classSafelist.add(cls);
|
|
114
|
+
}
|
|
115
115
|
for (const [propOrSlot, value] of Object.entries(entry.classes.byProp)) {
|
|
116
116
|
if (Array.isArray(value)) {
|
|
117
117
|
if (isPropUsed(propOrSlot, matchingUsages)) {
|
|
@@ -139,7 +139,7 @@ function buildSafelists(allUsages, manifest) {
|
|
|
139
139
|
for (const [propName, attrMap] of Object.entries(entry.attrs)) {
|
|
140
140
|
if (isPropUsed(propName, matchingUsages)) {
|
|
141
141
|
for (const [attr, val] of Object.entries(attrMap)) {
|
|
142
|
-
attrSafelist.add(
|
|
142
|
+
attrSafelist.add(`${attr}=${val}`);
|
|
143
143
|
}
|
|
144
144
|
}
|
|
145
145
|
}
|
|
@@ -148,8 +148,9 @@ function buildSafelists(allUsages, manifest) {
|
|
|
148
148
|
return { classSafelist, attrSafelist };
|
|
149
149
|
}
|
|
150
150
|
function findMatchingUsages(entryName, usageMap) {
|
|
151
|
-
if (usageMap.has(entryName))
|
|
151
|
+
if (usageMap.has(entryName)) {
|
|
152
152
|
return usageMap.get(entryName);
|
|
153
|
+
}
|
|
153
154
|
const results = [];
|
|
154
155
|
for (const [usageName, usages] of usageMap) {
|
|
155
156
|
if (usageName === entryName) {
|
|
@@ -192,10 +193,12 @@ function getUsedEnumValues(slotName, usages) {
|
|
|
192
193
|
}
|
|
193
194
|
async function scanConsumerSource(srcDir) {
|
|
194
195
|
const allUsages = [];
|
|
195
|
-
const
|
|
196
|
-
for (const relPath of
|
|
196
|
+
const glob = new Glob("**/*.{tsx,ts,jsx,js}");
|
|
197
|
+
for await (const relPath of glob.scan({ cwd: srcDir })) {
|
|
198
|
+
if (relPath.includes("node_modules"))
|
|
199
|
+
continue;
|
|
197
200
|
const fullPath = path.join(srcDir, relPath);
|
|
198
|
-
const code = await
|
|
201
|
+
const code = await Bun.file(fullPath).text();
|
|
199
202
|
if (!code.includes("@pathscale/ui"))
|
|
200
203
|
continue;
|
|
201
204
|
const isTsx = /\.[tj]sx$/.test(relPath);
|
|
@@ -207,131 +210,13 @@ async function scanConsumerSource(srcDir) {
|
|
|
207
210
|
}
|
|
208
211
|
return allUsages;
|
|
209
212
|
}
|
|
210
|
-
|
|
211
|
-
const root = postcss.parse(css);
|
|
212
|
-
root.walkRules((rule) => {
|
|
213
|
-
const selectors = rule.selectors;
|
|
214
|
-
const kept = [];
|
|
215
|
-
for (const sel of selectors) {
|
|
216
|
-
const attrMatches = sel.matchAll(/\[(data-[a-z-]+|aria-[a-z-]+)="([^"]+)"\]/g);
|
|
217
|
-
let shouldKeep = true;
|
|
218
|
-
for (const match of attrMatches) {
|
|
219
|
-
const attrSelector = `[${match[1]}="${match[2]}"]`;
|
|
220
|
-
if (match[1] === "data-slot")
|
|
221
|
-
continue;
|
|
222
|
-
if (!attrSafelist.has(attrSelector)) {
|
|
223
|
-
shouldKeep = false;
|
|
224
|
-
break;
|
|
225
|
-
}
|
|
226
|
-
}
|
|
227
|
-
if (shouldKeep)
|
|
228
|
-
kept.push(sel);
|
|
229
|
-
}
|
|
230
|
-
if (kept.length === 0) {
|
|
231
|
-
rule.remove();
|
|
232
|
-
} else if (kept.length < selectors.length) {
|
|
233
|
-
rule.selectors = kept;
|
|
234
|
-
}
|
|
235
|
-
});
|
|
236
|
-
return root.toString();
|
|
237
|
-
}
|
|
238
|
-
function cleanUnusedVars(css) {
|
|
239
|
-
let changed = true;
|
|
240
|
-
let result = css;
|
|
241
|
-
while (changed) {
|
|
242
|
-
changed = false;
|
|
243
|
-
const root = postcss.parse(result);
|
|
244
|
-
const declared = new Map;
|
|
245
|
-
root.walkDecls(/^--/, (decl) => {
|
|
246
|
-
const entries = declared.get(decl.prop) ?? [];
|
|
247
|
-
entries.push({ rule: decl.parent, prop: decl.prop, index: entries.length });
|
|
248
|
-
declared.set(decl.prop, entries);
|
|
249
|
-
});
|
|
250
|
-
const referenced = new Set;
|
|
251
|
-
root.walkDecls((decl) => {
|
|
252
|
-
const refs = decl.value.matchAll(/var\(\s*(--[a-zA-Z0-9_-]+)/g);
|
|
253
|
-
for (const ref of refs) {
|
|
254
|
-
referenced.add(ref[1]);
|
|
255
|
-
}
|
|
256
|
-
});
|
|
257
|
-
for (const [varName, entries] of declared) {
|
|
258
|
-
if (!referenced.has(varName)) {
|
|
259
|
-
for (const entry of entries) {
|
|
260
|
-
entry.rule.walkDecls(entry.prop, (decl) => {
|
|
261
|
-
decl.remove();
|
|
262
|
-
changed = true;
|
|
263
|
-
});
|
|
264
|
-
}
|
|
265
|
-
}
|
|
266
|
-
}
|
|
267
|
-
root.walkRules((rule) => {
|
|
268
|
-
if (rule.nodes && rule.nodes.length === 0)
|
|
269
|
-
rule.remove();
|
|
270
|
-
});
|
|
271
|
-
root.walkAtRules((atRule) => {
|
|
272
|
-
if (atRule.nodes && atRule.nodes.length === 0)
|
|
273
|
-
atRule.remove();
|
|
274
|
-
});
|
|
275
|
-
result = root.toString();
|
|
276
|
-
}
|
|
277
|
-
return result;
|
|
278
|
-
}
|
|
279
|
-
var pluginCssPurge = (options) => ({
|
|
280
|
-
name: "plugin-css-purge",
|
|
281
|
-
setup(api) {
|
|
282
|
-
const {
|
|
283
|
-
manifest: manifestPath,
|
|
284
|
-
srcDir = "src",
|
|
285
|
-
attrPurge = true,
|
|
286
|
-
cleanVars = true,
|
|
287
|
-
verbose = true
|
|
288
|
-
} = options;
|
|
289
|
-
api.processAssets({ stage: "optimize-size" }, async ({ assets, sources, compilation }) => {
|
|
290
|
-
const log = verbose ? console.log.bind(console) : () => {};
|
|
291
|
-
const manifest = JSON.parse(await nodeReadFile(path.resolve(manifestPath), "utf-8"));
|
|
292
|
-
log(`[css-purge] Manifest loaded: ${Object.keys(manifest).length} entries`);
|
|
293
|
-
const resolvedSrc = path.resolve(srcDir);
|
|
294
|
-
const usages = await scanConsumerSource(resolvedSrc);
|
|
295
|
-
log(`[css-purge] Scanned ${resolvedSrc}: ${usages.length} component usages`);
|
|
296
|
-
const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);
|
|
297
|
-
log(`[css-purge] Safelist: ${classSafelist.size} classes, ${attrSafelist.size} attrs`);
|
|
298
|
-
for (const [name, asset] of Object.entries(assets)) {
|
|
299
|
-
if (!name.endsWith(".css"))
|
|
300
|
-
continue;
|
|
301
|
-
const originalCss = asset.source().toString();
|
|
302
|
-
const originalSize = Buffer.byteLength(originalCss, "utf-8");
|
|
303
|
-
log(`[css-purge] Processing ${name} (${(originalSize / 1024).toFixed(1)} KB)`);
|
|
304
|
-
const purgeResult = await new PurgeCSS().purge({
|
|
305
|
-
content: [],
|
|
306
|
-
css: [{ raw: originalCss }],
|
|
307
|
-
safelist: [...classSafelist],
|
|
308
|
-
keyframes: false,
|
|
309
|
-
fontFace: false
|
|
310
|
-
});
|
|
311
|
-
let purgedCss = purgeResult[0]?.css ?? originalCss;
|
|
312
|
-
const afterL1 = Buffer.byteLength(purgedCss, "utf-8");
|
|
313
|
-
log(`[css-purge] L1 class purge: ${(originalSize / 1024).toFixed(1)} → ${(afterL1 / 1024).toFixed(1)} KB`);
|
|
314
|
-
if (attrPurge && attrSafelist.size > 0) {
|
|
315
|
-
purgedCss = purgeAttributes(purgedCss, attrSafelist);
|
|
316
|
-
const afterL2 = Buffer.byteLength(purgedCss, "utf-8");
|
|
317
|
-
log(`[css-purge] L2 attr purge: ${(afterL1 / 1024).toFixed(1)} → ${(afterL2 / 1024).toFixed(1)} KB`);
|
|
318
|
-
}
|
|
319
|
-
if (cleanVars) {
|
|
320
|
-
purgedCss = cleanUnusedVars(purgedCss);
|
|
321
|
-
const afterL3 = Buffer.byteLength(purgedCss, "utf-8");
|
|
322
|
-
log(`[css-purge] L3 var cleanup: → ${(afterL3 / 1024).toFixed(1)} KB`);
|
|
323
|
-
}
|
|
324
|
-
const finalSize = Buffer.byteLength(purgedCss, "utf-8");
|
|
325
|
-
log(`[css-purge] Final: ${(originalSize / 1024).toFixed(1)} → ${(finalSize / 1024).toFixed(1)} KB (${((1 - finalSize / originalSize) * 100).toFixed(1)}% reduction)`);
|
|
326
|
-
const source = new sources.RawSource(purgedCss);
|
|
327
|
-
compilation.updateAsset(name, source);
|
|
328
|
-
}
|
|
329
|
-
});
|
|
330
|
-
}
|
|
331
|
-
});
|
|
213
|
+
if (false) {}
|
|
332
214
|
export {
|
|
333
|
-
|
|
215
|
+
scanConsumerSource,
|
|
216
|
+
extractUIImports,
|
|
217
|
+
extractJSXUsages,
|
|
218
|
+
buildSafelists
|
|
334
219
|
};
|
|
335
220
|
|
|
336
|
-
//# debugId=
|
|
337
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/rsbuild-plugin.ts"],
  "sourcesContent": [
    "/**\n * rsbuild-plugin-css-purge\n *\n * Two-level CSS purge for @pathscale/ui consumers.\n *\n * Level 1: class-level purge via purgecss — removes entire rules whose selectors\n *          don't match the safelist built from consumer JSX analysis.\n * Level 2: attribute-level purge — within kept rules, strips compound selectors\n *          containing data-attr / aria-attr attribute selectors not in the attr safelist.\n *\n * Usage in rsbuild.config.ts:\n *   import { pluginCssPurge } from \"@pathscale/rebuild-plugin-ui-css-purge\";\n *   export default defineConfig({ plugins: [pluginCssPurge({ manifest: \"...\" })] });\n */\n\nimport type { RsbuildPlugin } from \"@rsbuild/core\";\nimport { PurgeCSS } from \"purgecss\";\nimport postcss from \"postcss\";\nimport type { Rule, AtRule } from \"postcss\";\nimport swc from \"@swc/core\";\nimport path from \"path\";\nimport { readFile as nodeReadFile } from \"node:fs/promises\";\nimport { glob as fastGlob } from \"fast-glob\";\n\n// ── Types ─────────────────────────────────────────────────────────────────────\n\ninterface ComponentManifest {\n  classes: {\n    always: string[];\n    byProp: Record<string, string[] | Record<string, string[]>>;\n  };\n  attrs?: Record<string, Record<string, string>>;\n}\n\ntype PurgeManifest = Record<string, ComponentManifest>;\n\ninterface PropUsage {\n  component: string;\n  props: Map<string, string | \"DYNAMIC\">;\n  booleanProps: Set<string>;\n  hasSpread: boolean;\n}\n\n// ── Plugin options ─────────────────────────────────────────────────────────────\n\nexport interface CssPurgeOptions {\n  /** Path to purge-manifest.json (generated by generate-manifest.ts) */\n  manifest: string;\n  /** Consumer source directory to scan for JSX usage (default: \"src\") */\n  srcDir?: string;\n  /** Enable Level 2 attribute purge (default: true) */\n  attrPurge?: boolean;\n  /** Enable CSS variable cleanup (default: true) */\n  cleanVars?: boolean;\n  /** Log purge stats (default: true) */\n  verbose?: boolean;\n}\n\n// ── AST utilities ─────────────────────────────────────────────────────────────\n\nfunction walkAST(node: any, visitor: (node: any) => void) {\n  if (!node || typeof node !== \"object\") return;\n  visitor(node);\n  for (const key of Object.keys(node)) {\n    if (key === \"span\") continue;\n    const val = node[key];\n    if (Array.isArray(val)) {\n      for (const item of val) walkAST(item, visitor);\n    } else if (val && typeof val === \"object\") {\n      walkAST(val, visitor);\n    }\n  }\n}\n\nfunction extractUIImports(ast: any): Map<string, string> {\n  const imports = new Map<string, string>();\n  for (const node of ast.body) {\n    if (node.type !== \"ImportDeclaration\") continue;\n    const src = node.source?.value as string;\n    if (!src || !src.startsWith(\"@pathscale/ui\")) continue;\n    for (const spec of node.specifiers) {\n      if (spec.type === \"ImportSpecifier\") {\n        const imported = spec.imported?.value ?? spec.local?.value;\n        const local = spec.local?.value;\n        if (local && imported) imports.set(local, imported);\n      } else if (spec.type === \"ImportDefaultSpecifier\") {\n        const local = spec.local?.value;\n        if (local) imports.set(local, local);\n      }\n    }\n  }\n  return imports;\n}\n\nfunction extractJSXUsages(ast: any, uiComponents: Map<string, string>): PropUsage[] {\n  const usages: PropUsage[] = [];\n  walkAST(ast, (node) => {\n    if (node.type !== \"JSXOpeningElement\") return;\n    let elementName: string | null = null;\n    let rootName: string | null = null;\n    if (node.name?.type === \"Identifier\") {\n      elementName = node.name.value;\n      rootName = elementName;\n    } else if (node.name?.type === \"JSXMemberExpression\") {\n      const parts: string[] = [];\n      let cursor = node.name;\n      while (cursor?.type === \"JSXMemberExpression\") {\n        parts.unshift(cursor.property?.value);\n        cursor = cursor.object;\n      }\n      if (cursor?.type === \"Identifier\") {\n        parts.unshift(cursor.value);\n        rootName = cursor.value;\n      }\n      elementName = parts.join(\".\");\n    }\n    if (!rootName || !uiComponents.has(rootName)) return;\n    const usage: PropUsage = {\n      component: elementName!,\n      props: new Map(),\n      booleanProps: new Set(),\n      hasSpread: false,\n    };\n    for (const attr of node.attributes || []) {\n      if (attr.type === \"SpreadElement\" || attr.type === \"JSXSpreadAttribute\") {\n        usage.hasSpread = true;\n        continue;\n      }\n      if (attr.type !== \"JSXAttribute\") continue;\n      const propName = attr.name?.value;\n      if (!propName) continue;\n      if (!attr.value) {\n        usage.booleanProps.add(propName);\n      } else if (attr.value.type === \"StringLiteral\") {\n        usage.props.set(propName, attr.value.value);\n      } else {\n        usage.props.set(propName, \"DYNAMIC\");\n      }\n    }\n    usages.push(usage);\n  });\n  return usages;\n}\n\n// ── Safelist builder ───────────────────────────────────────────────────────────\n\nfunction buildSafelists(\n  allUsages: PropUsage[],\n  manifest: PurgeManifest,\n): { classSafelist: Set<string>; attrSafelist: Set<string> } {\n  const classSafelist = new Set<string>();\n  const attrSafelist = new Set<string>();\n\n  const componentUsages = new Map<string, PropUsage[]>();\n  for (const usage of allUsages) {\n    const existing = componentUsages.get(usage.component) ?? [];\n    existing.push(usage);\n    componentUsages.set(usage.component, existing);\n  }\n\n  for (const [entryName, entry] of Object.entries(manifest)) {\n    const matchingUsages = findMatchingUsages(entryName, componentUsages);\n    if (matchingUsages.length === 0) continue;\n\n    for (const cls of entry.classes.always) classSafelist.add(cls);\n\n    for (const [propOrSlot, value] of Object.entries(entry.classes.byProp)) {\n      if (Array.isArray(value)) {\n        if (isPropUsed(propOrSlot, matchingUsages)) {\n          for (const cls of value) classSafelist.add(cls);\n        }\n      } else {\n        const usedValues = getUsedEnumValues(propOrSlot, matchingUsages);\n        if (usedValues === \"ALL\") {\n          for (const classes of Object.values(value)) {\n            for (const cls of classes) classSafelist.add(cls);\n          }\n        } else {\n          for (const val of usedValues) {\n            if (value[val]) {\n              for (const cls of value[val]) classSafelist.add(cls);\n            }\n          }\n        }\n      }\n    }\n\n    if (entry.attrs) {\n      for (const [propName, attrMap] of Object.entries(entry.attrs)) {\n        if (isPropUsed(propName, matchingUsages)) {\n          for (const [attr, val] of Object.entries(attrMap)) {\n            attrSafelist.add(`[${attr}=\"${val}\"]`);\n          }\n        }\n      }\n    }\n  }\n\n  return { classSafelist, attrSafelist };\n}\n\nfunction findMatchingUsages(entryName: string, usageMap: Map<string, PropUsage[]>): PropUsage[] {\n  if (usageMap.has(entryName)) return usageMap.get(entryName)!;\n  const results: PropUsage[] = [];\n  for (const [usageName, usages] of usageMap) {\n    if (usageName === entryName) {\n      results.push(...usages);\n    }\n    if (entryName.includes(\".\")) {\n      const [family, part] = entryName.split(\".\");\n      if (part === family && usageName === family) {\n        results.push(...usages);\n      }\n    }\n  }\n  return results;\n}\n\nfunction isPropUsed(propName: string, usages: PropUsage[]): boolean {\n  for (const usage of usages) {\n    if (usage.hasSpread) return true;\n    if (usage.booleanProps.has(propName)) return true;\n    if (usage.props.has(propName)) return true;\n  }\n  return false;\n}\n\nfunction getUsedEnumValues(slotName: string, usages: PropUsage[]): Set<string> | \"ALL\" {\n  const values = new Set<string>();\n  for (const usage of usages) {\n    if (usage.hasSpread) return \"ALL\";\n    const val = usage.props.get(slotName);\n    if (val === \"DYNAMIC\") return \"ALL\";\n    if (val !== undefined) values.add(val);\n    if (usage.booleanProps.has(slotName)) return \"ALL\";\n  }\n  return values;\n}\n\n// ── Consumer source scanning ───────────────────────────────────────────────────\n\nasync function scanConsumerSource(srcDir: string): Promise<PropUsage[]> {\n  const allUsages: PropUsage[] = [];\n  const files = await fastGlob(\"**/*.{tsx,ts,jsx,js}\", { cwd: srcDir, ignore: [\"**/node_modules/**\"] });\n\n  for (const relPath of files) {\n    const fullPath = path.join(srcDir, relPath);\n    const code = await nodeReadFile(fullPath, \"utf-8\");\n    if (!code.includes(\"@pathscale/ui\")) continue;\n\n    const isTsx = /\\.[tj]sx$/.test(relPath);\n    const ast = await swc.parse(code, { syntax: \"typescript\", tsx: isTsx });\n    const uiImports = extractUIImports(ast);\n    if (uiImports.size === 0) continue;\n\n    allUsages.push(...extractJSXUsages(ast, uiImports));\n  }\n\n  return allUsages;\n}\n\n// ── Level 2: attribute purge ───────────────────────────────────────────────────\n\nfunction purgeAttributes(css: string, attrSafelist: Set<string>): string {\n  const root = postcss.parse(css);\n\n  root.walkRules((rule) => {\n    const selectors = rule.selectors;\n    const kept: string[] = [];\n\n    for (const sel of selectors) {\n      const attrMatches = sel.matchAll(/\\[(data-[a-z-]+|aria-[a-z-]+)=\"([^\"]+)\"\\]/g);\n      let shouldKeep = true;\n\n      for (const match of attrMatches) {\n        const attrSelector = `[${match[1]}=\"${match[2]}\"]`;\n        if (match[1] === \"data-slot\") continue;\n        if (!attrSafelist.has(attrSelector)) {\n          shouldKeep = false;\n          break;\n        }\n      }\n\n      if (shouldKeep) kept.push(sel);\n    }\n\n    if (kept.length === 0) {\n      rule.remove();\n    } else if (kept.length < selectors.length) {\n      rule.selectors = kept;\n    }\n  });\n\n  return root.toString();\n}\n\n// ── Level 3: unused CSS variable cleanup ───────────────────────────────────────\n\nfunction cleanUnusedVars(css: string): string {\n  let changed = true;\n  let result = css;\n\n  while (changed) {\n    changed = false;\n    const root = postcss.parse(result);\n\n    const declared = new Map<string, { rule: Rule | AtRule; prop: string; index: number }[]>();\n    root.walkDecls(/^--/, (decl) => {\n      const entries = declared.get(decl.prop) ?? [];\n      entries.push({ rule: decl.parent as Rule, prop: decl.prop, index: entries.length });\n      declared.set(decl.prop, entries);\n    });\n\n    const referenced = new Set<string>();\n    root.walkDecls((decl) => {\n      const refs = decl.value.matchAll(/var\\(\\s*(--[a-zA-Z0-9_-]+)/g);\n      for (const ref of refs) {\n        referenced.add(ref[1]);\n      }\n    });\n\n    for (const [varName, entries] of declared) {\n      if (!referenced.has(varName)) {\n        for (const entry of entries) {\n          entry.rule.walkDecls(entry.prop, (decl) => {\n            decl.remove();\n            changed = true;\n          });\n        }\n      }\n    }\n\n    root.walkRules((rule) => {\n      if (rule.nodes && rule.nodes.length === 0) rule.remove();\n    });\n    root.walkAtRules((atRule) => {\n      if (atRule.nodes && atRule.nodes.length === 0) atRule.remove();\n    });\n\n    result = root.toString();\n  }\n\n  return result;\n}\n\n// ── The rsbuild plugin ─────────────────────────────────────────────────────────\n\nexport const pluginCssPurge = (options: CssPurgeOptions): RsbuildPlugin => ({\n  name: \"plugin-css-purge\",\n\n  setup(api) {\n    const {\n      manifest: manifestPath,\n      srcDir = \"src\",\n      attrPurge = true,\n      cleanVars = true,\n      verbose = true,\n    } = options;\n\n    api.processAssets(\n      { stage: \"optimize-size\" },\n      async ({ assets, sources, compilation }) => {\n        const log = verbose ? console.log.bind(console) : () => {};\n\n        // 1. Load manifest\n        const manifest: PurgeManifest = JSON.parse(\n          await nodeReadFile(path.resolve(manifestPath), \"utf-8\"),\n        );\n        log(`[css-purge] Manifest loaded: ${Object.keys(manifest).length} entries`);\n\n        // 2. Scan consumer source\n        const resolvedSrc = path.resolve(srcDir);\n        const usages = await scanConsumerSource(resolvedSrc);\n        log(`[css-purge] Scanned ${resolvedSrc}: ${usages.length} component usages`);\n\n        // 3. Build safelists\n        const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);\n        log(`[css-purge] Safelist: ${classSafelist.size} classes, ${attrSafelist.size} attrs`);\n\n        // 4. Process each CSS asset\n        for (const [name, asset] of Object.entries(assets)) {\n          if (!name.endsWith(\".css\")) continue;\n\n          const originalCss = asset.source().toString();\n          const originalSize = Buffer.byteLength(originalCss, \"utf-8\");\n          log(`[css-purge] Processing ${name} (${(originalSize / 1024).toFixed(1)} KB)`);\n\n          // Level 1: class-level purge\n          const purgeResult = await new PurgeCSS().purge({\n            content: [],\n            css: [{ raw: originalCss }],\n            safelist: [...classSafelist],\n            keyframes: false,\n            fontFace: false,\n          });\n\n          let purgedCss = purgeResult[0]?.css ?? originalCss;\n          const afterL1 = Buffer.byteLength(purgedCss, \"utf-8\");\n          log(`[css-purge]   L1 class purge: ${(originalSize / 1024).toFixed(1)} → ${(afterL1 / 1024).toFixed(1)} KB`);\n\n          // Level 2: attribute-level purge\n          if (attrPurge && attrSafelist.size > 0) {\n            purgedCss = purgeAttributes(purgedCss, attrSafelist);\n            const afterL2 = Buffer.byteLength(purgedCss, \"utf-8\");\n            log(`[css-purge]   L2 attr purge: ${(afterL1 / 1024).toFixed(1)} → ${(afterL2 / 1024).toFixed(1)} KB`);\n          }\n\n          // Level 3: unused CSS variable cleanup\n          if (cleanVars) {\n            purgedCss = cleanUnusedVars(purgedCss);\n            const afterL3 = Buffer.byteLength(purgedCss, \"utf-8\");\n            log(`[css-purge]   L3 var cleanup: → ${(afterL3 / 1024).toFixed(1)} KB`);\n          }\n\n          const finalSize = Buffer.byteLength(purgedCss, \"utf-8\");\n          log(`[css-purge]   Final: ${(originalSize / 1024).toFixed(1)} → ${(finalSize / 1024).toFixed(1)} KB (${((1 - finalSize / originalSize) * 100).toFixed(1)}% reduction)`);\n\n          // Write back\n          const source = new sources.RawSource(purgedCss);\n          compilation.updateAsset(name, source);\n        }\n      },\n    );\n  },\n});\n"
  ],
  "mappings": "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",
  "debugId": "CB5B2093063663E864756E2164756E21",
  "names": []
}
|
|
221
|
+
//# debugId=E6760AADCED89B2864756E2164756E21
|
|
222
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/scan-consumer.ts"],
  "sourcesContent": [
    "/**\n * Consumer-side JSX scanner.\n *\n * Walks a consumer's source tree, finds component imports from @pathscale/ui,\n * collects prop values, and cross-references with the purge manifest to build\n * Level 1 (class) and Level 2 (attribute) safelists.\n *\n * Usage:  bun run src/scan-consumer.ts <consumer-src-dir> <purge-manifest.json>\n */\n\nimport swc from \"@swc/core\";\nimport { Glob } from \"bun\";\nimport path from \"path\";\n\n// ── Types ──────────────────────────────────────────────────────────────────────\n\ninterface ComponentManifest {\n  classes: {\n    always: string[];\n    byProp: Record<string, string[] | Record<string, string[]>>;\n  };\n  attrs?: Record<string, Record<string, string>>;\n}\n\ntype PurgeManifest = Record<string, ComponentManifest>;\n\n/** What we collect per component usage from JSX */\ninterface PropUsage {\n  component: string;\n  props: Map<string, string | \"DYNAMIC\">; // propName → literal value or DYNAMIC\n  booleanProps: Set<string>; // props present without a value (truthy)\n  hasSpread: boolean;\n}\n\n// ── AST walker ─────────────────────────────────────────────────────────────────\n\nfunction walkAST(node: any, visitor: (node: any) => void) {\n  if (!node || typeof node !== \"object\") return;\n  visitor(node);\n  for (const key of Object.keys(node)) {\n    if (key === \"span\") continue;\n    const val = node[key];\n    if (Array.isArray(val)) {\n      for (const item of val) walkAST(item, visitor);\n    } else if (val && typeof val === \"object\") {\n      walkAST(val, visitor);\n    }\n  }\n}\n\n/** Extract @pathscale/ui imports from a parsed module */\nfunction extractUIImports(ast: any): Map<string, string> {\n  const imports = new Map<string, string>();\n  for (const node of ast.body) {\n    if (node.type !== \"ImportDeclaration\") continue;\n    const src = node.source?.value as string;\n    if (!src || !src.startsWith(\"@pathscale/ui\")) continue;\n\n    for (const spec of node.specifiers) {\n      if (spec.type === \"ImportSpecifier\") {\n        const imported = spec.imported?.value ?? spec.local?.value;\n        const local = spec.local?.value;\n        if (local && imported) imports.set(local, imported);\n      } else if (spec.type === \"ImportDefaultSpecifier\") {\n        const local = spec.local?.value;\n        if (local) imports.set(local, local);\n      }\n    }\n  }\n  return imports;\n}\n\n/** Extract JSX usages of UI components */\nfunction extractJSXUsages(ast: any, uiComponents: Map<string, string>): PropUsage[] {\n  const usages: PropUsage[] = [];\n\n  walkAST(ast, (node) => {\n    if (node.type !== \"JSXOpeningElement\") return;\n\n    let elementName: string | null = null;\n    let rootName: string | null = null;\n\n    if (node.name?.type === \"Identifier\") {\n      elementName = node.name.value;\n      rootName = elementName;\n    } else if (node.name?.type === \"JSXMemberExpression\") {\n      const parts: string[] = [];\n      let cursor = node.name;\n      while (cursor?.type === \"JSXMemberExpression\") {\n        parts.unshift(cursor.property?.value);\n        cursor = cursor.object;\n      }\n      if (cursor?.type === \"Identifier\") {\n        parts.unshift(cursor.value);\n        rootName = cursor.value;\n      }\n      elementName = parts.join(\".\");\n    }\n\n    if (!rootName || !uiComponents.has(rootName)) return;\n\n    const usage: PropUsage = {\n      component: elementName!,\n      props: new Map(),\n      booleanProps: new Set(),\n      hasSpread: false,\n    };\n\n    for (const attr of node.attributes || []) {\n      if (attr.type === \"SpreadElement\" || attr.type === \"JSXSpreadAttribute\") {\n        usage.hasSpread = true;\n        continue;\n      }\n      if (attr.type !== \"JSXAttribute\") continue;\n\n      const propName = attr.name?.value;\n      if (!propName) continue;\n\n      if (!attr.value) {\n        usage.booleanProps.add(propName);\n      } else if (attr.value.type === \"StringLiteral\") {\n        usage.props.set(propName, attr.value.value);\n      } else {\n        usage.props.set(propName, \"DYNAMIC\");\n      }\n    }\n\n    usages.push(usage);\n  });\n\n  return usages;\n}\n\n// ── Safelist builder ───────────────────────────────────────────────────────────\n\ninterface Safelists {\n  classSafelist: Set<string>;\n  attrSafelist: Set<string>;\n}\n\nfunction buildSafelists(\n  allUsages: PropUsage[],\n  manifest: PurgeManifest,\n): Safelists {\n  const classSafelist = new Set<string>();\n  const attrSafelist = new Set<string>();\n\n  const componentUsages = new Map<string, PropUsage[]>();\n  for (const usage of allUsages) {\n    const existing = componentUsages.get(usage.component) ?? [];\n    existing.push(usage);\n    componentUsages.set(usage.component, existing);\n  }\n\n  for (const [entryName, entry] of Object.entries(manifest)) {\n    const matchingUsages = findMatchingUsages(entryName, componentUsages);\n\n    if (matchingUsages.length === 0) {\n      continue;\n    }\n\n    for (const cls of entry.classes.always) {\n      classSafelist.add(cls);\n    }\n\n    for (const [propOrSlot, value] of Object.entries(entry.classes.byProp)) {\n      if (Array.isArray(value)) {\n        if (isPropUsed(propOrSlot, matchingUsages)) {\n          for (const cls of value) classSafelist.add(cls);\n        }\n      } else {\n        const usedValues = getUsedEnumValues(propOrSlot, matchingUsages);\n        if (usedValues === \"ALL\") {\n          for (const classes of Object.values(value)) {\n            for (const cls of classes) classSafelist.add(cls);\n          }\n        } else {\n          for (const val of usedValues) {\n            if (value[val]) {\n              for (const cls of value[val]) classSafelist.add(cls);\n            }\n          }\n        }\n      }\n    }\n\n    if (entry.attrs) {\n      for (const [propName, attrMap] of Object.entries(entry.attrs)) {\n        if (isPropUsed(propName, matchingUsages)) {\n          for (const [attr, val] of Object.entries(attrMap)) {\n            attrSafelist.add(`${attr}=${val}`);\n          }\n        }\n      }\n    }\n  }\n\n  return { classSafelist, attrSafelist };\n}\n\nfunction findMatchingUsages(\n  entryName: string,\n  usageMap: Map<string, PropUsage[]>,\n): PropUsage[] {\n  if (usageMap.has(entryName)) {\n    return usageMap.get(entryName)!;\n  }\n\n  const results: PropUsage[] = [];\n  for (const [usageName, usages] of usageMap) {\n    if (usageName === entryName) {\n      results.push(...usages);\n    }\n    if (entryName.includes(\".\")) {\n      const [family, part] = entryName.split(\".\");\n      if (part === family && usageName === family) {\n        results.push(...usages);\n      }\n    }\n  }\n  return results;\n}\n\nfunction isPropUsed(propName: string, usages: PropUsage[]): boolean {\n  for (const usage of usages) {\n    if (usage.hasSpread) return true;\n    if (usage.booleanProps.has(propName)) return true;\n    if (usage.props.has(propName)) return true;\n  }\n  return false;\n}\n\nfunction getUsedEnumValues(slotName: string, usages: PropUsage[]): Set<string> | \"ALL\" {\n  const values = new Set<string>();\n  for (const usage of usages) {\n    if (usage.hasSpread) return \"ALL\";\n    const val = usage.props.get(slotName);\n    if (val === \"DYNAMIC\") return \"ALL\";\n    if (val !== undefined) values.add(val);\n    if (usage.booleanProps.has(slotName)) return \"ALL\";\n  }\n  return values;\n}\n\n// ── Consumer source scanning ───────────────────────────────────────────────────\n\nasync function scanConsumerSource(srcDir: string): Promise<PropUsage[]> {\n  const allUsages: PropUsage[] = [];\n  const glob = new Glob(\"**/*.{tsx,ts,jsx,js}\");\n\n  for await (const relPath of glob.scan({ cwd: srcDir })) {\n    if (relPath.includes(\"node_modules\")) continue;\n    const fullPath = path.join(srcDir, relPath);\n    const code = await Bun.file(fullPath).text();\n    if (!code.includes(\"@pathscale/ui\")) continue;\n\n    const isTsx = /\\.[tj]sx$/.test(relPath);\n    const ast = await swc.parse(code, { syntax: \"typescript\", tsx: isTsx });\n    const uiImports = extractUIImports(ast);\n    if (uiImports.size === 0) continue;\n\n    allUsages.push(...extractJSXUsages(ast, uiImports));\n  }\n\n  return allUsages;\n}\n\n// ── Main (standalone CLI) ─────────────────────────────────────────────────────\n\nasync function main() {\n  const [srcDir, manifestPath] = process.argv.slice(2);\n  if (!srcDir || !manifestPath) {\n    console.error(\"Usage: bun run src/scan-consumer.ts <consumer-src-dir> <purge-manifest.json>\");\n    process.exit(1);\n  }\n\n  const manifest: PurgeManifest = JSON.parse(\n    await Bun.file(manifestPath).text(),\n  );\n  const resolvedSrc = path.resolve(srcDir);\n  console.log(`Scanning ${resolvedSrc} for @pathscale/ui component usage…`);\n  console.log(`Manifest: ${Object.keys(manifest).length} entries\\n`);\n\n  const usages = await scanConsumerSource(resolvedSrc);\n  const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);\n\n  console.log(\"=== Class Safelist ===\");\n  for (const cls of [...classSafelist].sort()) {\n    console.log(`  ${cls}`);\n  }\n\n  console.log(`\\n=== Attribute Safelist ===`);\n  for (const attr of [...attrSafelist].sort()) {\n    console.log(`  [${attr}]`);\n  }\n\n  console.log(`\\nTotal: ${classSafelist.size} classes, ${attrSafelist.size} attribute selectors`);\n}\n\n// Only run CLI when invoked directly (not when imported as a module)\nif (import.meta.main) {\n  main();\n}\n\nexport { extractUIImports, extractJSXUsages, buildSafelists, scanConsumerSource };\nexport type { PropUsage, PurgeManifest, ComponentManifest, Safelists };\n"
  ],
  "mappings": "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",
  "debugId": "E6760AADCED89B2864756E2164756E21",
  "names": []
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
#!/usr/bin/env bun
|
|
2
|
+
/**
|
|
3
|
+
* Postbuild CSS purge — standalone Bun script.
|
|
4
|
+
*
|
|
5
|
+
* Runs after rsbuild build, purges CSS files in dist/ using the purge manifest
|
|
6
|
+
* and consumer JSX analysis. Zero Node imports.
|
|
7
|
+
*
|
|
8
|
+
* Usage:
|
|
9
|
+
* bunx @pathscale/rebuild-plugin-ui-css-purge \
|
|
10
|
+
* --dist dist --src src --manifest node_modules/@pathscale/ui/dist/purge-manifest.json
|
|
11
|
+
*/
|
|
12
|
+
export {};
|
|
@@ -0,0 +1,362 @@
|
|
|
1
|
+
#!/usr/bin/env bun
|
|
2
|
+
// @bun
|
|
3
|
+
|
|
4
|
+
// src/postbuild-purge.ts
|
|
5
|
+
var {Glob: Glob2 } = globalThis.Bun;
|
|
6
|
+
import { PurgeCSS } from "purgecss";
|
|
7
|
+
import postcss from "postcss";
|
|
8
|
+
import path2 from "path";
|
|
9
|
+
|
|
10
|
+
// src/scan-consumer.ts
|
|
11
|
+
import swc from "@swc/core";
|
|
12
|
+
var {Glob } = globalThis.Bun;
|
|
13
|
+
import path from "path";
|
|
14
|
+
function walkAST(node, visitor) {
|
|
15
|
+
if (!node || typeof node !== "object")
|
|
16
|
+
return;
|
|
17
|
+
visitor(node);
|
|
18
|
+
for (const key of Object.keys(node)) {
|
|
19
|
+
if (key === "span")
|
|
20
|
+
continue;
|
|
21
|
+
const val = node[key];
|
|
22
|
+
if (Array.isArray(val)) {
|
|
23
|
+
for (const item of val)
|
|
24
|
+
walkAST(item, visitor);
|
|
25
|
+
} else if (val && typeof val === "object") {
|
|
26
|
+
walkAST(val, visitor);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
function extractUIImports(ast) {
|
|
31
|
+
const imports = new Map;
|
|
32
|
+
for (const node of ast.body) {
|
|
33
|
+
if (node.type !== "ImportDeclaration")
|
|
34
|
+
continue;
|
|
35
|
+
const src = node.source?.value;
|
|
36
|
+
if (!src || !src.startsWith("@pathscale/ui"))
|
|
37
|
+
continue;
|
|
38
|
+
for (const spec of node.specifiers) {
|
|
39
|
+
if (spec.type === "ImportSpecifier") {
|
|
40
|
+
const imported = spec.imported?.value ?? spec.local?.value;
|
|
41
|
+
const local = spec.local?.value;
|
|
42
|
+
if (local && imported)
|
|
43
|
+
imports.set(local, imported);
|
|
44
|
+
} else if (spec.type === "ImportDefaultSpecifier") {
|
|
45
|
+
const local = spec.local?.value;
|
|
46
|
+
if (local)
|
|
47
|
+
imports.set(local, local);
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
return imports;
|
|
52
|
+
}
|
|
53
|
+
function extractJSXUsages(ast, uiComponents) {
|
|
54
|
+
const usages = [];
|
|
55
|
+
walkAST(ast, (node) => {
|
|
56
|
+
if (node.type !== "JSXOpeningElement")
|
|
57
|
+
return;
|
|
58
|
+
let elementName = null;
|
|
59
|
+
let rootName = null;
|
|
60
|
+
if (node.name?.type === "Identifier") {
|
|
61
|
+
elementName = node.name.value;
|
|
62
|
+
rootName = elementName;
|
|
63
|
+
} else if (node.name?.type === "JSXMemberExpression") {
|
|
64
|
+
const parts = [];
|
|
65
|
+
let cursor = node.name;
|
|
66
|
+
while (cursor?.type === "JSXMemberExpression") {
|
|
67
|
+
parts.unshift(cursor.property?.value);
|
|
68
|
+
cursor = cursor.object;
|
|
69
|
+
}
|
|
70
|
+
if (cursor?.type === "Identifier") {
|
|
71
|
+
parts.unshift(cursor.value);
|
|
72
|
+
rootName = cursor.value;
|
|
73
|
+
}
|
|
74
|
+
elementName = parts.join(".");
|
|
75
|
+
}
|
|
76
|
+
if (!rootName || !uiComponents.has(rootName))
|
|
77
|
+
return;
|
|
78
|
+
const usage = {
|
|
79
|
+
component: elementName,
|
|
80
|
+
props: new Map,
|
|
81
|
+
booleanProps: new Set,
|
|
82
|
+
hasSpread: false
|
|
83
|
+
};
|
|
84
|
+
for (const attr of node.attributes || []) {
|
|
85
|
+
if (attr.type === "SpreadElement" || attr.type === "JSXSpreadAttribute") {
|
|
86
|
+
usage.hasSpread = true;
|
|
87
|
+
continue;
|
|
88
|
+
}
|
|
89
|
+
if (attr.type !== "JSXAttribute")
|
|
90
|
+
continue;
|
|
91
|
+
const propName = attr.name?.value;
|
|
92
|
+
if (!propName)
|
|
93
|
+
continue;
|
|
94
|
+
if (!attr.value) {
|
|
95
|
+
usage.booleanProps.add(propName);
|
|
96
|
+
} else if (attr.value.type === "StringLiteral") {
|
|
97
|
+
usage.props.set(propName, attr.value.value);
|
|
98
|
+
} else {
|
|
99
|
+
usage.props.set(propName, "DYNAMIC");
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
usages.push(usage);
|
|
103
|
+
});
|
|
104
|
+
return usages;
|
|
105
|
+
}
|
|
106
|
+
function buildSafelists(allUsages, manifest) {
|
|
107
|
+
const classSafelist = new Set;
|
|
108
|
+
const attrSafelist = new Set;
|
|
109
|
+
const componentUsages = new Map;
|
|
110
|
+
for (const usage of allUsages) {
|
|
111
|
+
const existing = componentUsages.get(usage.component) ?? [];
|
|
112
|
+
existing.push(usage);
|
|
113
|
+
componentUsages.set(usage.component, existing);
|
|
114
|
+
}
|
|
115
|
+
for (const [entryName, entry] of Object.entries(manifest)) {
|
|
116
|
+
const matchingUsages = findMatchingUsages(entryName, componentUsages);
|
|
117
|
+
if (matchingUsages.length === 0) {
|
|
118
|
+
continue;
|
|
119
|
+
}
|
|
120
|
+
for (const cls of entry.classes.always) {
|
|
121
|
+
classSafelist.add(cls);
|
|
122
|
+
}
|
|
123
|
+
for (const [propOrSlot, value] of Object.entries(entry.classes.byProp)) {
|
|
124
|
+
if (Array.isArray(value)) {
|
|
125
|
+
if (isPropUsed(propOrSlot, matchingUsages)) {
|
|
126
|
+
for (const cls of value)
|
|
127
|
+
classSafelist.add(cls);
|
|
128
|
+
}
|
|
129
|
+
} else {
|
|
130
|
+
const usedValues = getUsedEnumValues(propOrSlot, matchingUsages);
|
|
131
|
+
if (usedValues === "ALL") {
|
|
132
|
+
for (const classes of Object.values(value)) {
|
|
133
|
+
for (const cls of classes)
|
|
134
|
+
classSafelist.add(cls);
|
|
135
|
+
}
|
|
136
|
+
} else {
|
|
137
|
+
for (const val of usedValues) {
|
|
138
|
+
if (value[val]) {
|
|
139
|
+
for (const cls of value[val])
|
|
140
|
+
classSafelist.add(cls);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
if (entry.attrs) {
|
|
147
|
+
for (const [propName, attrMap] of Object.entries(entry.attrs)) {
|
|
148
|
+
if (isPropUsed(propName, matchingUsages)) {
|
|
149
|
+
for (const [attr, val] of Object.entries(attrMap)) {
|
|
150
|
+
attrSafelist.add(`${attr}=${val}`);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
return { classSafelist, attrSafelist };
|
|
157
|
+
}
|
|
158
|
+
function findMatchingUsages(entryName, usageMap) {
|
|
159
|
+
if (usageMap.has(entryName)) {
|
|
160
|
+
return usageMap.get(entryName);
|
|
161
|
+
}
|
|
162
|
+
const results = [];
|
|
163
|
+
for (const [usageName, usages] of usageMap) {
|
|
164
|
+
if (usageName === entryName) {
|
|
165
|
+
results.push(...usages);
|
|
166
|
+
}
|
|
167
|
+
if (entryName.includes(".")) {
|
|
168
|
+
const [family, part] = entryName.split(".");
|
|
169
|
+
if (part === family && usageName === family) {
|
|
170
|
+
results.push(...usages);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
return results;
|
|
175
|
+
}
|
|
176
|
+
function isPropUsed(propName, usages) {
|
|
177
|
+
for (const usage of usages) {
|
|
178
|
+
if (usage.hasSpread)
|
|
179
|
+
return true;
|
|
180
|
+
if (usage.booleanProps.has(propName))
|
|
181
|
+
return true;
|
|
182
|
+
if (usage.props.has(propName))
|
|
183
|
+
return true;
|
|
184
|
+
}
|
|
185
|
+
return false;
|
|
186
|
+
}
|
|
187
|
+
function getUsedEnumValues(slotName, usages) {
|
|
188
|
+
const values = new Set;
|
|
189
|
+
for (const usage of usages) {
|
|
190
|
+
if (usage.hasSpread)
|
|
191
|
+
return "ALL";
|
|
192
|
+
const val = usage.props.get(slotName);
|
|
193
|
+
if (val === "DYNAMIC")
|
|
194
|
+
return "ALL";
|
|
195
|
+
if (val !== undefined)
|
|
196
|
+
values.add(val);
|
|
197
|
+
if (usage.booleanProps.has(slotName))
|
|
198
|
+
return "ALL";
|
|
199
|
+
}
|
|
200
|
+
return values;
|
|
201
|
+
}
|
|
202
|
+
async function scanConsumerSource(srcDir) {
|
|
203
|
+
const allUsages = [];
|
|
204
|
+
const glob = new Glob("**/*.{tsx,ts,jsx,js}");
|
|
205
|
+
for await (const relPath of glob.scan({ cwd: srcDir })) {
|
|
206
|
+
if (relPath.includes("node_modules"))
|
|
207
|
+
continue;
|
|
208
|
+
const fullPath = path.join(srcDir, relPath);
|
|
209
|
+
const code = await Bun.file(fullPath).text();
|
|
210
|
+
if (!code.includes("@pathscale/ui"))
|
|
211
|
+
continue;
|
|
212
|
+
const isTsx = /\.[tj]sx$/.test(relPath);
|
|
213
|
+
const ast = await swc.parse(code, { syntax: "typescript", tsx: isTsx });
|
|
214
|
+
const uiImports = extractUIImports(ast);
|
|
215
|
+
if (uiImports.size === 0)
|
|
216
|
+
continue;
|
|
217
|
+
allUsages.push(...extractJSXUsages(ast, uiImports));
|
|
218
|
+
}
|
|
219
|
+
return allUsages;
|
|
220
|
+
}
|
|
221
|
+
if (false) {}
|
|
222
|
+
|
|
223
|
+
// src/postbuild-purge.ts
|
|
224
|
+
function parseArgs(argv) {
|
|
225
|
+
const args = argv.slice(2);
|
|
226
|
+
let distDir = "./dist";
|
|
227
|
+
let srcDir = "./src";
|
|
228
|
+
let manifestPath = "";
|
|
229
|
+
for (let i = 0;i < args.length; i++) {
|
|
230
|
+
if (args[i] === "--dist" && args[i + 1])
|
|
231
|
+
distDir = args[++i];
|
|
232
|
+
else if (args[i] === "--src" && args[i + 1])
|
|
233
|
+
srcDir = args[++i];
|
|
234
|
+
else if (args[i] === "--manifest" && args[i + 1])
|
|
235
|
+
manifestPath = args[++i];
|
|
236
|
+
}
|
|
237
|
+
if (!manifestPath) {
|
|
238
|
+
console.error("Usage: bunx @pathscale/rebuild-plugin-ui-css-purge --manifest <path> [--dist <path>] [--src <path>]");
|
|
239
|
+
process.exit(1);
|
|
240
|
+
}
|
|
241
|
+
return {
|
|
242
|
+
distDir: path2.resolve(distDir),
|
|
243
|
+
srcDir: path2.resolve(srcDir),
|
|
244
|
+
manifestPath: path2.resolve(manifestPath)
|
|
245
|
+
};
|
|
246
|
+
}
|
|
247
|
+
function purgeAttributes(css, attrSafelist) {
|
|
248
|
+
const root = postcss.parse(css);
|
|
249
|
+
root.walkRules((rule) => {
|
|
250
|
+
const selectors = rule.selectors;
|
|
251
|
+
const kept = [];
|
|
252
|
+
for (const sel of selectors) {
|
|
253
|
+
const attrMatches = sel.matchAll(/\[(data-[a-z-]+|aria-[a-z-]+)="([^"]+)"\]/g);
|
|
254
|
+
let shouldKeep = true;
|
|
255
|
+
for (const match of attrMatches) {
|
|
256
|
+
const attrSelector = `[${match[1]}="${match[2]}"]`;
|
|
257
|
+
if (match[1] === "data-slot")
|
|
258
|
+
continue;
|
|
259
|
+
if (!attrSafelist.has(attrSelector)) {
|
|
260
|
+
shouldKeep = false;
|
|
261
|
+
break;
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
if (shouldKeep)
|
|
265
|
+
kept.push(sel);
|
|
266
|
+
}
|
|
267
|
+
if (kept.length === 0) {
|
|
268
|
+
rule.remove();
|
|
269
|
+
} else if (kept.length < selectors.length) {
|
|
270
|
+
rule.selectors = kept;
|
|
271
|
+
}
|
|
272
|
+
});
|
|
273
|
+
return root.toString();
|
|
274
|
+
}
|
|
275
|
+
function cleanUnusedVars(css) {
|
|
276
|
+
let changed = true;
|
|
277
|
+
let result = css;
|
|
278
|
+
while (changed) {
|
|
279
|
+
changed = false;
|
|
280
|
+
const root = postcss.parse(result);
|
|
281
|
+
const declared = new Map;
|
|
282
|
+
root.walkDecls(/^--/, (decl) => {
|
|
283
|
+
const entries = declared.get(decl.prop) ?? [];
|
|
284
|
+
entries.push({ rule: decl.parent, prop: decl.prop, index: entries.length });
|
|
285
|
+
declared.set(decl.prop, entries);
|
|
286
|
+
});
|
|
287
|
+
const referenced = new Set;
|
|
288
|
+
root.walkDecls((decl) => {
|
|
289
|
+
const refs = decl.value.matchAll(/var\(\s*(--[a-zA-Z0-9_-]+)/g);
|
|
290
|
+
for (const ref of refs) {
|
|
291
|
+
referenced.add(ref[1]);
|
|
292
|
+
}
|
|
293
|
+
});
|
|
294
|
+
for (const [varName, entries] of declared) {
|
|
295
|
+
if (!referenced.has(varName)) {
|
|
296
|
+
for (const entry of entries) {
|
|
297
|
+
entry.rule.walkDecls(entry.prop, (decl) => {
|
|
298
|
+
decl.remove();
|
|
299
|
+
changed = true;
|
|
300
|
+
});
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
root.walkRules((rule) => {
|
|
305
|
+
if (rule.nodes && rule.nodes.length === 0)
|
|
306
|
+
rule.remove();
|
|
307
|
+
});
|
|
308
|
+
root.walkAtRules((atRule) => {
|
|
309
|
+
if (atRule.nodes && atRule.nodes.length === 0)
|
|
310
|
+
atRule.remove();
|
|
311
|
+
});
|
|
312
|
+
result = root.toString();
|
|
313
|
+
}
|
|
314
|
+
return result;
|
|
315
|
+
}
|
|
316
|
+
async function main() {
|
|
317
|
+
const { distDir, srcDir, manifestPath } = parseArgs(process.argv);
|
|
318
|
+
const manifest = JSON.parse(await Bun.file(manifestPath).text());
|
|
319
|
+
console.log(`[css-purge] Manifest loaded: ${Object.keys(manifest).length} entries`);
|
|
320
|
+
const usages = await scanConsumerSource(srcDir);
|
|
321
|
+
console.log(`[css-purge] Scanned ${srcDir}: ${usages.length} component usages`);
|
|
322
|
+
const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);
|
|
323
|
+
console.log(`[css-purge] Safelist: ${classSafelist.size} classes, ${attrSafelist.size} attrs`);
|
|
324
|
+
const glob = new Glob2("**/*.css");
|
|
325
|
+
let totalBefore = 0;
|
|
326
|
+
let totalAfter = 0;
|
|
327
|
+
for await (const relPath of glob.scan({ cwd: distDir })) {
|
|
328
|
+
const fullPath = path2.join(distDir, relPath);
|
|
329
|
+
const originalCss = await Bun.file(fullPath).text();
|
|
330
|
+
const originalSize = Buffer.byteLength(originalCss, "utf-8");
|
|
331
|
+
totalBefore += originalSize;
|
|
332
|
+
console.log(`[css-purge] Processing ${relPath} (${(originalSize / 1024).toFixed(1)} KB)`);
|
|
333
|
+
const purgeResult = await new PurgeCSS().purge({
|
|
334
|
+
content: [],
|
|
335
|
+
css: [{ raw: originalCss }],
|
|
336
|
+
safelist: [...classSafelist],
|
|
337
|
+
keyframes: false,
|
|
338
|
+
fontFace: false
|
|
339
|
+
});
|
|
340
|
+
let purgedCss = purgeResult[0]?.css ?? originalCss;
|
|
341
|
+
const afterL1 = Buffer.byteLength(purgedCss, "utf-8");
|
|
342
|
+
console.log(`[css-purge] L1 class purge: ${(originalSize / 1024).toFixed(1)} \u2192 ${(afterL1 / 1024).toFixed(1)} KB`);
|
|
343
|
+
if (attrSafelist.size > 0) {
|
|
344
|
+
purgedCss = purgeAttributes(purgedCss, attrSafelist);
|
|
345
|
+
const afterL2 = Buffer.byteLength(purgedCss, "utf-8");
|
|
346
|
+
console.log(`[css-purge] L2 attr purge: ${(afterL1 / 1024).toFixed(1)} \u2192 ${(afterL2 / 1024).toFixed(1)} KB`);
|
|
347
|
+
}
|
|
348
|
+
purgedCss = cleanUnusedVars(purgedCss);
|
|
349
|
+
const afterL3 = Buffer.byteLength(purgedCss, "utf-8");
|
|
350
|
+
console.log(`[css-purge] L3 var cleanup: \u2192 ${(afterL3 / 1024).toFixed(1)} KB`);
|
|
351
|
+
const finalSize = Buffer.byteLength(purgedCss, "utf-8");
|
|
352
|
+
totalAfter += finalSize;
|
|
353
|
+
console.log(`[css-purge] Final: ${(originalSize / 1024).toFixed(1)} \u2192 ${(finalSize / 1024).toFixed(1)} KB (${((1 - finalSize / originalSize) * 100).toFixed(1)}% reduction)`);
|
|
354
|
+
await Bun.write(fullPath, purgedCss);
|
|
355
|
+
}
|
|
356
|
+
console.log(`
|
|
357
|
+
[css-purge] Total: ${(totalBefore / 1024).toFixed(1)} \u2192 ${(totalAfter / 1024).toFixed(1)} KB (${((1 - totalAfter / totalBefore) * 100).toFixed(1)}% reduction)`);
|
|
358
|
+
}
|
|
359
|
+
main();
|
|
360
|
+
|
|
361
|
+
//# debugId=9846F414E5F0A23D64756E2164756E21
|
|
362
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/postbuild-purge.ts", "../src/scan-consumer.ts"],
  "sourcesContent": [
    "#!/usr/bin/env bun\n/**\n * Postbuild CSS purge — standalone Bun script.\n *\n * Runs after rsbuild build, purges CSS files in dist/ using the purge manifest\n * and consumer JSX analysis. Zero Node imports.\n *\n * Usage:\n *   bunx @pathscale/rebuild-plugin-ui-css-purge \\\n *     --dist dist --src src --manifest node_modules/@pathscale/ui/dist/purge-manifest.json\n */\n\nimport { Glob } from \"bun\";\nimport { PurgeCSS } from \"purgecss\";\nimport postcss from \"postcss\";\nimport type { Rule, AtRule } from \"postcss\";\nimport path from \"path\";\nimport { scanConsumerSource, buildSafelists } from \"./scan-consumer\";\nimport type { PurgeManifest } from \"./scan-consumer\";\n\n// ── CLI args ──────────────────────────────────────────────────────────────────\n\nfunction parseArgs(argv: string[]): { distDir: string; srcDir: string; manifestPath: string } {\n  const args = argv.slice(2);\n  let distDir = \"./dist\";\n  let srcDir = \"./src\";\n  let manifestPath = \"\";\n\n  for (let i = 0; i < args.length; i++) {\n    if (args[i] === \"--dist\" && args[i + 1]) distDir = args[++i];\n    else if (args[i] === \"--src\" && args[i + 1]) srcDir = args[++i];\n    else if (args[i] === \"--manifest\" && args[i + 1]) manifestPath = args[++i];\n  }\n\n  if (!manifestPath) {\n    console.error(\n      \"Usage: bunx @pathscale/rebuild-plugin-ui-css-purge --manifest <path> [--dist <path>] [--src <path>]\",\n    );\n    process.exit(1);\n  }\n\n  return {\n    distDir: path.resolve(distDir),\n    srcDir: path.resolve(srcDir),\n    manifestPath: path.resolve(manifestPath),\n  };\n}\n\n// ── Level 2: attribute purge ──────────────────────────────────────────────────\n\nfunction purgeAttributes(css: string, attrSafelist: Set<string>): string {\n  const root = postcss.parse(css);\n\n  root.walkRules((rule) => {\n    const selectors = rule.selectors;\n    const kept: string[] = [];\n\n    for (const sel of selectors) {\n      const attrMatches = sel.matchAll(/\\[(data-[a-z-]+|aria-[a-z-]+)=\"([^\"]+)\"\\]/g);\n      let shouldKeep = true;\n\n      for (const match of attrMatches) {\n        const attrSelector = `[${match[1]}=\"${match[2]}\"]`;\n        if (match[1] === \"data-slot\") continue;\n        if (!attrSafelist.has(attrSelector)) {\n          shouldKeep = false;\n          break;\n        }\n      }\n\n      if (shouldKeep) kept.push(sel);\n    }\n\n    if (kept.length === 0) {\n      rule.remove();\n    } else if (kept.length < selectors.length) {\n      rule.selectors = kept;\n    }\n  });\n\n  return root.toString();\n}\n\n// ── Level 3: unused CSS variable cleanup ──────────────────────────────────────\n\nfunction cleanUnusedVars(css: string): string {\n  let changed = true;\n  let result = css;\n\n  while (changed) {\n    changed = false;\n    const root = postcss.parse(result);\n\n    const declared = new Map<string, { rule: Rule | AtRule; prop: string; index: number }[]>();\n    root.walkDecls(/^--/, (decl) => {\n      const entries = declared.get(decl.prop) ?? [];\n      entries.push({ rule: decl.parent as Rule, prop: decl.prop, index: entries.length });\n      declared.set(decl.prop, entries);\n    });\n\n    const referenced = new Set<string>();\n    root.walkDecls((decl) => {\n      const refs = decl.value.matchAll(/var\\(\\s*(--[a-zA-Z0-9_-]+)/g);\n      for (const ref of refs) {\n        referenced.add(ref[1]);\n      }\n    });\n\n    for (const [varName, entries] of declared) {\n      if (!referenced.has(varName)) {\n        for (const entry of entries) {\n          entry.rule.walkDecls(entry.prop, (decl) => {\n            decl.remove();\n            changed = true;\n          });\n        }\n      }\n    }\n\n    root.walkRules((rule) => {\n      if (rule.nodes && rule.nodes.length === 0) rule.remove();\n    });\n    root.walkAtRules((atRule) => {\n      if (atRule.nodes && atRule.nodes.length === 0) atRule.remove();\n    });\n\n    result = root.toString();\n  }\n\n  return result;\n}\n\n// ── Main ──────────────────────────────────────────────────────────────────────\n\nasync function main() {\n  const { distDir, srcDir, manifestPath } = parseArgs(process.argv);\n\n  // 1. Load manifest\n  const manifest: PurgeManifest = JSON.parse(\n    await Bun.file(manifestPath).text(),\n  );\n  console.log(`[css-purge] Manifest loaded: ${Object.keys(manifest).length} entries`);\n\n  // 2. Scan consumer source\n  const usages = await scanConsumerSource(srcDir);\n  console.log(`[css-purge] Scanned ${srcDir}: ${usages.length} component usages`);\n\n  // 3. Build safelists\n  const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);\n  console.log(`[css-purge] Safelist: ${classSafelist.size} classes, ${attrSafelist.size} attrs`);\n\n  // 4. Glob CSS files in dist\n  const glob = new Glob(\"**/*.css\");\n  let totalBefore = 0;\n  let totalAfter = 0;\n\n  for await (const relPath of glob.scan({ cwd: distDir })) {\n    const fullPath = path.join(distDir, relPath);\n    const originalCss = await Bun.file(fullPath).text();\n    const originalSize = Buffer.byteLength(originalCss, \"utf-8\");\n    totalBefore += originalSize;\n\n    console.log(`[css-purge] Processing ${relPath} (${(originalSize / 1024).toFixed(1)} KB)`);\n\n    // Level 1: class-level purge\n    const purgeResult = await new PurgeCSS().purge({\n      content: [],\n      css: [{ raw: originalCss }],\n      safelist: [...classSafelist],\n      keyframes: false,\n      fontFace: false,\n    });\n\n    let purgedCss = purgeResult[0]?.css ?? originalCss;\n    const afterL1 = Buffer.byteLength(purgedCss, \"utf-8\");\n    console.log(`[css-purge]   L1 class purge: ${(originalSize / 1024).toFixed(1)} → ${(afterL1 / 1024).toFixed(1)} KB`);\n\n    // Level 2: attribute-level purge\n    if (attrSafelist.size > 0) {\n      purgedCss = purgeAttributes(purgedCss, attrSafelist);\n      const afterL2 = Buffer.byteLength(purgedCss, \"utf-8\");\n      console.log(`[css-purge]   L2 attr purge: ${(afterL1 / 1024).toFixed(1)} → ${(afterL2 / 1024).toFixed(1)} KB`);\n    }\n\n    // Level 3: unused CSS variable cleanup\n    purgedCss = cleanUnusedVars(purgedCss);\n    const afterL3 = Buffer.byteLength(purgedCss, \"utf-8\");\n    console.log(`[css-purge]   L3 var cleanup: → ${(afterL3 / 1024).toFixed(1)} KB`);\n\n    const finalSize = Buffer.byteLength(purgedCss, \"utf-8\");\n    totalAfter += finalSize;\n    console.log(`[css-purge]   Final: ${(originalSize / 1024).toFixed(1)} → ${(finalSize / 1024).toFixed(1)} KB (${((1 - finalSize / originalSize) * 100).toFixed(1)}% reduction)`);\n\n    // Write back\n    await Bun.write(fullPath, purgedCss);\n  }\n\n  console.log(`\\n[css-purge] Total: ${(totalBefore / 1024).toFixed(1)} → ${(totalAfter / 1024).toFixed(1)} KB (${((1 - totalAfter / totalBefore) * 100).toFixed(1)}% reduction)`);\n}\n\nmain();\n",
    "/**\n * Consumer-side JSX scanner.\n *\n * Walks a consumer's source tree, finds component imports from @pathscale/ui,\n * collects prop values, and cross-references with the purge manifest to build\n * Level 1 (class) and Level 2 (attribute) safelists.\n *\n * Usage:  bun run src/scan-consumer.ts <consumer-src-dir> <purge-manifest.json>\n */\n\nimport swc from \"@swc/core\";\nimport { Glob } from \"bun\";\nimport path from \"path\";\n\n// ── Types ──────────────────────────────────────────────────────────────────────\n\ninterface ComponentManifest {\n  classes: {\n    always: string[];\n    byProp: Record<string, string[] | Record<string, string[]>>;\n  };\n  attrs?: Record<string, Record<string, string>>;\n}\n\ntype PurgeManifest = Record<string, ComponentManifest>;\n\n/** What we collect per component usage from JSX */\ninterface PropUsage {\n  component: string;\n  props: Map<string, string | \"DYNAMIC\">; // propName → literal value or DYNAMIC\n  booleanProps: Set<string>; // props present without a value (truthy)\n  hasSpread: boolean;\n}\n\n// ── AST walker ─────────────────────────────────────────────────────────────────\n\nfunction walkAST(node: any, visitor: (node: any) => void) {\n  if (!node || typeof node !== \"object\") return;\n  visitor(node);\n  for (const key of Object.keys(node)) {\n    if (key === \"span\") continue;\n    const val = node[key];\n    if (Array.isArray(val)) {\n      for (const item of val) walkAST(item, visitor);\n    } else if (val && typeof val === \"object\") {\n      walkAST(val, visitor);\n    }\n  }\n}\n\n/** Extract @pathscale/ui imports from a parsed module */\nfunction extractUIImports(ast: any): Map<string, string> {\n  const imports = new Map<string, string>();\n  for (const node of ast.body) {\n    if (node.type !== \"ImportDeclaration\") continue;\n    const src = node.source?.value as string;\n    if (!src || !src.startsWith(\"@pathscale/ui\")) continue;\n\n    for (const spec of node.specifiers) {\n      if (spec.type === \"ImportSpecifier\") {\n        const imported = spec.imported?.value ?? spec.local?.value;\n        const local = spec.local?.value;\n        if (local && imported) imports.set(local, imported);\n      } else if (spec.type === \"ImportDefaultSpecifier\") {\n        const local = spec.local?.value;\n        if (local) imports.set(local, local);\n      }\n    }\n  }\n  return imports;\n}\n\n/** Extract JSX usages of UI components */\nfunction extractJSXUsages(ast: any, uiComponents: Map<string, string>): PropUsage[] {\n  const usages: PropUsage[] = [];\n\n  walkAST(ast, (node) => {\n    if (node.type !== \"JSXOpeningElement\") return;\n\n    let elementName: string | null = null;\n    let rootName: string | null = null;\n\n    if (node.name?.type === \"Identifier\") {\n      elementName = node.name.value;\n      rootName = elementName;\n    } else if (node.name?.type === \"JSXMemberExpression\") {\n      const parts: string[] = [];\n      let cursor = node.name;\n      while (cursor?.type === \"JSXMemberExpression\") {\n        parts.unshift(cursor.property?.value);\n        cursor = cursor.object;\n      }\n      if (cursor?.type === \"Identifier\") {\n        parts.unshift(cursor.value);\n        rootName = cursor.value;\n      }\n      elementName = parts.join(\".\");\n    }\n\n    if (!rootName || !uiComponents.has(rootName)) return;\n\n    const usage: PropUsage = {\n      component: elementName!,\n      props: new Map(),\n      booleanProps: new Set(),\n      hasSpread: false,\n    };\n\n    for (const attr of node.attributes || []) {\n      if (attr.type === \"SpreadElement\" || attr.type === \"JSXSpreadAttribute\") {\n        usage.hasSpread = true;\n        continue;\n      }\n      if (attr.type !== \"JSXAttribute\") continue;\n\n      const propName = attr.name?.value;\n      if (!propName) continue;\n\n      if (!attr.value) {\n        usage.booleanProps.add(propName);\n      } else if (attr.value.type === \"StringLiteral\") {\n        usage.props.set(propName, attr.value.value);\n      } else {\n        usage.props.set(propName, \"DYNAMIC\");\n      }\n    }\n\n    usages.push(usage);\n  });\n\n  return usages;\n}\n\n// ── Safelist builder ───────────────────────────────────────────────────────────\n\ninterface Safelists {\n  classSafelist: Set<string>;\n  attrSafelist: Set<string>;\n}\n\nfunction buildSafelists(\n  allUsages: PropUsage[],\n  manifest: PurgeManifest,\n): Safelists {\n  const classSafelist = new Set<string>();\n  const attrSafelist = new Set<string>();\n\n  const componentUsages = new Map<string, PropUsage[]>();\n  for (const usage of allUsages) {\n    const existing = componentUsages.get(usage.component) ?? [];\n    existing.push(usage);\n    componentUsages.set(usage.component, existing);\n  }\n\n  for (const [entryName, entry] of Object.entries(manifest)) {\n    const matchingUsages = findMatchingUsages(entryName, componentUsages);\n\n    if (matchingUsages.length === 0) {\n      continue;\n    }\n\n    for (const cls of entry.classes.always) {\n      classSafelist.add(cls);\n    }\n\n    for (const [propOrSlot, value] of Object.entries(entry.classes.byProp)) {\n      if (Array.isArray(value)) {\n        if (isPropUsed(propOrSlot, matchingUsages)) {\n          for (const cls of value) classSafelist.add(cls);\n        }\n      } else {\n        const usedValues = getUsedEnumValues(propOrSlot, matchingUsages);\n        if (usedValues === \"ALL\") {\n          for (const classes of Object.values(value)) {\n            for (const cls of classes) classSafelist.add(cls);\n          }\n        } else {\n          for (const val of usedValues) {\n            if (value[val]) {\n              for (const cls of value[val]) classSafelist.add(cls);\n            }\n          }\n        }\n      }\n    }\n\n    if (entry.attrs) {\n      for (const [propName, attrMap] of Object.entries(entry.attrs)) {\n        if (isPropUsed(propName, matchingUsages)) {\n          for (const [attr, val] of Object.entries(attrMap)) {\n            attrSafelist.add(`${attr}=${val}`);\n          }\n        }\n      }\n    }\n  }\n\n  return { classSafelist, attrSafelist };\n}\n\nfunction findMatchingUsages(\n  entryName: string,\n  usageMap: Map<string, PropUsage[]>,\n): PropUsage[] {\n  if (usageMap.has(entryName)) {\n    return usageMap.get(entryName)!;\n  }\n\n  const results: PropUsage[] = [];\n  for (const [usageName, usages] of usageMap) {\n    if (usageName === entryName) {\n      results.push(...usages);\n    }\n    if (entryName.includes(\".\")) {\n      const [family, part] = entryName.split(\".\");\n      if (part === family && usageName === family) {\n        results.push(...usages);\n      }\n    }\n  }\n  return results;\n}\n\nfunction isPropUsed(propName: string, usages: PropUsage[]): boolean {\n  for (const usage of usages) {\n    if (usage.hasSpread) return true;\n    if (usage.booleanProps.has(propName)) return true;\n    if (usage.props.has(propName)) return true;\n  }\n  return false;\n}\n\nfunction getUsedEnumValues(slotName: string, usages: PropUsage[]): Set<string> | \"ALL\" {\n  const values = new Set<string>();\n  for (const usage of usages) {\n    if (usage.hasSpread) return \"ALL\";\n    const val = usage.props.get(slotName);\n    if (val === \"DYNAMIC\") return \"ALL\";\n    if (val !== undefined) values.add(val);\n    if (usage.booleanProps.has(slotName)) return \"ALL\";\n  }\n  return values;\n}\n\n// ── Consumer source scanning ───────────────────────────────────────────────────\n\nasync function scanConsumerSource(srcDir: string): Promise<PropUsage[]> {\n  const allUsages: PropUsage[] = [];\n  const glob = new Glob(\"**/*.{tsx,ts,jsx,js}\");\n\n  for await (const relPath of glob.scan({ cwd: srcDir })) {\n    if (relPath.includes(\"node_modules\")) continue;\n    const fullPath = path.join(srcDir, relPath);\n    const code = await Bun.file(fullPath).text();\n    if (!code.includes(\"@pathscale/ui\")) continue;\n\n    const isTsx = /\\.[tj]sx$/.test(relPath);\n    const ast = await swc.parse(code, { syntax: \"typescript\", tsx: isTsx });\n    const uiImports = extractUIImports(ast);\n    if (uiImports.size === 0) continue;\n\n    allUsages.push(...extractJSXUsages(ast, uiImports));\n  }\n\n  return allUsages;\n}\n\n// ── Main (standalone CLI) ─────────────────────────────────────────────────────\n\nasync function main() {\n  const [srcDir, manifestPath] = process.argv.slice(2);\n  if (!srcDir || !manifestPath) {\n    console.error(\"Usage: bun run src/scan-consumer.ts <consumer-src-dir> <purge-manifest.json>\");\n    process.exit(1);\n  }\n\n  const manifest: PurgeManifest = JSON.parse(\n    await Bun.file(manifestPath).text(),\n  );\n  const resolvedSrc = path.resolve(srcDir);\n  console.log(`Scanning ${resolvedSrc} for @pathscale/ui component usage…`);\n  console.log(`Manifest: ${Object.keys(manifest).length} entries\\n`);\n\n  const usages = await scanConsumerSource(resolvedSrc);\n  const { classSafelist, attrSafelist } = buildSafelists(usages, manifest);\n\n  console.log(\"=== Class Safelist ===\");\n  for (const cls of [...classSafelist].sort()) {\n    console.log(`  ${cls}`);\n  }\n\n  console.log(`\\n=== Attribute Safelist ===`);\n  for (const attr of [...attrSafelist].sort()) {\n    console.log(`  [${attr}]`);\n  }\n\n  console.log(`\\nTotal: ${classSafelist.size} classes, ${attrSafelist.size} attribute selectors`);\n}\n\n// Only run CLI when invoked directly (not when imported as a module)\nif (import.meta.main) {\n  main();\n}\n\nexport { extractUIImports, extractJSXUsages, buildSafelists, scanConsumerSource };\nexport type { PropUsage, PurgeManifest, ComponentManifest, Safelists };\n"
  ],
  "mappings": "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",
  "debugId": "9846F414E5F0A23D64756E2164756E21",
  "names": []
}
|
package/dist/scan-consumer.d.ts
CHANGED
|
@@ -31,5 +31,6 @@ interface Safelists {
|
|
|
31
31
|
attrSafelist: Set<string>;
|
|
32
32
|
}
|
|
33
33
|
declare function buildSafelists(allUsages: PropUsage[], manifest: PurgeManifest): Safelists;
|
|
34
|
-
|
|
34
|
+
declare function scanConsumerSource(srcDir: string): Promise<PropUsage[]>;
|
|
35
|
+
export { extractUIImports, extractJSXUsages, buildSafelists, scanConsumerSource };
|
|
35
36
|
export type { PropUsage, PurgeManifest, ComponentManifest, Safelists };
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pathscale/rebuild-plugin-ui-css-purge",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.2.0",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"main": "dist/index.js",
|
|
6
6
|
"types": "dist/index.d.ts",
|
|
@@ -11,25 +11,21 @@
|
|
|
11
11
|
}
|
|
12
12
|
},
|
|
13
13
|
"bin": {
|
|
14
|
+
"rebuild-plugin-ui-css-purge": "dist/postbuild-purge.js",
|
|
14
15
|
"generate-manifest": "src/generate-manifest.ts"
|
|
15
16
|
},
|
|
16
17
|
"files": ["dist", "src/generate-manifest.ts"],
|
|
17
18
|
"dependencies": {
|
|
18
19
|
"@swc/core": "^1.15.24",
|
|
19
|
-
"fast-glob": "^3.3.3",
|
|
20
20
|
"postcss": "^8.5.9",
|
|
21
21
|
"purgecss": "^8.0.0"
|
|
22
22
|
},
|
|
23
|
-
"peerDependencies": {
|
|
24
|
-
"@rsbuild/core": "^1.7.5"
|
|
25
|
-
},
|
|
26
23
|
"scripts": {
|
|
27
24
|
"build": "bun run build.ts",
|
|
28
25
|
"lint": "biome check .",
|
|
29
26
|
"format": "biome format . --write"
|
|
30
27
|
},
|
|
31
28
|
"devDependencies": {
|
|
32
|
-
"@rsbuild/core": "^1.7.5",
|
|
33
29
|
"bun-types": "^1.3.12"
|
|
34
30
|
}
|
|
35
31
|
}
|
package/src/generate-manifest.ts
CHANGED
package/dist/rsbuild-plugin.d.ts
DELETED
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* rsbuild-plugin-css-purge
|
|
3
|
-
*
|
|
4
|
-
* Two-level CSS purge for @pathscale/ui consumers.
|
|
5
|
-
*
|
|
6
|
-
* Level 1: class-level purge via purgecss — removes entire rules whose selectors
|
|
7
|
-
* don't match the safelist built from consumer JSX analysis.
|
|
8
|
-
* Level 2: attribute-level purge — within kept rules, strips compound selectors
|
|
9
|
-
* containing data-attr / aria-attr attribute selectors not in the attr safelist.
|
|
10
|
-
*
|
|
11
|
-
* Usage in rsbuild.config.ts:
|
|
12
|
-
* import { pluginCssPurge } from "@pathscale/rebuild-plugin-ui-css-purge";
|
|
13
|
-
* export default defineConfig({ plugins: [pluginCssPurge({ manifest: "..." })] });
|
|
14
|
-
*/
|
|
15
|
-
import type { RsbuildPlugin } from "@rsbuild/core";
|
|
16
|
-
export interface CssPurgeOptions {
|
|
17
|
-
/** Path to purge-manifest.json (generated by generate-manifest.ts) */
|
|
18
|
-
manifest: string;
|
|
19
|
-
/** Consumer source directory to scan for JSX usage (default: "src") */
|
|
20
|
-
srcDir?: string;
|
|
21
|
-
/** Enable Level 2 attribute purge (default: true) */
|
|
22
|
-
attrPurge?: boolean;
|
|
23
|
-
/** Enable CSS variable cleanup (default: true) */
|
|
24
|
-
cleanVars?: boolean;
|
|
25
|
-
/** Log purge stats (default: true) */
|
|
26
|
-
verbose?: boolean;
|
|
27
|
-
}
|
|
28
|
-
export declare const pluginCssPurge: (options: CssPurgeOptions) => RsbuildPlugin;
|