@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 CHANGED
@@ -1,2 +1,2 @@
1
- export { pluginCssPurge } from "./rsbuild-plugin";
2
- export type { CssPurgeOptions } from "./rsbuild-plugin";
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
- // src/rsbuild-plugin.ts
2
- import { PurgeCSS } from "purgecss";
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
- for (const cls of entry.classes.always)
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(`[${attr}="${val}"]`);
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 files = await fastGlob("**/*.{tsx,ts,jsx,js}", { cwd: srcDir, ignore: ["**/node_modules/**"] });
196
- for (const relPath of files) {
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 nodeReadFile(fullPath, "utf-8");
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
- function purgeAttributes(css, attrSafelist) {
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
- pluginCssPurge
215
+ scanConsumerSource,
216
+ extractUIImports,
217
+ extractJSXUsages,
218
+ buildSafelists
334
219
  };
335
220
 
336
- //# debugId=CB5B2093063663E864756E2164756E21
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": []
}
@@ -31,5 +31,6 @@ interface Safelists {
31
31
  attrSafelist: Set<string>;
32
32
  }
33
33
  declare function buildSafelists(allUsages: PropUsage[], manifest: PurgeManifest): Safelists;
34
- export { extractUIImports, extractJSXUsages, buildSafelists };
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.1.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
  }
@@ -152,4 +152,6 @@ async function main() {
152
152
  console.log(`\nWrote ${outPath} (${Object.keys(manifest).length} entries, ${json.length} bytes)`);
153
153
  }
154
154
 
155
- main();
155
+ if (import.meta.main) {
156
+ main();
157
+ }
@@ -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;