@yahoo/uds 3.134.0 → 3.134.2

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.
Files changed (63) hide show
  1. package/dist/cli/dist/lib/args.cjs +7 -3
  2. package/dist/cli/dist/lib/args.js +7 -3
  3. package/dist/components/client/Menu/Menu.ItemCheckbox.d.cts +1 -1
  4. package/dist/components/client/Menu/Menu.ItemCheckbox.d.ts +1 -1
  5. package/dist/components/client/Toast/UDSToastConfigProvider.d.cts +1 -1
  6. package/dist/components/client/Toast/UDSToastConfigProvider.d.ts +1 -1
  7. package/dist/index.d.cts +2 -2
  8. package/dist/index.d.ts +2 -2
  9. package/dist/styles/styler.d.cts +21 -21
  10. package/dist/styles/styler.d.ts +21 -21
  11. package/dist/tailwind/dist/commands/css.cjs +1 -0
  12. package/dist/tailwind/dist/commands/css.d.cts.map +1 -1
  13. package/dist/tailwind/dist/commands/css.d.ts.map +1 -1
  14. package/dist/tailwind/dist/commands/css.helpers.cjs +8 -1
  15. package/dist/tailwind/dist/commands/css.helpers.js +8 -1
  16. package/dist/tailwind/dist/commands/css.helpers.js.map +1 -1
  17. package/dist/tailwind/dist/commands/css.js +1 -0
  18. package/dist/tailwind/dist/commands/css.js.map +1 -1
  19. package/dist/tailwind/dist/css/generate.cjs +7 -4
  20. package/dist/tailwind/dist/css/generate.d.cts.map +1 -1
  21. package/dist/tailwind/dist/css/generate.d.ts.map +1 -1
  22. package/dist/tailwind/dist/css/generate.js +7 -4
  23. package/dist/tailwind/dist/css/generate.js.map +1 -1
  24. package/dist/tailwind/dist/css/nodeUtils.cjs +19 -8
  25. package/dist/tailwind/dist/css/nodeUtils.js +19 -8
  26. package/dist/tailwind/dist/css/nodeUtils.js.map +1 -1
  27. package/dist/tailwind/dist/css/perf.cjs +92 -0
  28. package/dist/tailwind/dist/css/perf.js +89 -0
  29. package/dist/tailwind/dist/css/perf.js.map +1 -0
  30. package/dist/tailwind/dist/css/purgeWorker.cjs +47 -0
  31. package/dist/tailwind/dist/css/purgeWorker.d.cts +2 -0
  32. package/dist/tailwind/dist/css/purgeWorker.d.ts +2 -0
  33. package/dist/tailwind/dist/css/purgeWorker.js +48 -0
  34. package/dist/tailwind/dist/css/purgeWorker.js.map +1 -0
  35. package/dist/tailwind/dist/css/runner.cjs +158 -145
  36. package/dist/tailwind/dist/css/runner.js +158 -145
  37. package/dist/tailwind/dist/css/runner.js.map +1 -1
  38. package/dist/tailwind/dist/css/theme.d.cts +1 -1
  39. package/dist/tailwind/dist/css/theme.d.ts +1 -1
  40. package/dist/tailwind/dist/css/workerPool.cjs +89 -0
  41. package/dist/tailwind/dist/css/workerPool.js +90 -0
  42. package/dist/tailwind/dist/css/workerPool.js.map +1 -0
  43. package/dist/tailwind/dist/index.d.cts +2 -2
  44. package/dist/tailwind/dist/index.d.ts +2 -2
  45. package/dist/tailwind/dist/purger/optimized/ast/expressions.cjs +95 -15
  46. package/dist/tailwind/dist/purger/optimized/ast/expressions.js +95 -15
  47. package/dist/tailwind/dist/purger/optimized/ast/expressions.js.map +1 -1
  48. package/dist/tailwind/dist/purger/optimized/purgeFromCode.cjs +38 -14
  49. package/dist/tailwind/dist/purger/optimized/purgeFromCode.d.cts.map +1 -1
  50. package/dist/tailwind/dist/purger/optimized/purgeFromCode.d.ts.map +1 -1
  51. package/dist/tailwind/dist/purger/optimized/purgeFromCode.js +39 -15
  52. package/dist/tailwind/dist/purger/optimized/purgeFromCode.js.map +1 -1
  53. package/dist/tailwind/dist/purger/optimized/types.d.cts +10 -0
  54. package/dist/tailwind/dist/purger/optimized/types.d.cts.map +1 -1
  55. package/dist/tailwind/dist/purger/optimized/types.d.ts +10 -0
  56. package/dist/tailwind/dist/purger/optimized/types.d.ts.map +1 -1
  57. package/dist/uds/generated/componentData.cjs +271 -271
  58. package/dist/uds/generated/componentData.js +271 -271
  59. package/generated/componentData.json +361 -361
  60. package/package.json +1 -1
  61. package/dist/tailwind/dist/purger/optimized/ast/jsx.cjs +0 -16
  62. package/dist/tailwind/dist/purger/optimized/ast/jsx.js +0 -17
  63. package/dist/tailwind/dist/purger/optimized/ast/jsx.js.map +0 -1
@@ -0,0 +1,89 @@
1
+ /*! © 2026 Yahoo, Inc. UDS Tailwind and Purger v0.0.0-development */
2
+ require("../_virtual/_rolldown/runtime.cjs");
3
+ let node_os = require("node:os");
4
+ let node_worker_threads = require("node:worker_threads");
5
+ //#region src/css/workerPool.ts
6
+ const createWorkerPool = async (config, numWorkers) => {
7
+ const workerCount = numWorkers ?? Math.max(1, (0, node_os.cpus)().length - 1);
8
+ const workerUrl = new URL("./purgeWorker.js", require("url").pathToFileURL(__filename).href);
9
+ const workers = [];
10
+ const available = [];
11
+ const pending = /* @__PURE__ */ new Map();
12
+ const queue = [];
13
+ let nextId = 0;
14
+ await Promise.all(Array.from({ length: workerCount }, () => {
15
+ return new Promise((resolve, reject) => {
16
+ const worker = new node_worker_threads.Worker(workerUrl);
17
+ let initialized = false;
18
+ worker.on("message", (msg) => {
19
+ if (msg.type === "ready") {
20
+ initialized = true;
21
+ workers.push(worker);
22
+ available.push(worker);
23
+ resolve();
24
+ return;
25
+ }
26
+ if (msg.type === "result" || msg.type === "error") {
27
+ const task = msg.id != null ? pending.get(msg.id) : void 0;
28
+ if (task) {
29
+ pending.delete(msg.id);
30
+ if (msg.type === "result") task.resolve(msg.result);
31
+ else task.reject(new Error(msg.error ?? "Unknown worker error"));
32
+ }
33
+ available.push(worker);
34
+ if (queue.length > 0 && available.length > 0) {
35
+ const next = queue.shift();
36
+ const nextWorker = available.pop();
37
+ pending.set(next.msg.id, next.task);
38
+ nextWorker.postMessage(next.msg);
39
+ }
40
+ }
41
+ });
42
+ worker.on("error", (err) => {
43
+ if (!initialized) reject(err);
44
+ });
45
+ worker.postMessage({
46
+ type: "init",
47
+ config: JSON.parse(JSON.stringify(config))
48
+ });
49
+ });
50
+ }));
51
+ const processFile = (options) => {
52
+ return new Promise((resolve, reject) => {
53
+ const id = nextId++;
54
+ const msg = {
55
+ type: "task",
56
+ id,
57
+ code: options.code,
58
+ filePath: options.filePath,
59
+ colorModes: options.colorModes,
60
+ variantDefaults: options.variantDefaults,
61
+ runtimeConfigValues: options.runtimeConfigValues,
62
+ includeAllClassNamePrimitives: options.includeAllClassNamePrimitives
63
+ };
64
+ const task = {
65
+ resolve,
66
+ reject
67
+ };
68
+ if (available.length > 0) {
69
+ const worker = available.pop();
70
+ pending.set(id, task);
71
+ worker.postMessage(msg);
72
+ } else queue.push({
73
+ msg,
74
+ task
75
+ });
76
+ });
77
+ };
78
+ const destroy = async () => {
79
+ for (const worker of workers) worker.postMessage({ type: "done" });
80
+ await Promise.all(workers.map((w) => new Promise((res) => w.on("exit", () => res()))));
81
+ };
82
+ return {
83
+ processFile,
84
+ destroy,
85
+ workerCount
86
+ };
87
+ };
88
+ //#endregion
89
+ exports.createWorkerPool = createWorkerPool;
@@ -0,0 +1,90 @@
1
+ /*! © 2026 Yahoo, Inc. UDS Tailwind and Purger v0.0.0-development */
2
+ import { cpus } from "node:os";
3
+ import { Worker } from "node:worker_threads";
4
+ //#region src/css/workerPool.ts
5
+ const createWorkerPool = async (config, numWorkers) => {
6
+ const workerCount = numWorkers ?? Math.max(1, cpus().length - 1);
7
+ const workerUrl = new URL("./purgeWorker.js", import.meta.url);
8
+ const workers = [];
9
+ const available = [];
10
+ const pending = /* @__PURE__ */ new Map();
11
+ const queue = [];
12
+ let nextId = 0;
13
+ await Promise.all(Array.from({ length: workerCount }, () => {
14
+ return new Promise((resolve, reject) => {
15
+ const worker = new Worker(workerUrl);
16
+ let initialized = false;
17
+ worker.on("message", (msg) => {
18
+ if (msg.type === "ready") {
19
+ initialized = true;
20
+ workers.push(worker);
21
+ available.push(worker);
22
+ resolve();
23
+ return;
24
+ }
25
+ if (msg.type === "result" || msg.type === "error") {
26
+ const task = msg.id != null ? pending.get(msg.id) : void 0;
27
+ if (task) {
28
+ pending.delete(msg.id);
29
+ if (msg.type === "result") task.resolve(msg.result);
30
+ else task.reject(new Error(msg.error ?? "Unknown worker error"));
31
+ }
32
+ available.push(worker);
33
+ if (queue.length > 0 && available.length > 0) {
34
+ const next = queue.shift();
35
+ const nextWorker = available.pop();
36
+ pending.set(next.msg.id, next.task);
37
+ nextWorker.postMessage(next.msg);
38
+ }
39
+ }
40
+ });
41
+ worker.on("error", (err) => {
42
+ if (!initialized) reject(err);
43
+ });
44
+ worker.postMessage({
45
+ type: "init",
46
+ config: JSON.parse(JSON.stringify(config))
47
+ });
48
+ });
49
+ }));
50
+ const processFile = (options) => {
51
+ return new Promise((resolve, reject) => {
52
+ const id = nextId++;
53
+ const msg = {
54
+ type: "task",
55
+ id,
56
+ code: options.code,
57
+ filePath: options.filePath,
58
+ colorModes: options.colorModes,
59
+ variantDefaults: options.variantDefaults,
60
+ runtimeConfigValues: options.runtimeConfigValues,
61
+ includeAllClassNamePrimitives: options.includeAllClassNamePrimitives
62
+ };
63
+ const task = {
64
+ resolve,
65
+ reject
66
+ };
67
+ if (available.length > 0) {
68
+ const worker = available.pop();
69
+ pending.set(id, task);
70
+ worker.postMessage(msg);
71
+ } else queue.push({
72
+ msg,
73
+ task
74
+ });
75
+ });
76
+ };
77
+ const destroy = async () => {
78
+ for (const worker of workers) worker.postMessage({ type: "done" });
79
+ await Promise.all(workers.map((w) => new Promise((res) => w.on("exit", () => res()))));
80
+ };
81
+ return {
82
+ processFile,
83
+ destroy,
84
+ workerCount
85
+ };
86
+ };
87
+ //#endregion
88
+ export { createWorkerPool };
89
+
90
+ //# sourceMappingURL=workerPool.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workerPool.js","names":[],"sources":["../../src/css/workerPool.ts"],"sourcesContent":["import { cpus } from 'node:os';\nimport { Worker } from 'node:worker_threads';\n\nimport type { PurgeFromCodeOptions, PurgeFromCodeResult } from '../purger/optimized/purgeFromCode';\n\ntype SharedConfig = Pick<PurgeFromCodeOptions, 'variants' | 'autoVariants' | 'componentData'>;\n\ninterface TaskOptions {\n code: string;\n filePath: string;\n colorModes?: ('dark' | 'light')[];\n variantDefaults?: PurgeFromCodeOptions['variantDefaults'];\n runtimeConfigValues?: PurgeFromCodeOptions['runtimeConfigValues'];\n includeAllClassNamePrimitives?: boolean;\n}\n\ninterface PendingTask {\n resolve: (result: PurgeFromCodeResult) => void;\n reject: (error: Error) => void;\n}\n\nexport interface WorkerPool {\n processFile: (options: TaskOptions) => Promise<PurgeFromCodeResult>;\n destroy: () => Promise<void>;\n workerCount: number;\n}\n\nexport const createWorkerPool = async (\n config: SharedConfig,\n numWorkers?: number,\n): Promise<WorkerPool> => {\n const workerCount = numWorkers ?? Math.max(1, cpus().length - 1);\n const workerUrl = new URL('./purgeWorker.js', import.meta.url);\n\n const workers: Worker[] = [];\n const available: Worker[] = [];\n const pending = new Map<number, PendingTask>();\n const queue: Array<{ msg: Record<string, unknown>; task: PendingTask }> = [];\n let nextId = 0;\n\n // Create workers, init with shared config, and set up message handlers\n await Promise.all(\n Array.from({ length: workerCount }, () => {\n return new Promise<void>((resolve, reject) => {\n const worker = new Worker(workerUrl);\n let initialized = false;\n\n worker.on(\n 'message',\n (msg: { type: string; id?: number; result?: PurgeFromCodeResult; error?: string }) => {\n if (msg.type === 'ready') {\n initialized = true;\n workers.push(worker);\n available.push(worker);\n resolve();\n return;\n }\n\n if (msg.type === 'result' || msg.type === 'error') {\n const task = msg.id != null ? pending.get(msg.id) : undefined;\n if (task) {\n pending.delete(msg.id!);\n if (msg.type === 'result') {\n task.resolve(msg.result!);\n } else {\n task.reject(new Error(msg.error ?? 'Unknown worker error'));\n }\n }\n\n available.push(worker);\n\n // Drain queue\n if (queue.length > 0 && available.length > 0) {\n const next = queue.shift()!;\n const nextWorker = available.pop()!;\n pending.set(next.msg.id as number, next.task);\n nextWorker.postMessage(next.msg);\n }\n }\n },\n );\n\n worker.on('error', (err) => {\n if (!initialized) {\n reject(err);\n }\n });\n\n // JSON round-trip ensures config is a plain object (ES module namespaces can't be cloned)\n worker.postMessage({ type: 'init', config: JSON.parse(JSON.stringify(config)) });\n });\n }),\n );\n\n const processFile = (options: TaskOptions): Promise<PurgeFromCodeResult> => {\n return new Promise((resolve, reject) => {\n const id = nextId++;\n const msg = {\n type: 'task' as const,\n id,\n code: options.code,\n filePath: options.filePath,\n colorModes: options.colorModes,\n variantDefaults: options.variantDefaults,\n runtimeConfigValues: options.runtimeConfigValues,\n includeAllClassNamePrimitives: options.includeAllClassNamePrimitives,\n };\n const task = { resolve, reject };\n\n if (available.length > 0) {\n const worker = available.pop()!;\n pending.set(id, task);\n worker.postMessage(msg);\n } else {\n queue.push({ msg, task });\n }\n });\n };\n\n const destroy = async () => {\n for (const worker of workers) {\n worker.postMessage({ type: 'done' });\n }\n await Promise.all(workers.map((w) => new Promise<void>((res) => w.on('exit', () => res()))));\n };\n\n return { processFile, destroy, workerCount };\n};\n"],"mappings":";;;;AA2BA,MAAa,mBAAmB,OAC9B,QACA,eACwB;CACxB,MAAM,cAAc,cAAc,KAAK,IAAI,GAAG,MAAM,CAAC,SAAS,EAAE;CAChE,MAAM,YAAY,IAAI,IAAI,oBAAoB,OAAO,KAAK,IAAI;CAE9D,MAAM,UAAoB,EAAE;CAC5B,MAAM,YAAsB,EAAE;CAC9B,MAAM,0BAAU,IAAI,KAA0B;CAC9C,MAAM,QAAoE,EAAE;CAC5E,IAAI,SAAS;AAGb,OAAM,QAAQ,IACZ,MAAM,KAAK,EAAE,QAAQ,aAAa,QAAQ;AACxC,SAAO,IAAI,SAAe,SAAS,WAAW;GAC5C,MAAM,SAAS,IAAI,OAAO,UAAU;GACpC,IAAI,cAAc;AAElB,UAAO,GACL,YACC,QAAqF;AACpF,QAAI,IAAI,SAAS,SAAS;AACxB,mBAAc;AACd,aAAQ,KAAK,OAAO;AACpB,eAAU,KAAK,OAAO;AACtB,cAAS;AACT;;AAGF,QAAI,IAAI,SAAS,YAAY,IAAI,SAAS,SAAS;KACjD,MAAM,OAAO,IAAI,MAAM,OAAO,QAAQ,IAAI,IAAI,GAAG,GAAG,KAAA;AACpD,SAAI,MAAM;AACR,cAAQ,OAAO,IAAI,GAAI;AACvB,UAAI,IAAI,SAAS,SACf,MAAK,QAAQ,IAAI,OAAQ;UAEzB,MAAK,OAAO,IAAI,MAAM,IAAI,SAAS,uBAAuB,CAAC;;AAI/D,eAAU,KAAK,OAAO;AAGtB,SAAI,MAAM,SAAS,KAAK,UAAU,SAAS,GAAG;MAC5C,MAAM,OAAO,MAAM,OAAO;MAC1B,MAAM,aAAa,UAAU,KAAK;AAClC,cAAQ,IAAI,KAAK,IAAI,IAAc,KAAK,KAAK;AAC7C,iBAAW,YAAY,KAAK,IAAI;;;KAIvC;AAED,UAAO,GAAG,UAAU,QAAQ;AAC1B,QAAI,CAAC,YACH,QAAO,IAAI;KAEb;AAGF,UAAO,YAAY;IAAE,MAAM;IAAQ,QAAQ,KAAK,MAAM,KAAK,UAAU,OAAO,CAAC;IAAE,CAAC;IAChF;GACF,CACH;CAED,MAAM,eAAe,YAAuD;AAC1E,SAAO,IAAI,SAAS,SAAS,WAAW;GACtC,MAAM,KAAK;GACX,MAAM,MAAM;IACV,MAAM;IACN;IACA,MAAM,QAAQ;IACd,UAAU,QAAQ;IAClB,YAAY,QAAQ;IACpB,iBAAiB,QAAQ;IACzB,qBAAqB,QAAQ;IAC7B,+BAA+B,QAAQ;IACxC;GACD,MAAM,OAAO;IAAE;IAAS;IAAQ;AAEhC,OAAI,UAAU,SAAS,GAAG;IACxB,MAAM,SAAS,UAAU,KAAK;AAC9B,YAAQ,IAAI,IAAI,KAAK;AACrB,WAAO,YAAY,IAAI;SAEvB,OAAM,KAAK;IAAE;IAAK;IAAM,CAAC;IAE3B;;CAGJ,MAAM,UAAU,YAAY;AAC1B,OAAK,MAAM,UAAU,QACnB,QAAO,YAAY,EAAE,MAAM,QAAQ,CAAC;AAEtC,QAAM,QAAQ,IAAI,QAAQ,KAAK,MAAM,IAAI,SAAe,QAAQ,EAAE,GAAG,cAAc,KAAK,CAAC,CAAC,CAAC,CAAC;;AAG9F,QAAO;EAAE;EAAa;EAAS;EAAa"}
@@ -1,5 +1,5 @@
1
1
 
2
- import { defineTheme } from "./css/theme.cjs";
2
+ import { UDSThemeConfig, UDSThemeConfigInput, UDSThemeContext, defineTheme } from "./css/theme.cjs";
3
3
  import { textVariantsSafe } from "./tailwind/components/getResponsiveTextStyles.cjs";
4
4
  import { WebTokens, parseTokens } from "./utils/parseTokens.cjs";
5
- export { type WebTokens, defineTheme, parseTokens, textVariantsSafe };
5
+ export { type UDSThemeConfig, type UDSThemeConfigInput, type UDSThemeContext, type WebTokens, defineTheme, parseTokens, textVariantsSafe };
@@ -1,5 +1,5 @@
1
1
 
2
- import { defineTheme } from "./css/theme.js";
2
+ import { UDSThemeConfig, UDSThemeConfigInput, UDSThemeContext, defineTheme } from "./css/theme.js";
3
3
  import { textVariantsSafe } from "./tailwind/components/getResponsiveTextStyles.js";
4
4
  import { WebTokens, parseTokens } from "./utils/parseTokens.js";
5
- export { type WebTokens, defineTheme, parseTokens, textVariantsSafe };
5
+ export { type UDSThemeConfig, type UDSThemeConfigInput, type UDSThemeContext, type WebTokens, defineTheme, parseTokens, textVariantsSafe };
@@ -24,11 +24,7 @@ const getLocalDefinitionNodes = (node) => {
24
24
  ].sort((left, right) => right.getStart() - left.getStart()).slice(0, 1);
25
25
  };
26
26
  const getDefinitionNodesSafe = (node) => {
27
- try {
28
- return node.getDefinitionNodes();
29
- } catch {
30
- return getLocalDefinitionNodes(node);
31
- }
27
+ return getLocalDefinitionNodes(node);
32
28
  };
33
29
  const isExistingFile = (candidate) => node_fs.default.existsSync(candidate) && node_fs.default.statSync(candidate).isFile();
34
30
  const resolveRelativeImportPath = (sourceFilePath, moduleSpecifier) => {
@@ -230,7 +226,7 @@ const extractArrayElementValues = (node, visited) => {
230
226
  if (ts_morph.Node.isArrayLiteralExpression(node)) return node.getElements().flatMap((element) => extractStringLiterals(element, visited)).filter((value, index, values) => values.indexOf(value) === index);
231
227
  if (ts_morph.Node.isParenthesizedExpression(node)) return extractArrayElementValues(node.getExpression(), visited);
232
228
  if (ts_morph.Node.isAsExpression(node) || ts_morph.Node.isTypeAssertion(node) || ts_morph.Node.isSatisfiesExpression(node)) return extractArrayElementValues(node.getExpression(), visited);
233
- if (ts_morph.Node.isIdentifier(node)) return node.getDefinitionNodes().flatMap((definition) => {
229
+ if (ts_morph.Node.isIdentifier(node)) return getLocalDefinitionNodes(node).flatMap((definition) => {
234
230
  if (ts_morph.Node.isVariableDeclaration(definition)) {
235
231
  const initializer = definition.getInitializer();
236
232
  return initializer ? extractArrayElementValues(initializer, visited) : [];
@@ -258,7 +254,22 @@ const extractIdentifierValues = (node, visited) => {
258
254
  if (mappedParameterValues.length > 0) return mappedParameterValues;
259
255
  if (ts_morph.Node.isVariableDeclaration(definition) || ts_morph.Node.isParameterDeclaration(definition) || ts_morph.Node.isBindingElement(definition)) {
260
256
  const initializer = definition.getInitializer();
261
- return initializer ? extractStringLiterals(initializer, visited) : [];
257
+ if (initializer) return extractStringLiterals(initializer, visited);
258
+ if (ts_morph.Node.isBindingElement(definition)) {
259
+ const bindingPattern = definition.getParent();
260
+ if (ts_morph.Node.isObjectBindingPattern(bindingPattern)) {
261
+ const parentDecl = bindingPattern.getParent();
262
+ if (ts_morph.Node.isVariableDeclaration(parentDecl)) {
263
+ const parentInit = parentDecl.getInitializer();
264
+ if (parentInit) {
265
+ const propName = definition.getNameNode().getText();
266
+ const propValues = extractObjectValues(parentInit, new Set(visited)).filter((obj) => typeof obj === "object" && obj !== null && propName in obj).map((obj) => obj[propName]).filter((v) => typeof v === "string");
267
+ if (propValues.length > 0) return propValues;
268
+ }
269
+ }
270
+ }
271
+ }
272
+ return [];
262
273
  }
263
274
  return [];
264
275
  });
@@ -321,6 +332,35 @@ const extractObjectValues = (node, visited) => {
321
332
  });
322
333
  return values;
323
334
  }
335
+ if (ts_morph.Node.isCallExpression(node)) {
336
+ const expression = node.getExpression();
337
+ if (ts_morph.Node.isIdentifier(expression)) getDefinitionNodesSafe(expression).forEach((definition) => {
338
+ if (ts_morph.Node.isFunctionDeclaration(definition)) definition.getDescendantsOfKind(ts_morph.SyntaxKind.ReturnStatement).forEach((returnStmt) => {
339
+ const returnExpr = returnStmt.getExpression();
340
+ if (returnExpr) values.push(...extractObjectValues(returnExpr, visited));
341
+ });
342
+ if (ts_morph.Node.isVariableDeclaration(definition)) {
343
+ const initializer = definition.getInitializer();
344
+ if (initializer) {
345
+ const extractFromFn = (fn) => {
346
+ if (ts_morph.Node.isArrowFunction(fn)) {
347
+ const body = fn.getBody();
348
+ if (ts_morph.Node.isBlock(body)) body.getDescendantsOfKind(ts_morph.SyntaxKind.ReturnStatement).forEach((returnStmt) => {
349
+ const returnExpr = returnStmt.getExpression();
350
+ if (returnExpr) values.push(...extractObjectValues(returnExpr, visited));
351
+ });
352
+ else values.push(...extractObjectValues(body, visited));
353
+ } else if (ts_morph.Node.isFunctionExpression(fn)) fn.getDescendantsOfKind(ts_morph.SyntaxKind.ReturnStatement).forEach((returnStmt) => {
354
+ const returnExpr = returnStmt.getExpression();
355
+ if (returnExpr) values.push(...extractObjectValues(returnExpr, visited));
356
+ });
357
+ };
358
+ extractFromFn(initializer);
359
+ }
360
+ }
361
+ });
362
+ return values;
363
+ }
324
364
  if (ts_morph.Node.isElementAccessExpression(node)) {
325
365
  extractObjectValues(node.getExpression(), visited).forEach((obj) => {
326
366
  if (typeof obj === "object" && obj !== null) Object.values(obj).forEach((value) => values.push(value));
@@ -330,16 +370,56 @@ const extractObjectValues = (node, visited) => {
330
370
  return values;
331
371
  };
332
372
  /**
333
- * Extract literal values from a TypeScript type (for union types like 'brand' | 'secondary')
373
+ * Extract string literal values from a TypeScript type annotation AST node.
374
+ * Pure AST traversal — avoids expensive TS Language Service `getType()` calls.
375
+ */
376
+ const extractStringLiteralsFromTypeNode = (typeNode) => {
377
+ if (ts_morph.Node.isLiteralTypeNode(typeNode)) {
378
+ const literal = typeNode.getLiteral();
379
+ if (ts_morph.Node.isStringLiteral(literal)) return [literal.getLiteralValue()];
380
+ return [];
381
+ }
382
+ if (ts_morph.Node.isUnionTypeNode(typeNode)) return typeNode.getTypeNodes().flatMap(extractStringLiteralsFromTypeNode);
383
+ return [];
384
+ };
385
+ /**
386
+ * Extract literal values from a TypeScript type annotation (for union types like 'brand' | 'secondary').
387
+ * Uses pure AST traversal instead of the TS type checker to avoid expensive LS initialization.
334
388
  */
335
389
  const extractLiteralValuesFromType = (node) => {
336
- const values = [];
337
- const nodeType = node.getType();
338
- if (nodeType.isUnion()) nodeType.getUnionTypes().forEach((unionMember) => {
339
- if (unionMember.isStringLiteral()) values.push(unionMember.getLiteralValue());
340
- });
341
- else if (nodeType.isStringLiteral()) values.push(nodeType.getLiteralValue());
342
- return values;
390
+ if (ts_morph.Node.isIdentifier(node)) {
391
+ const definitions = getLocalDefinitionNodes(node);
392
+ for (const def of definitions) {
393
+ if (ts_morph.Node.isParameterDeclaration(def) || ts_morph.Node.isVariableDeclaration(def) || ts_morph.Node.isPropertyDeclaration(def) || ts_morph.Node.isPropertySignature(def)) {
394
+ const typeNode = def.getTypeNode?.();
395
+ if (typeNode) {
396
+ const values = extractStringLiteralsFromTypeNode(typeNode);
397
+ if (values.length > 0) return values;
398
+ }
399
+ }
400
+ if (ts_morph.Node.isBindingElement(def)) {
401
+ const bindingPattern = def.getParent();
402
+ if (bindingPattern) {
403
+ const parentDecl = bindingPattern.getParent();
404
+ if (parentDecl && ts_morph.Node.isVariableDeclaration(parentDecl)) {
405
+ const parentType = parentDecl.getTypeNode();
406
+ if (parentType && ts_morph.Node.isTypeLiteral(parentType)) {
407
+ const propName = def.getNameNode().getText();
408
+ for (const member of parentType.getMembers()) if (ts_morph.Node.isPropertySignature(member) && member.getName() === propName) {
409
+ const memberType = member.getTypeNode();
410
+ if (memberType) {
411
+ const values = extractStringLiteralsFromTypeNode(memberType);
412
+ if (values.length > 0) return values;
413
+ }
414
+ }
415
+ }
416
+ }
417
+ }
418
+ }
419
+ }
420
+ }
421
+ if (ts_morph.Node.isPropertyAccessExpression(node) || ts_morph.Node.isElementAccessExpression(node)) return [];
422
+ return [];
343
423
  };
344
424
  //#endregion
345
425
  exports.extractStringLiterals = extractStringLiterals;
@@ -21,11 +21,7 @@ const getLocalDefinitionNodes = (node) => {
21
21
  ].sort((left, right) => right.getStart() - left.getStart()).slice(0, 1);
22
22
  };
23
23
  const getDefinitionNodesSafe = (node) => {
24
- try {
25
- return node.getDefinitionNodes();
26
- } catch {
27
- return getLocalDefinitionNodes(node);
28
- }
24
+ return getLocalDefinitionNodes(node);
29
25
  };
30
26
  const isExistingFile = (candidate) => fs.existsSync(candidate) && fs.statSync(candidate).isFile();
31
27
  const resolveRelativeImportPath = (sourceFilePath, moduleSpecifier) => {
@@ -227,7 +223,7 @@ const extractArrayElementValues = (node, visited) => {
227
223
  if (Node.isArrayLiteralExpression(node)) return node.getElements().flatMap((element) => extractStringLiterals(element, visited)).filter((value, index, values) => values.indexOf(value) === index);
228
224
  if (Node.isParenthesizedExpression(node)) return extractArrayElementValues(node.getExpression(), visited);
229
225
  if (Node.isAsExpression(node) || Node.isTypeAssertion(node) || Node.isSatisfiesExpression(node)) return extractArrayElementValues(node.getExpression(), visited);
230
- if (Node.isIdentifier(node)) return node.getDefinitionNodes().flatMap((definition) => {
226
+ if (Node.isIdentifier(node)) return getLocalDefinitionNodes(node).flatMap((definition) => {
231
227
  if (Node.isVariableDeclaration(definition)) {
232
228
  const initializer = definition.getInitializer();
233
229
  return initializer ? extractArrayElementValues(initializer, visited) : [];
@@ -255,7 +251,22 @@ const extractIdentifierValues = (node, visited) => {
255
251
  if (mappedParameterValues.length > 0) return mappedParameterValues;
256
252
  if (Node.isVariableDeclaration(definition) || Node.isParameterDeclaration(definition) || Node.isBindingElement(definition)) {
257
253
  const initializer = definition.getInitializer();
258
- return initializer ? extractStringLiterals(initializer, visited) : [];
254
+ if (initializer) return extractStringLiterals(initializer, visited);
255
+ if (Node.isBindingElement(definition)) {
256
+ const bindingPattern = definition.getParent();
257
+ if (Node.isObjectBindingPattern(bindingPattern)) {
258
+ const parentDecl = bindingPattern.getParent();
259
+ if (Node.isVariableDeclaration(parentDecl)) {
260
+ const parentInit = parentDecl.getInitializer();
261
+ if (parentInit) {
262
+ const propName = definition.getNameNode().getText();
263
+ const propValues = extractObjectValues(parentInit, new Set(visited)).filter((obj) => typeof obj === "object" && obj !== null && propName in obj).map((obj) => obj[propName]).filter((v) => typeof v === "string");
264
+ if (propValues.length > 0) return propValues;
265
+ }
266
+ }
267
+ }
268
+ }
269
+ return [];
259
270
  }
260
271
  return [];
261
272
  });
@@ -318,6 +329,35 @@ const extractObjectValues = (node, visited) => {
318
329
  });
319
330
  return values;
320
331
  }
332
+ if (Node.isCallExpression(node)) {
333
+ const expression = node.getExpression();
334
+ if (Node.isIdentifier(expression)) getDefinitionNodesSafe(expression).forEach((definition) => {
335
+ if (Node.isFunctionDeclaration(definition)) definition.getDescendantsOfKind(SyntaxKind.ReturnStatement).forEach((returnStmt) => {
336
+ const returnExpr = returnStmt.getExpression();
337
+ if (returnExpr) values.push(...extractObjectValues(returnExpr, visited));
338
+ });
339
+ if (Node.isVariableDeclaration(definition)) {
340
+ const initializer = definition.getInitializer();
341
+ if (initializer) {
342
+ const extractFromFn = (fn) => {
343
+ if (Node.isArrowFunction(fn)) {
344
+ const body = fn.getBody();
345
+ if (Node.isBlock(body)) body.getDescendantsOfKind(SyntaxKind.ReturnStatement).forEach((returnStmt) => {
346
+ const returnExpr = returnStmt.getExpression();
347
+ if (returnExpr) values.push(...extractObjectValues(returnExpr, visited));
348
+ });
349
+ else values.push(...extractObjectValues(body, visited));
350
+ } else if (Node.isFunctionExpression(fn)) fn.getDescendantsOfKind(SyntaxKind.ReturnStatement).forEach((returnStmt) => {
351
+ const returnExpr = returnStmt.getExpression();
352
+ if (returnExpr) values.push(...extractObjectValues(returnExpr, visited));
353
+ });
354
+ };
355
+ extractFromFn(initializer);
356
+ }
357
+ }
358
+ });
359
+ return values;
360
+ }
321
361
  if (Node.isElementAccessExpression(node)) {
322
362
  extractObjectValues(node.getExpression(), visited).forEach((obj) => {
323
363
  if (typeof obj === "object" && obj !== null) Object.values(obj).forEach((value) => values.push(value));
@@ -327,16 +367,56 @@ const extractObjectValues = (node, visited) => {
327
367
  return values;
328
368
  };
329
369
  /**
330
- * Extract literal values from a TypeScript type (for union types like 'brand' | 'secondary')
370
+ * Extract string literal values from a TypeScript type annotation AST node.
371
+ * Pure AST traversal — avoids expensive TS Language Service `getType()` calls.
372
+ */
373
+ const extractStringLiteralsFromTypeNode = (typeNode) => {
374
+ if (Node.isLiteralTypeNode(typeNode)) {
375
+ const literal = typeNode.getLiteral();
376
+ if (Node.isStringLiteral(literal)) return [literal.getLiteralValue()];
377
+ return [];
378
+ }
379
+ if (Node.isUnionTypeNode(typeNode)) return typeNode.getTypeNodes().flatMap(extractStringLiteralsFromTypeNode);
380
+ return [];
381
+ };
382
+ /**
383
+ * Extract literal values from a TypeScript type annotation (for union types like 'brand' | 'secondary').
384
+ * Uses pure AST traversal instead of the TS type checker to avoid expensive LS initialization.
331
385
  */
332
386
  const extractLiteralValuesFromType = (node) => {
333
- const values = [];
334
- const nodeType = node.getType();
335
- if (nodeType.isUnion()) nodeType.getUnionTypes().forEach((unionMember) => {
336
- if (unionMember.isStringLiteral()) values.push(unionMember.getLiteralValue());
337
- });
338
- else if (nodeType.isStringLiteral()) values.push(nodeType.getLiteralValue());
339
- return values;
387
+ if (Node.isIdentifier(node)) {
388
+ const definitions = getLocalDefinitionNodes(node);
389
+ for (const def of definitions) {
390
+ if (Node.isParameterDeclaration(def) || Node.isVariableDeclaration(def) || Node.isPropertyDeclaration(def) || Node.isPropertySignature(def)) {
391
+ const typeNode = def.getTypeNode?.();
392
+ if (typeNode) {
393
+ const values = extractStringLiteralsFromTypeNode(typeNode);
394
+ if (values.length > 0) return values;
395
+ }
396
+ }
397
+ if (Node.isBindingElement(def)) {
398
+ const bindingPattern = def.getParent();
399
+ if (bindingPattern) {
400
+ const parentDecl = bindingPattern.getParent();
401
+ if (parentDecl && Node.isVariableDeclaration(parentDecl)) {
402
+ const parentType = parentDecl.getTypeNode();
403
+ if (parentType && Node.isTypeLiteral(parentType)) {
404
+ const propName = def.getNameNode().getText();
405
+ for (const member of parentType.getMembers()) if (Node.isPropertySignature(member) && member.getName() === propName) {
406
+ const memberType = member.getTypeNode();
407
+ if (memberType) {
408
+ const values = extractStringLiteralsFromTypeNode(memberType);
409
+ if (values.length > 0) return values;
410
+ }
411
+ }
412
+ }
413
+ }
414
+ }
415
+ }
416
+ }
417
+ }
418
+ if (Node.isPropertyAccessExpression(node) || Node.isElementAccessExpression(node)) return [];
419
+ return [];
340
420
  };
341
421
  //#endregion
342
422
  export { extractStringLiterals };