@wise/wds-codemods 0.0.1-experimental-b3cbbc0 → 0.0.1-experimental-42663ed
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.js +1873 -49
- package/dist/index.js.map +1 -1
- package/dist/reportManualReview-Bf-sT2Ka.js +61 -0
- package/dist/reportManualReview-Bf-sT2Ka.js.map +1 -0
- package/dist/transforms/button/config.json +1 -23
- package/dist/transforms/button/transformer.js +34 -46
- package/dist/transforms/button/transformer.js.map +1 -1
- package/package.json +2 -2
- package/dist/helpers-25o8ZWHe.js +0 -2005
- package/dist/helpers-25o8ZWHe.js.map +0 -1
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
Object.defineProperty(exports, '__esModule', { value: true });
|
|
2
|
-
const
|
|
2
|
+
const require_reportManualReview = require('../../reportManualReview-Bf-sT2Ka.js');
|
|
3
3
|
|
|
4
4
|
//#region src/helpers/hasImport.ts
|
|
5
5
|
/**
|
|
@@ -216,8 +216,7 @@ var CodemodReporter = class {
|
|
|
216
216
|
* Auto-detects and reports common attribute issues
|
|
217
217
|
*/
|
|
218
218
|
reportAttributeIssues(element) {
|
|
219
|
-
const
|
|
220
|
-
const { attributes } = node.openingElement;
|
|
219
|
+
const { attributes } = this.getNode(element).openingElement;
|
|
221
220
|
if (!attributes) return;
|
|
222
221
|
if (attributes.some((attr) => attr.type === "JSXSpreadAttribute")) this.reportSpreadProps(element);
|
|
223
222
|
attributes.forEach((attr) => {
|
|
@@ -266,27 +265,22 @@ const createReporter = (j, issues) => {
|
|
|
266
265
|
//#endregion
|
|
267
266
|
//#region src/transforms/button/transformer.ts
|
|
268
267
|
const parser = "tsx";
|
|
269
|
-
const
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
primary: "primary",
|
|
286
|
-
secondary: "secondary",
|
|
287
|
-
tertiary: "secondary"
|
|
288
|
-
}
|
|
289
|
-
};
|
|
268
|
+
const priorityMapping = {
|
|
269
|
+
accent: {
|
|
270
|
+
primary: "primary",
|
|
271
|
+
secondary: "secondary-neutral",
|
|
272
|
+
tertiary: "tertiary"
|
|
273
|
+
},
|
|
274
|
+
positive: {
|
|
275
|
+
primary: "primary",
|
|
276
|
+
secondary: "secondary-neutral",
|
|
277
|
+
tertiary: "secondary-neutral"
|
|
278
|
+
},
|
|
279
|
+
negative: {
|
|
280
|
+
primary: "primary",
|
|
281
|
+
secondary: "secondary",
|
|
282
|
+
tertiary: "secondary"
|
|
283
|
+
}
|
|
290
284
|
};
|
|
291
285
|
const sizeMap = {
|
|
292
286
|
EXTRA_SMALL: "xs",
|
|
@@ -306,9 +300,13 @@ const resolveSize = (size) => {
|
|
|
306
300
|
if (match) return sizeMap[match[1]];
|
|
307
301
|
return sizeMap[size] || size;
|
|
308
302
|
};
|
|
303
|
+
const resolvePriority = (type, priority) => {
|
|
304
|
+
if (type && priority) return priorityMapping[type]?.[priority] || priority;
|
|
305
|
+
return priority;
|
|
306
|
+
};
|
|
309
307
|
const resolveType = (type, htmlType) => {
|
|
310
308
|
if (htmlType) return htmlType;
|
|
311
|
-
|
|
309
|
+
return type && [
|
|
312
310
|
"accent",
|
|
313
311
|
"negative",
|
|
314
312
|
"positive",
|
|
@@ -317,21 +315,19 @@ const resolveType = (type, htmlType) => {
|
|
|
317
315
|
"secondary",
|
|
318
316
|
"danger",
|
|
319
317
|
"link"
|
|
320
|
-
];
|
|
321
|
-
return type && legacyButtonTypes.includes(type) ? type : null;
|
|
318
|
+
].includes(type) ? type : null;
|
|
322
319
|
};
|
|
323
320
|
const convertEnumValue = (value) => {
|
|
324
321
|
if (!value) return value;
|
|
325
322
|
const strippedValue = value.replace(/^['"]|['"]$/gu, "");
|
|
326
|
-
|
|
323
|
+
return {
|
|
327
324
|
"Priority.SECONDARY": "secondary",
|
|
328
325
|
"Priority.PRIMARY": "primary",
|
|
329
326
|
"Priority.TERTIARY": "tertiary",
|
|
330
327
|
"ControlType.NEGATIVE": "negative",
|
|
331
328
|
"ControlType.POSITIVE": "positive",
|
|
332
329
|
"ControlType.ACCENT": "accent"
|
|
333
|
-
};
|
|
334
|
-
return enumMapping[strippedValue] || strippedValue;
|
|
330
|
+
}[strippedValue] || strippedValue;
|
|
335
331
|
};
|
|
336
332
|
/**
|
|
337
333
|
* This transform function modifies the Button and ActionButton components from the @transferwise/components library.
|
|
@@ -347,11 +343,6 @@ const transformer = (file, api, options) => {
|
|
|
347
343
|
const j = api.jscodeshift;
|
|
348
344
|
const root = j(file.source);
|
|
349
345
|
const manualReviewIssues = [];
|
|
350
|
-
const priorityMapping = buildPriorityMapping(options);
|
|
351
|
-
const resolvePriority = (type, priority) => {
|
|
352
|
-
if (type && priority) return priorityMapping[type]?.[priority] || priority;
|
|
353
|
-
return priority;
|
|
354
|
-
};
|
|
355
346
|
const reporter = createReporter(j, manualReviewIssues);
|
|
356
347
|
const { exists: hasButtonImport } = hasImport_default(root, "@transferwise/components", "Button", j);
|
|
357
348
|
const { exists: hasActionButtonImport, remove: removeActionButtonImport } = hasImport_default(root, "@transferwise/components", "ActionButton", j);
|
|
@@ -432,14 +423,13 @@ const transformer = (file, api, options) => {
|
|
|
432
423
|
if ("size" in legacyProps) {
|
|
433
424
|
const rawValue = legacyProps.size;
|
|
434
425
|
const resolved = resolveSize(rawValue);
|
|
435
|
-
|
|
426
|
+
if (typeof rawValue === "string" && typeof resolved === "string" && [
|
|
436
427
|
"xs",
|
|
437
428
|
"sm",
|
|
438
429
|
"md",
|
|
439
430
|
"lg",
|
|
440
431
|
"xl"
|
|
441
|
-
];
|
|
442
|
-
if (typeof rawValue === "string" && typeof resolved === "string" && supportedSizes.includes(resolved)) openingElement.attributes?.push(j.jsxAttribute(j.jsxIdentifier("size"), j.literal(resolved)));
|
|
432
|
+
].includes(resolved)) openingElement.attributes?.push(j.jsxAttribute(j.jsxIdentifier("size"), j.literal(resolved)));
|
|
443
433
|
else if (typeof rawValue === "string") reporter.reportUnsupportedValue(path, "size", rawValue);
|
|
444
434
|
else if (rawValue !== void 0) reporter.reportAmbiguousExpression(path, "size");
|
|
445
435
|
}
|
|
@@ -447,13 +437,12 @@ const transformer = (file, api, options) => {
|
|
|
447
437
|
const rawValue = legacyProps.priority;
|
|
448
438
|
const converted = convertEnumValue(rawValue);
|
|
449
439
|
const mapped = resolvePriority(legacyProps.type, converted);
|
|
450
|
-
|
|
440
|
+
if (typeof rawValue === "string" && typeof mapped === "string" && [
|
|
451
441
|
"primary",
|
|
452
442
|
"secondary",
|
|
453
443
|
"tertiary",
|
|
454
444
|
"secondary-neutral"
|
|
455
|
-
];
|
|
456
|
-
if (typeof rawValue === "string" && typeof mapped === "string" && supportedPriorities.includes(mapped)) openingElement.attributes?.push(j.jsxAttribute(j.jsxIdentifier("priority"), j.literal(mapped)));
|
|
445
|
+
].includes(mapped)) openingElement.attributes?.push(j.jsxAttribute(j.jsxIdentifier("priority"), j.literal(mapped)));
|
|
457
446
|
else if (typeof rawValue === "string") reporter.reportUnsupportedValue(path, "priority", rawValue);
|
|
458
447
|
else if (rawValue !== void 0) reporter.reportAmbiguousExpression(path, "priority");
|
|
459
448
|
}
|
|
@@ -462,7 +451,7 @@ const transformer = (file, api, options) => {
|
|
|
462
451
|
const rawHtmlType = legacyProps.htmlType;
|
|
463
452
|
const resolvedType = typeof rawType === "string" ? rawType : rawType && typeof rawType === "object" ? convertEnumValue(j(rawType).toSource()) : void 0;
|
|
464
453
|
const resolved = resolveType(resolvedType, rawHtmlType);
|
|
465
|
-
|
|
454
|
+
if (typeof resolved === "string" && [
|
|
466
455
|
"accent",
|
|
467
456
|
"negative",
|
|
468
457
|
"positive",
|
|
@@ -474,8 +463,7 @@ const transformer = (file, api, options) => {
|
|
|
474
463
|
"submit",
|
|
475
464
|
"button",
|
|
476
465
|
"reset"
|
|
477
|
-
]
|
|
478
|
-
if (typeof resolved === "string" && supportedTypes.includes(resolved)) {
|
|
466
|
+
].includes(resolved)) {
|
|
479
467
|
openingElement.attributes?.push(j.jsxAttribute(j.jsxIdentifier("type"), j.literal(resolved)));
|
|
480
468
|
if (resolved === "negative") openingElement.attributes?.push(j.jsxAttribute(j.jsxIdentifier("sentiment"), j.literal("negative")));
|
|
481
469
|
} else if (typeof rawType === "string" || typeof rawHtmlType === "string") reporter.reportUnsupportedValue(path, "type", rawType ?? rawHtmlType ?? "");
|
|
@@ -507,13 +495,13 @@ const transformer = (file, api, options) => {
|
|
|
507
495
|
});
|
|
508
496
|
if (asValue && asValue !== "a") reporter.reportUnsupportedValue(path, "as", asValue);
|
|
509
497
|
if (asValue === "a") {
|
|
510
|
-
if (asIndex !== -1) openingElement.attributes = openingElement.attributes?.filter((
|
|
498
|
+
if (asIndex !== -1) openingElement.attributes = openingElement.attributes?.filter((_, idx) => idx !== asIndex);
|
|
511
499
|
if (!hrefExists) openingElement.attributes = [...openingElement.attributes ?? [], j.jsxAttribute(j.jsxIdentifier("href"), j.literal("#"))];
|
|
512
500
|
}
|
|
513
501
|
if ((openingElement.attributes ?? []).some((attr) => attr.type === "JSXSpreadAttribute")) reporter.reportSpreadProps(path);
|
|
514
502
|
});
|
|
515
503
|
if (manualReviewIssues.length > 0) manualReviewIssues.forEach(async (issue) => {
|
|
516
|
-
await
|
|
504
|
+
await require_reportManualReview.reportManualReview_default(file.path, issue);
|
|
517
505
|
});
|
|
518
506
|
return root.toSource();
|
|
519
507
|
};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transformer.js","names":["sizeMap: Record<string, string>","enumMapping: Record<string, string>","j: JSCodeshift","manualReviewIssues: string[]","hasImport","legacyProps: LegacyProps","asValue: string | null","reportManualReview"],"sources":["../../../src/helpers/hasImport.ts","../../../src/helpers/iconUtils.ts","../../../src/helpers/jsxElementUtils.ts","../../../src/helpers/jsxReportingUtils.ts","../../../src/transforms/button/transformer.ts"],"sourcesContent":["import type { Collection, JSCodeshift } from 'jscodeshift';\n\n/**\n * Checks if a specific import exists in the given root collection and provides\n * a method to remove it if found.\n */\nfunction hasImport(\n root: Collection,\n sourceValue: string,\n importName: string,\n j: JSCodeshift,\n): { exists: boolean; remove: () => void } {\n const importDeclarations = root.find(j.ImportDeclaration, {\n source: { value: sourceValue },\n });\n\n if (importDeclarations.size() === 0) {\n return {\n exists: false,\n remove: () => {},\n };\n }\n\n const namedImport = importDeclarations.find(j.ImportSpecifier, {\n imported: { name: importName },\n });\n\n const defaultImport = importDeclarations.find(j.ImportDefaultSpecifier, {\n local: { name: importName },\n });\n\n const exists = namedImport.size() > 0 || defaultImport.size() > 0;\n\n const remove = () => {\n importDeclarations.forEach((path) => {\n const filteredSpecifiers =\n path.node.specifiers?.filter((specifier) => {\n if (specifier.type === 'ImportSpecifier' && specifier.imported.name === importName) {\n return false;\n }\n if (specifier.type === 'ImportDefaultSpecifier' && specifier.local?.name === importName) {\n return false;\n }\n return true;\n }) ?? [];\n\n if (filteredSpecifiers.length === 0) {\n path.prune();\n } else {\n j(path).replaceWith(\n j.importDeclaration(filteredSpecifiers, path.node.source, path.node.importKind),\n );\n }\n });\n };\n\n return { exists, remove };\n}\n\nexport default hasImport;\n","import type { JSCodeshift, JSXElement, JSXExpressionContainer } from 'jscodeshift';\n\n/**\n * Process children of a JSX element to detect icon components and add iconStart or iconEnd attributes accordingly.\n * This is specific to icon handling but can be reused in codemods dealing with icon children.\n */\nconst processIconChildren = (\n j: JSCodeshift,\n children: (JSXElement | JSXExpressionContainer | unknown)[] | undefined,\n iconImports: Set<string>,\n openingElement: JSXElement['openingElement'],\n) => {\n if (!children || !openingElement.attributes) return;\n\n const unwrapJsxElement = (node: unknown): JSXElement | unknown => {\n if (\n typeof node === 'object' &&\n node !== null &&\n 'type' in node &&\n node.type === 'JSXExpressionContainer' &&\n j.JSXElement.check((node as JSXExpressionContainer).expression)\n ) {\n return (node as JSXExpressionContainer).expression;\n }\n return node;\n };\n\n const totalChildren = children.length;\n\n // Find index of icon child\n const iconChildIndex = children.findIndex((child) => {\n const unwrapped = unwrapJsxElement(child);\n return (\n j.JSXElement.check(unwrapped) &&\n unwrapped.openingElement.name.type === 'JSXIdentifier' &&\n iconImports.has(unwrapped.openingElement.name.name)\n );\n });\n\n if (iconChildIndex === -1) return;\n\n const iconChild = unwrapJsxElement(children[iconChildIndex]) as JSXElement;\n\n if (!iconChild || iconChild.openingElement.name.type !== 'JSXIdentifier') return;\n\n const iconName = iconChild.openingElement.name.name;\n\n // Determine if icon is closer to start or end\n const distanceToStart = iconChildIndex;\n const distanceToEnd = totalChildren - 1 - iconChildIndex;\n const iconPropName = distanceToStart <= distanceToEnd ? 'addonStart' : 'addonEnd';\n\n // Build: { type: 'icon', value: <IconName /> }\n const iconObject = j.objectExpression([\n j.property('init', j.identifier('type'), j.literal('icon')),\n j.property('init', j.identifier('value'), iconChild),\n ]);\n const iconProp = j.jsxAttribute(\n j.jsxIdentifier(iconPropName),\n j.jsxExpressionContainer(iconObject),\n );\n\n openingElement.attributes.push(iconProp);\n\n // Remove the icon child\n children.splice(iconChildIndex, 1);\n\n // Helper to check if a child is whitespace-only JSXText\n const isWhitespaceJsxText = (node: unknown): boolean => {\n return (\n typeof node === 'object' &&\n node !== null &&\n (node as { type?: unknown }).type === 'JSXText' &&\n typeof (node as { value?: string }).value === 'string' &&\n (node as { value?: string }).value!.trim() === ''\n );\n };\n\n // Remove adjacent whitespace-only JSXText node if any\n if (iconChildIndex - 1 >= 0 && isWhitespaceJsxText(children[iconChildIndex - 1])) {\n children.splice(iconChildIndex - 1, 1);\n } else if (isWhitespaceJsxText(children[iconChildIndex])) {\n children.splice(iconChildIndex, 1);\n }\n};\n\nexport default processIconChildren;\n","import type {\n JSCodeshift,\n JSXAttribute,\n JSXElement,\n JSXIdentifier,\n JSXMemberExpression,\n JSXNamespacedName,\n JSXSpreadAttribute,\n} from 'jscodeshift';\n\n/**\n * Rename a JSX element name if it is a JSXIdentifier.\n */\nexport const setNameIfJSXIdentifier = (\n elementName: JSXIdentifier | JSXNamespacedName | JSXMemberExpression | undefined,\n newName: string,\n): JSXIdentifier | JSXNamespacedName | JSXMemberExpression | undefined => {\n if (elementName && elementName.type === 'JSXIdentifier') {\n return { ...elementName, name: newName };\n }\n return elementName;\n};\n\n/**\n * Check if a list of attributes contains a specific attribute by name.\n */\nexport const hasAttribute = (\n attributes: (JSXAttribute | JSXSpreadAttribute)[] | undefined,\n attributeName: string,\n): boolean => {\n return (\n Array.isArray(attributes) &&\n attributes.some(\n (attr): attr is JSXAttribute =>\n attr.type === 'JSXAttribute' &&\n attr.name.type === 'JSXIdentifier' &&\n attr.name.name === attributeName,\n )\n );\n};\n\n/**\n * Check if a JSX element's openingElement has a specific attribute.\n */\nexport const hasAttributeOnElement = (\n element: JSXElement['openingElement'],\n attributeName: string,\n): boolean => {\n return hasAttribute(element.attributes, attributeName);\n};\n\n/**\n * Add specified attributes to a JSX element's openingElement if they are not already present.\n */\nexport const addAttributesIfMissing = (\n j: JSCodeshift,\n openingElement: JSXElement['openingElement'],\n attributesToAdd: { attribute: JSXAttribute; name: string }[],\n) => {\n if (!Array.isArray(openingElement.attributes)) return;\n const attrs = openingElement.attributes;\n attributesToAdd.forEach(({ attribute, name }) => {\n if (!hasAttributeOnElement(openingElement, name)) {\n attrs.push(attribute);\n }\n });\n};\n","import type { ASTPath, JSCodeshift, JSXAttribute, JSXElement, Node } from 'jscodeshift';\n\nexport interface ReporterOptions {\n jscodeshift: JSCodeshift;\n issues: string[];\n}\n\n/**\n * CodemodReporter is a utility class for reporting issues found during codemod transformations.\n * It provides methods to report issues related to JSX elements, props, and attributes.\n *\n * @example\n * ```typescript\n * const issues: string[] = [];\n * const reporter = createReporter(j, issues);\n *\n * // Report a deprecated prop\n * reporter.reportDeprecatedProp(buttonElement, 'flat', 'variant=\"text\"');\n *\n * // Report complex expression that needs review\n * reporter.reportAmbiguousExpression(element, 'size');\n *\n * // Auto-detect common issues\n * reporter.reportAttributeIssues(element);\n * ```\n */\nexport class CodemodReporter {\n private readonly j: JSCodeshift;\n private readonly issues: string[];\n\n constructor(options: ReporterOptions) {\n this.j = options.jscodeshift;\n this.issues = options.issues;\n }\n\n /**\n * Reports an issue with a JSX element\n */\n reportElement(element: JSXElement | ASTPath<JSXElement>, reason: string): void {\n const node = this.getNode(element);\n const componentName = this.getComponentName(node);\n const line = this.getLineNumber(node);\n\n this.addIssue(`Manual review required: <${componentName}> at line ${line} ${reason}.`);\n }\n\n /**\n * Reports an issue with a specific prop\n */\n reportProp(element: JSXElement | ASTPath<JSXElement>, propName: string, reason: string): void {\n const node = this.getNode(element);\n const componentName = this.getComponentName(node);\n const line = this.getLineNumber(node);\n\n this.addIssue(\n `Manual review required: prop \"${propName}\" on <${componentName}> at line ${line} ${reason}.`,\n );\n }\n\n /**\n * Reports an issue with a JSX attribute directly\n */\n reportAttribute(\n attr: JSXAttribute,\n element: JSXElement | ASTPath<JSXElement>,\n reason?: string,\n ): void {\n const node = this.getNode(element);\n const componentName = this.getComponentName(node);\n const propName = this.getAttributeName(attr);\n const line = this.getLineNumber(attr) || this.getLineNumber(node);\n\n const defaultReason = this.getAttributeReason(attr);\n const finalReason = reason || defaultReason;\n\n this.addIssue(\n `Manual review required: prop \"${propName}\" on <${componentName}> at line ${line} ${finalReason}.`,\n );\n }\n\n /**\n * Reports spread props on an element\n */\n reportSpreadProps(element: JSXElement | ASTPath<JSXElement>): void {\n this.reportElement(element, 'contains spread props that need manual review');\n }\n\n /**\n * Reports conflicting prop and children\n */\n reportPropWithChildren(element: JSXElement | ASTPath<JSXElement>, propName: string): void {\n this.reportProp(\n element,\n propName,\n `conflicts with children - both \"${propName}\" prop and children are present`,\n );\n }\n\n /**\n * Reports unsupported prop value\n */\n reportUnsupportedValue(\n element: JSXElement | ASTPath<JSXElement>,\n propName: string,\n value: string,\n ): void {\n this.reportProp(element, propName, `has unsupported value \"${value}\"`);\n }\n\n /**\n * Reports ambiguous expression in prop\n */\n reportAmbiguousExpression(element: JSXElement | ASTPath<JSXElement>, propName: string): void {\n this.reportProp(element, propName, 'contains a complex expression that needs manual review');\n }\n\n /**\n * Reports ambiguous children (like dynamic icons)\n */\n reportAmbiguousChildren(element: JSXElement | ASTPath<JSXElement>, childType = 'content'): void {\n this.reportElement(element, `contains ambiguous ${childType} that needs manual review`);\n }\n\n /**\n * Reports deprecated prop usage\n */\n reportDeprecatedProp(\n element: JSXElement | ASTPath<JSXElement>,\n propName: string,\n alternative?: string,\n ): void {\n const suggestion = alternative ? ` Use ${alternative} instead` : '';\n this.reportProp(element, propName, `is deprecated${suggestion}`);\n }\n\n /**\n * Reports missing required prop\n */\n reportMissingRequiredProp(element: JSXElement | ASTPath<JSXElement>, propName: string): void {\n this.reportProp(element, propName, 'is required but missing');\n }\n\n /**\n * Reports conflicting props\n */\n reportConflictingProps(element: JSXElement | ASTPath<JSXElement>, propNames: string[]): void {\n const propList = propNames.map((name) => `\"${name}\"`).join(', ');\n this.reportElement(element, `has conflicting props: ${propList} cannot be used together`);\n }\n\n /**\n * Auto-detects and reports common attribute issues\n */\n reportAttributeIssues(element: JSXElement | ASTPath<JSXElement>): void {\n const node = this.getNode(element);\n const { attributes } = node.openingElement;\n\n if (!attributes) return;\n\n // Check for spread props\n if (attributes.some((attr) => attr.type === 'JSXSpreadAttribute')) {\n this.reportSpreadProps(element);\n }\n\n // Check for complex expressions in attributes\n attributes.forEach((attr) => {\n if (attr.type === 'JSXAttribute' && attr.value?.type === 'JSXExpressionContainer') {\n this.reportAttribute(attr, element);\n }\n });\n }\n\n // Private helper methods\n private getNode(element: JSXElement | ASTPath<JSXElement>): JSXElement {\n return 'node' in element ? element.node : element;\n }\n\n private getComponentName(node: JSXElement): string {\n const { name } = node.openingElement;\n if (name.type === 'JSXIdentifier') {\n return name.name;\n }\n // Handle JSXMemberExpression, JSXNamespacedName, etc.\n return this.j(name).toSource();\n }\n\n private getLineNumber(node: JSXElement | JSXAttribute | Node): string {\n return node.loc?.start.line?.toString() || 'unknown';\n }\n\n private getAttributeName(attr: JSXAttribute): string {\n if (attr.name.type === 'JSXIdentifier') {\n return attr.name.name;\n }\n return this.j(attr.name).toSource();\n }\n\n private getAttributeReason(attr: JSXAttribute): string {\n if (!attr.value) return 'has no value';\n\n if (attr.value.type === 'JSXExpressionContainer') {\n const expr = attr.value.expression;\n const expressionType = expr.type.replace('Expression', '').toLowerCase();\n\n // Show actual value for simple cases\n if (expr.type === 'Identifier' || expr.type === 'MemberExpression') {\n const valueText = this.j(expr).toSource();\n return `contains a ${expressionType} (${valueText})`;\n }\n\n return `contains a complex ${expressionType} expression`;\n }\n\n return 'needs manual review';\n }\n\n private addIssue(message: string): void {\n this.issues.push(message);\n }\n}\n\nexport const createReporter = (j: JSCodeshift, issues: string[]): CodemodReporter => {\n return new CodemodReporter({ jscodeshift: j, issues });\n};\n","import type { API, FileInfo, JSCodeshift, JSXIdentifier, Options } from 'jscodeshift';\n\nimport { reportManualReview } from '../../controller/helpers';\nimport {\n addAttributesIfMissing,\n createReporter,\n hasAttributeOnElement,\n hasImport,\n processIconChildren,\n setNameIfJSXIdentifier,\n} from '../../helpers';\n\nexport const parser = 'tsx';\n\ninterface LegacyProps {\n priority?: string;\n size?: string;\n type?: string;\n htmlType?: string;\n sentiment?: string;\n [key: string]: unknown;\n}\n\ninterface ExtendedOptions extends Options {\n accentSecondaryMapping?: string;\n positiveSecondaryMapping?: string;\n}\n\ntype PriorityMapping = Record<string, Record<string, string>>;\n\nconst buildPriorityMapping = (opts: Options): PriorityMapping => {\n const extendedOpts = opts as ExtendedOptions;\n const accentSecondary = extendedOpts.accentSecondaryMapping || 'secondary-neutral';\n const positiveSecondary = extendedOpts.positiveSecondaryMapping || 'secondary-neutral';\n return {\n accent: {\n primary: 'primary',\n secondary: accentSecondary,\n tertiary: 'tertiary',\n },\n positive: {\n primary: 'primary',\n secondary: positiveSecondary,\n tertiary: positiveSecondary,\n },\n negative: {\n primary: 'primary',\n secondary: 'secondary',\n tertiary: 'secondary',\n },\n };\n};\n\nconst sizeMap: Record<string, string> = {\n EXTRA_SMALL: 'xs',\n SMALL: 'sm',\n MEDIUM: 'md',\n LARGE: 'lg',\n EXTRA_LARGE: 'xl',\n xs: 'sm',\n sm: 'sm',\n md: 'md',\n lg: 'lg',\n xl: 'xl',\n};\n\nconst resolveSize = (size?: string): string | undefined => {\n if (!size) return size;\n const match = /^Size\\.(EXTRA_SMALL|SMALL|MEDIUM|LARGE|EXTRA_LARGE)$/u.exec(size);\n if (match) {\n return sizeMap[match[1]];\n }\n return sizeMap[size] || size;\n};\n\nconst resolveType = (type?: string, htmlType?: string): string | null => {\n if (htmlType) {\n return htmlType;\n }\n const legacyButtonTypes = [\n 'accent',\n 'negative',\n 'positive',\n 'primary',\n 'pay',\n 'secondary',\n 'danger',\n 'link',\n ];\n return type && legacyButtonTypes.includes(type) ? type : null;\n};\n\nconst convertEnumValue = (value?: string): string | undefined => {\n if (!value) return value;\n const strippedValue = value.replace(/^['\"]|['\"]$/gu, '');\n const enumMapping: Record<string, string> = {\n 'Priority.SECONDARY': 'secondary',\n 'Priority.PRIMARY': 'primary',\n 'Priority.TERTIARY': 'tertiary',\n 'ControlType.NEGATIVE': 'negative',\n 'ControlType.POSITIVE': 'positive',\n 'ControlType.ACCENT': 'accent',\n };\n return enumMapping[strippedValue] || strippedValue;\n};\n\n/**\n * This transform function modifies the Button and ActionButton components from the @transferwise/components library.\n * It updates the ActionButton component to use the Button component with specific attributes and mappings.\n * It also processes icon children and removes legacy props.\n *\n * @param {FileInfo} file - The file information object.\n * @param {API} api - The API object for jscodeshift.\n * @param {Options} options - The options object for jscodeshift.\n * @returns {string} - The transformed source code.\n */\nconst transformer = (file: FileInfo, api: API, options: Options) => {\n const j: JSCodeshift = api.jscodeshift;\n const root = j(file.source);\n const manualReviewIssues: string[] = [];\n const priorityMapping = buildPriorityMapping(options);\n\n const resolvePriority = (type?: string, priority?: string): string | undefined => {\n if (type && priority) {\n return priorityMapping[type]?.[priority] || priority;\n }\n return priority;\n };\n\n // Create reporter instance\n const reporter = createReporter(j, manualReviewIssues);\n\n const { exists: hasButtonImport } = hasImport(root, '@transferwise/components', 'Button', j);\n const { exists: hasActionButtonImport, remove: removeActionButtonImport } = hasImport(\n root,\n '@transferwise/components',\n 'ActionButton',\n j,\n );\n\n const iconImports = new Set<string>();\n root.find(j.ImportDeclaration, { source: { value: '@transferwise/icons' } }).forEach((path) => {\n path.node.specifiers?.forEach((specifier) => {\n if (\n (specifier.type === 'ImportDefaultSpecifier' || specifier.type === 'ImportSpecifier') &&\n specifier.local\n ) {\n const localName = (specifier.local as { name: string }).name;\n iconImports.add(localName);\n }\n });\n });\n\n if (hasActionButtonImport) {\n root.findJSXElements('ActionButton').forEach((path) => {\n const { openingElement, closingElement } = path.node;\n\n openingElement.name = setNameIfJSXIdentifier(openingElement.name, 'Button')!;\n if (closingElement) {\n closingElement.name = setNameIfJSXIdentifier(closingElement.name, 'Button')!;\n }\n\n addAttributesIfMissing(j, openingElement, [\n { attribute: j.jsxAttribute(j.jsxIdentifier('v2')), name: 'v2' },\n { attribute: j.jsxAttribute(j.jsxIdentifier('size'), j.literal('sm')), name: 'size' },\n ]);\n\n processIconChildren(j, path.node.children, iconImports, openingElement);\n\n if ((openingElement.attributes ?? []).some((attr) => attr.type === 'JSXSpreadAttribute')) {\n reporter.reportSpreadProps(path);\n }\n\n const legacyPropNames = ['priority', 'text'];\n const legacyProps: LegacyProps = {};\n\n openingElement.attributes?.forEach((attr) => {\n if (attr.type === 'JSXAttribute' && attr.name && attr.name.type === 'JSXIdentifier') {\n const { name } = attr.name;\n if (legacyPropNames.includes(name)) {\n if (attr.value) {\n if (attr.value.type === 'StringLiteral') {\n legacyProps[name] = attr.value.value;\n } else if (attr.value.type === 'JSXExpressionContainer') {\n reporter.reportAttribute(attr, path);\n }\n }\n }\n }\n });\n\n const hasTextProp = openingElement.attributes?.some(\n (attr) =>\n attr.type === 'JSXAttribute' &&\n attr.name.type === 'JSXIdentifier' &&\n attr.name.name === 'text',\n );\n const hasChildren = path.node.children?.some(\n (child) =>\n (child.type === 'JSXText' && child.value.trim() !== '') ||\n child.type === 'JSXElement' ||\n child.type === 'JSXExpressionContainer',\n );\n\n if (hasTextProp && hasChildren) {\n reporter.reportPropWithChildren(path, 'text');\n }\n\n (path.node.children || []).forEach((child) => {\n if (child.type === 'JSXExpressionContainer') {\n const expr = child.expression;\n if (\n expr.type === 'ConditionalExpression' ||\n expr.type === 'CallExpression' ||\n expr.type === 'Identifier' ||\n expr.type === 'MemberExpression'\n ) {\n reporter.reportAmbiguousChildren(path, 'icon');\n }\n }\n });\n });\n\n removeActionButtonImport();\n }\n\n if (hasButtonImport) {\n root.findJSXElements('Button').forEach((path) => {\n const { openingElement } = path.node;\n\n if (hasAttributeOnElement(openingElement, 'v2')) return;\n\n addAttributesIfMissing(j, openingElement, [\n { attribute: j.jsxAttribute(j.jsxIdentifier('v2')), name: 'v2' },\n ]);\n processIconChildren(j, path.node.children, iconImports, openingElement);\n\n const legacyProps: LegacyProps = {};\n const legacyPropNames = ['priority', 'size', 'type', 'htmlType', 'sentiment'];\n\n openingElement.attributes?.forEach((attr) => {\n if (attr.type === 'JSXAttribute' && attr.name && attr.name.type === 'JSXIdentifier') {\n const { name } = attr.name;\n if (legacyPropNames.includes(name)) {\n if (attr.value) {\n if (attr.value.type === 'StringLiteral') {\n legacyProps[name] = attr.value.value;\n } else if (attr.value.type === 'JSXExpressionContainer') {\n legacyProps[name] = convertEnumValue(String(j(attr.value.expression).toSource()));\n }\n } else {\n legacyProps[name] = undefined;\n }\n }\n }\n });\n\n if (openingElement.attributes) {\n openingElement.attributes = openingElement.attributes.filter(\n (attr) =>\n !(\n attr.type === 'JSXAttribute' &&\n attr.name &&\n legacyPropNames.includes((attr.name as JSXIdentifier).name)\n ),\n );\n }\n\n if ('size' in legacyProps) {\n const rawValue = legacyProps.size;\n const resolved = resolveSize(rawValue);\n const supportedSizes = ['xs', 'sm', 'md', 'lg', 'xl'];\n if (\n typeof rawValue === 'string' &&\n typeof resolved === 'string' &&\n supportedSizes.includes(resolved)\n ) {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('size'), j.literal(resolved)),\n );\n } else if (typeof rawValue === 'string') {\n reporter.reportUnsupportedValue(path, 'size', rawValue);\n } else if (rawValue !== undefined) {\n reporter.reportAmbiguousExpression(path, 'size');\n }\n }\n\n if ('priority' in legacyProps) {\n const rawValue = legacyProps.priority;\n const converted = convertEnumValue(rawValue);\n const mapped = resolvePriority(legacyProps.type, converted);\n const supportedPriorities = ['primary', 'secondary', 'tertiary', 'secondary-neutral'];\n if (\n typeof rawValue === 'string' &&\n typeof mapped === 'string' &&\n supportedPriorities.includes(mapped)\n ) {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('priority'), j.literal(mapped)),\n );\n } else if (typeof rawValue === 'string') {\n reporter.reportUnsupportedValue(path, 'priority', rawValue);\n } else if (rawValue !== undefined) {\n reporter.reportAmbiguousExpression(path, 'priority');\n }\n }\n\n if ('type' in legacyProps || 'htmlType' in legacyProps) {\n const rawType = legacyProps.type;\n const rawHtmlType = legacyProps.htmlType;\n const resolvedType =\n typeof rawType === 'string'\n ? rawType\n : rawType && typeof rawType === 'object'\n ? convertEnumValue(j(rawType).toSource())\n : undefined;\n const resolved = resolveType(resolvedType, rawHtmlType);\n\n const supportedTypes = [\n 'accent',\n 'negative',\n 'positive',\n 'primary',\n 'pay',\n 'secondary',\n 'danger',\n 'link',\n 'submit',\n 'button',\n 'reset',\n ];\n\n if (typeof resolved === 'string' && supportedTypes.includes(resolved)) {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('type'), j.literal(resolved)),\n );\n if (resolved === 'negative') {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('sentiment'), j.literal('negative')),\n );\n }\n } else if (typeof rawType === 'string' || typeof rawHtmlType === 'string') {\n reporter.reportUnsupportedValue(path, 'type', rawType ?? rawHtmlType ?? '');\n } else if (rawType !== undefined || rawHtmlType !== undefined) {\n reporter.reportAmbiguousExpression(path, 'type');\n }\n }\n\n if ('sentiment' in legacyProps) {\n const rawValue = legacyProps.sentiment;\n if (rawValue === 'negative') {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('sentiment'), j.literal('negative')),\n );\n } else if (typeof rawValue === 'string') {\n reporter.reportUnsupportedValue(path, 'sentiment', rawValue);\n } else if (rawValue !== undefined) {\n reporter.reportAmbiguousExpression(path, 'sentiment');\n }\n }\n\n let asIndex = -1;\n let asValue: string | null = null;\n let hrefExists = false;\n let asAmbiguous = false;\n let hrefAmbiguous = false;\n\n openingElement.attributes?.forEach((attr, index) => {\n if (attr.type === 'JSXAttribute' && attr.name) {\n if (attr.name.name === 'as') {\n if (attr.value) {\n if (attr.value.type === 'StringLiteral') {\n asValue = attr.value.value;\n } else if (attr.value.type === 'JSXExpressionContainer') {\n asAmbiguous = true;\n reporter.reportAttribute(attr, path);\n }\n }\n asIndex = index;\n }\n if (attr.name.name === 'href') {\n hrefExists = true;\n if (attr.value && attr.value.type !== 'StringLiteral') {\n hrefAmbiguous = true;\n reporter.reportAttribute(attr, path);\n }\n }\n }\n });\n\n if (asValue && asValue !== 'a') {\n reporter.reportUnsupportedValue(path, 'as', asValue);\n }\n\n if (asValue === 'a') {\n if (asIndex !== -1) {\n openingElement.attributes = openingElement.attributes?.filter(\n (_attr, idx) => idx !== asIndex,\n );\n }\n if (!hrefExists) {\n openingElement.attributes = [\n ...(openingElement.attributes ?? []),\n j.jsxAttribute(j.jsxIdentifier('href'), j.literal('#')),\n ];\n }\n }\n\n if ((openingElement.attributes ?? []).some((attr) => attr.type === 'JSXSpreadAttribute')) {\n reporter.reportSpreadProps(path);\n }\n });\n }\n\n if (manualReviewIssues.length > 0) {\n manualReviewIssues.forEach(async (issue) => {\n await reportManualReview(file.path, issue);\n });\n }\n\n return root.toSource();\n};\n\nexport default transformer;\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"transformer.js","names":["priorityMapping: Record<string, Record<string, string>>","sizeMap: Record<string, string>","j: JSCodeshift","manualReviewIssues: string[]","hasImport","legacyProps: LegacyProps","asValue: string | null","reportManualReview"],"sources":["../../../src/helpers/hasImport.ts","../../../src/helpers/iconUtils.ts","../../../src/helpers/jsxElementUtils.ts","../../../src/helpers/jsxReportingUtils.ts","../../../src/transforms/button/transformer.ts"],"sourcesContent":["import type { Collection, JSCodeshift } from 'jscodeshift';\n\n/**\n * Checks if a specific import exists in the given root collection and provides\n * a method to remove it if found.\n */\nfunction hasImport(\n root: Collection,\n sourceValue: string,\n importName: string,\n j: JSCodeshift,\n): { exists: boolean; remove: () => void } {\n const importDeclarations = root.find(j.ImportDeclaration, {\n source: { value: sourceValue },\n });\n\n if (importDeclarations.size() === 0) {\n return {\n exists: false,\n remove: () => {},\n };\n }\n\n const namedImport = importDeclarations.find(j.ImportSpecifier, {\n imported: { name: importName },\n });\n\n const defaultImport = importDeclarations.find(j.ImportDefaultSpecifier, {\n local: { name: importName },\n });\n\n const exists = namedImport.size() > 0 || defaultImport.size() > 0;\n\n const remove = () => {\n importDeclarations.forEach((path) => {\n const filteredSpecifiers =\n path.node.specifiers?.filter((specifier) => {\n if (specifier.type === 'ImportSpecifier' && specifier.imported.name === importName) {\n return false;\n }\n if (specifier.type === 'ImportDefaultSpecifier' && specifier.local?.name === importName) {\n return false;\n }\n return true;\n }) ?? [];\n\n if (filteredSpecifiers.length === 0) {\n path.prune();\n } else {\n j(path).replaceWith(\n j.importDeclaration(filteredSpecifiers, path.node.source, path.node.importKind),\n );\n }\n });\n };\n\n return { exists, remove };\n}\n\nexport default hasImport;\n","import type { JSCodeshift, JSXElement, JSXExpressionContainer } from 'jscodeshift';\n\n/**\n * Process children of a JSX element to detect icon components and add iconStart or iconEnd attributes accordingly.\n * This is specific to icon handling but can be reused in codemods dealing with icon children.\n */\nconst processIconChildren = (\n j: JSCodeshift,\n children: (JSXElement | JSXExpressionContainer | unknown)[] | undefined,\n iconImports: Set<string>,\n openingElement: JSXElement['openingElement'],\n) => {\n if (!children || !openingElement.attributes) return;\n\n const unwrapJsxElement = (node: unknown): JSXElement | unknown => {\n if (\n typeof node === 'object' &&\n node !== null &&\n 'type' in node &&\n node.type === 'JSXExpressionContainer' &&\n j.JSXElement.check((node as JSXExpressionContainer).expression)\n ) {\n return (node as JSXExpressionContainer).expression;\n }\n return node;\n };\n\n const totalChildren = children.length;\n\n // Find index of icon child\n const iconChildIndex = children.findIndex((child) => {\n const unwrapped = unwrapJsxElement(child);\n return (\n j.JSXElement.check(unwrapped) &&\n unwrapped.openingElement.name.type === 'JSXIdentifier' &&\n iconImports.has(unwrapped.openingElement.name.name)\n );\n });\n\n if (iconChildIndex === -1) return;\n\n const iconChild = unwrapJsxElement(children[iconChildIndex]) as JSXElement;\n\n if (!iconChild || iconChild.openingElement.name.type !== 'JSXIdentifier') return;\n\n const iconName = iconChild.openingElement.name.name;\n\n // Determine if icon is closer to start or end\n const distanceToStart = iconChildIndex;\n const distanceToEnd = totalChildren - 1 - iconChildIndex;\n const iconPropName = distanceToStart <= distanceToEnd ? 'addonStart' : 'addonEnd';\n\n // Build: { type: 'icon', value: <IconName /> }\n const iconObject = j.objectExpression([\n j.property('init', j.identifier('type'), j.literal('icon')),\n j.property('init', j.identifier('value'), iconChild),\n ]);\n const iconProp = j.jsxAttribute(\n j.jsxIdentifier(iconPropName),\n j.jsxExpressionContainer(iconObject),\n );\n\n openingElement.attributes.push(iconProp);\n\n // Remove the icon child\n children.splice(iconChildIndex, 1);\n\n // Helper to check if a child is whitespace-only JSXText\n const isWhitespaceJsxText = (node: unknown): boolean => {\n return (\n typeof node === 'object' &&\n node !== null &&\n (node as { type?: unknown }).type === 'JSXText' &&\n typeof (node as { value?: string }).value === 'string' &&\n (node as { value?: string }).value!.trim() === ''\n );\n };\n\n // Remove adjacent whitespace-only JSXText node if any\n if (iconChildIndex - 1 >= 0 && isWhitespaceJsxText(children[iconChildIndex - 1])) {\n children.splice(iconChildIndex - 1, 1);\n } else if (isWhitespaceJsxText(children[iconChildIndex])) {\n children.splice(iconChildIndex, 1);\n }\n};\n\nexport default processIconChildren;\n","import type {\n JSCodeshift,\n JSXAttribute,\n JSXElement,\n JSXIdentifier,\n JSXMemberExpression,\n JSXNamespacedName,\n JSXSpreadAttribute,\n} from 'jscodeshift';\n\n/**\n * Rename a JSX element name if it is a JSXIdentifier.\n */\nexport const setNameIfJSXIdentifier = (\n elementName: JSXIdentifier | JSXNamespacedName | JSXMemberExpression | undefined,\n newName: string,\n): JSXIdentifier | JSXNamespacedName | JSXMemberExpression | undefined => {\n if (elementName && elementName.type === 'JSXIdentifier') {\n return { ...elementName, name: newName };\n }\n return elementName;\n};\n\n/**\n * Check if a list of attributes contains a specific attribute by name.\n */\nexport const hasAttribute = (\n attributes: (JSXAttribute | JSXSpreadAttribute)[] | undefined,\n attributeName: string,\n): boolean => {\n return (\n Array.isArray(attributes) &&\n attributes.some(\n (attr): attr is JSXAttribute =>\n attr.type === 'JSXAttribute' &&\n attr.name.type === 'JSXIdentifier' &&\n attr.name.name === attributeName,\n )\n );\n};\n\n/**\n * Check if a JSX element's openingElement has a specific attribute.\n */\nexport const hasAttributeOnElement = (\n element: JSXElement['openingElement'],\n attributeName: string,\n): boolean => {\n return hasAttribute(element.attributes, attributeName);\n};\n\n/**\n * Add specified attributes to a JSX element's openingElement if they are not already present.\n */\nexport const addAttributesIfMissing = (\n j: JSCodeshift,\n openingElement: JSXElement['openingElement'],\n attributesToAdd: { attribute: JSXAttribute; name: string }[],\n) => {\n if (!Array.isArray(openingElement.attributes)) return;\n const attrs = openingElement.attributes;\n attributesToAdd.forEach(({ attribute, name }) => {\n if (!hasAttributeOnElement(openingElement, name)) {\n attrs.push(attribute);\n }\n });\n};\n","import type { ASTPath, JSCodeshift, JSXAttribute, JSXElement, Node } from 'jscodeshift';\n\nexport interface ReporterOptions {\n jscodeshift: JSCodeshift;\n issues: string[];\n}\n\n/**\n * CodemodReporter is a utility class for reporting issues found during codemod transformations.\n * It provides methods to report issues related to JSX elements, props, and attributes.\n *\n * @example\n * ```typescript\n * const issues: string[] = [];\n * const reporter = createReporter(j, issues);\n *\n * // Report a deprecated prop\n * reporter.reportDeprecatedProp(buttonElement, 'flat', 'variant=\"text\"');\n *\n * // Report complex expression that needs review\n * reporter.reportAmbiguousExpression(element, 'size');\n *\n * // Auto-detect common issues\n * reporter.reportAttributeIssues(element);\n * ```\n */\nexport class CodemodReporter {\n private readonly j: JSCodeshift;\n private readonly issues: string[];\n\n constructor(options: ReporterOptions) {\n this.j = options.jscodeshift;\n this.issues = options.issues;\n }\n\n /**\n * Reports an issue with a JSX element\n */\n reportElement(element: JSXElement | ASTPath<JSXElement>, reason: string): void {\n const node = this.getNode(element);\n const componentName = this.getComponentName(node);\n const line = this.getLineNumber(node);\n\n this.addIssue(`Manual review required: <${componentName}> at line ${line} ${reason}.`);\n }\n\n /**\n * Reports an issue with a specific prop\n */\n reportProp(element: JSXElement | ASTPath<JSXElement>, propName: string, reason: string): void {\n const node = this.getNode(element);\n const componentName = this.getComponentName(node);\n const line = this.getLineNumber(node);\n\n this.addIssue(\n `Manual review required: prop \"${propName}\" on <${componentName}> at line ${line} ${reason}.`,\n );\n }\n\n /**\n * Reports an issue with a JSX attribute directly\n */\n reportAttribute(\n attr: JSXAttribute,\n element: JSXElement | ASTPath<JSXElement>,\n reason?: string,\n ): void {\n const node = this.getNode(element);\n const componentName = this.getComponentName(node);\n const propName = this.getAttributeName(attr);\n const line = this.getLineNumber(attr) || this.getLineNumber(node);\n\n const defaultReason = this.getAttributeReason(attr);\n const finalReason = reason || defaultReason;\n\n this.addIssue(\n `Manual review required: prop \"${propName}\" on <${componentName}> at line ${line} ${finalReason}.`,\n );\n }\n\n /**\n * Reports spread props on an element\n */\n reportSpreadProps(element: JSXElement | ASTPath<JSXElement>): void {\n this.reportElement(element, 'contains spread props that need manual review');\n }\n\n /**\n * Reports conflicting prop and children\n */\n reportPropWithChildren(element: JSXElement | ASTPath<JSXElement>, propName: string): void {\n this.reportProp(\n element,\n propName,\n `conflicts with children - both \"${propName}\" prop and children are present`,\n );\n }\n\n /**\n * Reports unsupported prop value\n */\n reportUnsupportedValue(\n element: JSXElement | ASTPath<JSXElement>,\n propName: string,\n value: string,\n ): void {\n this.reportProp(element, propName, `has unsupported value \"${value}\"`);\n }\n\n /**\n * Reports ambiguous expression in prop\n */\n reportAmbiguousExpression(element: JSXElement | ASTPath<JSXElement>, propName: string): void {\n this.reportProp(element, propName, 'contains a complex expression that needs manual review');\n }\n\n /**\n * Reports ambiguous children (like dynamic icons)\n */\n reportAmbiguousChildren(element: JSXElement | ASTPath<JSXElement>, childType = 'content'): void {\n this.reportElement(element, `contains ambiguous ${childType} that needs manual review`);\n }\n\n /**\n * Reports deprecated prop usage\n */\n reportDeprecatedProp(\n element: JSXElement | ASTPath<JSXElement>,\n propName: string,\n alternative?: string,\n ): void {\n const suggestion = alternative ? ` Use ${alternative} instead` : '';\n this.reportProp(element, propName, `is deprecated${suggestion}`);\n }\n\n /**\n * Reports missing required prop\n */\n reportMissingRequiredProp(element: JSXElement | ASTPath<JSXElement>, propName: string): void {\n this.reportProp(element, propName, 'is required but missing');\n }\n\n /**\n * Reports conflicting props\n */\n reportConflictingProps(element: JSXElement | ASTPath<JSXElement>, propNames: string[]): void {\n const propList = propNames.map((name) => `\"${name}\"`).join(', ');\n this.reportElement(element, `has conflicting props: ${propList} cannot be used together`);\n }\n\n /**\n * Auto-detects and reports common attribute issues\n */\n reportAttributeIssues(element: JSXElement | ASTPath<JSXElement>): void {\n const node = this.getNode(element);\n const { attributes } = node.openingElement;\n\n if (!attributes) return;\n\n // Check for spread props\n if (attributes.some((attr) => attr.type === 'JSXSpreadAttribute')) {\n this.reportSpreadProps(element);\n }\n\n // Check for complex expressions in attributes\n attributes.forEach((attr) => {\n if (attr.type === 'JSXAttribute' && attr.value?.type === 'JSXExpressionContainer') {\n this.reportAttribute(attr, element);\n }\n });\n }\n\n // Private helper methods\n private getNode(element: JSXElement | ASTPath<JSXElement>): JSXElement {\n return 'node' in element ? element.node : element;\n }\n\n private getComponentName(node: JSXElement): string {\n const { name } = node.openingElement;\n if (name.type === 'JSXIdentifier') {\n return name.name;\n }\n // Handle JSXMemberExpression, JSXNamespacedName, etc.\n return this.j(name).toSource();\n }\n\n private getLineNumber(node: JSXElement | JSXAttribute | Node): string {\n return node.loc?.start.line?.toString() || 'unknown';\n }\n\n private getAttributeName(attr: JSXAttribute): string {\n if (attr.name.type === 'JSXIdentifier') {\n return attr.name.name;\n }\n return this.j(attr.name).toSource();\n }\n\n private getAttributeReason(attr: JSXAttribute): string {\n if (!attr.value) return 'has no value';\n\n if (attr.value.type === 'JSXExpressionContainer') {\n const expr = attr.value.expression;\n const expressionType = expr.type.replace('Expression', '').toLowerCase();\n\n // Show actual value for simple cases\n if (expr.type === 'Identifier' || expr.type === 'MemberExpression') {\n const valueText = this.j(expr).toSource();\n return `contains a ${expressionType} (${valueText})`;\n }\n\n return `contains a complex ${expressionType} expression`;\n }\n\n return 'needs manual review';\n }\n\n private addIssue(message: string): void {\n this.issues.push(message);\n }\n}\n\nexport const createReporter = (j: JSCodeshift, issues: string[]): CodemodReporter => {\n return new CodemodReporter({ jscodeshift: j, issues });\n};\n","import type { API, FileInfo, JSCodeshift, JSXIdentifier, Options } from 'jscodeshift';\n\nimport reportManualReview from '../../controller/helpers/reportManualReview';\nimport {\n addAttributesIfMissing,\n createReporter,\n hasAttributeOnElement,\n hasImport,\n processIconChildren,\n setNameIfJSXIdentifier,\n} from '../../helpers';\n\nexport const parser = 'tsx';\n\ninterface LegacyProps {\n priority?: string;\n size?: string;\n type?: string;\n htmlType?: string;\n sentiment?: string;\n [key: string]: unknown;\n}\n\nconst priorityMapping: Record<string, Record<string, string>> = {\n accent: {\n primary: 'primary',\n secondary: 'secondary-neutral',\n tertiary: 'tertiary',\n },\n positive: {\n primary: 'primary',\n secondary: 'secondary-neutral',\n tertiary: 'secondary-neutral',\n },\n negative: {\n primary: 'primary',\n secondary: 'secondary',\n tertiary: 'secondary',\n },\n};\n\nconst sizeMap: Record<string, string> = {\n EXTRA_SMALL: 'xs',\n SMALL: 'sm',\n MEDIUM: 'md',\n LARGE: 'lg',\n EXTRA_LARGE: 'xl',\n xs: 'sm',\n sm: 'sm',\n md: 'md',\n lg: 'lg',\n xl: 'xl',\n};\n\nconst resolveSize = (size?: string): string | undefined => {\n if (!size) return size;\n const match = /^Size\\.(EXTRA_SMALL|SMALL|MEDIUM|LARGE|EXTRA_LARGE)$/u.exec(size);\n if (match) {\n return sizeMap[match[1]];\n }\n return sizeMap[size] || size;\n};\n\nconst resolvePriority = (type?: string, priority?: string): string | undefined => {\n if (type && priority) {\n return priorityMapping[type]?.[priority] || priority;\n }\n return priority;\n};\n\nconst resolveType = (type?: string, htmlType?: string): string | null => {\n if (htmlType) {\n return htmlType;\n }\n\n const legacyButtonTypes = [\n 'accent',\n 'negative',\n 'positive',\n 'primary',\n 'pay',\n 'secondary',\n 'danger',\n 'link',\n ];\n return type && legacyButtonTypes.includes(type) ? type : null;\n};\n\nconst convertEnumValue = (value?: string): string | undefined => {\n if (!value) return value;\n const strippedValue = value.replace(/^['\"]|['\"]$/gu, '');\n const enumMapping: Record<string, string> = {\n 'Priority.SECONDARY': 'secondary',\n 'Priority.PRIMARY': 'primary',\n 'Priority.TERTIARY': 'tertiary',\n 'ControlType.NEGATIVE': 'negative',\n 'ControlType.POSITIVE': 'positive',\n 'ControlType.ACCENT': 'accent',\n };\n return enumMapping[strippedValue] || strippedValue;\n};\n\n/**\n * This transform function modifies the Button and ActionButton components from the @transferwise/components library.\n * It updates the ActionButton component to use the Button component with specific attributes and mappings.\n * It also processes icon children and removes legacy props.\n *\n * @param {FileInfo} file - The file information object.\n * @param {API} api - The API object for jscodeshift.\n * @param {Options} options - The options object for jscodeshift.\n * @returns {string} - The transformed source code.\n */\nconst transformer = (file: FileInfo, api: API, options: Options) => {\n const j: JSCodeshift = api.jscodeshift;\n const root = j(file.source);\n const manualReviewIssues: string[] = [];\n\n // Create reporter instance\n const reporter = createReporter(j, manualReviewIssues);\n\n const { exists: hasButtonImport } = hasImport(root, '@transferwise/components', 'Button', j);\n const { exists: hasActionButtonImport, remove: removeActionButtonImport } = hasImport(\n root,\n '@transferwise/components',\n 'ActionButton',\n j,\n );\n\n const iconImports = new Set<string>();\n root.find(j.ImportDeclaration, { source: { value: '@transferwise/icons' } }).forEach((path) => {\n path.node.specifiers?.forEach((specifier) => {\n if (\n (specifier.type === 'ImportDefaultSpecifier' || specifier.type === 'ImportSpecifier') &&\n specifier.local\n ) {\n const localName = (specifier.local as { name: string }).name;\n iconImports.add(localName);\n }\n });\n });\n\n if (hasActionButtonImport) {\n root.findJSXElements('ActionButton').forEach((path) => {\n const { openingElement, closingElement } = path.node;\n\n openingElement.name = setNameIfJSXIdentifier(openingElement.name, 'Button')!;\n if (closingElement) {\n closingElement.name = setNameIfJSXIdentifier(closingElement.name, 'Button')!;\n }\n\n addAttributesIfMissing(j, openingElement, [\n { attribute: j.jsxAttribute(j.jsxIdentifier('v2')), name: 'v2' },\n { attribute: j.jsxAttribute(j.jsxIdentifier('size'), j.literal('sm')), name: 'size' },\n ]);\n\n processIconChildren(j, path.node.children, iconImports, openingElement);\n\n if ((openingElement.attributes ?? []).some((attr) => attr.type === 'JSXSpreadAttribute')) {\n reporter.reportSpreadProps(path);\n }\n\n const legacyPropNames = ['priority', 'text'];\n const legacyProps: LegacyProps = {};\n\n openingElement.attributes?.forEach((attr) => {\n if (attr.type === 'JSXAttribute' && attr.name && attr.name.type === 'JSXIdentifier') {\n const { name } = attr.name;\n if (legacyPropNames.includes(name)) {\n if (attr.value) {\n if (attr.value.type === 'StringLiteral') {\n legacyProps[name] = attr.value.value;\n } else if (attr.value.type === 'JSXExpressionContainer') {\n reporter.reportAttribute(attr, path);\n }\n }\n }\n }\n });\n\n const hasTextProp = openingElement.attributes?.some(\n (attr) =>\n attr.type === 'JSXAttribute' &&\n attr.name.type === 'JSXIdentifier' &&\n attr.name.name === 'text',\n );\n const hasChildren = path.node.children?.some(\n (child) =>\n (child.type === 'JSXText' && child.value.trim() !== '') ||\n child.type === 'JSXElement' ||\n child.type === 'JSXExpressionContainer',\n );\n\n if (hasTextProp && hasChildren) {\n reporter.reportPropWithChildren(path, 'text');\n }\n\n (path.node.children || []).forEach((child) => {\n if (child.type === 'JSXExpressionContainer') {\n const expr = child.expression;\n if (\n expr.type === 'ConditionalExpression' ||\n expr.type === 'CallExpression' ||\n expr.type === 'Identifier' ||\n expr.type === 'MemberExpression'\n ) {\n reporter.reportAmbiguousChildren(path, 'icon');\n }\n }\n });\n });\n\n removeActionButtonImport();\n }\n\n if (hasButtonImport) {\n root.findJSXElements('Button').forEach((path) => {\n const { openingElement } = path.node;\n\n if (hasAttributeOnElement(openingElement, 'v2')) return;\n\n addAttributesIfMissing(j, openingElement, [\n { attribute: j.jsxAttribute(j.jsxIdentifier('v2')), name: 'v2' },\n ]);\n processIconChildren(j, path.node.children, iconImports, openingElement);\n\n const legacyProps: LegacyProps = {};\n const legacyPropNames = ['priority', 'size', 'type', 'htmlType', 'sentiment'];\n\n openingElement.attributes?.forEach((attr) => {\n if (attr.type === 'JSXAttribute' && attr.name && attr.name.type === 'JSXIdentifier') {\n const { name } = attr.name;\n if (legacyPropNames.includes(name)) {\n if (attr.value) {\n if (attr.value.type === 'StringLiteral') {\n legacyProps[name] = attr.value.value;\n } else if (attr.value.type === 'JSXExpressionContainer') {\n legacyProps[name] = convertEnumValue(String(j(attr.value.expression).toSource()));\n }\n } else {\n legacyProps[name] = undefined;\n }\n }\n }\n });\n\n if (openingElement.attributes) {\n openingElement.attributes = openingElement.attributes.filter(\n (attr) =>\n !(\n attr.type === 'JSXAttribute' &&\n attr.name &&\n legacyPropNames.includes((attr.name as JSXIdentifier).name)\n ),\n );\n }\n\n if ('size' in legacyProps) {\n const rawValue = legacyProps.size;\n const resolved = resolveSize(rawValue);\n const supportedSizes = ['xs', 'sm', 'md', 'lg', 'xl'];\n\n if (\n typeof rawValue === 'string' &&\n typeof resolved === 'string' &&\n supportedSizes.includes(resolved)\n ) {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('size'), j.literal(resolved)),\n );\n } else if (typeof rawValue === 'string') {\n reporter.reportUnsupportedValue(path, 'size', rawValue);\n } else if (rawValue !== undefined) {\n reporter.reportAmbiguousExpression(path, 'size');\n }\n }\n\n if ('priority' in legacyProps) {\n const rawValue = legacyProps.priority;\n const converted = convertEnumValue(rawValue);\n const mapped = resolvePriority(legacyProps.type, converted);\n const supportedPriorities = ['primary', 'secondary', 'tertiary', 'secondary-neutral'];\n\n if (\n typeof rawValue === 'string' &&\n typeof mapped === 'string' &&\n supportedPriorities.includes(mapped)\n ) {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('priority'), j.literal(mapped)),\n );\n } else if (typeof rawValue === 'string') {\n reporter.reportUnsupportedValue(path, 'priority', rawValue);\n } else if (rawValue !== undefined) {\n reporter.reportAmbiguousExpression(path, 'priority');\n }\n }\n\n if ('type' in legacyProps || 'htmlType' in legacyProps) {\n const rawType = legacyProps.type;\n const rawHtmlType = legacyProps.htmlType;\n\n const resolvedType =\n typeof rawType === 'string'\n ? rawType\n : rawType && typeof rawType === 'object'\n ? convertEnumValue(j(rawType).toSource())\n : undefined;\n\n const resolved = resolveType(resolvedType, rawHtmlType);\n\n const supportedTypes = [\n 'accent',\n 'negative',\n 'positive',\n 'primary',\n 'pay',\n 'secondary',\n 'danger',\n 'link',\n 'submit',\n 'button',\n 'reset',\n ];\n\n if (typeof resolved === 'string' && supportedTypes.includes(resolved)) {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('type'), j.literal(resolved)),\n );\n\n if (resolved === 'negative') {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('sentiment'), j.literal('negative')),\n );\n }\n } else if (typeof rawType === 'string' || typeof rawHtmlType === 'string') {\n reporter.reportUnsupportedValue(path, 'type', rawType ?? rawHtmlType ?? '');\n } else if (rawType !== undefined || rawHtmlType !== undefined) {\n reporter.reportAmbiguousExpression(path, 'type');\n }\n }\n\n if ('sentiment' in legacyProps) {\n const rawValue = legacyProps.sentiment;\n if (rawValue === 'negative') {\n openingElement.attributes?.push(\n j.jsxAttribute(j.jsxIdentifier('sentiment'), j.literal('negative')),\n );\n } else if (typeof rawValue === 'string') {\n reporter.reportUnsupportedValue(path, 'sentiment', rawValue);\n } else if (rawValue !== undefined) {\n reporter.reportAmbiguousExpression(path, 'sentiment');\n }\n }\n\n let asIndex = -1;\n let asValue: string | null = null;\n let hrefExists = false;\n let asAmbiguous = false;\n let hrefAmbiguous = false;\n\n openingElement.attributes?.forEach((attr, index) => {\n if (attr.type === 'JSXAttribute' && attr.name) {\n if (attr.name.name === 'as') {\n if (attr.value) {\n if (attr.value.type === 'StringLiteral') {\n asValue = attr.value.value;\n } else if (attr.value.type === 'JSXExpressionContainer') {\n asAmbiguous = true;\n reporter.reportAttribute(attr, path);\n }\n }\n asIndex = index;\n }\n\n if (attr.name.name === 'href') {\n hrefExists = true;\n if (attr.value && attr.value.type !== 'StringLiteral') {\n hrefAmbiguous = true;\n reporter.reportAttribute(attr, path);\n }\n }\n }\n });\n\n if (asValue && asValue !== 'a') {\n reporter.reportUnsupportedValue(path, 'as', asValue);\n }\n\n if (asValue === 'a') {\n if (asIndex !== -1) {\n openingElement.attributes = openingElement.attributes?.filter(\n (_, idx) => idx !== asIndex,\n );\n }\n if (!hrefExists) {\n openingElement.attributes = [\n ...(openingElement.attributes ?? []),\n j.jsxAttribute(j.jsxIdentifier('href'), j.literal('#')),\n ];\n }\n }\n\n if ((openingElement.attributes ?? []).some((attr) => attr.type === 'JSXSpreadAttribute')) {\n reporter.reportSpreadProps(path);\n }\n });\n }\n\n if (manualReviewIssues.length > 0) {\n manualReviewIssues.forEach(async (issue) => {\n await reportManualReview(file.path, issue);\n });\n }\n\n return root.toSource();\n};\n\nexport default transformer;\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@wise/wds-codemods",
|
|
3
|
-
"version": "0.0.1-experimental-
|
|
3
|
+
"version": "0.0.1-experimental-42663ed",
|
|
4
4
|
"license": "UNLICENSED",
|
|
5
5
|
"author": "Wise Payments Ltd.",
|
|
6
6
|
"repository": {
|
|
@@ -56,7 +56,7 @@
|
|
|
56
56
|
"@wise/eslint-config": "^13.1.0",
|
|
57
57
|
"babel-jest": "^30.1.2",
|
|
58
58
|
"babel-plugin-transform-import-meta": "^2.3.3",
|
|
59
|
-
"eslint": "^9.
|
|
59
|
+
"eslint": "^9.35.0",
|
|
60
60
|
"husky": "^9.1.7",
|
|
61
61
|
"jest": "^30.1.3",
|
|
62
62
|
"jest-file-snapshot": "^0.7.0",
|